isowmi.c
上传用户:leituo004
上传日期:2014-11-03
资源大小:159k
文件大小:16k
源码类别:

驱动编程

开发平台:

Visual C++

  1. /*++
  2. Copyright (c) 2000 Microsoft Corporation
  3. Module Name:
  4.     isowmi.c
  5. Abstract:
  6. Environment:
  7.     Kernel mode
  8. Notes:
  9.     Copyright (c) 2000 Microsoft Corporation.  
  10.     All Rights Reserved.
  11. --*/
  12. #include "isousb.h"
  13. #include "isopnp.h"
  14. #include "isopwr.h"
  15. #include "isodev.h"
  16. #include "isowmi.h"
  17. #include "isousr.h"
  18. #include "isorwr.h"
  19. #include "isostrm.h"
  20. #define MOFRESOURCENAME L"MofResourceName"
  21. #define WMI_ISOUSB_DRIVER_INFORMATION 0
  22. DEFINE_GUID (ISOUSB_WMI_STD_DATA_GUID, 
  23. 0xBBA21300, 0x6DD3, 0x11d2, 0xB8, 0x44, 0x00, 0xC0, 0x4F, 0xAD, 0x51, 0x71);
  24. WMIGUIDREGINFO IsoWmiGuidList[1] = { {
  25.         &ISOUSB_WMI_STD_DATA_GUID, 1, 0 // driver information
  26.     }
  27. };
  28. NTSTATUS
  29. IsoUsb_WmiRegistration(
  30.     IN OUT PDEVICE_EXTENSION DeviceExtension
  31.     )
  32. /*++
  33. Routine Description:
  34.     Registers with WMI as a data provider for this
  35.     instance of the device
  36. Arguments:
  37. Return Value:
  38. --*/
  39. {
  40.     NTSTATUS ntStatus;
  41.     
  42.     PAGED_CODE();
  43.     DeviceExtension->WmiLibInfo.GuidCount = 
  44.           sizeof (IsoWmiGuidList) / sizeof (WMIGUIDREGINFO);
  45.     DeviceExtension->WmiLibInfo.GuidList           = IsoWmiGuidList;
  46.     DeviceExtension->WmiLibInfo.QueryWmiRegInfo    = IsoUsb_QueryWmiRegInfo;
  47.     DeviceExtension->WmiLibInfo.QueryWmiDataBlock  = IsoUsb_QueryWmiDataBlock;
  48.     DeviceExtension->WmiLibInfo.SetWmiDataBlock    = IsoUsb_SetWmiDataBlock;
  49.     DeviceExtension->WmiLibInfo.SetWmiDataItem     = IsoUsb_SetWmiDataItem;
  50.     DeviceExtension->WmiLibInfo.ExecuteWmiMethod   = NULL;
  51.     DeviceExtension->WmiLibInfo.WmiFunctionControl = NULL;
  52.     //
  53.     // Register with WMI
  54.     //
  55.     
  56.     ntStatus = IoWMIRegistrationControl(DeviceExtension->FunctionalDeviceObject,
  57.                                         WMIREG_ACTION_REGISTER);
  58.     return ntStatus;
  59.     
  60. }
  61. NTSTATUS
  62. IsoUsb_WmiDeRegistration(
  63.     IN OUT PDEVICE_EXTENSION DeviceExtension
  64.     )
  65. /*++
  66. Routine Description:
  67.      Inform WMI to remove this DeviceObject from its 
  68.      list of providers. This function also 
  69.      decrements the reference count of the deviceobject.
  70. Arguments:
  71. Return Value:
  72. --*/
  73. {
  74.     PAGED_CODE();
  75.     return IoWMIRegistrationControl(DeviceExtension->FunctionalDeviceObject,
  76.                                     WMIREG_ACTION_DEREGISTER);
  77. }
  78. NTSTATUS
  79. IsoUsb_DispatchSysCtrl(
  80.     IN PDEVICE_OBJECT DeviceObject,
  81.     IN PIRP           Irp
  82.     )
  83. /*++
  84.  
  85. Routine Description:
  86. Arguments:
  87. Return Value:
  88. --*/
  89. {
  90.     PDEVICE_EXTENSION       deviceExtension;
  91.     SYSCTL_IRP_DISPOSITION  disposition;
  92.     NTSTATUS                ntStatus;
  93.     PIO_STACK_LOCATION      irpStack;
  94.     
  95.     PAGED_CODE();
  96.     irpStack = IoGetCurrentIrpStackLocation (Irp);
  97.     deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
  98.     IsoUsb_DbgPrint(3, (WMIMinorFunctionString(irpStack->MinorFunction)));
  99.     if(Removed == deviceExtension->DeviceState) {
  100.         ntStatus = STATUS_DELETE_PENDING;
  101.         Irp->IoStatus.Status = ntStatus;
  102.         Irp->IoStatus.Information = 0;
  103.         IoCompleteRequest(Irp, IO_NO_INCREMENT);
  104.         return ntStatus;
  105.     }
  106.     IsoUsb_DbgPrint(3, ("IsoUsb_DispatchSysCtrl::"));
  107.     IsoUsb_IoIncrement(deviceExtension);
  108.     ntStatus = WmiSystemControl(&deviceExtension->WmiLibInfo, 
  109.                                 DeviceObject, 
  110.                                 Irp,
  111.                                 &disposition);
  112.     switch(disposition) {
  113.         case IrpProcessed: 
  114.         {
  115.             //
  116.             // This irp has been processed and may be completed or pending.
  117.             //
  118.             break;
  119.         }
  120.         
  121.         case IrpNotCompleted:
  122.         {
  123.             //
  124.             // This irp has not been completed, but has been fully processed.
  125.             // we will complete it now
  126.             //
  127.             IoCompleteRequest(Irp, IO_NO_INCREMENT);                
  128.             break;
  129.         }
  130.         
  131.         case IrpForward:
  132.         case IrpNotWmi:
  133.         {
  134.             //
  135.             // This irp is either not a WMI irp or is a WMI irp targeted
  136.             // at a device lower in the stack.
  137.             //
  138.             IoSkipCurrentIrpStackLocation (Irp);
  139.             ntStatus = IoCallDriver(deviceExtension->TopOfStackDeviceObject, 
  140.                                     Irp);
  141.             break;
  142.         }
  143.                                     
  144.         default:
  145.         {
  146.             //
  147.             // We really should never get here, but if we do just forward....
  148.             //
  149.             ASSERT(FALSE);
  150.             IoSkipCurrentIrpStackLocation (Irp);
  151.             ntStatus = IoCallDriver(deviceExtension->TopOfStackDeviceObject, 
  152.                                   Irp);
  153.             break;
  154.         }        
  155.     }
  156.     IsoUsb_DbgPrint(3, ("IsoUsb_DispatchSysCtrl::"));
  157.     IsoUsb_IoDecrement(deviceExtension);
  158.     return ntStatus;
  159. }
  160. NTSTATUS
  161. IsoUsb_QueryWmiRegInfo(
  162.     IN  PDEVICE_OBJECT  DeviceObject,
  163.     OUT ULONG           *RegFlags,
  164.     OUT PUNICODE_STRING InstanceName,
  165.     OUT PUNICODE_STRING *RegistryPath,
  166.     OUT PUNICODE_STRING MofResourceName,
  167.     OUT PDEVICE_OBJECT  *Pdo     
  168.     )
  169. /*++
  170. Routine Description:
  171.     This routine is a callback into the driver to retrieve the list of
  172.     guids or data blocks that the driver wants to register with WMI. This
  173.     routine may not pend or block. Driver should NOT call
  174.     WmiCompleteRequest.
  175. Arguments:
  176.     DeviceObject is the device whose data block is being queried
  177.     *RegFlags returns with a set of flags that describe the guids being
  178.         registered for this device. If the device wants enable and disable
  179.         collection callbacks before receiving queries for the registered
  180.         guids then it should return the WMIREG_FLAG_EXPENSIVE flag. Also the
  181.         returned flags may specify WMIREG_FLAG_INSTANCE_PDO in which case
  182.         the instance name is determined from the PDO associated with the
  183.         device object. Note that the PDO must have an associated devnode. If
  184.         WMIREG_FLAG_INSTANCE_PDO is not set then Name must return a unique
  185.         name for the device.
  186.     InstanceName returns with the instance name for the guids if
  187.         WMIREG_FLAG_INSTANCE_PDO is not set in the returned *RegFlags. The
  188.         caller will call ExFreePool with the buffer returned.
  189.     *RegistryPath returns with the registry path of the driver
  190.     *MofResourceName returns with the name of the MOF resource attached to
  191.         the binary file. If the driver does not have a mof resource attached
  192.         then this can be returned as NULL.
  193.     *Pdo returns with the device object for the PDO associated with this
  194.         device if the WMIREG_FLAG_INSTANCE_PDO flag is returned in 
  195.         *RegFlags.
  196. Return Value:
  197.     status
  198. --*/
  199. {
  200.     PDEVICE_EXTENSION deviceExtension;
  201.     PAGED_CODE();
  202.     IsoUsb_DbgPrint(3, ("IsoUsb_QueryWmiRegInfo - beginsn"));
  203.     deviceExtension = DeviceObject->DeviceExtension;
  204.     *RegFlags     = WMIREG_FLAG_INSTANCE_PDO;
  205.     *RegistryPath = &Globals.IsoUsb_RegistryPath;
  206.     *Pdo          = deviceExtension->PhysicalDeviceObject;
  207.     RtlInitUnicodeString(MofResourceName, MOFRESOURCENAME);
  208.     IsoUsb_DbgPrint(3, ("IsoUsb_QueryWmiRegInfo - endsn"));
  209.     
  210.     return STATUS_SUCCESS;
  211. }
  212. NTSTATUS
  213. IsoUsb_QueryWmiDataBlock(
  214.     IN PDEVICE_OBJECT DeviceObject,
  215.     IN PIRP           Irp,
  216.     IN ULONG          GuidIndex,
  217.     IN ULONG          InstanceIndex,
  218.     IN ULONG          InstanceCount,
  219.     IN OUT PULONG     InstanceLengthArray,
  220.     IN ULONG          OutBufferSize,
  221.     OUT PUCHAR        Buffer
  222.     )
  223. /*++
  224. Routine Description:
  225.     This routine is a callback into the driver to query for the contents of
  226.     a data block. When the driver has finished filling the data block it
  227.     must call WmiCompleteRequest to complete the irp. The driver can
  228.     return STATUS_PENDING if the irp cannot be completed immediately.
  229. Arguments:
  230.     DeviceObject is the device whose data block is being queried
  231.     Irp is the Irp that makes this request
  232.     GuidIndex is the index into the list of guids provided when the
  233.         device registered
  234.     InstanceIndex is the index that denotes which instance of the data block
  235.         is being queried.
  236.             
  237.     InstanceCount is the number of instances expected to be returned for
  238.         the data block.
  239.             
  240.     InstanceLengthArray is a pointer to an array of ULONG that returns the 
  241.         lengths of each instance of the data block. If this is NULL then
  242.         there was not enough space in the output buffer to fulfill the request
  243.         so the irp should be completed with the buffer needed.        
  244.             
  245.     OutBufferSize has the maximum size available to write the data
  246.         block.
  247.     Buffer on return is filled with the returned data block
  248. Return Value:
  249.     status
  250. --*/
  251. {
  252.     PDEVICE_EXTENSION deviceExtension;
  253.     NTSTATUS          ntStatus;
  254.     ULONG             size;
  255.     WCHAR             modelName[] = L"Aishverya";
  256.     USHORT            modelNameLen;
  257.     PAGED_CODE();
  258.     IsoUsb_DbgPrint(3, ("IsoUsb_QueryWmiDataBlock - beginsn"));
  259.     size = 0;
  260.     modelNameLen = (wcslen(modelName) + 1) * sizeof(WCHAR);
  261.     //
  262.     // Only ever registers 1 instance per guid
  263.     //
  264.     ASSERT((InstanceIndex == 0) &&
  265.            (InstanceCount == 1));
  266.     
  267.     deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
  268.     switch (GuidIndex) {
  269.     case WMI_ISOUSB_DRIVER_INFORMATION:
  270.         size = sizeof(ULONG) + modelNameLen + sizeof(USHORT);
  271.         if (OutBufferSize < size ) {
  272.             IsoUsb_DbgPrint(3, ("OutBuffer too smalln"));
  273.             ntStatus = STATUS_INVALID_BUFFER_SIZE;
  274.             break;
  275.         }
  276.         * (PULONG) Buffer = DebugLevel;
  277.         Buffer += sizeof(ULONG);
  278.         //
  279.         // put length of string ahead of string
  280.         //
  281.         *((PUSHORT)Buffer) = modelNameLen;
  282.         Buffer = (PUCHAR)Buffer + sizeof(USHORT);
  283.         RtlCopyBytes((PVOID)Buffer, (PVOID)modelName, modelNameLen);
  284.         *InstanceLengthArray = size ;
  285.         ntStatus = STATUS_SUCCESS;
  286.         break;
  287.     default:
  288.         ntStatus = STATUS_WMI_GUID_NOT_FOUND;
  289.     }
  290.     ntStatus = WmiCompleteRequest(DeviceObject,
  291.                                 Irp,
  292.                                 ntStatus,
  293.                                 size,
  294.                                 IO_NO_INCREMENT);
  295.     IsoUsb_DbgPrint(3, ("IsoUsb_QueryWmiDataBlock - endsn"));
  296.     return ntStatus;
  297. }
  298. NTSTATUS
  299. IsoUsb_SetWmiDataItem(
  300.     IN PDEVICE_OBJECT DeviceObject,
  301.     IN PIRP           Irp,
  302.     IN ULONG          GuidIndex,
  303.     IN ULONG          InstanceIndex,
  304.     IN ULONG          DataItemId,
  305.     IN ULONG          BufferSize,
  306.     IN PUCHAR         Buffer
  307.     )
  308. /*++
  309. Routine Description:
  310.     This routine is a callback into the driver to set for the contents of
  311.     a data block. When the driver has finished filling the data block it
  312.     must call WmiCompleteRequest to complete the irp. The driver can
  313.     return STATUS_PENDING if the irp cannot be completed immediately.
  314. Arguments:
  315.     DeviceObject is the device whose data block is being queried
  316.     Irp is the Irp that makes this request
  317.     GuidIndex is the index into the list of guids provided when the
  318.         device registered
  319.     InstanceIndex is the index that denotes which instance of the data block
  320.         is being queried.
  321.             
  322.     DataItemId has the id of the data item being set
  323.     BufferSize has the size of the data item passed
  324.     Buffer has the new values for the data item
  325. Return Value:
  326.     status
  327. --*/
  328. {
  329.     PDEVICE_EXTENSION deviceExtension;
  330.     NTSTATUS          ntStatus;
  331.     ULONG             info;
  332.     
  333.     PAGED_CODE();
  334.     IsoUsb_DbgPrint(3, ("IsoUsb_SetWmiDataItem - beginsn"));
  335.     deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
  336.     info = 0;
  337.     switch(GuidIndex) {
  338.     
  339.     case WMI_ISOUSB_DRIVER_INFORMATION:
  340.         if(DataItemId == 1) {
  341.             if(BufferSize == sizeof(ULONG)) {
  342.                 DebugLevel = *((PULONG)Buffer);
  343.                 ntStatus = STATUS_SUCCESS;
  344.                 info = sizeof(ULONG);
  345.             }
  346.             else {
  347.                 ntStatus = STATUS_INFO_LENGTH_MISMATCH;
  348.             }
  349.         }
  350.         else {
  351.             ntStatus = STATUS_WMI_READ_ONLY;
  352.         }
  353.         break;
  354.     default:
  355.         ntStatus = STATUS_WMI_GUID_NOT_FOUND;
  356.     }
  357.     ntStatus = WmiCompleteRequest(DeviceObject,
  358.                                 Irp,
  359.                                 ntStatus,
  360.                                 info,
  361.                                 IO_NO_INCREMENT);
  362.     IsoUsb_DbgPrint(3, ("IsoUsb_SetWmiDataItem - endsn"));
  363.     return ntStatus;
  364. }
  365. NTSTATUS
  366. IsoUsb_SetWmiDataBlock(
  367.     IN PDEVICE_OBJECT DeviceObject,
  368.     IN PIRP           Irp,
  369.     IN ULONG          GuidIndex,
  370.     IN ULONG          InstanceIndex,
  371.     IN ULONG          BufferSize,
  372.     IN PUCHAR         Buffer
  373.     )
  374. /*++
  375. Routine Description:
  376.     This routine is a callback into the driver to set the contents of
  377.     a data block. When the driver has finished filling the data block it
  378.     must call WmiCompleteRequest to complete the irp. The driver can
  379.     return STATUS_PENDING if the irp cannot be completed immediately.
  380. Arguments:
  381.     DeviceObject is the device whose data block is being queried
  382.     Irp is the Irp that makes this request
  383.     GuidIndex is the index into the list of guids provided when the
  384.         device registered
  385.     InstanceIndex is the index that denotes which instance of the data block
  386.         is being queried.
  387.             
  388.     BufferSize has the size of the data block passed
  389.     Buffer has the new values for the data block
  390. Return Value:
  391.     status
  392. --*/
  393. {
  394.     PDEVICE_EXTENSION deviceExtension;
  395.     NTSTATUS          ntStatus;
  396.     ULONG             info;
  397.     PAGED_CODE();
  398.     deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
  399.     info = 0;
  400.     IsoUsb_DbgPrint(3, ("IsoUsb_SetWmiDataBlock - beginsn"));
  401.     switch(GuidIndex) {
  402.     
  403.     case WMI_ISOUSB_DRIVER_INFORMATION:
  404.         if(BufferSize == sizeof(ULONG)) {
  405.             DebugLevel = *(PULONG) Buffer;
  406.                     
  407.             ntStatus = STATUS_SUCCESS;
  408.             info = sizeof(ULONG);
  409.         }
  410.         else {
  411.             ntStatus = STATUS_INFO_LENGTH_MISMATCH;
  412.         }
  413.         break;
  414.     default:
  415.         ntStatus = STATUS_WMI_GUID_NOT_FOUND;
  416.     }
  417.     ntStatus = WmiCompleteRequest(DeviceObject,
  418.                                 Irp,
  419.                                 ntStatus,
  420.                                 info,
  421.                                 IO_NO_INCREMENT);
  422.     IsoUsb_DbgPrint(3, ("IsoUsb_SetWmiDataBlock - endsn"));
  423.     return ntStatus;
  424. }
  425. PCHAR
  426. WMIMinorFunctionString (
  427.     UCHAR MinorFunction
  428.     )
  429. /*++
  430.  
  431. Routine Description:
  432. Arguments:
  433. Return Value:
  434. --*/
  435. {
  436.     switch (MinorFunction) {
  437.         case IRP_MN_CHANGE_SINGLE_INSTANCE:
  438.             return "IRP_MN_CHANGE_SINGLE_INSTANCEn";
  439.         case IRP_MN_CHANGE_SINGLE_ITEM:
  440.             return "IRP_MN_CHANGE_SINGLE_ITEMn";
  441.         case IRP_MN_DISABLE_COLLECTION:
  442.             return "IRP_MN_DISABLE_COLLECTIONn";
  443.         case IRP_MN_DISABLE_EVENTS:
  444.             return "IRP_MN_DISABLE_EVENTSn";
  445.         case IRP_MN_ENABLE_COLLECTION:
  446.             return "IRP_MN_ENABLE_COLLECTIONn";
  447.         case IRP_MN_ENABLE_EVENTS:
  448.             return "IRP_MN_ENABLE_EVENTSn";
  449.         case IRP_MN_EXECUTE_METHOD:
  450.             return "IRP_MN_EXECUTE_METHODn";
  451.         case IRP_MN_QUERY_ALL_DATA:
  452.             return "IRP_MN_QUERY_ALL_DATAn";
  453.         case IRP_MN_QUERY_SINGLE_INSTANCE:
  454.             return "IRP_MN_QUERY_SINGLE_INSTANCEn";
  455.         case IRP_MN_REGINFO:
  456.             return "IRP_MN_REGINFOn";
  457.         default:
  458.             return "IRP_MN_?????n";
  459.     }
  460. }