// // UIOTEST.C // // Test program for ndisuio.sys // // usage: UIOTEST [options] // // options: // -e: Enumerate devices // -r: Read // -w: Write (default) // -l : length of each packet (default: %d)\n", PacketLength // -n : number of packets (defaults to infinity) // -m (defaults to local MAC) // #pragma warning(disable:4201) // nameless struct/union #pragma warning(disable:4127) // conditional expression is constant #include #include #include #include #include #include #include #include #include #include #include #include "nuiouser.h" #ifndef NDIS_STATUS #define NDIS_STATUS ULONG #endif #if DBG #define DEBUGP(stmt) printf stmt #else #define DEBUGP(stmt) #endif #define PRINTF(stmt) printf stmt #ifndef MAC_ADDR_LEN #define MAC_ADDR_LEN 6 #endif #define MAX_NDIS_DEVICE_NAME_LEN 256 CHAR NdisuioDevice[] = "\\\\.\\\\Ndisuio"; CHAR * pNdisuioDevice = &NdisuioDevice[0]; BOOLEAN DoEnumerate = FALSE; BOOLEAN DoReads = FALSE; INT NumberOfPackets = -1; ULONG PacketLength = 100; UCHAR SrcMacAddr[MAC_ADDR_LEN]; UCHAR DstMacAddr[MAC_ADDR_LEN]; BOOLEAN bDstMacSpecified = FALSE; CHAR * pNdisDeviceName = "JUNK"; USHORT EthType = 0x8e88; BOOLEAN bUseFakeAddress = FALSE; UCHAR FakeSrcMacAddr[MAC_ADDR_LEN] = {0}; #include typedef struct _ETH_HEADER { UCHAR DstAddr[MAC_ADDR_LEN]; UCHAR SrcAddr[MAC_ADDR_LEN]; USHORT EthType; } ETH_HEADER, *PETH_HEADER; #include VOID PrintUsage() { PRINTF(("usage: UIOTEST [options] \n")); PRINTF(("options:\n")); PRINTF((" -e: Enumerate devices\n")); PRINTF((" -r: Read\n")); PRINTF((" -w: Write (default)\n")); PRINTF((" -l : length of each packet (default: %d)\n", PacketLength)); PRINTF((" -n : number of packets (defaults to infinity)\n")); PRINTF((" -m (defaults to local MAC)\n")); PRINTF((" -f Use a fake address to send out the packets.\n")); } BOOL GetOptions( INT argc, CHAR *argv[] ) { BOOL bOkay; INT i, j, increment; CHAR *pOption; ULONG DstMacAddrUlong[MAC_ADDR_LEN]; INT RetVal; bOkay = TRUE; do { if (argc < 2) { PRINTF(("Missing argument\n")); bOkay = FALSE; break; } i = 1; while (i < argc) { increment = 1; pOption = argv[i]; if ((*pOption == '-') || (*pOption == '/')) { pOption++; if (*pOption == '\0') { DEBUGP(("Badly formed option\n")); return (FALSE); } } else { break; } switch (*pOption) { case 'e': DoEnumerate = TRUE; break; case 'f': bUseFakeAddress = TRUE; break; case 'r': DoReads = TRUE; break; case 'w': DoReads = FALSE; break; case 'l': if (i+1 < argc-1) { RetVal = atoi(argv[i+1]); if (RetVal != 0) { PacketLength = RetVal; DEBUGP((" Option: PacketLength = %d\n", PacketLength)); increment = 2; break; } } PRINTF(("Option l needs PacketLength parameter\n")); return (FALSE); case 'n': if (i+1 < argc-1) { RetVal = atoi(argv[i+1]); if (RetVal != 0) { NumberOfPackets = RetVal; DEBUGP((" Option: NumberOfPackets = %d\n", NumberOfPackets)); increment = 2; break; } } PRINTF(("Option n needs NumberOfPackets parameter\n")); return (FALSE); case 'm': if (i+1 < argc-1) { RetVal = sscanf(argv[i+1], "%2x:%2x:%2x:%2x:%2x:%2x", &DstMacAddrUlong[0], &DstMacAddrUlong[1], &DstMacAddrUlong[2], &DstMacAddrUlong[3], &DstMacAddrUlong[4], &DstMacAddrUlong[5]); if (RetVal == 6) { for (j = 0; j < MAC_ADDR_LEN; j++) { DstMacAddr[j] = (UCHAR)DstMacAddrUlong[j]; } DEBUGP((" Option: Dest MAC Addr: %02x:%02x:%02x:%02x:%02x:%02x\n", DstMacAddr[0], DstMacAddr[1], DstMacAddr[2], DstMacAddr[3], DstMacAddr[4], DstMacAddr[5])); bDstMacSpecified = TRUE; increment = 2; break; } } PRINTF(("Option m needs MAC address parameter\n")); return (FALSE); case '?': return (FALSE); default: PRINTF(("Unknown option %c\n", *pOption)); return (FALSE); } i+= increment; } pNdisDeviceName = argv[argc-1]; } while (FALSE); return (bOkay); } HANDLE OpenHandle( CHAR *pDeviceName ) { DWORD DesiredAccess; DWORD ShareMode; LPSECURITY_ATTRIBUTES lpSecurityAttributes = NULL; DWORD CreationDistribution; DWORD FlagsAndAttributes; HANDLE TemplateFile; HANDLE Handle; DWORD BytesReturned; DesiredAccess = GENERIC_READ|GENERIC_WRITE; ShareMode = 0; CreationDistribution = OPEN_EXISTING; FlagsAndAttributes = FILE_ATTRIBUTE_NORMAL; TemplateFile = (HANDLE)INVALID_HANDLE_VALUE; Handle = CreateFile( pDeviceName, DesiredAccess, ShareMode, lpSecurityAttributes, CreationDistribution, FlagsAndAttributes, TemplateFile ); if (Handle == INVALID_HANDLE_VALUE) { DEBUGP(("Creating file failed, error %x\n", GetLastError())); return Handle; } // // Wait for the driver to finish binding. // if (!DeviceIoControl( Handle, IOCTL_NDISUIO_BIND_WAIT, NULL, 0, NULL, 0, &BytesReturned, NULL)) { DEBUGP(("IOCTL_NDISIO_BIND_WAIT failed, error %x\n", GetLastError())); CloseHandle(Handle); Handle = INVALID_HANDLE_VALUE; } return (Handle); } BOOL OpenNdisDevice( HANDLE Handle, CHAR *pDeviceName ) { WCHAR wNdisDeviceName[MAX_NDIS_DEVICE_NAME_LEN]; INT wNameLength; INT NameLength = strlen(pDeviceName); DWORD BytesReturned; INT i; // // Convert to unicode string - non-localized... // wNameLength = 0; for (i = 0; i < NameLength && i < MAX_NDIS_DEVICE_NAME_LEN-1; i++) { wNdisDeviceName[i] = (WCHAR)pDeviceName[i]; wNameLength++; } wNdisDeviceName[i] = L'\0'; DEBUGP(("Trying to access NDIS Device: %ws\n", wNdisDeviceName)); return (DeviceIoControl( Handle, IOCTL_NDISUIO_OPEN_DEVICE, (LPVOID)&wNdisDeviceName[0], wNameLength*sizeof(WCHAR), NULL, 0, &BytesReturned, NULL)); } BOOL GetSrcMac( HANDLE Handle, PUCHAR pSrcMacAddr ) { DWORD BytesReturned; BOOLEAN bSuccess; UCHAR QueryBuffer[sizeof(NDISUIO_QUERY_OID) + MAC_ADDR_LEN]; PNDISUIO_QUERY_OID pQueryOid; DEBUGP(("Trying to get src mac address\n")); pQueryOid = (PNDISUIO_QUERY_OID)&QueryBuffer[0]; pQueryOid->Oid = OID_802_3_CURRENT_ADDRESS; bSuccess = (BOOLEAN)DeviceIoControl( Handle, IOCTL_NDISUIO_QUERY_OID_VALUE, (LPVOID)&QueryBuffer[0], sizeof(QueryBuffer), (LPVOID)&QueryBuffer[0], sizeof(QueryBuffer), &BytesReturned, NULL); if (bSuccess) { DEBUGP(("GetSrcMac: IoControl success, BytesReturned = %d\n", BytesReturned)); memcpy(pSrcMacAddr, pQueryOid->Data, MAC_ADDR_LEN); } else { DEBUGP(("GetSrcMac: IoControl failed: %d\n", GetLastError())); } return (bSuccess); } VOID DoReadProc( HANDLE Handle ) { PUCHAR pReadBuf = NULL; INT ReadCount = 0; BOOLEAN bSuccess; ULONG BytesRead; DEBUGP(("DoReadProc\n")); do { pReadBuf = malloc(PacketLength); if (pReadBuf == NULL) { PRINTF(("DoReadProc: failed to alloc %d bytes\n", PacketLength)); break; } ReadCount = 0; while (TRUE) { bSuccess = (BOOLEAN)ReadFile( Handle, (LPVOID)pReadBuf, PacketLength, &BytesRead, NULL); if (!bSuccess) { PRINTF(("DoReadProc: ReadFile failed on Handle %p, error %x\n", Handle, GetLastError())); break; } ReadCount++; DEBUGP(("DoReadProc: read pkt # %d, %d bytes\n", ReadCount, BytesRead)); if ((NumberOfPackets != -1) && (ReadCount == NumberOfPackets)) { break; } } } while (FALSE); if (pReadBuf) { free(pReadBuf); } PRINTF(("DoReadProc finished: read %d packets\n", ReadCount)); } VOID DoWriteProc( HANDLE Handle ) { PUCHAR pWriteBuf = NULL; PUCHAR pData; UINT i; INT SendCount; PETH_HEADER pEthHeader; DWORD BytesWritten; BOOLEAN bSuccess; DEBUGP(("DoWriteProc\n")); SendCount = 0; do { pWriteBuf = malloc(PacketLength); if (pWriteBuf == NULL) { DEBUGP(("DoWriteProc: Failed to malloc %d bytes\n", PacketLength)); break; } pEthHeader = (PETH_HEADER)pWriteBuf; pEthHeader->EthType = EthType; if (bUseFakeAddress) { memcpy(pEthHeader->SrcAddr, FakeSrcMacAddr, MAC_ADDR_LEN); } else { memcpy(pEthHeader->SrcAddr, SrcMacAddr, MAC_ADDR_LEN); } memcpy(pEthHeader->DstAddr, DstMacAddr, MAC_ADDR_LEN); pData = (PUCHAR)(pEthHeader + 1); for (i = 0; i < PacketLength - sizeof(ETH_HEADER); i++) { *pData++ = (UCHAR)i; } SendCount = 0; while (TRUE) { bSuccess = (BOOLEAN)WriteFile( Handle, pWriteBuf, PacketLength, &BytesWritten, NULL); if (!bSuccess) { PRINTF(("DoWriteProc: WriteFile failed on Handle %p\n", Handle)); break; } SendCount++; DEBUGP(("DoWriteProc: sent %d bytes\n", BytesWritten)); if ((NumberOfPackets != -1) && (SendCount == NumberOfPackets)) { break; } } } while (FALSE); if (pWriteBuf) { free(pWriteBuf); } PRINTF(("DoWriteProc: finished sending %d packets of %d bytes each\n", SendCount, PacketLength)); } VOID EnumerateDevices( HANDLE Handle ) { CHAR Buf[1024]; DWORD BufLength = sizeof(Buf); DWORD BytesWritten; DWORD i; PNDISUIO_QUERY_BINDING pQueryBinding; pQueryBinding = (PNDISUIO_QUERY_BINDING)Buf; i = 0; for (pQueryBinding->BindingIndex = i; /* NOTHING */; pQueryBinding->BindingIndex = ++i) { if (DeviceIoControl( Handle, IOCTL_NDISUIO_QUERY_BINDING, pQueryBinding, sizeof(NDISUIO_QUERY_BINDING), Buf, BufLength, &BytesWritten, NULL)) { PRINTF(("%2d. %ws\n - %ws\n", pQueryBinding->BindingIndex, (WCHAR *)((PUCHAR)pQueryBinding + pQueryBinding->DeviceNameOffset), (WCHAR *)((PUCHAR )pQueryBinding + pQueryBinding->DeviceDescrOffset))); memset(Buf, 0, BufLength); } else { ULONG rc = GetLastError(); if (rc != ERROR_NO_MORE_ITEMS) { PRINTF(("EnumerateDevices: terminated abnormally, error %d\n", rc)); } break; } } } VOID __cdecl main( INT argc, CHAR *argv[] ) { HANDLE DeviceHandle; DeviceHandle = INVALID_HANDLE_VALUE; do { if (!GetOptions(argc, argv)) { PrintUsage(); break; } DeviceHandle = OpenHandle(pNdisuioDevice); if (DeviceHandle == INVALID_HANDLE_VALUE) { PRINTF(("Failed to open %s\n", pNdisuioDevice)); break; } if (DoEnumerate) { EnumerateDevices(DeviceHandle); break; } if (!OpenNdisDevice(DeviceHandle, pNdisDeviceName)) { PRINTF(("Failed to access %s\n", pNdisDeviceName)); break; } DEBUGP(("Opened device %s successfully!\n", pNdisDeviceName)); if (!GetSrcMac(DeviceHandle, SrcMacAddr)) { PRINTF(("Failed to obtain local MAC address\n")); break; } DEBUGP(("Got local MAC: %02x:%02x:%02x:%02x:%02x:%02x\n", SrcMacAddr[0], SrcMacAddr[1], SrcMacAddr[2], SrcMacAddr[3], SrcMacAddr[4], SrcMacAddr[5])); if (!bDstMacSpecified) { memcpy(DstMacAddr, SrcMacAddr, MAC_ADDR_LEN); } if (DoReads) { DoReadProc(DeviceHandle); } else { DoWriteProc(DeviceHandle); DoReadProc(DeviceHandle); } } while (FALSE); if (DeviceHandle != INVALID_HANDLE_VALUE) { CloseHandle(DeviceHandle); } }