1 /*
2  * PROJECT:         ReactOS kernel-mode tests
3  * LICENSE:         LGPLv2+ - See COPYING.LIB in the top level directory
4  * PURPOSE:         Kernel-Mode Test Suite for TCPIP.sys
5  * PROGRAMMER:      Jérôme Gardou <jerome.gardou@reactos.org>
6  */
7 
8 #include <kmt_test.h>
9 #include <tdikrnl.h>
10 #include <ndk/rtlfuncs.h>
11 
12 #include <sys/param.h>
13 
14 #include "tcpip.h"
15 
16 #define TAG_TEST 'tseT'
17 
18 #if BYTE_ORDER == LITTLE_ENDIAN
19 USHORT
20 htons(USHORT x)
21 {
22     return ((x & 0x00FF) << 8) | ((x & 0xFF00) >> 8);
23 }
24 #else
25 #define htons(x) (x)
26 #endif
27 
28 static
29 NTSTATUS
30 NTAPI
31 IrpCompletionRoutine(
32     _In_ PDEVICE_OBJECT    DeviceObject,
33     _In_ PIRP              Irp,
34     _In_ PVOID             Context)
35 {
36     UNREFERENCED_PARAMETER(DeviceObject);
37     UNREFERENCED_PARAMETER(Irp);
38 
39     KeSetEvent((PKEVENT)Context, IO_NETWORK_INCREMENT, FALSE);
40 
41     return STATUS_MORE_PROCESSING_REQUIRED;
42 }
43 
44 static
45 VOID
46 TestTcpConnect(void)
47 {
48     PIRP Irp;
49     HANDLE AddressHandle, ConnectionHandle;
50     FILE_OBJECT* ConnectionFileObject;
51     DEVICE_OBJECT* DeviceObject;
52     UNICODE_STRING TcpDeviceName = RTL_CONSTANT_STRING(L"\\Device\\Tcp");
53     NTSTATUS Status;
54     PFILE_FULL_EA_INFORMATION FileInfo;
55     TA_IP_ADDRESS* IpAddress;
56     TA_IP_ADDRESS ConnectAddress, ReturnAddress;
57     OBJECT_ATTRIBUTES ObjectAttributes;
58     IO_STATUS_BLOCK StatusBlock;
59     ULONG FileInfoSize;
60     IN_ADDR InAddr;
61     LPCWSTR AddressTerminator;
62     CONNECTION_CONTEXT ConnectionContext = (CONNECTION_CONTEXT)0xC0CAC01A;
63     KEVENT Event;
64     TDI_CONNECTION_INFORMATION RequestInfo, ReturnInfo;
65 
66     /* Create a TCP address file */
67     FileInfoSize = FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName[TDI_TRANSPORT_ADDRESS_LENGTH]) + 1 + sizeof(TA_IP_ADDRESS);
68     FileInfo = ExAllocatePoolWithTag(NonPagedPool,
69             FileInfoSize,
70             TAG_TEST);
71     ok(FileInfo != NULL, "FileInfo is NULL!\n");
72     RtlZeroMemory(FileInfo, FileInfoSize);
73 
74     FileInfo->EaNameLength = TDI_TRANSPORT_ADDRESS_LENGTH;
75     FileInfo->EaValueLength = sizeof(TA_IP_ADDRESS);
76     RtlCopyMemory(&FileInfo->EaName[0], TdiTransportAddress, TDI_TRANSPORT_ADDRESS_LENGTH);
77 
78     IpAddress = (PTA_IP_ADDRESS)(&FileInfo->EaName[TDI_TRANSPORT_ADDRESS_LENGTH + 1]);
79     IpAddress->TAAddressCount = 1;
80     IpAddress->Address[0].AddressType = TDI_ADDRESS_TYPE_IP;
81     IpAddress->Address[0].AddressLength = TDI_ADDRESS_LENGTH_IP;
82     IpAddress->Address[0].Address[0].sin_port = htons(TEST_CONNECT_CLIENT_PORT);
83     Status = RtlIpv4StringToAddressW(L"127.0.0.1", TRUE, &AddressTerminator, &InAddr);
84     ok_eq_hex(Status, STATUS_SUCCESS);
85     IpAddress->Address[0].Address[0].in_addr = InAddr.S_un.S_addr;
86 
87     InitializeObjectAttributes(&ObjectAttributes,
88             &TcpDeviceName,
89             OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
90             NULL,
91             NULL);
92 
93     Status = ZwCreateFile(
94         &AddressHandle,
95         GENERIC_READ | GENERIC_WRITE,
96         &ObjectAttributes,
97         &StatusBlock,
98         0,
99         FILE_ATTRIBUTE_NORMAL,
100         FILE_SHARE_READ | FILE_SHARE_WRITE,
101         FILE_OPEN_IF,
102         0L,
103         FileInfo,
104         FileInfoSize);
105     ok_eq_hex(Status, STATUS_SUCCESS);
106 
107     ExFreePoolWithTag(FileInfo, TAG_TEST);
108 
109     /* Create a TCP connection file */
110     FileInfoSize = FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName[TDI_CONNECTION_CONTEXT_LENGTH]) + 1 + sizeof(CONNECTION_CONTEXT);
111     FileInfo = ExAllocatePoolWithTag(NonPagedPool,
112             FileInfoSize,
113             TAG_TEST);
114     ok(FileInfo != NULL, "FileInfo is NULL!\n");
115     RtlZeroMemory(FileInfo, FileInfoSize);
116 
117     FileInfo->EaNameLength = TDI_CONNECTION_CONTEXT_LENGTH;
118     FileInfo->EaValueLength = sizeof(CONNECTION_CONTEXT);
119     RtlCopyMemory(&FileInfo->EaName[0], TdiConnectionContext, TDI_CONNECTION_CONTEXT_LENGTH);
120     *((CONNECTION_CONTEXT*)&FileInfo->EaName[TDI_CONNECTION_CONTEXT_LENGTH + 1]) = ConnectionContext;
121 
122     Status = ZwCreateFile(
123         &ConnectionHandle,
124         GENERIC_READ | GENERIC_WRITE,
125         &ObjectAttributes,
126         &StatusBlock,
127         0,
128         FILE_ATTRIBUTE_NORMAL,
129         FILE_SHARE_READ | FILE_SHARE_WRITE,
130         FILE_OPEN_IF,
131         0L,
132         FileInfo,
133         FileInfoSize);
134     ok_eq_hex(Status, STATUS_SUCCESS);
135 
136     ExFreePoolWithTag(FileInfo, TAG_TEST);
137 
138     /* Get the file and device object for the upcoming IRPs */
139     Status = ObReferenceObjectByHandle(
140         ConnectionHandle,
141         GENERIC_READ,
142         *IoFileObjectType,
143         KernelMode,
144         (PVOID*)&ConnectionFileObject,
145         NULL);
146     ok_eq_hex(Status, STATUS_SUCCESS);
147     DeviceObject = IoGetRelatedDeviceObject(ConnectionFileObject);
148     ok(DeviceObject != NULL, "Device object is NULL!\n");
149 
150     /* Associate the connection file and the address */
151     KeInitializeEvent(&Event, NotificationEvent, FALSE);
152     Irp = IoAllocateIrp(DeviceObject->StackSize, FALSE);
153     ok(Irp != NULL, "IoAllocateIrp failed.\n");
154 
155     TdiBuildAssociateAddress(Irp, DeviceObject, ConnectionFileObject, NULL, NULL, AddressHandle);
156     IoSetCompletionRoutine(Irp, IrpCompletionRoutine, &Event, TRUE, TRUE, TRUE);
157 
158     Status = IoCallDriver(DeviceObject, Irp);
159     if (Status == STATUS_PENDING)
160     {
161         trace("Associate address IRP is pending.\n");
162         KeWaitForSingleObject(
163             &Event,
164             Executive,
165             KernelMode,
166             FALSE,
167             NULL);
168         Status = Irp->IoStatus.Status;
169     }
170     ok_eq_hex(Status, STATUS_SUCCESS);
171     IoFreeIrp(Irp);
172 
173 
174     KeClearEvent(&Event);
175 
176     /* Build the connect IRP. */
177     Irp = IoAllocateIrp(DeviceObject->StackSize, FALSE);
178     ok(Irp != NULL, "IoAllocateIrp failed.\n");
179 
180     /* Prepare the request */
181     RtlZeroMemory(&RequestInfo, sizeof(RequestInfo));
182     RtlZeroMemory(&ConnectAddress, sizeof(ConnectAddress));
183     RequestInfo.RemoteAddressLength = sizeof(TA_IP_ADDRESS);
184     RequestInfo.RemoteAddress = &ConnectAddress;
185     ConnectAddress.TAAddressCount = 1;
186     ConnectAddress.Address[0].AddressType = TDI_ADDRESS_TYPE_IP;
187     ConnectAddress.Address[0].AddressLength = TDI_ADDRESS_LENGTH_IP;
188     ConnectAddress.Address[0].Address[0].sin_port = htons(TEST_CONNECT_SERVER_PORT);
189     Status = RtlIpv4StringToAddressW(L"127.0.0.1", TRUE, &AddressTerminator, &InAddr);
190     ConnectAddress.Address[0].Address[0].in_addr = InAddr.S_un.S_addr;
191 
192     /* See what we will get in exchange */
193     RtlZeroMemory(&ReturnInfo, sizeof(ReturnInfo));
194     RtlZeroMemory(&ReturnAddress, sizeof(ReturnAddress));
195     ReturnInfo.RemoteAddressLength = sizeof(TA_IP_ADDRESS);
196     ReturnInfo.RemoteAddress = &ReturnAddress;
197 
198     TdiBuildConnect(Irp,
199         DeviceObject,
200         ConnectionFileObject,
201         NULL,
202         NULL,
203         NULL,
204         &RequestInfo,
205         &ReturnInfo);
206     IoSetCompletionRoutine(Irp, IrpCompletionRoutine, &Event, TRUE, TRUE, TRUE);
207 
208     Status = IoCallDriver(DeviceObject, Irp);
209     if (Status == STATUS_PENDING)
210     {
211         trace("Connect IRP is pending.\n");
212         KeWaitForSingleObject(
213             &Event,
214             Executive,
215             KernelMode,
216             FALSE,
217             NULL);
218         Status = Irp->IoStatus.Status;
219         trace("Connect IRP completed.\n");
220     }
221     ok_eq_hex(Status, STATUS_SUCCESS);
222     IoFreeIrp(Irp);
223 
224     /* The IRP doesn't touch the return info */
225     ok_eq_long(ReturnInfo.RemoteAddressLength, sizeof(TA_IP_ADDRESS));
226     ok_eq_pointer(ReturnInfo.RemoteAddress, &ReturnAddress);
227     ok_eq_long(ReturnInfo.OptionsLength, 0);
228     ok_eq_pointer(ReturnInfo.Options, NULL);
229     ok_eq_long(ReturnInfo.UserDataLength, 0);
230     ok_eq_pointer(ReturnInfo.UserData, NULL);
231 
232     ok_eq_long(ReturnAddress.TAAddressCount, 0);
233     ok_eq_hex(ReturnAddress.Address[0].AddressType, 0);
234     ok_eq_hex(ReturnAddress.Address[0].AddressLength, 0);
235     ok_eq_hex(ReturnAddress.Address[0].Address[0].sin_port, 0);
236     ok_eq_hex(ReturnAddress.Address[0].Address[0].in_addr, 0);
237 
238     ObDereferenceObject(ConnectionFileObject);
239 
240     ZwClose(ConnectionHandle);
241     ZwClose(AddressHandle);
242 }
243 
244 static KSTART_ROUTINE RunTest;
245 static
246 VOID
247 NTAPI
248 RunTest(
249     _In_ PVOID Context)
250 {
251     UNREFERENCED_PARAMETER(Context);
252 
253     TestTcpConnect();
254 }
255 
256 KMT_MESSAGE_HANDLER TestConnect;
257 NTSTATUS
258 TestConnect(
259     _In_ PDEVICE_OBJECT DeviceObject,
260     _In_ ULONG ControlCode,
261     _In_opt_ PVOID Buffer,
262     _In_ SIZE_T InLength,
263     _Inout_ PSIZE_T OutLength
264 )
265 {
266     PKTHREAD Thread;
267 
268     Thread = KmtStartThread(RunTest, NULL);
269     KmtFinishThread(Thread, NULL);
270 
271     return STATUS_SUCCESS;
272 }
273