2010-05-20 36 views
2

我試圖編寫一個允許「現金抽屜」附件的銷售點系統。代碼在打開錢箱的手冊中提供(使用IOCTL在C++中)。由於我使用C#.NET進行編碼,是否可以在C#中執行類似的操作,還是必須編寫一些非託管代碼?C# - 可以使用IOCTL

我能從C#中獲得「\\。\ ADVANSYS」的句柄嗎?我需要使用DLLImport嗎?

如果有人能指出我正確的方向,我將不勝感激。

// IOCTL Codes 
#define GPD_TYPE 56053 
#define ADV_OPEN_CTL_CODE CTL_CODE(GPD_TYPE, 0x920, METHOD_BUFFERED, FILE_ANY_ACCESS) 
#define ADV_STATUS_CTL_CODE CTL_CODE(GPD_TYPE, 0x900, METHOD_BUFFERED, FILE_ANY_ACCESS) 
void OpenDrawer(UCHAR uWhichDrawer) // uWhichDrawer = 1 => CD#1, uWhichDrawer = 2 => CD#2 
{ 
    HANDLE hFile; 
    BOOL bRet 
    UCHAR uDrawer = uWhichDrawer; 

    // Open the driver 
    hFile = CreateFile(TEXT("\\\\.\\ADVSYS"), 
    GENERIC_WRITE | GENERIC_READ, 
    FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, 
    OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); 

    if (m_hFile == INVALID_HANDLE_VALUE) 
    { 
     AfxMessageBox("Unable to open Cash Drawer Device Driver!"); 
     return; 
    } 

    // Turn on the Cash Drawer Output (Fire the required solenoid) 
    bRet = DeviceIoControl(hFile, ADV_CD_OPEN_CTL_CODE, 
    &uDrawer, sizeof(uDrawer), 
    NULL, 0, 
    &ulBytesReturned, NULL); 

    if (bRet == FALSE || ulBytesReturned != 1) 
    { 
     AfxMessageBox("Failed to write to cash drawer driver"); 
     CloseHandle(hFile); 
     return; 
    } 
    CloseHandle(hFile); 
} 

回答

5

C++充滿了錯誤,不知道我是否正確。最好的辦法是用改變的參數類型聲明DeviceIoControl(),以便調用。您還必須P/Invoke CreateFile,因爲FileStream無法打開設備。它應該與此類似:

using System; 
using System.IO; 
using System.ComponentModel; 
using System.Runtime.InteropServices; 

class Program { 
    static void Main(string[] args) { 
     IntPtr hdl = CreateFile("\\\\.\\ADVSYS", FileAccess.ReadWrite, 
      FileShare.None, IntPtr.Zero, FileMode.Open, 
      FileOptions.None, IntPtr.Zero); 
     if (hdl == (IntPtr)(-1)) throw new Win32Exception(); 
     try { 
      byte drawer = 1; 
      bool ok = DeviceIoControl(hdl, CTLCODE, ref drawer, 1, IntPtr.Zero, 
       0, IntPtr.Zero, IntPtr.Zero); 
      if (!ok) throw new Win32Exception(); 
     } 
     finally { 
      CloseHandle(hdl); 
     } 
    } 
    // P/Invoke: 
    private const uint CTLCODE = 0xdaf52480; 
    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)] 
    private static extern IntPtr CreateFile(string filename, FileAccess access, 
      FileShare sharing, IntPtr SecurityAttributes, FileMode mode, 
      FileOptions options, IntPtr template 
    ); 
    [DllImport("kernel32.dll", SetLastError = true)] 
    private static extern bool DeviceIoControl(IntPtr device, uint ctlcode, 
     ref byte inbuffer, int inbuffersize, 
     IntPtr outbuffer, int outbufferSize, 
     IntPtr bytesreturned, IntPtr overlapped 
    ); 
    [DllImport("kernel32.dll")] 
    private static extern void CloseHandle(IntPtr hdl); 
} 
+0

感謝Hans Passant。我現在可以成功打開設備驅動程序,但調用DeviceIoControl會產生「Win32Exception:連接到系統的設備無法運行。」我相信它可能是傳入的變量之一(或類型)或緩衝區的長度。我會繼續看看我能否實現它。 – theblip 2010-05-24 00:27:29

+0

知道現在所有的工作。謝謝你的幫助。看來,我的文檔中指定的GPD_TYPE 56053不正確。 – theblip 2010-05-24 12:15:00

2

您可以使用Pinvoke;

[return: MarshalAs(UnmanagedType.Bool)] 
[DllImport("kernel32.dll", CharSet=CharSet.Unicode, SetLastError=true)] 
internal static extern bool DeviceIoControl([In] SafeFileHandle hDevice, [In] int dwIoControlCode, [In] IntPtr lpInBuffer, [In] int nInBufferSize, [Out] IntPtr lpOutBuffer, [In] int nOutBufferSize, out int lpBytesReturned, [In] IntPtr lpOverlapped); 

This example也可能有所幫助。

1

你寫道:

看來,該GPD_TYPE 56053 如文檔中指定 我是不正確的

您可以發佈適當的值GPD_TYPE

最佳regadrs!