2012-10-09 6 views
9

Ich spielte mit einem Projekt von Codeprojekt, das im Grunde die Druckaktivität auf dem Computer überwacht. Es funktioniert jedoch nicht richtig für 64-Bit-Konfiguration. Der folgende Teil des Codes war das Problem. Dieser Code wird immer dann aufgerufen, wenn gedruckt wird.Wie definiert man PRINT_NOTIFY_INFO_DATA korrekt?

PRINTER_NOTIFY_INFO info = (PRINTER_NOTIFY_INFO)Marshal.PtrToStructure(pNotifyInfo, typeof(PRINTER_NOTIFY_INFO));       
int pData = (int)pNotifyInfo + Marshal.SizeOf(typeof(PRINTER_NOTIFY_INFO)); 
PRINTER_NOTIFY_INFO_DATA[] data = new PRINTER_NOTIFY_INFO_DATA[info.Count]; 
for (uint i = 0; i < info.Count; i++) 
{ 
    data[i] = (PRINTER_NOTIFY_INFO_DATA)Marshal.PtrToStructure((IntPtr)pData, typeof(PRINTER_NOTIFY_INFO_DATA)); 
    pData += Marshal.SizeOf(typeof(PRINTER_NOTIFY_INFO_DATA)); 
} 

Debugging zeigt, dass der Datenwert [i] .field immer 0 ist. In 32 Bit funktioniert es jedoch korrekt. Ich denke, die PRINTER_NOTIFY_INFO_DATA ist nicht korrekt definiert. Derzeit verwende ich den folgenden Code. Kann jemand das reparieren, um in 64 Bit auch richtig zu arbeiten?

[StructLayout(LayoutKind.Sequential)] 
public struct PRINTER_NOTIFY_INFO 
{ 
    public uint Version; 
    public uint Flags; 
    public uint Count; 
} 


[StructLayout(LayoutKind.Sequential)] 
public struct PRINTER_NOTIFY_INFO_DATA_DATA 
{ 
    public uint cbBuf; 
    public IntPtr pBuf; 
} 

[StructLayout(LayoutKind.Explicit)] 
public struct PRINTER_NOTIFY_INFO_DATA_UNION 
{ 
    [FieldOffset(0)] 
    private uint adwData0; 
    [FieldOffset(4)] 
    private uint adwData1; 
    [FieldOffset(0)] 
    public PRINTER_NOTIFY_INFO_DATA_DATA Data; 
    public uint[] adwData 
    { 
     get 
     { 
      return new uint[] { this.adwData0, this.adwData1 }; 
     } 
    } 
} 

// Structure borrowed from http://lifeandtimesofadeveloper.blogspot.com/2007/10/unmanaged-structures-padding-and-c-part_18.html. 
[StructLayout(LayoutKind.Sequential)] 
public struct PRINTER_NOTIFY_INFO_DATA 
{ 
    public ushort Type; 
    public ushort Field; 
    public uint Reserved; 
    public uint Id; 
    public PRINTER_NOTIFY_INFO_DATA_UNION NotifyData; 
} 

Ich habe Drucken mit dem MS XPS-Treiber getestet. Der Code Projektartikel ist Here

Antwort

14

Es funktioniert nicht richtig für 64-Bit-Konfiguration wegen Data Alignment. So

Ich schlage vor, Sie PRINTER_NOTIFY_INFO wie folgt zu ändern:

[StructLayout(LayoutKind.Sequential)] 
public struct PRINTER_NOTIFY_INFO 
{ 
    public uint Version; 
    public uint Flags; 
    public uint Count; 
    public PRINTER_NOTIFY_INFO_DATA_UNION aData; 
} 

Und dann verwenden Marshal.OffsetOf statt Marshal.SizeOf:

PRINTER_NOTIFY_INFO info = (PRINTER_NOTIFY_INFO)Marshal.PtrToStructure(pNotifyInfo, typeof(PRINTER_NOTIFY_INFO));       
long pData = (long)pNotifyInfo + (long)Marshal.OffsetOf(typeof(PRINTER_NOTIFY_INFO), "aData"); 
PRINTER_NOTIFY_INFO_DATA[] data = new PRINTER_NOTIFY_INFO_DATA[info.Count]; 
for (uint i = 0; i < info.Count; i++) 
{ 
    data[i] = (PRINTER_NOTIFY_INFO_DATA)Marshal.PtrToStructure((IntPtr)pData, typeof(PRINTER_NOTIFY_INFO_DATA)); 
    pData += (long)Marshal.SizeOf(typeof(PRINTER_NOTIFY_INFO_DATA)); 
} 
+0

Sie mir das Leben gerettet, thanx eine Million. –

+0

Stellen Sie außerdem sicher, dass die beteiligten Projekte zum Erstellen von ausführbaren x86-Dateien konfiguriert sind. –

+0

Einige Dinge zu beachten. 1) Marshal.OffsetOf() gibt ein IntPtr-Objekt zurück, das meines Wissens nicht direkt zu einem Long-Objekt umgewandelt werden kann. Sie müssen die IntPtr.ToInt32() -Methode verwenden. 2) Dieser Code kann Ihren Code möglicherweise zum Absturz bringen, wenn PRINTER_NOTIFY_INFO.Count 0 ist. Dies würde bedeuten, dass keine Strukturen in der aData [] - Liste vorhanden sind. Indem Sie es auf die beschriebene Weise deklarieren, garantieren Sie mindestens 1 Objekt im Array. Wenn count 0 ist, ist das ein potenzieller Segmentierungsfehler. Ich würde stattdessen aData als IntPtr deklarieren, da es im unverwalteten Code sowieso nur einen Zeiger gibt. – Ultratrunks