1 /* 2 * PROJECT: ReactOS kernel-mode tests 3 * LICENSE: LGPLv2+ - See COPYING.LIB in the top level directory 4 * PURPOSE: Kernel-Mode Test Suite Device Interface functions test 5 * PROGRAMMER: Filip Navara <xnavara@volny.cz> 6 */ 7 8 /* TODO: what's with the prototypes at the top, what's with the if-ed out part? Doesn't process most results */ 9 10 #include <kmt_test.h> 11 #include <poclass.h> 12 13 #define NDEBUG 14 #include <debug.h> 15 16 #if 0 17 NTSTATUS 18 (NTAPI *IoGetDeviceInterfaces_Func)( 19 IN CONST GUID *InterfaceClassGuid, 20 IN PDEVICE_OBJECT PhysicalDeviceObject OPTIONAL, 21 IN ULONG Flags, 22 OUT PWSTR *SymbolicLinkList); 23 24 NTSTATUS NTAPI 25 ReactOS_IoGetDeviceInterfaces( 26 IN CONST GUID *InterfaceClassGuid, 27 IN PDEVICE_OBJECT PhysicalDeviceObject OPTIONAL, 28 IN ULONG Flags, 29 OUT PWSTR *SymbolicLinkList); 30 #endif /* 0 */ 31 32 static VOID DeviceInterfaceTest_Func() 33 { 34 NTSTATUS Status; 35 PWSTR SymbolicLinkList; 36 PWSTR SymbolicLinkListPtr; 37 GUID Guid = {0x378de44c, 0x56ef, 0x11d1, {0xbc, 0x8c, 0x00, 0xa0, 0xc9, 0x14, 0x05, 0xdd}}; 38 39 Status = IoGetDeviceInterfaces( 40 &Guid, 41 NULL, 42 0, 43 &SymbolicLinkList); 44 45 ok(NT_SUCCESS(Status), 46 "IoGetDeviceInterfaces failed with status 0x%X\n", 47 (unsigned int)Status); 48 if (!NT_SUCCESS(Status)) 49 { 50 return; 51 } 52 53 DPRINT("IoGetDeviceInterfaces results:\n"); 54 for (SymbolicLinkListPtr = SymbolicLinkList; 55 SymbolicLinkListPtr[0] != 0 && SymbolicLinkListPtr[1] != 0; 56 SymbolicLinkListPtr += wcslen(SymbolicLinkListPtr) + 1) 57 { 58 DPRINT1("Symbolic Link: %S\n", SymbolicLinkListPtr); 59 } 60 61 #if 0 62 DPRINT("[PnP Test] Trying to get aliases\n"); 63 64 for (SymbolicLinkListPtr = SymbolicLinkList; 65 SymbolicLinkListPtr[0] != 0 && SymbolicLinkListPtr[1] != 0; 66 SymbolicLinkListPtr += wcslen(SymbolicLinkListPtr) + 1) 67 { 68 UNICODE_STRING SymbolicLink; 69 UNICODE_STRING AliasSymbolicLink; 70 71 SymbolicLink.Buffer = SymbolicLinkListPtr; 72 SymbolicLink.Length = SymbolicLink.MaximumLength = wcslen(SymbolicLinkListPtr); 73 RtlInitUnicodeString(&AliasSymbolicLink, NULL); 74 IoGetDeviceInterfaceAlias( 75 &SymbolicLink, 76 &AliasGuid, 77 &AliasSymbolicLink); 78 if (AliasSymbolicLink.Buffer != NULL) 79 { 80 DPRINT("[PnP Test] Original: %S\n", SymbolicLinkListPtr); 81 DPRINT("[PnP Test] Alias: %S\n", AliasSymbolicLink.Buffer); 82 } 83 } 84 #endif 85 86 ExFreePool(SymbolicLinkList); 87 } 88 89 static 90 VOID 91 Test_IoRegisterDeviceInterface(VOID) 92 { 93 GUID Guid = {0x378de44c, 0x56ef, 0x11d1, {0xbc, 0x8c, 0x00, 0xa0, 0xc9, 0x14, 0x05, 0xdd}}; 94 DEVICE_OBJECT DeviceObject; 95 EXTENDED_DEVOBJ_EXTENSION DeviceObjectExtension; 96 DEVICE_NODE DeviceNode; 97 UNICODE_STRING SymbolicLinkName; 98 NTSTATUS Status; 99 100 RtlInitUnicodeString(&SymbolicLinkName, L""); 101 102 // Prepare our surrogate of a Device Object 103 DeviceObject.DeviceObjectExtension = (PDEVOBJ_EXTENSION)&DeviceObjectExtension; 104 105 // 1. DeviceNode = NULL 106 DeviceObjectExtension.DeviceNode = NULL; 107 Status = IoRegisterDeviceInterface(&DeviceObject, &Guid, NULL, 108 &SymbolicLinkName); 109 110 ok(Status == STATUS_INVALID_DEVICE_REQUEST, 111 "IoRegisterDeviceInterface returned 0x%08lX\n", Status); 112 113 // 2. DeviceNode->InstancePath is of a null length 114 DeviceObjectExtension.DeviceNode = &DeviceNode; 115 DeviceNode.InstancePath.Length = 0; 116 Status = IoRegisterDeviceInterface(&DeviceObject, &Guid, NULL, 117 &SymbolicLinkName); 118 119 ok(Status == STATUS_INVALID_DEVICE_REQUEST, 120 "IoRegisterDeviceInterface returned 0x%08lX\n", Status); 121 122 DeviceInterfaceTest_Func(); 123 } 124 125 static UCHAR NotificationContext; 126 127 static DRIVER_NOTIFICATION_CALLBACK_ROUTINE NotificationCallback; 128 static 129 NTSTATUS 130 NTAPI 131 NotificationCallback( 132 _In_ PVOID NotificationStructure, 133 _Inout_opt_ PVOID Context) 134 { 135 PDEVICE_INTERFACE_CHANGE_NOTIFICATION Notification = NotificationStructure; 136 NTSTATUS Status; 137 OBJECT_ATTRIBUTES ObjectAttributes; 138 HANDLE Handle; 139 140 ok_irql(PASSIVE_LEVEL); 141 ok_eq_pointer(Context, &NotificationContext); 142 ok_eq_uint(Notification->Version, 1); 143 ok_eq_uint(Notification->Size, sizeof(*Notification)); 144 145 /* symbolic link must exist */ 146 trace("Interface change: %wZ\n", Notification->SymbolicLinkName); 147 InitializeObjectAttributes(&ObjectAttributes, 148 Notification->SymbolicLinkName, 149 OBJ_KERNEL_HANDLE, 150 NULL, 151 NULL); 152 Status = ZwOpenSymbolicLinkObject(&Handle, GENERIC_READ, &ObjectAttributes); 153 ok_eq_hex(Status, STATUS_SUCCESS); 154 if (!skip(NT_SUCCESS(Status), "No symbolic link\n")) 155 { 156 Status = ObCloseHandle(Handle, KernelMode); 157 ok_eq_hex(Status, STATUS_SUCCESS); 158 } 159 return STATUS_SUCCESS; 160 } 161 162 static 163 VOID 164 Test_IoRegisterPlugPlayNotification(VOID) 165 { 166 NTSTATUS Status; 167 PVOID NotificationEntry; 168 169 Status = IoRegisterPlugPlayNotification(EventCategoryDeviceInterfaceChange, 170 PNPNOTIFY_DEVICE_INTERFACE_INCLUDE_EXISTING_INTERFACES, 171 (PVOID)&GUID_DEVICE_SYS_BUTTON, 172 KmtDriverObject, 173 NotificationCallback, 174 &NotificationContext, 175 &NotificationEntry); 176 ok_eq_hex(Status, STATUS_SUCCESS); 177 if (!skip(NT_SUCCESS(Status), "PlugPlayNotification not registered\n")) 178 { 179 Status = IoUnregisterPlugPlayNotification(NotificationEntry); 180 ok_eq_hex(Status, STATUS_SUCCESS); 181 } 182 } 183 184 static 185 VOID 186 Test_IoSetDeviceInterface(VOID) 187 { 188 NTSTATUS Status; 189 UNICODE_STRING SymbolicLinkName; 190 PWCHAR Buffer; 191 ULONG BufferSize; 192 193 /* Invalid prefix or GUID */ 194 KmtStartSeh() 195 Status = IoSetDeviceInterfaceState(NULL, TRUE); 196 KmtEndSeh(STATUS_SUCCESS) 197 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 198 199 RtlInitEmptyUnicodeString(&SymbolicLinkName, NULL, 0); 200 KmtStartSeh() 201 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 202 KmtEndSeh(STATUS_SUCCESS) 203 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 204 205 RtlInitUnicodeString(&SymbolicLinkName, L"\\??"); 206 KmtStartSeh() 207 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 208 KmtEndSeh(STATUS_SUCCESS) 209 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 210 211 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\"); 212 KmtStartSeh() 213 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 214 KmtEndSeh(STATUS_SUCCESS) 215 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 216 217 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\\\"); 218 KmtStartSeh() 219 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 220 KmtEndSeh(STATUS_SUCCESS) 221 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 222 223 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\{aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}"); 224 KmtStartSeh() 225 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 226 KmtEndSeh(STATUS_SUCCESS) 227 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 228 229 /* Valid prefix & GUID, invalid device node */ 230 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\X{aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}"); 231 KmtStartSeh() 232 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 233 KmtEndSeh(STATUS_SUCCESS) 234 ok_eq_hex(Status, STATUS_OBJECT_NAME_NOT_FOUND); 235 236 RtlInitUnicodeString(&SymbolicLinkName, L"\\\\?\\X{aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}"); 237 KmtStartSeh() 238 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 239 KmtEndSeh(STATUS_SUCCESS) 240 ok_eq_hex(Status, STATUS_OBJECT_NAME_NOT_FOUND); 241 242 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\X{aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}\\"); 243 KmtStartSeh() 244 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 245 KmtEndSeh(STATUS_SUCCESS) 246 ok_eq_hex(Status, STATUS_OBJECT_NAME_NOT_FOUND); 247 248 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\#{aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}"); 249 KmtStartSeh() 250 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 251 KmtEndSeh(STATUS_SUCCESS) 252 ok_eq_hex(Status, STATUS_OBJECT_NAME_NOT_FOUND); 253 254 /* Must not read past the buffer */ 255 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\#{aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}"); 256 BufferSize = SymbolicLinkName.Length; 257 Buffer = KmtAllocateGuarded(BufferSize); 258 if (!skip(Buffer != NULL, "Failed to allocate %lu bytes\n", BufferSize)) 259 { 260 RtlCopyMemory(Buffer, SymbolicLinkName.Buffer, BufferSize); 261 SymbolicLinkName.Buffer = Buffer; 262 SymbolicLinkName.MaximumLength = BufferSize; 263 KmtStartSeh() 264 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 265 KmtEndSeh(STATUS_SUCCESS) 266 ok_eq_hex(Status, STATUS_OBJECT_NAME_NOT_FOUND); 267 KmtFreeGuarded(Buffer); 268 } 269 270 RtlInitUnicodeString(&SymbolicLinkName, L"\\??\\#aaaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa}"); 271 BufferSize = SymbolicLinkName.Length; 272 Buffer = KmtAllocateGuarded(BufferSize); 273 if (!skip(Buffer != NULL, "Failed to allocate %lu bytes\n", BufferSize)) 274 { 275 RtlCopyMemory(Buffer, SymbolicLinkName.Buffer, BufferSize); 276 SymbolicLinkName.Buffer = Buffer; 277 SymbolicLinkName.MaximumLength = BufferSize; 278 KmtStartSeh() 279 Status = IoSetDeviceInterfaceState(&SymbolicLinkName, TRUE); 280 KmtEndSeh(STATUS_SUCCESS) 281 ok_eq_hex(Status, STATUS_INVALID_PARAMETER); 282 KmtFreeGuarded(Buffer); 283 } 284 } 285 286 START_TEST(IoDeviceInterface) 287 { 288 // FIXME: This test crashes in Windows 289 (void)Test_IoRegisterDeviceInterface; 290 Test_IoRegisterPlugPlayNotification(); 291 Test_IoSetDeviceInterface(); 292 } 293