xref: /reactos/drivers/processor/processr/pnp.c (revision 682f85ad)
1 /*
2  * PROJECT:        ReactOS Generic CPU Driver
3  * LICENSE:        GNU GPLv2 only as published by the Free Software Foundation
4  * FILE:           drivers/processor/processr/pnp.c
5  * PURPOSE:        Plug N Play routines
6  * PROGRAMMERS:    Eric Kohl <eric.kohl@reactos.org>
7  */
8 
9 /* INCLUDES *******************************************************************/
10 
11 #include "processr.h"
12 
13 #include <stdio.h>
14 #define NDEBUG
15 #include <debug.h>
16 
17 /* FUNCTIONS ******************************************************************/
18 
19 static
20 NTSTATUS
21 GetDeviceId(
22     PDEVICE_OBJECT DeviceObject,
23     BUS_QUERY_ID_TYPE IdType,
24     PWSTR *DeviceId)
25 {
26     PIO_STACK_LOCATION IrpStack;
27     IO_STATUS_BLOCK IoStatus;
28     PDEVICE_OBJECT TargetObject;
29     KEVENT Event;
30     PIRP Irp;
31     NTSTATUS Status;
32 
33     PAGED_CODE();
34 
35     /* Initialize the event */
36     KeInitializeEvent(&Event, NotificationEvent, FALSE);
37 
38     TargetObject = IoGetAttachedDeviceReference(DeviceObject);
39 
40     /* Build the IRP */
41     Irp = IoBuildSynchronousFsdRequest(IRP_MJ_PNP,
42                                        TargetObject,
43                                        NULL,
44                                        0,
45                                        NULL,
46                                        &Event,
47                                        &IoStatus);
48     if (Irp == NULL)
49     {
50         Status = STATUS_INSUFFICIENT_RESOURCES;
51         goto done;
52     }
53 
54     /* PNP IRPs all begin life as STATUS_NOT_SUPPORTED */
55     Irp->IoStatus.Status = STATUS_NOT_SUPPORTED;
56 
57     /* Get the top of stack */
58     IrpStack = IoGetNextIrpStackLocation(Irp);
59 
60     /* Set the top of stack */
61     RtlZeroMemory(IrpStack, sizeof(IO_STACK_LOCATION));
62     IrpStack->MajorFunction = IRP_MJ_PNP;
63     IrpStack->MinorFunction = IRP_MN_QUERY_ID;
64     IrpStack->Parameters.QueryId.IdType = IdType;
65 
66     /* Call the driver */
67     Status = IoCallDriver(TargetObject, Irp);
68     if (Status == STATUS_PENDING)
69     {
70         KeWaitForSingleObject(&Event,
71                               Executive,
72                               KernelMode,
73                               FALSE,
74                               NULL);
75 
76         Status = IoStatus.Status;
77     }
78 
79     if (NT_SUCCESS(Status))
80     {
81         *DeviceId = (PWSTR)IoStatus.Information;
82     }
83 
84 done:
85     /* Dereference the target device object */
86     ObDereferenceObject(TargetObject);
87 
88     return Status;
89 }
90 
91 
92 
93 static
94 VOID
95 ProcessorSetFriendlyName(
96     PDEVICE_OBJECT DeviceObject)
97 {
98     KEY_VALUE_PARTIAL_INFORMATION *Buffer = NULL;
99     OBJECT_ATTRIBUTES ObjectAttributes;
100     UNICODE_STRING HardwareKeyName, ValueName, EnumKeyName;
101     HANDLE KeyHandle = NULL;
102     ULONG DataLength = 0;
103     ULONG BufferLength = 0;
104     NTSTATUS Status;
105     PWSTR KeyNameBuffer = NULL;
106     PWSTR DeviceId = NULL;
107     PWSTR InstanceId = NULL;
108     PWSTR pszPrefix = L"\\Registry\\Machine\\System\\CurrentControlSet\\Enum";
109 
110     RtlInitUnicodeString(&HardwareKeyName,
111                          L"\\Registry\\Machine\\HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0");
112     InitializeObjectAttributes(&ObjectAttributes,
113                                &HardwareKeyName,
114                                OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
115                                NULL,
116                                NULL);
117     Status = ZwOpenKey(&KeyHandle,
118                        KEY_READ,
119                        &ObjectAttributes);
120     if (!NT_SUCCESS(Status))
121     {
122         DPRINT1("ZwOpenKey() failed (Status 0x%08lx)\n", Status);
123         return;
124     }
125 
126     RtlInitUnicodeString(&ValueName,
127                          L"ProcessorNameString");
128     Status = ZwQueryValueKey(KeyHandle,
129                              &ValueName,
130                              KeyValuePartialInformation,
131                              NULL,
132                              0,
133                              &DataLength);
134     if (Status != STATUS_BUFFER_OVERFLOW && Status != STATUS_BUFFER_TOO_SMALL && Status != STATUS_SUCCESS)
135     {
136         DPRINT1("ZwQueryValueKey() failed (Status 0x%08lx)\n", Status);
137         goto done;
138     }
139 
140     Buffer = ExAllocatePool(PagedPool,
141                             DataLength + sizeof(KEY_VALUE_PARTIAL_INFORMATION));
142     if (Buffer == NULL)
143     {
144         DPRINT1("ExAllocatePool() failed\n");
145         goto done;
146     }
147 
148     Status = ZwQueryValueKey(KeyHandle,
149                              &ValueName,
150                              KeyValuePartialInformation,
151                              Buffer,
152                              DataLength + sizeof(KEY_VALUE_PARTIAL_INFORMATION),
153                              &DataLength);
154     if (!NT_SUCCESS(Status))
155     {
156         DPRINT1("ZwQueryValueKey() failed (Status 0x%08lx)\n", Status);
157         goto done;
158     }
159 
160     DPRINT("ProcessorNameString: %S\n", (PWSTR)&Buffer->Data[0]);
161 
162     ZwClose(KeyHandle);
163     KeyHandle = NULL;
164 
165     Status = GetDeviceId(DeviceObject,
166                          BusQueryDeviceID,
167                          &DeviceId);
168     if (!NT_SUCCESS(Status))
169     {
170         DPRINT1("GetDeviceId() failed (Status 0x%08lx)\n", Status);
171         goto done;
172     }
173 
174     DPRINT("DeviceId: %S\n", DeviceId);
175 
176     Status = GetDeviceId(DeviceObject,
177                          BusQueryInstanceID,
178                          &InstanceId);
179     if (!NT_SUCCESS(Status))
180     {
181         DPRINT1("GetDeviceId() failed (Status 0x%08lx)\n", Status);
182         goto done;
183     }
184 
185     DPRINT("InstanceId: %S\n", InstanceId);
186 
187     BufferLength = wcslen(pszPrefix) + 1 + wcslen(DeviceId) + 1 + wcslen(InstanceId) + 1;
188 
189     KeyNameBuffer = ExAllocatePool(PagedPool, BufferLength * sizeof(WCHAR));
190     if (KeyNameBuffer == NULL)
191     {
192         DPRINT1("ExAllocatePool() failed\n");
193         goto done;
194     }
195 
196     swprintf(KeyNameBuffer, L"%s\\%s\\%s", pszPrefix, DeviceId, InstanceId);
197 
198     RtlInitUnicodeString(&EnumKeyName, KeyNameBuffer);
199     InitializeObjectAttributes(&ObjectAttributes,
200                                &EnumKeyName,
201                                OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
202                                NULL,
203                                NULL);
204     Status = ZwOpenKey(&KeyHandle,
205                        KEY_WRITE,
206                        &ObjectAttributes);
207     if (!NT_SUCCESS(Status))
208     {
209         DPRINT1("ZwOpenKey() failed (Status 0x%08lx)\n", Status);
210         goto done;
211     }
212 
213     RtlInitUnicodeString(&ValueName,
214                          L"FriendlyName");
215     Status = ZwSetValueKey(KeyHandle,
216                            &ValueName,
217                            0,
218                            REG_SZ,
219                            (PVOID)&Buffer->Data[0],
220                            Buffer->DataLength);
221     if (!NT_SUCCESS(Status))
222     {
223         DPRINT1("ZwSetValueKey() failed (Status 0x%08lx)\n", Status);
224         goto done;
225     }
226 
227 done:
228     if (KeyHandle != NULL)
229         ZwClose(KeyHandle);
230 
231     if (KeyNameBuffer != NULL)
232         ExFreePool(KeyNameBuffer);
233 
234     if (InstanceId != NULL)
235         ExFreePool(InstanceId);
236 
237     if (DeviceId != NULL)
238         ExFreePool(DeviceId);
239 
240     if (Buffer != NULL)
241         ExFreePool(Buffer);
242 }
243 
244 
245 static
246 NTSTATUS
247 ProcessorStartDevice(
248     IN PDEVICE_OBJECT DeviceObject,
249     IN PCM_RESOURCE_LIST ResourceList,
250     IN PCM_RESOURCE_LIST ResourceListTranslated)
251 {
252     DPRINT("ProcessorStartDevice()\n");
253 
254     ProcessorSetFriendlyName(DeviceObject);
255 
256     return STATUS_SUCCESS;
257 }
258 
259 
260 NTSTATUS
261 NTAPI
262 ProcessorPnp(
263     IN PDEVICE_OBJECT DeviceObject,
264     IN PIRP Irp)
265 {
266     PIO_STACK_LOCATION IrpSp;
267     ULONG_PTR Information = 0;
268     NTSTATUS Status = STATUS_NOT_SUPPORTED;
269 
270     DPRINT("ProcessorPnp()\n");
271 
272     IrpSp = IoGetCurrentIrpStackLocation(Irp);
273 
274     switch (IrpSp->MinorFunction)
275     {
276         case IRP_MN_START_DEVICE:
277             DPRINT("  IRP_MN_START_DEVICE received\n");
278 
279             /* Call lower driver */
280             Status = ForwardIrpAndWait(DeviceObject, Irp);
281             if (NT_SUCCESS(Status))
282             {
283                 Status = ProcessorStartDevice(DeviceObject,
284                                               IrpSp->Parameters.StartDevice.AllocatedResources,
285                                               IrpSp->Parameters.StartDevice.AllocatedResourcesTranslated);
286             }
287             break;
288 
289         case IRP_MN_QUERY_REMOVE_DEVICE:
290             DPRINT("  IRP_MN_QUERY_REMOVE_DEVICE\n");
291             return ForwardIrpAndForget(DeviceObject, Irp);
292 
293         case IRP_MN_REMOVE_DEVICE:
294             DPRINT("  IRP_MN_REMOVE_DEVICE received\n");
295             return ForwardIrpAndForget(DeviceObject, Irp);
296 
297         case IRP_MN_CANCEL_REMOVE_DEVICE:
298             DPRINT("  IRP_MN_CANCEL_REMOVE_DEVICE\n");
299             return ForwardIrpAndForget(DeviceObject, Irp);
300 
301         case IRP_MN_STOP_DEVICE:
302             DPRINT("  IRP_MN_STOP_DEVICE received\n");
303             return ForwardIrpAndForget(DeviceObject, Irp);
304 
305         case IRP_MN_QUERY_STOP_DEVICE:
306             DPRINT("  IRP_MN_QUERY_STOP_DEVICE received\n");
307             return ForwardIrpAndForget(DeviceObject, Irp);
308 
309         case IRP_MN_CANCEL_STOP_DEVICE:
310             DPRINT("  IRP_MN_CANCEL_STOP_DEVICE\n");
311             return ForwardIrpAndForget(DeviceObject, Irp);
312 
313         case IRP_MN_QUERY_DEVICE_RELATIONS:
314             DPRINT("  IRP_MN_QUERY_DEVICE_RELATIONS\n");
315 
316             switch (IrpSp->Parameters.QueryDeviceRelations.Type)
317             {
318                 case BusRelations:
319                     DPRINT("    IRP_MJ_PNP / IRP_MN_QUERY_DEVICE_RELATIONS / BusRelations\n");
320                     return ForwardIrpAndForget(DeviceObject, Irp);
321                     break;
322 
323                 case RemovalRelations:
324                     DPRINT("    IRP_MJ_PNP / IRP_MN_QUERY_DEVICE_RELATIONS / RemovalRelations\n");
325                     return ForwardIrpAndForget(DeviceObject, Irp);
326 
327                 default:
328                     DPRINT("    IRP_MJ_PNP / IRP_MN_QUERY_DEVICE_RELATIONS / Unknown type 0x%lx\n",
329                             IrpSp->Parameters.QueryDeviceRelations.Type);
330                     return ForwardIrpAndForget(DeviceObject, Irp);
331             }
332             break;
333 
334         case IRP_MN_SURPRISE_REMOVAL:
335             DPRINT("  IRP_MN_SURPRISE_REMOVAL received\n");
336             return ForwardIrpAndForget(DeviceObject, Irp);
337 
338         case IRP_MN_FILTER_RESOURCE_REQUIREMENTS: /* (optional) 0xd */
339             DPRINT("  IRP_MN_FILTER_RESOURCE_REQUIREMENTS\n");
340             return ForwardIrpAndForget(DeviceObject, Irp);
341 
342         default:
343             DPRINT("  Unknown IOCTL 0x%lx\n", IrpSp->MinorFunction);
344             return ForwardIrpAndForget(DeviceObject, Irp);
345     }
346 
347     Irp->IoStatus.Information = Information;
348     Irp->IoStatus.Status = Status;
349     IoCompleteRequest(Irp, IO_NO_INCREMENT);
350 
351     return Status;
352 }
353 
354 
355 NTSTATUS
356 NTAPI
357 ProcessorAddDevice(
358     IN PDRIVER_OBJECT DriverObject,
359     IN PDEVICE_OBJECT Pdo)
360 {
361     PDEVICE_EXTENSION DeviceExtension = NULL;
362     PDEVICE_OBJECT Fdo = NULL;
363     NTSTATUS Status;
364 
365     DPRINT("ProcessorAddDevice()\n");
366 
367     ASSERT(DriverObject);
368     ASSERT(Pdo);
369 
370     /* Create functional device object */
371     Status = IoCreateDevice(DriverObject,
372                             sizeof(DEVICE_EXTENSION),
373                             NULL,
374                             FILE_DEVICE_UNKNOWN,
375                             FILE_DEVICE_SECURE_OPEN,
376                             FALSE,
377                             &Fdo);
378     if (NT_SUCCESS(Status))
379     {
380         DeviceExtension = (PDEVICE_EXTENSION)Fdo->DeviceExtension;
381         RtlZeroMemory(DeviceExtension, sizeof(DEVICE_EXTENSION));
382 
383         DeviceExtension->DeviceObject = Fdo;
384 
385         Status = IoAttachDeviceToDeviceStackSafe(Fdo, Pdo, &DeviceExtension->LowerDevice);
386         if (!NT_SUCCESS(Status))
387         {
388             DPRINT1("IoAttachDeviceToDeviceStackSafe() failed with status 0x%08lx\n", Status);
389             IoDeleteDevice(Fdo);
390             return Status;
391         }
392 
393         Fdo->Flags |= DO_DIRECT_IO;
394         Fdo->Flags |= DO_POWER_PAGABLE;
395 
396         Fdo->Flags &= ~DO_DEVICE_INITIALIZING;
397     }
398 
399     return Status;
400 }
401 
402 /* EOF */
403