1 #include <windows.h>
2 #include <net/lan.h>
3 #include <iostream>
4 #include <string>
5 #include <ddk/ntddk.h>
6 #include <rosrtl/string.h>
7
8 using std::string;
9 using std::cin;
10 using std::cout;
11 using std::cerr;
12 using std::endl;
13
display_row(char * data,int off,int len)14 void display_row( char *data, int off, int len ) {
15 int i;
16
17 printf( "%08x:", off );
18 for( i = off; i < len && i < off + 16; i++ ) {
19 printf( " %02x", data[i] & 0xff );
20 }
21
22 for( ; i < off + 16; i++ )
23 printf(" ");
24
25 printf( " -- " );
26
27 for( i = off; i < len && i < off + 16; i++ ) {
28 printf( "%c", (data[i] >= ' ' && data[i] <= '~') ? data[i] : '.' );
29 }
30
31 printf( "\n" );
32 }
33
display_buffer(char * Packet,int ReadLen)34 void display_buffer( char *Packet, int ReadLen ) {
35 UINT PktLen;
36 for( PktLen = 0; PktLen < ReadLen; PktLen += 16 )
37 display_row( Packet, PktLen, ReadLen );
38 }
39
byte_till_end(char * Packet,int PktLen)40 int byte_till_end( char *Packet, int PktLen ) {
41 int byte;
42 std::string word;
43
44 cin >> word;
45 while( word != "end" ) {
46 byte = strtoul( (string("0x") + word).c_str(), 0, 0 );
47 fprintf( stderr, "Byte[%d]: %x\n", PktLen, byte & 0xff );
48 Packet[PktLen++] = byte;
49 cin >> word;
50 }
51
52 return PktLen;
53 }
54
55 /* Ethernet types. We swap constants so we can compare values at runtime
56 without swapping them there */
57 #define ETYPE_IPv4 WH2N(0x0800)
58 #define ETYPE_IPv6 WH2N(0x86DD)
59 #define ETYPE_ARP WH2N(0x0806)
60
61 extern "C"
62 NTSTATUS NTAPI NtCreateFile(
63 PHANDLE FileHandle,
64 ACCESS_MASK DesiredAccess,
65 POBJECT_ATTRIBUTES ObjectAttributes,
66 PIO_STATUS_BLOCK IoStatusBlock,
67 PLARGE_INTEGER AllocationSize,
68 ULONG FileAttributes,
69 ULONG ShareAccess,
70 ULONG CreateDisposition,
71 ULONG CreateOptions,
72 PVOID EaBuffer,
73 ULONG EaLength);
74
main(int argc,char ** argv)75 int main( int argc, char **argv ) {
76 string word;
77 HANDLE LanFile;
78 OBJECT_ATTRIBUTES Attributes;
79 UNICODE_STRING LanDevice;
80 IO_STATUS_BLOCK Iosb;
81 HANDLE Event;
82 PFILE_FULL_EA_INFORMATION EaBuffer;
83 NTSTATUS Status;
84 DWORD On = 1, PktLen;
85 CHAR Packet[1600];
86 PLAN_PACKET_HEADER Hdr = (PLAN_PACKET_HEADER)Packet;
87 PLAN_ADDRESS Addr = (PLAN_ADDRESS)Packet;
88 USHORT TypesToListen[] = { ETYPE_IPv4, ETYPE_IPv6, ETYPE_ARP };
89 UINT EaLength = LAN_EA_INFO_SIZE(sizeof(TypesToListen)/sizeof(USHORT));
90
91 Status = NtCreateEvent(&Event,
92 EVENT_ALL_ACCESS,
93 NULL,
94 0,
95 0 );
96
97 RtlInitUnicodeString( &LanDevice, L"\\Device\\Lan" );
98
99 InitializeObjectAttributes( &Attributes,
100 &LanDevice,
101 OBJ_CASE_INSENSITIVE,
102 NULL,
103 NULL );
104
105 EaBuffer = (PFILE_FULL_EA_INFORMATION)calloc( EaLength, 1 );
106 LAN_FILL_EA_INFO(EaBuffer,sizeof(TypesToListen)/sizeof(USHORT),
107 TypesToListen);
108
109 Status = ZwCreateFile( &LanFile,
110 SYNCHRONIZE | GENERIC_READ | GENERIC_WRITE |
111 GENERIC_EXECUTE,
112 &Attributes,
113 &Iosb,
114 NULL,
115 FILE_ATTRIBUTE_NORMAL,
116 FILE_SHARE_READ | FILE_SHARE_WRITE,
117 FILE_OPEN_IF,
118 FILE_SYNCHRONOUS_IO_NONALERT,
119 EaBuffer,
120 EaLength );
121
122 if( !NT_SUCCESS(Status) ) {
123 cerr << "Could not open lan device " << Status << "\n";
124 return 1;
125 }
126
127 Status = DeviceIoControl( LanFile,
128 IOCTL_IF_BUFFERED_MODE,
129 &On,
130 sizeof(On),
131 0,
132 0,
133 &PktLen,
134 NULL );
135
136 if( !Status ) {
137 cerr << "Could not turn on buffered mode " << Status << "\n";
138 return 1;
139 }
140
141 while( cin >> word ) {
142 if( word == "end" ) {
143 NtClose( LanFile );
144 return 0;
145 } else if( word == "enum" ) {
146 Status = DeviceIoControl( LanFile,
147 IOCTL_IF_ENUM_ADAPTERS,
148 NULL,
149 0,
150 Packet,
151 sizeof(Packet),
152 &PktLen,
153 NULL );
154
155 cout << "EnumAdapters: " << Status << "\n";
156 if( Status )
157 display_buffer( Packet, PktLen );
158 } else if( word == "query" ) {
159 cin >> PktLen;
160
161 Status = DeviceIoControl( LanFile,
162 IOCTL_IF_ADAPTER_INFO,
163 &PktLen,
164 sizeof(PktLen),
165 Packet,
166 sizeof(Packet),
167 &PktLen,
168 NULL );
169
170 cout << "QueryAdapterInfo: " << Status << "\n";
171 if( Status )
172 display_buffer( Packet, PktLen );
173 } else if( word == "send" ) {
174 cin >> Hdr->Fixed.Adapter
175 >> Hdr->Fixed.AddressType
176 >> Hdr->Fixed.AddressLen
177 >> Hdr->Fixed.PacketType;
178 Hdr->Fixed.Mdl = NULL;
179 PktLen = byte_till_end( Packet, Hdr->Address - (PCHAR)Hdr );
180 Status = NtWriteFile( LanFile,
181 NULL,
182 NULL,
183 NULL,
184 &Iosb,
185 Packet,
186 PktLen,
187 NULL,
188 NULL );
189
190 cout << "NtWriteFile: " << Status << "\n";
191 } else if( word == "recv" ) {
192 ULONG ReadLen;
193 Status = NtReadFile( LanFile,
194 Event,
195 NULL,
196 NULL,
197 &Iosb,
198 Packet,
199 sizeof(Packet),
200 NULL,
201 NULL );
202 cout << "NtReadFile: " << Status << "\n";
203 if( Status == STATUS_PENDING ) {
204 LARGE_INTEGER Timeout = { 0 };
205 Status = NtWaitForSingleObject( Event, 1, &Timeout );
206 }
207
208 ReadLen = Iosb.Information;
209
210 if( Status == STATUS_SUCCESS ) {
211 cout << "Read " << ReadLen << " bytes\n";
212 display_buffer( Packet, ReadLen );
213 }
214 }
215 }
216
217 return 0;
218 }
219