12

私は、基本的にコンピューター上の印刷アクティビティを監視する codeproject のプロジェクトで遊んでいました。ただし、64 ビット構成では正しく機能しません。コードの以下の部分が問題でした。このコードは、印刷が完了するたびに呼び出されます。

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));
}

デバッグは、data[i].field 値が常に 0 であることを示しています。ただし、32 ビットでは正しく動作します。PRINTER_NOTIFY_INFO_DATA が正しく定義されていないと思います。現在、次のコードを使用しています。これを 64 ビットでも正しく動作するように修正できる人はいますか?

[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;
}

MS XPS ドライバーを使用して印刷をテストしていました。コードプロジェクトの記事はこちら

4

2 に答える 2

16

データアライメントのため、64ビット構成では正しく機能しません。

したがって、PRINTER_NOTIFY_INFOを次のように変更することをお勧めします。

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

そして、Marshal.OffsetOfMarshal.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));
}
于 2012-10-09T08:07:35.430 に答える
0

もう 1 つの解決策は、作成後に 4 バイトを追加することです。PRINTER_NOTIFY_INFO は、データの配置を埋めるために使用されます (32int は 64 ビットの半分にすぎません) + 4 バイトで 64 ビットになります。

また、注意してください: PRINTER_NOTIFY_INFO_DATA_DATA => IntPtr pBuf; は inptr であるため、Windows 64 ビットでは 64 ビットです。これにより、このクラス全体が 128 ビット相当のデータになります。(したがって、ここでも 4 バイトを追加することを忘れないでください。

予約済みまたは Id にも 4 バイト余分にあるように見えるので、最後に info_data ごとに 8 バイトを追加して機能させました。

于 2018-10-29T20:25:38.410 に答える