Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

6796 lines
210 KiB

  1. /*++
  2. Copyright (c) 1995 Microsoft Corporation
  3. Module Name:
  4. mofcheck.c
  5. Abstract:
  6. TODO: Enable localization
  7. Tool to validate that a binary MOF is valid for use with WMI
  8. Usage:
  9. wmimofck <binary mof file>
  10. Rules that are enforced by this program:
  11. * Any classes that do not have the WMI qualifier are ignored. Any class
  12. with the qualifier is a WMI class.
  13. * All WMI classes must have guids, including embedded classes.
  14. * All WMI classes that do not have special HMOM qualifiers [Dynamic,
  15. Provider("WMIProv")] are embedded only classes.
  16. * All non embedded WMI classes must have a property with the [key]
  17. qualifier named InstanceName, be of type string and NOT have a
  18. WmiDataId qualifier.
  19. * Embedded only classes should not have InstanceName or Active properties
  20. * All other properties in a WMI class must have a WmiDataId qualifier
  21. that specifies the position of the data item represented by the property
  22. within the data block represented by the class.
  23. * The property for the first data item in the data block must have
  24. WmiDataId(1). WmiDataId(0) is reserved.
  25. * WmiDataId qualifier values must be contiguous, ie, 1,2,3,4,5... There may
  26. not be any duplicate or missing WmiDataId values in a class.
  27. * The order of the properties as specified in the mof does not need to
  28. follow that of the WmiDataId
  29. * A property with a greater WmiDataId can not be marked with a WmiVersion
  30. qualifier whose value is lower than any properties with a lower WmiDataId
  31. * All embedded classes must be defined in the same mof
  32. * Only the following types are valid for properties:
  33. string, sint32, uint32, sint64, uint64, bool, sint16, uint16, char16
  34. sint8, uint8, datetime
  35. * Any variable length array must have a WmiSizeIs qualifier that specifies
  36. the property that holds the number of elements in the array. This
  37. property must be part of the same class as the variable length array,
  38. and this property must be an unsigned integer type (uint8, uint16,
  39. uint32, uint64)
  40. * Fixed length arrays must have the max qualifier.
  41. * An array may not be both fixed and variable length
  42. * Methods must have WmiMethodId qualifier with a unique value
  43. * Methods must have the Implemented qualifier
  44. * Methods must have a void return
  45. * Classes derrived from WmiEvent may not be abstract
  46. Author:
  47. 16-Jan-1997 AlanWar
  48. Revision History:
  49. --*/
  50. #define WMI_USER_MODE
  51. #include <stdio.h>
  52. #include <fcntl.h>
  53. #include <io.h>
  54. #include <ctype.h>
  55. #include "wmiump.h"
  56. #include "bmof.h"
  57. #include "mrcicode.h"
  58. #ifdef WmipUnreferenceMC
  59. #undef WmipUnreferenceMC
  60. #endif
  61. #define WmipUnreferenceMC(MofClass)
  62. #ifdef WmipUnreferenceMR
  63. #undef WmipUnreferenceMR
  64. #endif
  65. #define WmipUnreferenceMR(MofClass)
  66. #ifdef WmipAllocMofResource
  67. #undef WmipAllocMofResource
  68. #endif
  69. #define WmipAllocMofResource() WmipAlloc(sizeof(MOFRESOURCE))
  70. #ifdef WmipAllocMofClass
  71. #undef WmipAllocMofClass
  72. #endif
  73. #define WmipAllocMofClass() WmipAlloc(sizeof(MOFCLASS))
  74. BOOL wGUIDFromString(LPCWSTR lpsz, LPGUID pguid);
  75. #if DBG
  76. BOOLEAN WmipLoggingEnabled = FALSE;
  77. #endif
  78. #ifdef MEMPHIS
  79. #if DBG
  80. void __cdecl DebugOut(char *Format, ...)
  81. {
  82. char Buffer[1024];
  83. va_list pArg;
  84. ULONG i;
  85. va_start(pArg, Format);
  86. i = _vsnprintf(Buffer, sizeof(Buffer), Format, pArg);
  87. OutputDebugString(Buffer);
  88. }
  89. #endif
  90. #endif
  91. TCHAR *MessageText[ERROR_WMIMOF_COUNT + 5] =
  92. {
  93. TEXT("This file is not a valid binary mof file"),
  94. TEXT("There was not enough memory to complete an operation"),
  95. TEXT("Binary Mof file %s could not be opened"),
  96. TEXT("Unknown error code %d\n"),
  97. //
  98. // ERROR_WMIMOF_ messages start here
  99. TEXT("ERROR_WMIMOF_INCORRECT_DATA_TYPE"),
  100. TEXT("ERROR_WMIMOF_NO_DATA"),
  101. TEXT("ERROR_WMIMOF_NOT_FOUND"),
  102. TEXT("ERROR_WMIMOF_UNUSED"),
  103. TEXT("Property %ws in class %ws has no embedded class name"),
  104. TEXT("Property %ws in class %ws has an unknown data type"),
  105. TEXT("Property %ws in class %ws has no syntax qualifier"),
  106. TEXT("ERROR_WMIMOF_NO_SYNTAX_QUALIFIER"),
  107. TEXT("ERROR_WMIMOF_NO_CLASS_NAME"),
  108. TEXT("ERROR_WMIMOF_BAD_DATA_FORMAT"),
  109. TEXT("Property %ws in class %ws has the same WmiDataId %d as property %ws"),
  110. TEXT("Property %ws in class %ws has a WmiDataId of %d which is out of range"),
  111. TEXT("ERROR_WMIMOF_MISSING_DATAITEM"),
  112. TEXT("Property for WmiDataId %d is not defined in class %ws"),
  113. TEXT("Embedded class %ws not defined for Property %ws in Class %ws"),
  114. TEXT("Property %ws in class %ws has an incorrect [WmiVersion] qualifier"),
  115. TEXT("ERROR_WMIMOF_NO_PROPERTY_QUALIFERS"),
  116. TEXT("Class %ws has a badly formed or missing [guid] qualifier"),
  117. TEXT("Could not find property %ws which is the array size for property %ws in class %ws"),
  118. TEXT("A class could not be parsed properly"),
  119. TEXT("Wmi class %ws requires the qualifiers [Dynamic, Provider(\"WmiProv\")]"),
  120. TEXT("Error accessing binary mof file %s, code %d"),
  121. TEXT("Property InstanceName in class %ws must be type string and not %ws"),
  122. TEXT("Property Active in class %ws must be type bool and not %ws"),
  123. TEXT("Property %ws in class %ws does not have [WmiDataId()] qualifier"),
  124. TEXT("Property InstanceName in class %ws must have [key] qualifier"),
  125. TEXT("Class %ws and all its base classes do not have an InstanceName property"),
  126. TEXT("Class %ws and all its base classes do not have an Active qualifier"),
  127. TEXT("Property %ws in class %ws is an array, but doesn't specify a dimension"),
  128. TEXT("The element count property %ws for the variable length array %ws in class %ws is not an integral type"),
  129. TEXT("Property %ws in class %ws is both a fixed and variable length array"),
  130. TEXT("Embedded class %ws should be abstract or not have InstaneName or Active properties"),
  131. TEXT("Implemented qualifier required on method %ws in class %ws"),
  132. TEXT("WmiMethodId for method %ws in class %ws must be unique"),
  133. TEXT("WmiMethodId for method %ws in class %ws must be specified"),
  134. TEXT("WmiMethodId for method %ws in class %ws must not be 0"),
  135. TEXT("Class %ws is derived from WmiEvent and may not be [abstract]"),
  136. TEXT("The element count property for the variable length array %ws in class %ws is not a property of the class"),
  137. TEXT("An error occured resolving the variable length array property %ws in class %ws to element count property"),
  138. TEXT("Method %ws in class %ws must have return type void\n")
  139. };
  140. HANDLE FileHandle, MappingHandle;
  141. TCHAR *BMofFileName;
  142. BOOLEAN DoMethodHeaderGeneration;
  143. BOOLEAN ForceHeaderGeneration;
  144. //
  145. // These global variables hold the compressed buffer and size
  146. PVOID CompressedFileBuffer;
  147. ULONG CompressedSize;
  148. //
  149. // These global variables hold the uncompressed buffer and size
  150. PVOID FileBuffer;
  151. ULONG UncompressedSize;
  152. BOOLEAN SkipEmbedClassCheck;
  153. void __cdecl ErrorMessage(
  154. BOOLEAN ExitProgram,
  155. ULONG ErrorCode,
  156. ...
  157. )
  158. {
  159. va_list pArg;
  160. LONG Index;
  161. TCHAR *ErrorText;
  162. TCHAR Buffer[1024];
  163. UnmapViewOfFile(CompressedFileBuffer);
  164. CloseHandle(MappingHandle);
  165. CloseHandle(FileHandle);
  166. DeleteFile(BMofFileName);
  167. if (ErrorCode == ERROR_WMI_INVALID_MOF)
  168. {
  169. Index = 0;
  170. } else if (ErrorCode == ERROR_NOT_ENOUGH_MEMORY) {
  171. Index = 1;
  172. } else if (ErrorCode == ERROR_FILE_NOT_FOUND) {
  173. Index = 2;
  174. } else {
  175. Index = (-1 * ((LONG)ErrorCode)) + 4;
  176. }
  177. fprintf(stderr, "%s (0) : error RC2135 : ", BMofFileName);
  178. if ( (Index < 0) || (Index > (ERROR_WMIMOF_COUNT+4)))
  179. {
  180. ErrorText = MessageText[3];
  181. fprintf(stderr, ErrorText, ErrorCode);
  182. } else {
  183. ErrorText = MessageText[Index];
  184. va_start(pArg, ErrorCode);
  185. _vsnprintf(Buffer, sizeof(Buffer), ErrorText, pArg);
  186. fprintf(stderr, Buffer);
  187. fprintf(stderr, "\n");
  188. }
  189. if (ExitProgram)
  190. {
  191. ExitProcess(ErrorCode);
  192. }
  193. }
  194. typedef struct
  195. {
  196. DWORD Signature;
  197. DWORD CompressionType;
  198. DWORD CompressedSize;
  199. DWORD UncompressedSize;
  200. BYTE Buffer[];
  201. } COMPRESSEDHEADER, *PCOMPRESSEDHEADER;
  202. ULONG WmipDecompressBuffer(
  203. IN PVOID CompressedBuffer,
  204. OUT PVOID *UncompressedBuffer,
  205. OUT ULONG *UncompressedSize
  206. )
  207. /*++
  208. Routine Description:
  209. This routine will decompress a compressed MOF blob into a buffer
  210. that can be used to interpert the blob.
  211. Arguments:
  212. CompressedBuffer points at the compressed MOF blob
  213. *UncompressedBuffer returns with a pointer to the uncompressed
  214. MOF blob
  215. Return Value:
  216. ERROR_SUCCESS or an error code
  217. --*/
  218. {
  219. PCOMPRESSEDHEADER CompressedHeader = (PCOMPRESSEDHEADER)CompressedBuffer;
  220. BYTE *Buffer;
  221. ULONG Status;
  222. if ((CompressedHeader->Signature != BMOF_SIG) ||
  223. (CompressedHeader->CompressionType != 1))
  224. {
  225. WmipDebugPrint(("WMI: Invalid compressed mof header\n"));
  226. Status = ERROR_WMI_INVALID_MOF;
  227. } else {
  228. Buffer = WmipAlloc(CompressedHeader->UncompressedSize);
  229. if (Buffer == NULL)
  230. {
  231. Status = ERROR_NOT_ENOUGH_MEMORY;
  232. } else {
  233. *UncompressedSize = Mrci1Decompress(&CompressedHeader->Buffer[0],
  234. CompressedHeader->CompressedSize,
  235. Buffer,
  236. CompressedHeader->UncompressedSize);
  237. if (*UncompressedSize != CompressedHeader->UncompressedSize)
  238. {
  239. WmipDebugPrint(("WMI: Invalid compressed mof buffer\n"));
  240. WmipFree(Buffer);
  241. Status = ERROR_WMI_INVALID_MOF;
  242. } else {
  243. *UncompressedBuffer = Buffer;
  244. Status = ERROR_SUCCESS;
  245. }
  246. }
  247. }
  248. return(Status);
  249. }
  250. ULONG WmipGetDataItemIdInMofClass(
  251. PMOFCLASSINFOW MofClassInfo,
  252. PWCHAR PropertyName,
  253. ULONG *DataItemIndex
  254. )
  255. {
  256. PMOFDATAITEMW MofDataItem;
  257. ULONG i;
  258. for (i = 0; i < MofClassInfo->DataItemCount; i++)
  259. {
  260. MofDataItem = &MofClassInfo->DataItems[i];
  261. WmipAssert(MofDataItem->Name != NULL);
  262. if (_wcsicmp(PropertyName, MofDataItem->Name) == 0)
  263. {
  264. //
  265. // data item ids are 0 or 1 based depending if they are parameters
  266. // for a method or part of a data class. They match the
  267. // value in the MOF file while data item indexes within
  268. // the MofClassInfo structure are 0 based.
  269. *DataItemIndex = i;
  270. return(ERROR_SUCCESS);
  271. }
  272. }
  273. return(ERROR_WMIMOF_DATAITEM_NOT_FOUND);
  274. }
  275. PMOFCLASS WmipFindClassInMofResourceByGuid(
  276. PMOFRESOURCE MofResource,
  277. LPGUID Guid
  278. )
  279. {
  280. PLIST_ENTRY MofClassList;
  281. PMOFCLASS MofClass;
  282. MofClassList = MofResource->MRMCHead.Flink;
  283. while (MofClassList != &MofResource->MRMCHead)
  284. {
  285. MofClass = CONTAINING_RECORD(MofClassList,
  286. MOFCLASS,
  287. MCMRList);
  288. if (IsEqualGUID(&MofClass->MofClassInfo->Guid, Guid))
  289. {
  290. return(MofClass);
  291. }
  292. MofClassList = MofClassList->Flink;
  293. }
  294. return(NULL);
  295. }
  296. PMOFCLASS WmipFindClassInMofResourceByName(
  297. PMOFRESOURCE MofResource,
  298. PWCHAR ClassName
  299. )
  300. {
  301. PLIST_ENTRY MofClassList;
  302. PMOFCLASS MofClass;
  303. MofClassList = MofResource->MRMCHead.Flink;
  304. while (MofClassList != &MofResource->MRMCHead)
  305. {
  306. MofClass = CONTAINING_RECORD(MofClassList,
  307. MOFCLASS,
  308. MCMRList);
  309. if (_wcsicmp(MofClass->MofClassInfo->Name, ClassName) == 0)
  310. {
  311. return(MofClass);
  312. }
  313. MofClassList = MofClassList->Flink;
  314. }
  315. return(NULL);
  316. }
  317. ULONG WmipFillEmbeddedClasses(
  318. PMOFRESOURCE MofResource,
  319. PMOFCLASSINFOW MofClassInfo,
  320. BOOLEAN CleanupOnly
  321. )
  322. {
  323. PWCHAR ClassName;
  324. PMOFDATAITEMW MofDataItem;
  325. ULONG i;
  326. PMOFCLASS EmbeddedMofClass;
  327. ULONG Status = ERROR_SUCCESS;
  328. WCHAR *EmbeddedClassName;
  329. for (i = 0; i < MofClassInfo->DataItemCount; i++)
  330. {
  331. MofDataItem = &MofClassInfo->DataItems[i];
  332. if (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  333. {
  334. ClassName = (PWCHAR)MofDataItem->EcTempPtr;
  335. #if DBG
  336. printf("Look for embdedded class %ws (%x) for %ws in class %ws\n",
  337. ClassName, (ULONG_PTR)ClassName, MofDataItem->Name, MofClassInfo->Name);
  338. #endif
  339. if (! CleanupOnly)
  340. {
  341. EmbeddedClassName = ClassName+(sizeof(L"object") / sizeof(WCHAR));
  342. EmbeddedMofClass = WmipFindClassInMofResourceByName(
  343. MofResource,
  344. EmbeddedClassName);
  345. if (EmbeddedMofClass != NULL)
  346. {
  347. memcpy(&MofDataItem->EmbeddedClassGuid,
  348. &EmbeddedMofClass->MofClassInfo->Guid,
  349. sizeof(GUID));
  350. } else if (SkipEmbedClassCheck) {
  351. MofDataItem->Flags |= MOFDI_FLAG_EC_GUID_NOT_SET;
  352. } else {
  353. Status = ERROR_WMIMOF_EMBEDDED_CLASS_NOT_FOUND;
  354. ErrorMessage(TRUE,
  355. ERROR_WMIMOF_EMBEDDED_CLASS_NOT_FOUND,
  356. EmbeddedClassName,
  357. MofDataItem->Name,
  358. MofClassInfo->Name);
  359. }
  360. }
  361. // Don't free ClassName since we may use MofDataItem->EcTempPtr later
  362. }
  363. }
  364. //
  365. // Resolve guids for embedded classes in any method parameter classinfo
  366. //
  367. for (i = 0; i < MofClassInfo->MethodCount; i++)
  368. {
  369. MofDataItem = &MofClassInfo->DataItems[i+MofClassInfo->DataItemCount];
  370. WmipFillEmbeddedClasses(MofResource,
  371. MofDataItem->MethodClassInfo,
  372. CleanupOnly);
  373. }
  374. return(Status);
  375. }
  376. void WmipFreeMofClassInfo(
  377. PMOFCLASSINFOW MofClassInfo
  378. )
  379. /*++
  380. Routine Description:
  381. This routine frees any memory allocated for the MofClassInfo and then
  382. the MofClassInfo itself
  383. Arguments:
  384. MofClassInfo is a pointer to a MofClassInfo structure
  385. Return Value:
  386. --*/
  387. {
  388. ULONG i;
  389. PMOFDATAITEMW MofDataItem;
  390. WmipAssert(MofClassInfo != NULL);
  391. if (MofClassInfo != NULL)
  392. {
  393. if (MofClassInfo->Name != NULL)
  394. {
  395. WmipFree(MofClassInfo->Name);
  396. }
  397. if (MofClassInfo->Description != NULL)
  398. {
  399. WmipFree(MofClassInfo->Description);
  400. }
  401. for (i = 0; i < MofClassInfo->DataItemCount + MofClassInfo->MethodCount; i++)
  402. {
  403. MofDataItem = &MofClassInfo->DataItems[i];
  404. if (MofDataItem->Name != NULL)
  405. {
  406. WmipFree(MofDataItem->Name);
  407. }
  408. if (MofDataItem->Description != NULL)
  409. {
  410. WmipFree(MofDataItem->Description);
  411. }
  412. }
  413. }
  414. }
  415. ULONG WmipFindProperty(
  416. CBMOFObj * ClassObject,
  417. WCHAR * PropertyName,
  418. CBMOFDataItem *MofPropertyData,
  419. DWORD *ValueType,
  420. PVOID ValueBuffer
  421. )
  422. /*++
  423. Routine Description:
  424. This routine will find a named property within a class object
  425. Arguments:
  426. ClassObject is the class object in which to search
  427. PropertyName is the name of the property to search for
  428. MofPropertyData returns with the property data
  429. *ValueType on entry has the property data type being searched for. On exit
  430. it has the actual qualifier type for the qualifier value. If on entry
  431. *ValueType is 0xffffffff then any data type is acceptable
  432. ValueBuffer points to a buffer that returns the value of the
  433. property. If the property is a simple type (int or int64) then
  434. the value is returned in the buffer. If qualifier value is a string
  435. then a pointer to the string is returned in the buffer
  436. Return Value:
  437. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  438. --*/
  439. {
  440. ULONG Status;
  441. if (FindProp(ClassObject, PropertyName, MofPropertyData))
  442. {
  443. if ((*ValueType != 0xffffffff) &&
  444. (MofPropertyData->m_dwType != *ValueType))
  445. {
  446. Status = ERROR_WMIMOF_INCORRECT_DATA_TYPE;
  447. }
  448. if (GetData(MofPropertyData, (BYTE *)ValueBuffer, 0) == 0)
  449. {
  450. Status = ERROR_WMIMOF_NO_DATA;
  451. } else {
  452. *ValueType = MofPropertyData->m_dwType;
  453. Status = ERROR_SUCCESS;
  454. }
  455. } else {
  456. Status = ERROR_WMIMOF_NOT_FOUND;
  457. }
  458. return(Status);
  459. }
  460. ULONG WmipFindMofQualifier(
  461. CBMOFQualList *QualifierList,
  462. LPCWSTR QualifierName,
  463. DWORD *QualifierType,
  464. DWORD *NumberElements,
  465. PVOID QualifierValueBuffer
  466. )
  467. /*++
  468. Routine Description:
  469. This routine will find a MOF qualifier within the qualifier list passed,
  470. ensure that its type matches the type requested and return the qualifier's
  471. value
  472. Arguments:
  473. QualifierList is the MOF qualifier list
  474. QualifierName is the name of the qualifier to search for
  475. *QualifierType on entry has the qualifier type being searched for. On exit
  476. it has the actual qualifier type for the qualifier value. If on entry
  477. *QualifierType is 0xffffffff then any qualifier type is acceptable
  478. *NumberElements returns the number of elements in the array if the result
  479. of the qualifier is an array
  480. QualifierValueBuffer points to a buffer that returns the value of the
  481. qualifier. If the qualifier is a simple type (int or int64) then
  482. the value is returned in the buffer. If qualifier value is a string
  483. then a pointer to the string is returned in the buffer
  484. Return Value:
  485. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  486. --*/
  487. {
  488. CBMOFDataItem MofDataItem;
  489. ULONG Status;
  490. PUCHAR List, ListPtr;
  491. ULONG BaseTypeSize;
  492. ULONG ElementCount;
  493. ULONG i;
  494. if (FindQual(QualifierList, (PWCHAR)QualifierName, &MofDataItem))
  495. {
  496. if ((*QualifierType != 0xffffffff) &&
  497. (MofDataItem.m_dwType != *QualifierType))
  498. {
  499. Status = ERROR_WMIMOF_INCORRECT_DATA_TYPE;
  500. }
  501. if (MofDataItem.m_dwType & VT_ARRAY)
  502. {
  503. if (MofDataItem.m_dwType == (VT_BSTR | VT_ARRAY))
  504. {
  505. BaseTypeSize = sizeof(PWCHAR);
  506. } else {
  507. BaseTypeSize = iTypeSize(MofDataItem.m_dwType);
  508. }
  509. ElementCount = GetNumElements(&MofDataItem, 0);
  510. if (NumberElements != NULL)
  511. {
  512. *NumberElements = ElementCount;
  513. }
  514. if (ElementCount != -1)
  515. {
  516. List = WmipAlloc(ElementCount * BaseTypeSize);
  517. if (List != NULL)
  518. {
  519. ListPtr = List;
  520. for (i = 0; i < ElementCount; i++)
  521. {
  522. if ((GetData(&MofDataItem,
  523. (BYTE *)ListPtr,
  524. &i)) == 0)
  525. {
  526. WmipFree(List);
  527. Status = ERROR_WMIMOF_NO_DATA;
  528. return(Status);
  529. }
  530. ListPtr += BaseTypeSize;
  531. }
  532. Status = ERROR_SUCCESS;
  533. *QualifierType = MofDataItem.m_dwType;
  534. *((PVOID *)QualifierValueBuffer) = List;
  535. } else {
  536. Status = ERROR_NOT_ENOUGH_MEMORY;
  537. }
  538. } else {
  539. Status = ERROR_WMIMOF_NOT_FOUND;
  540. }
  541. } else {
  542. if (GetData(&MofDataItem, (BYTE *)QualifierValueBuffer, 0) == 0)
  543. {
  544. Status = ERROR_WMIMOF_NO_DATA;
  545. } else {
  546. *QualifierType = MofDataItem.m_dwType;
  547. Status = ERROR_SUCCESS;
  548. }
  549. }
  550. } else {
  551. Status = ERROR_WMIMOF_NOT_FOUND;
  552. }
  553. return(Status);
  554. }
  555. MOFDATATYPE VTToMofDataTypeMap[] =
  556. {
  557. MOFUnknown, /* VT_EMPTY= 0, */
  558. MOFUnknown, /* VT_NULL = 1, */
  559. MOFInt16, /* VT_I2 = 2, */
  560. MOFInt32, /* VT_I4 = 3, */
  561. MOFUnknown, /* VT_R4 = 4, */
  562. MOFUnknown, /* VT_R8 = 5, */
  563. MOFUnknown, /* VT_CY = 6, */
  564. MOFUnknown, /* VT_DATE = 7, */
  565. MOFString, /* VT_BSTR = 8, */
  566. MOFUnknown, /* VT_DISPATCH = 9, */
  567. MOFUnknown, /* VT_ERROR = 10, */
  568. MOFBoolean, /* VT_BOOL = 11, */
  569. MOFUnknown, /* VT_VARIANT = 12, */
  570. MOFUnknown, /* VT_UNKNOWN = 13, */
  571. MOFUnknown, /* VT_DECIMAL = 14, */
  572. MOFChar, /* VT_I1 = 16, */
  573. MOFByte, /* VT_UI1 = 17, */
  574. MOFUInt16, /* VT_UI2 = 18, */
  575. MOFUInt32, /* VT_UI4 = 19, */
  576. MOFInt64, /* VT_I8 = 20, */
  577. MOFUInt64, /* VT_UI8 = 21, */
  578. };
  579. MOFDATATYPE WmipVTToMofDataType(
  580. DWORD VariantType
  581. )
  582. {
  583. MOFDATATYPE MofDataType;
  584. if (VariantType < (sizeof(VTToMofDataTypeMap) / sizeof(MOFDATATYPE)))
  585. {
  586. MofDataType = VTToMofDataTypeMap[VariantType];
  587. } else {
  588. MofDataType = MOFUnknown;
  589. }
  590. return(MofDataType);
  591. }
  592. ULONG WmipGetClassDataItemCount(
  593. CBMOFObj * ClassObject,
  594. PWCHAR QualifierToFind
  595. )
  596. /*++
  597. Routine Description:
  598. This routine will count the number of WMI data items in the class and
  599. the total number of properties in the class.
  600. Arguments:
  601. ClassObject is class for which we count the number of data items
  602. *TotalCount returns the total number of properties
  603. Return Value:
  604. Count of methods
  605. --*/
  606. {
  607. CBMOFQualList *PropQualifierList;
  608. CBMOFDataItem MofPropertyData;
  609. DWORD QualifierType;
  610. ULONG Counter = 0;
  611. WCHAR *PropertyName;
  612. ULONG Status;
  613. ULONG Index;
  614. ResetObj(ClassObject);
  615. while ((NextProp(ClassObject, &PropertyName, &MofPropertyData)) &&
  616. PropertyName != NULL)
  617. {
  618. PropQualifierList = GetPropQualList(ClassObject, PropertyName);
  619. if (PropQualifierList != NULL)
  620. {
  621. //
  622. // Get the id of the property so we know it order in class
  623. QualifierType = VT_I4;
  624. Status = WmipFindMofQualifier(PropQualifierList,
  625. QualifierToFind,
  626. &QualifierType,
  627. NULL,
  628. (PVOID)&Index);
  629. if (Status == ERROR_SUCCESS)
  630. {
  631. Counter++;
  632. }
  633. BMOFFree(PropQualifierList);
  634. }
  635. BMOFFree(PropertyName);
  636. }
  637. return(Counter);
  638. }
  639. ULONG WmipGetClassMethodCount(
  640. CBMOFObj * ClassObject
  641. )
  642. /*++
  643. Routine Description:
  644. This routine will count the number of WMI data items in the class
  645. Arguments:
  646. ClassObject is class for which we count the number of data items
  647. Return Value:
  648. Count of methods
  649. --*/
  650. {
  651. ULONG MethodCount;
  652. WCHAR *MethodName;
  653. CBMOFDataItem MofMethodData;
  654. ResetObj(ClassObject);
  655. MethodCount = 0;
  656. while(NextMeth(ClassObject, &MethodName, &MofMethodData))
  657. {
  658. MethodCount++;
  659. if (MethodName != NULL)
  660. {
  661. BMOFFree(MethodName);
  662. }
  663. }
  664. return(MethodCount);
  665. }
  666. ULONG WmipParsePropertyObject(
  667. OUT PMOFDATAITEMW MofDataItem,
  668. IN CBMOFDataItem *MofPropertyData,
  669. IN CBMOFQualList *PropQualifierList
  670. )
  671. /*++
  672. Routine Description:
  673. This routine will parse a BMOF object that is known to be an object that
  674. contains a property and fill int the MofDataItem that holds its
  675. information. If the routine detects an error then parsing will stop as
  676. the whole class will need to be rejected.
  677. No error messages are generated from this routine. The return error
  678. code is checked and an appropriate message generated.
  679. Arguments:
  680. MofDataItem is the mof data item information structure to fill in
  681. MofPropertyData has the property information for the data item
  682. PropQualifierList has the qualifier list for the property
  683. Return Value:
  684. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  685. --*/
  686. {
  687. short BooleanValue;
  688. DWORD QualifierType;
  689. ULONG Status;
  690. WCHAR *StringPtr;
  691. ULONG FixedArraySize;
  692. DWORD VTDataType;
  693. DWORD VersionValue;
  694. BOOLEAN FreeString = TRUE;
  695. PMOFCLASSINFOW MethodClassInfo;
  696. ULONG MaxLen;
  697. //
  698. // Keep track of property and property qualifier objects
  699. MofDataItem->PropertyQualifierHandle = (ULONG_PTR)PropQualifierList;
  700. //
  701. // Get the description string which is not required
  702. QualifierType = VT_BSTR;
  703. Status = WmipFindMofQualifier(PropQualifierList,
  704. L"description",
  705. &QualifierType,
  706. NULL,
  707. (PVOID)&MofDataItem->Description);
  708. //
  709. // Get the version value which is not required
  710. QualifierType = VT_I4;
  711. Status = WmipFindMofQualifier(PropQualifierList,
  712. L"WmiVersion",
  713. &QualifierType,
  714. NULL,
  715. (PVOID)&VersionValue);
  716. if (Status == ERROR_SUCCESS)
  717. {
  718. MofDataItem->Version = VersionValue;
  719. }
  720. //
  721. // Get read qualifier which is not required
  722. QualifierType = VT_BOOL;
  723. Status = WmipFindMofQualifier(PropQualifierList,
  724. L"read",
  725. &QualifierType,
  726. NULL,
  727. (PVOID)&BooleanValue);
  728. if ((Status == ERROR_SUCCESS) && BooleanValue)
  729. {
  730. MofDataItem->Flags |= MOFDI_FLAG_READABLE;
  731. }
  732. //
  733. // Get write qualifier which is not required
  734. QualifierType = VT_BOOL;
  735. Status = WmipFindMofQualifier(PropQualifierList,
  736. L"write",
  737. &QualifierType,
  738. NULL,
  739. (PVOID)&BooleanValue);
  740. if ((Status == ERROR_SUCCESS) && BooleanValue)
  741. {
  742. MofDataItem->Flags |= MOFDI_FLAG_WRITEABLE;
  743. }
  744. //
  745. // Get WmiEvent qualifier which is not required
  746. QualifierType = VT_BOOL;
  747. Status = WmipFindMofQualifier(PropQualifierList,
  748. L"WmiEvent",
  749. &QualifierType,
  750. NULL,
  751. (PVOID)&BooleanValue);
  752. if ((Status == ERROR_SUCCESS) && BooleanValue)
  753. {
  754. MofDataItem->Flags |= MOFDI_FLAG_EVENT;
  755. }
  756. //
  757. // See if this is a fixed length array
  758. QualifierType = VT_I4;
  759. Status = WmipFindMofQualifier(PropQualifierList,
  760. L"max",
  761. &QualifierType,
  762. NULL,
  763. (PVOID)&FixedArraySize);
  764. if (Status == ERROR_SUCCESS)
  765. {
  766. MofDataItem->Flags |= MOFDI_FLAG_FIXED_ARRAY;
  767. MofDataItem->FixedArrayElements = FixedArraySize;
  768. #if DBG
  769. printf(" Fixed Array");
  770. #endif
  771. }
  772. //
  773. // See if this is a fixed length array
  774. QualifierType = VT_I4;
  775. Status = WmipFindMofQualifier(PropQualifierList,
  776. L"MaxLen",
  777. &QualifierType,
  778. NULL,
  779. (PVOID)&MaxLen);
  780. if (Status == ERROR_SUCCESS)
  781. {
  782. MofDataItem->MaxLen = MaxLen;
  783. }
  784. //
  785. // See if maxmium length
  786. QualifierType = VT_BSTR;
  787. Status = WmipFindMofQualifier(PropQualifierList,
  788. L"WmiSizeIs",
  789. &QualifierType,
  790. NULL,
  791. (PVOID)&StringPtr);
  792. if (Status == ERROR_SUCCESS)
  793. {
  794. if (MofDataItem->Flags & MOFDI_FLAG_FIXED_ARRAY)
  795. {
  796. BMOFFree(StringPtr);
  797. return(ERROR_WMIMOF_BOTH_FIXED_AND_VARIABLE_ARRAY);
  798. }
  799. MofDataItem->Flags |= MOFDI_FLAG_VARIABLE_ARRAY;
  800. //
  801. // When all properties are parsed we will come back and compute the
  802. // data item id for the data item that holds the number of elements
  803. // in the array. For now we'll hang onto the string pointer
  804. MofDataItem->VarArrayTempPtr = StringPtr;
  805. #if DBG
  806. printf(" Variable Array of %ws", StringPtr);
  807. #endif
  808. }
  809. if ((MofPropertyData->m_dwType & VT_ARRAY) &&
  810. ((MofDataItem->Flags & (MOFDI_FLAG_VARIABLE_ARRAY | MOFDI_FLAG_FIXED_ARRAY)) == 0))
  811. {
  812. return(ERROR_WMIMOF_MUST_DIM_ARRAY);
  813. }
  814. //
  815. // Now figure out the data type and size of the data item
  816. VTDataType = MofPropertyData->m_dwType & ~(VT_ARRAY | VT_BYREF);
  817. QualifierType = VT_BSTR;
  818. Status = WmipFindMofQualifier(PropQualifierList,
  819. L"CIMTYPE",
  820. &QualifierType,
  821. NULL,
  822. (PVOID)&StringPtr);
  823. if (VTDataType == VT_DISPATCH)
  824. {
  825. //
  826. // This is an embedded class
  827. MofDataItem->DataType = MOFEmbedded;
  828. if (Status == ERROR_SUCCESS)
  829. {
  830. //
  831. // We will resolve the class name to its guid later so we
  832. // just hang onto the embedded class name here.
  833. MofDataItem->EcTempPtr = StringPtr;
  834. MofDataItem->Flags |= MOFDI_FLAG_EMBEDDED_CLASS;
  835. #if DBG
  836. printf(" Embedded Class of %ws (%x)", StringPtr, (ULONG_PTR)StringPtr);
  837. #endif
  838. } else {
  839. return(ERROR_WMIMOF_NO_EMBEDDED_CLASS_NAME);
  840. }
  841. } else {
  842. if (Status == ERROR_SUCCESS)
  843. {
  844. if (_wcsnicmp(StringPtr, L"object:", 7) == 0) {
  845. MofDataItem->DataType = MOFEmbedded;
  846. MofDataItem->EcTempPtr = StringPtr;
  847. MofDataItem->Flags |= MOFDI_FLAG_EMBEDDED_CLASS;
  848. FreeString = FALSE;
  849. #if DBG
  850. printf(" Embedded Class of %ws (%x)", StringPtr, (ULONG_PTR)StringPtr);
  851. #endif
  852. } else if (_wcsicmp(StringPtr, L"string") == 0) {
  853. MofDataItem->DataType = MOFString;
  854. } else if (_wcsicmp(StringPtr, L"sint32") == 0) {
  855. MofDataItem->DataType = MOFInt32;
  856. MofDataItem->SizeInBytes = 4;
  857. } else if (_wcsicmp(StringPtr, L"uint32") == 0) {
  858. MofDataItem->DataType = MOFUInt32;
  859. MofDataItem->SizeInBytes = 4;
  860. } else if (_wcsicmp(StringPtr, L"boolean") == 0) {
  861. MofDataItem->DataType = MOFBoolean;
  862. MofDataItem->SizeInBytes = 1;
  863. } else if (_wcsicmp(StringPtr, L"sint64") == 0) {
  864. MofDataItem->DataType = MOFInt64;
  865. MofDataItem->SizeInBytes = 8;
  866. } else if (_wcsicmp(StringPtr, L"uint64") == 0) {
  867. MofDataItem->DataType = MOFUInt64;
  868. MofDataItem->SizeInBytes = 8;
  869. } else if ((_wcsicmp(StringPtr, L"sint16") == 0) ||
  870. (_wcsicmp(StringPtr, L"char16") == 0)) {
  871. MofDataItem->DataType = MOFInt16;
  872. MofDataItem->SizeInBytes = 2;
  873. } else if (_wcsicmp(StringPtr, L"uint16") == 0) {
  874. MofDataItem->DataType = MOFUInt16;
  875. MofDataItem->SizeInBytes = 2;
  876. } else if (_wcsicmp(StringPtr, L"sint8") == 0) {
  877. MofDataItem->DataType = MOFChar;
  878. MofDataItem->SizeInBytes = 1;
  879. } else if (_wcsicmp(StringPtr, L"uint8") == 0) {
  880. MofDataItem->DataType = MOFByte;
  881. MofDataItem->SizeInBytes = 1;
  882. } else if (_wcsicmp(StringPtr, L"datetime") == 0) {
  883. MofDataItem->DataType = MOFDate;
  884. MofDataItem->SizeInBytes = 25;
  885. } else {
  886. WmipDebugPrint(("WMI: Unknown data item syntax %ws\n",
  887. StringPtr));
  888. BMOFFree(StringPtr);
  889. return(ERROR_WMIMOF_UNKNOWN_DATA_TYPE);
  890. }
  891. if (FreeString)
  892. {
  893. BMOFFree(StringPtr);
  894. }
  895. //
  896. // If fixed array then multiply number elements by element size
  897. if ((MofDataItem->SizeInBytes != 0) &&
  898. (MofDataItem->Flags & MOFDI_FLAG_FIXED_ARRAY))
  899. {
  900. MofDataItem->SizeInBytes *= MofDataItem->FixedArrayElements;
  901. }
  902. } else {
  903. WmipDebugPrint(("WMI: No Syntax qualifier for %ws\n",
  904. MofDataItem->Name));
  905. return(ERROR_WMIMOF_NO_SYNTAX_QUALIFIER);
  906. }
  907. }
  908. return(ERROR_SUCCESS);
  909. }
  910. ULONG WmipParseMethodObject(
  911. CBMOFDataItem *MofMethodData,
  912. CBMOFQualList *PropQualifierList,
  913. PULONG MethodId
  914. )
  915. {
  916. ULONG UlongValue;
  917. ULONG Status;
  918. DWORD QualifierType;
  919. short BooleanValue;
  920. QualifierType = VT_BOOL;
  921. Status = WmipFindMofQualifier(PropQualifierList,
  922. L"Implemented",
  923. &QualifierType,
  924. NULL,
  925. (PVOID)&BooleanValue);
  926. if ((Status != ERROR_SUCCESS) || (! BooleanValue))
  927. {
  928. return(ERROR_WMIMOF_IMPLEMENTED_REQUIRED);
  929. }
  930. QualifierType = VT_I4;
  931. Status = WmipFindMofQualifier(PropQualifierList,
  932. L"WmiMethodId",
  933. &QualifierType,
  934. NULL,
  935. (PVOID)&UlongValue);
  936. if (Status != ERROR_SUCCESS)
  937. {
  938. return(ERROR_WMIMOF_IMPLEMENTED_REQUIRED);
  939. }
  940. if (UlongValue == 0)
  941. {
  942. return(ERROR_WMIMOF_METHODID_ZERO);
  943. }
  944. *MethodId = UlongValue;
  945. return(ERROR_SUCCESS);
  946. }
  947. ULONG WmipResolveVLArray(
  948. IN PMOFCLASSINFOW MofClassInfo,
  949. IN OUT PMOFDATAITEMW MofDataItem,
  950. IN ULONG FinalStatus
  951. )
  952. /*++
  953. Routine Description:
  954. This routine will resolve the array index for a variable length array
  955. Arguments:
  956. MofCLassInfo is the class info for the class
  957. MofDataItem is the mof data item that is a variable length array and
  958. whole index needs to be resolved.
  959. FinalStatus is the status of the mof parsing previously done for the
  960. class
  961. Return Value:
  962. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  963. --*/
  964. {
  965. MOFDATATYPE ArraySizeDataType;
  966. PWCHAR PropertyName;
  967. ULONG Status;
  968. ULONG Index;
  969. PropertyName = (PWCHAR)MofDataItem->VarArrayTempPtr;
  970. if (FinalStatus == ERROR_SUCCESS)
  971. {
  972. //
  973. // Only resolve this in the case where the class parsing
  974. // has not failed. We kept the name of the property containing the
  975. // number of elements in the array handy, so we need to
  976. // resolve it to its data item id and free the name.
  977. //
  978. Status = WmipGetDataItemIdInMofClass(MofClassInfo,
  979. PropertyName,
  980. &Index);
  981. if (Status != ERROR_SUCCESS)
  982. {
  983. FinalStatus = Status;
  984. } else {
  985. if ((MofClassInfo->Flags & MOFCI_FLAG_METHOD_PARAMS) ==
  986. MOFCI_FLAG_METHOD_PARAMS)
  987. {
  988. MofDataItem->VariableArraySizeId = Index;
  989. } else {
  990. MofDataItem->VariableArraySizeId = Index + 1;
  991. }
  992. ArraySizeDataType = MofClassInfo->DataItems[Index].DataType;
  993. if ((ArraySizeDataType != MOFInt32) &&
  994. (ArraySizeDataType != MOFUInt32) &&
  995. (ArraySizeDataType != MOFInt64) &&
  996. (ArraySizeDataType != MOFUInt64) &&
  997. (ArraySizeDataType != MOFInt16) &&
  998. (ArraySizeDataType != MOFUInt16))
  999. {
  1000. FinalStatus = ERROR_WMIMOF_BAD_VL_ARRAY_SIZE_TYPE;
  1001. }
  1002. }
  1003. }
  1004. BMOFFree(PropertyName);
  1005. PropertyName = NULL;
  1006. return(FinalStatus);
  1007. }
  1008. ULONG WmipParseMethodInOutObject(
  1009. CBMOFObj *ClassObject,
  1010. PMOFCLASSINFOW ClassInfo,
  1011. ULONG DataItemCount
  1012. )
  1013. /*++
  1014. Routine Description:
  1015. This routine will parse a class object that is either the in or out
  1016. parameters for a method.
  1017. Arguments:
  1018. ClassObject is the in or out parameter class object to parse
  1019. ClassInfo returns updated with information from ClassObject
  1020. DataItemCount is the number of data items in the ClassInfo
  1021. Return Value:
  1022. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  1023. --*/
  1024. {
  1025. ULONG Status;
  1026. CBMOFDataItem MofPropertyData;
  1027. PWCHAR PropertyName = NULL;
  1028. ULONG Index;
  1029. PMOFDATAITEMW MofDataItem;
  1030. CBMOFQualList *PropQualifierList = NULL;
  1031. DWORD QualifierType;
  1032. short BooleanValue;
  1033. ResetObj(ClassObject);
  1034. while (NextProp(ClassObject, &PropertyName, &MofPropertyData))
  1035. {
  1036. PropQualifierList = GetPropQualList(ClassObject, PropertyName);
  1037. if (PropQualifierList != NULL)
  1038. {
  1039. //
  1040. // Get the id of the property so we know its order in class
  1041. //
  1042. QualifierType = VT_I4;
  1043. Status = WmipFindMofQualifier(PropQualifierList,
  1044. L"Id",
  1045. &QualifierType,
  1046. NULL,
  1047. (PVOID)&Index);
  1048. if (Status == ERROR_SUCCESS)
  1049. {
  1050. //
  1051. // Method ids are 0 based
  1052. //
  1053. if (Index < DataItemCount)
  1054. {
  1055. //
  1056. // Valid data item id, make sure it already isn't
  1057. // in use. Note that we could have the same property
  1058. // be in both the in and the out class objects
  1059. //
  1060. MofDataItem = &ClassInfo->DataItems[Index];
  1061. //
  1062. // See if this is an input, output or both
  1063. //
  1064. QualifierType = VT_BOOL;
  1065. Status = WmipFindMofQualifier(PropQualifierList,
  1066. L"in",
  1067. &QualifierType,
  1068. NULL,
  1069. (PVOID)&BooleanValue);
  1070. if ((Status == ERROR_SUCCESS) && BooleanValue)
  1071. {
  1072. MofDataItem->Flags |= MOFDI_FLAG_INPUT_METHOD;
  1073. }
  1074. QualifierType = VT_BOOL;
  1075. Status = WmipFindMofQualifier(PropQualifierList,
  1076. L"out",
  1077. &QualifierType,
  1078. NULL,
  1079. (PVOID)&BooleanValue);
  1080. if ((Status == ERROR_SUCCESS) && BooleanValue)
  1081. {
  1082. MofDataItem->Flags |= MOFDI_FLAG_OUTPUT_METHOD;
  1083. }
  1084. if ((MofDataItem->Name != NULL) &&
  1085. (wcscmp(MofDataItem->Name, PropertyName) != 0))
  1086. {
  1087. //
  1088. // id already in use
  1089. //
  1090. Status = ERROR_WMIMOF_DUPLICATE_ID;
  1091. goto done;
  1092. }
  1093. if (MofDataItem->Name == NULL)
  1094. {
  1095. MofDataItem->Name = PropertyName;
  1096. } else {
  1097. BMOFFree(PropertyName);
  1098. }
  1099. PropertyName = NULL;
  1100. Status = WmipParsePropertyObject(
  1101. MofDataItem,
  1102. &MofPropertyData,
  1103. PropQualifierList);
  1104. if (Status != ERROR_SUCCESS)
  1105. {
  1106. goto done;
  1107. }
  1108. } else {
  1109. //
  1110. // Method ID qualifier is out of range
  1111. //
  1112. Status = ERROR_WMIMOF_BAD_DATAITEM_ID;
  1113. goto done;
  1114. }
  1115. } else {
  1116. //
  1117. // property is supposed to have a method id !!
  1118. //
  1119. Status = ERROR_WMIMOF_METHOD_RETURN_NOT_VOID;
  1120. goto done;
  1121. }
  1122. }
  1123. }
  1124. done:
  1125. if (PropertyName != NULL)
  1126. {
  1127. BMOFFree(PropertyName);
  1128. PropertyName = NULL;
  1129. }
  1130. if (PropQualifierList != NULL)
  1131. {
  1132. BMOFFree(PropQualifierList);
  1133. }
  1134. return(Status);
  1135. }
  1136. ULONG WmipParseMethodParameterObjects(
  1137. IN CBMOFObj *InObject,
  1138. IN CBMOFObj *OutObject,
  1139. OUT PMOFCLASSINFOW *ClassInfo
  1140. )
  1141. /*++
  1142. Routine Description:
  1143. This routine will parse the in and out method parameter obejcts to create
  1144. a MOFCLASSINFO that describes the method call.
  1145. Arguments:
  1146. InObject is the object with the input parameters
  1147. OutObject is the object with the output parameters
  1148. *ClassInfo returns with the class info for the method call
  1149. Return Value:
  1150. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  1151. --*/
  1152. {
  1153. PMOFCLASSINFOW MofClassInfo;
  1154. ULONG Status, FinalStatus;
  1155. DWORD QualifierType;
  1156. ULONG DataItemCount;
  1157. ULONG MofClassInfoSize;
  1158. WCHAR *StringPtr;
  1159. ULONG i, Index;
  1160. ULONG InItemCount, OutItemCount;
  1161. ULONG Size;
  1162. PMOFDATAITEMW MofDataItem;
  1163. ULONG Count;
  1164. FinalStatus = ERROR_SUCCESS;
  1165. if (InObject != NULL)
  1166. {
  1167. ResetObj(InObject);
  1168. InItemCount = WmipGetClassDataItemCount(InObject, L"Id");
  1169. } else {
  1170. InItemCount = 0;
  1171. }
  1172. if (OutObject != NULL)
  1173. {
  1174. ResetObj(OutObject);
  1175. OutItemCount = WmipGetClassDataItemCount(OutObject, L"Id");
  1176. } else {
  1177. OutItemCount = 0;
  1178. }
  1179. DataItemCount = InItemCount + OutItemCount;
  1180. Size = sizeof(MOFCLASSINFOW) + DataItemCount * sizeof(MOFDATAITEMW);
  1181. MofClassInfo = WmipAlloc(Size);
  1182. if (MofClassInfo == NULL)
  1183. {
  1184. return(ERROR_NOT_ENOUGH_MEMORY);
  1185. }
  1186. //
  1187. // Get the essential information to fill in the MOF class info
  1188. memset(MofClassInfo, 0, Size);
  1189. MofClassInfo->Flags |= MOFCI_FLAG_METHOD_PARAMS;
  1190. MofClassInfo->DataItems = (PMOFDATAITEMW) ((PUCHAR)MofClassInfo +
  1191. sizeof(MOFCLASSINFOW));
  1192. //
  1193. // number of properties/data items in class
  1194. MofClassInfo->DataItemCount = DataItemCount;
  1195. MofClassInfo->MethodCount = 0;
  1196. //
  1197. // Parse the input parameter class object
  1198. //
  1199. if (InObject != NULL)
  1200. {
  1201. Status = WmipParseMethodInOutObject(InObject,
  1202. MofClassInfo,
  1203. DataItemCount);
  1204. if (Status == ERROR_SUCCESS)
  1205. {
  1206. if (OutObject != NULL)
  1207. {
  1208. //
  1209. // Parse the output parameter class object
  1210. //
  1211. Status = WmipParseMethodInOutObject(OutObject,
  1212. MofClassInfo,
  1213. DataItemCount);
  1214. }
  1215. //
  1216. // Now do any post parsing validation and fixup any variable
  1217. // length array properties
  1218. //
  1219. FinalStatus = Status;
  1220. Count = MofClassInfo->DataItemCount;
  1221. for (Index = 0; Index < Count; Index++)
  1222. {
  1223. MofDataItem = &MofClassInfo->DataItems[Index];
  1224. if (MofDataItem->Name == NULL)
  1225. {
  1226. MofClassInfo->DataItemCount--;
  1227. }
  1228. if (MofDataItem->Flags & MOFDI_FLAG_VARIABLE_ARRAY)
  1229. {
  1230. FinalStatus = WmipResolveVLArray(MofClassInfo,
  1231. MofDataItem,
  1232. FinalStatus);
  1233. }
  1234. }
  1235. }
  1236. }
  1237. *ClassInfo = MofClassInfo;
  1238. return(FinalStatus);
  1239. }
  1240. ULONG WmipParseMethodParameters(
  1241. CBMOFDataItem *MofMethodData,
  1242. PMOFCLASSINFOW *MethodClassInfo
  1243. )
  1244. {
  1245. ULONG Status = ERROR_SUCCESS;
  1246. CBMOFObj *InObject;
  1247. CBMOFObj *OutObject;
  1248. ULONG i;
  1249. ULONG NumberDimensions;
  1250. ULONG NumberElements;
  1251. VARIANT InVar, OutVar;
  1252. DWORD SimpleType;
  1253. SimpleType = MofMethodData->m_dwType & ~VT_ARRAY & ~VT_BYREF;
  1254. NumberDimensions = GetNumDimensions(MofMethodData);
  1255. if (NumberDimensions > 0)
  1256. {
  1257. NumberElements = GetNumElements(MofMethodData, 0);
  1258. WmipAssert(NumberDimensions == 1);
  1259. WmipAssert((NumberElements == 1) || (NumberElements == 2));
  1260. i = 0;
  1261. memset((void *)&InVar.lVal, 0, 8);
  1262. if (GetData(MofMethodData, (BYTE *)&(InVar.lVal), &i))
  1263. {
  1264. InObject = (CBMOFObj *)InVar.bstrVal;
  1265. InVar.vt = (VARTYPE)SimpleType;
  1266. WmipAssert(InVar.vt == VT_UNKNOWN);
  1267. if (NumberElements == 2)
  1268. {
  1269. i = 1;
  1270. memset((void *)&OutVar.lVal, 0, 8);
  1271. if (GetData(MofMethodData, (BYTE *)&(OutVar.lVal), &i))
  1272. {
  1273. OutVar.vt = (VARTYPE)SimpleType;
  1274. WmipAssert(OutVar.vt == VT_UNKNOWN);
  1275. OutObject = (CBMOFObj *)OutVar.bstrVal;
  1276. } else {
  1277. Status = ERROR_WMIMOF_NOT_FOUND;
  1278. }
  1279. } else {
  1280. OutObject = NULL;
  1281. }
  1282. } else {
  1283. Status = ERROR_WMIMOF_NOT_FOUND;
  1284. }
  1285. } else {
  1286. InObject = NULL;
  1287. OutObject = NULL;
  1288. }
  1289. if (Status == ERROR_SUCCESS)
  1290. {
  1291. Status = WmipParseMethodParameterObjects(InObject,
  1292. OutObject,
  1293. MethodClassInfo);
  1294. }
  1295. return(Status);
  1296. }
  1297. ULONG WmipParseClassObject(
  1298. PMOFRESOURCE MofResource,
  1299. CBMOFObj * ClassObject
  1300. )
  1301. /*++
  1302. Routine Description:
  1303. This routine will parse a BMOF object that is known to be an object that
  1304. contains a class. If we run into a parsing error then we immediate
  1305. quit parsing the class and return an error.
  1306. Arguments:
  1307. MofResource is the Mof Resource structure
  1308. ClassObject is BMOF object to parse
  1309. Return Value:
  1310. ERROR_SUCCESS or a WMI Mof error code (see wmiump.h)
  1311. --*/
  1312. {
  1313. PMOFCLASSINFOW MofClassInfo;
  1314. CBMOFQualList *ClassQualifierList = NULL;
  1315. CBMOFQualList *PropQualifierList = NULL;
  1316. CBMOFQualList *MethQualifierList = NULL;
  1317. CBMOFDataItem MofPropertyData, MofMethodData;
  1318. ULONG Status;
  1319. DWORD QualifierType;
  1320. ULONG DataItemCount;
  1321. ULONG MofClassInfoSize;
  1322. WCHAR *PropertyName = NULL;
  1323. WCHAR *MethodName = NULL;
  1324. ULONG Index;
  1325. PMOFDATAITEMW MofDataItem;
  1326. WCHAR *StringPtr;
  1327. PMOFCLASS MofClass;
  1328. ULONG FailedStatus;
  1329. ULONG Version;
  1330. short BooleanValue;
  1331. WCHAR *ClassName;
  1332. BOOLEAN DynamicQualifier, ProviderQualifier;
  1333. PULONG MethodList = 0;
  1334. ULONG MethodCount, MethodId;
  1335. ULONG i, Size;
  1336. BOOLEAN IsEvent;
  1337. PMOFCLASSINFOW MethodClassInfo;
  1338. BOOLEAN AbstractClass = FALSE;
  1339. //
  1340. // Get the class name which is required
  1341. if (! GetName(ClassObject, &ClassName))
  1342. {
  1343. WmipDebugPrint(("WMI: MofClass does not have a name\n"));
  1344. Status = ERROR_WMIMOF_NO_CLASS_NAME;
  1345. ErrorMessage(TRUE, ERROR_WMI_INVALID_MOF);
  1346. return(Status);
  1347. }
  1348. #if DBG
  1349. printf("Parsing class %ws\n", ClassName);
  1350. #endif
  1351. ResetObj(ClassObject);
  1352. ClassQualifierList = GetQualList(ClassObject);
  1353. if (ClassQualifierList == NULL)
  1354. {
  1355. WmipDebugPrint(("WMI: MofClass %ws does not have a qualifier list\n",
  1356. ClassName));
  1357. Status = ERROR_WMIMOF_NO_CLASS_NAME;
  1358. ErrorMessage(TRUE, ERROR_WMI_INVALID_MOF);
  1359. return(Status);
  1360. }
  1361. //
  1362. // Classes derived from WmiEvent may not be [abstract]
  1363. QualifierType = VT_BSTR;
  1364. Status = WmipFindProperty(ClassObject,
  1365. L"__SUPERCLASS",
  1366. &MofPropertyData,
  1367. &QualifierType,
  1368. (PVOID)&StringPtr);
  1369. if (Status == ERROR_SUCCESS)
  1370. {
  1371. IsEvent = (_wcsicmp(StringPtr, L"WmiEvent") == 0);
  1372. } else {
  1373. IsEvent = FALSE;
  1374. }
  1375. QualifierType = VT_BOOL;
  1376. Status = WmipFindMofQualifier(ClassQualifierList,
  1377. L"Abstract",
  1378. &QualifierType,
  1379. NULL,
  1380. (PVOID)&BooleanValue);
  1381. if ((Status == ERROR_SUCCESS) && BooleanValue)
  1382. {
  1383. //
  1384. // This is an abstract class - make sure it is not derived from
  1385. // WmiEvent
  1386. AbstractClass = TRUE;
  1387. QualifierType = VT_BSTR;
  1388. Status = WmipFindProperty(ClassObject,
  1389. L"__SUPERCLASS",
  1390. &MofPropertyData,
  1391. &QualifierType,
  1392. (PVOID)&StringPtr);
  1393. if (Status == ERROR_SUCCESS)
  1394. {
  1395. if (_wcsicmp(StringPtr, L"WmiEvent") == 0)
  1396. {
  1397. ErrorMessage(TRUE,
  1398. ERROR_WMIMOF_WMIEVENT_ABSTRACT,
  1399. ClassName);
  1400. return(ERROR_WMIMOF_WMIEVENT_ABSTRACT);
  1401. }
  1402. BMOFFree(StringPtr);
  1403. }
  1404. }
  1405. //
  1406. // See if this is a WMI class. Wmi classes have the [WMI] qualifier
  1407. QualifierType = VT_BOOL;
  1408. Status = WmipFindMofQualifier(ClassQualifierList,
  1409. L"WMI",
  1410. &QualifierType,
  1411. NULL,
  1412. (PVOID)&BooleanValue);
  1413. if (! ((Status == ERROR_SUCCESS) && BooleanValue))
  1414. {
  1415. //
  1416. // Skip this non-wmi class
  1417. return(ERROR_SUCCESS);
  1418. }
  1419. //
  1420. // Now check for WBEM required qualifiers
  1421. QualifierType = VT_BOOL;
  1422. Status = WmipFindMofQualifier(ClassQualifierList,
  1423. L"Dynamic",
  1424. &QualifierType,
  1425. NULL,
  1426. (PVOID)&BooleanValue);
  1427. DynamicQualifier = ((Status == ERROR_SUCCESS) && BooleanValue);
  1428. QualifierType = VT_BSTR;
  1429. Status = WmipFindMofQualifier(ClassQualifierList,
  1430. L"Provider",
  1431. &QualifierType,
  1432. NULL,
  1433. (PVOID)&StringPtr);
  1434. if (Status == ERROR_SUCCESS)
  1435. {
  1436. if (_wcsicmp(StringPtr, L"WmiProv") != 0)
  1437. {
  1438. Status = ERROR_WMIMOF_MISSING_HMOM_QUALIFIERS;
  1439. }
  1440. BMOFFree(StringPtr);
  1441. ProviderQualifier = TRUE;
  1442. } else {
  1443. ProviderQualifier = FALSE;
  1444. }
  1445. if ((ProviderQualifier && ! DynamicQualifier) ||
  1446. (! ProviderQualifier && DynamicQualifier))
  1447. {
  1448. //
  1449. // Both or neither [Dynamic, Provider(WmiProv)] qualifiers are required
  1450. ErrorMessage(TRUE,
  1451. ERROR_WMIMOF_MISSING_HMOM_QUALIFIERS,
  1452. ClassName);
  1453. return(ERROR_WMIMOF_MISSING_HMOM_QUALIFIERS);
  1454. }
  1455. MofClass = WmipAllocMofClass();
  1456. if (MofClass == NULL)
  1457. {
  1458. //
  1459. // No memory for MofClass so give up
  1460. return(ERROR_NOT_ENOUGH_MEMORY);
  1461. }
  1462. //
  1463. // Reference the MofResource so it stays around while the MofClass
  1464. // stays around
  1465. MofClass->MofResource = MofResource;
  1466. WmipReferenceMR(MofResource);
  1467. DataItemCount = WmipGetClassDataItemCount(ClassObject, L"WmiDataId");
  1468. MethodCount = WmipGetClassMethodCount(ClassObject);
  1469. ResetObj(ClassObject);
  1470. Size = sizeof(MOFCLASSINFO);
  1471. MofClassInfoSize = Size +
  1472. (DataItemCount+MethodCount) * sizeof(MOFDATAITEMW);
  1473. MofClassInfo = WmipAlloc(MofClassInfoSize);
  1474. if (MofClassInfo == NULL)
  1475. {
  1476. // WmipMCCleanup will unreference the MofResource
  1477. WmipUnreferenceMC(MofClass);
  1478. return(ERROR_NOT_ENOUGH_MEMORY);
  1479. } else {
  1480. MofClass->MofClassInfo = MofClassInfo;
  1481. MofClass->ClassObjectHandle = (ULONG_PTR)ClassObject;
  1482. //
  1483. // Get the essential information to fill in the MOF class info
  1484. memset(MofClassInfo, 0, MofClassInfoSize);
  1485. MofClassInfo->Flags = MOFCI_FLAG_READONLY;
  1486. MofClassInfo->Flags |= IsEvent ? MOFCI_FLAG_EVENT : 0;
  1487. MofClassInfo->Flags |= ProviderQualifier ? MOFCI_RESERVED0 : 0;
  1488. if (!ProviderQualifier && !DynamicQualifier)
  1489. {
  1490. //
  1491. // If neither the provider qualifier and Dynamic qualifier are
  1492. // specified then this is an embedded class
  1493. //
  1494. MofClassInfo->Flags |= MOFCI_FLAG_EMBEDDED_CLASS;
  1495. }
  1496. MofClassInfo->DataItems = (PMOFDATAITEMW) ((PUCHAR)MofClassInfo +
  1497. Size);
  1498. //
  1499. // number of properties/data items in class
  1500. MofClassInfo->DataItemCount = DataItemCount;
  1501. MofClassInfo->MethodCount = MethodCount;
  1502. MofClassInfo->Name = ClassName;
  1503. ClassQualifierList = GetQualList(ClassObject);
  1504. if (ClassQualifierList == NULL)
  1505. {
  1506. WmipDebugPrint(("WMI: MofClass %ws does not have a qualifier list\n",
  1507. MofClassInfo->Name));
  1508. Status = ERROR_WMIMOF_NO_CLASS_NAME;
  1509. ErrorMessage(TRUE, ERROR_WMI_INVALID_MOF);
  1510. goto done;
  1511. }
  1512. //
  1513. // Get the class guid which is required. Then convert it into
  1514. // binary form.
  1515. QualifierType = VT_BSTR;
  1516. Status = WmipFindMofQualifier(ClassQualifierList,
  1517. L"guid",
  1518. &QualifierType,
  1519. NULL,
  1520. (PVOID)&StringPtr);
  1521. if (Status == ERROR_SUCCESS)
  1522. {
  1523. Status = wGUIDFromString(StringPtr , &MofClassInfo->Guid) ?
  1524. ERROR_SUCCESS :
  1525. ERROR_WMIMOF_BAD_DATA_FORMAT;
  1526. BMOFFree((PVOID)StringPtr);
  1527. }
  1528. if (Status != ERROR_SUCCESS)
  1529. {
  1530. WmipDebugPrint(("WMI: MofClass %ws guid not found or in incorrect format\n",
  1531. MofClassInfo->Name));
  1532. ErrorMessage(TRUE,
  1533. ERROR_WMIMOF_BAD_OR_MISSING_GUID,
  1534. MofClassInfo->Name);
  1535. goto done;
  1536. }
  1537. //
  1538. // Get the description string which is not required
  1539. QualifierType = VT_BSTR;
  1540. Status = WmipFindMofQualifier(ClassQualifierList,
  1541. L"description",
  1542. &QualifierType,
  1543. NULL,
  1544. (PVOID)&MofClassInfo->Description);
  1545. //
  1546. // Get the header name string which is not required
  1547. QualifierType = VT_BSTR;
  1548. Status = WmipFindMofQualifier(ClassQualifierList,
  1549. L"HeaderName",
  1550. &QualifierType,
  1551. NULL,
  1552. (PVOID)&MofClassInfo->HeaderName);
  1553. //
  1554. // Get the header name string which is not required
  1555. QualifierType = VT_BSTR;
  1556. Status = WmipFindMofQualifier(ClassQualifierList,
  1557. L"GuidName1",
  1558. &QualifierType,
  1559. NULL,
  1560. (PVOID)&MofClassInfo->GuidName1);
  1561. //
  1562. // Get the header name string which is not required
  1563. QualifierType = VT_BSTR;
  1564. Status = WmipFindMofQualifier(ClassQualifierList,
  1565. L"GuidName2",
  1566. &QualifierType,
  1567. NULL,
  1568. (PVOID)&MofClassInfo->GuidName2);
  1569. //
  1570. // Now gather all of the information about the data items/properties
  1571. ResetObj(ClassObject);
  1572. Status = ERROR_SUCCESS;
  1573. while (NextProp(ClassObject, &PropertyName, &MofPropertyData))
  1574. {
  1575. #if DBG
  1576. printf(" %ws - ", PropertyName);
  1577. #endif
  1578. PropQualifierList = GetPropQualList(ClassObject, PropertyName);
  1579. if (PropQualifierList != NULL)
  1580. {
  1581. //
  1582. // Get the id of the property so we know its order in class
  1583. // If it doesn't have an id then we ignore it
  1584. QualifierType = VT_I4;
  1585. Status = WmipFindMofQualifier(PropQualifierList,
  1586. L"WmiDataId",
  1587. &QualifierType,
  1588. NULL,
  1589. (PVOID)&Index);
  1590. if (Status == ERROR_SUCCESS)
  1591. {
  1592. //
  1593. // Wmi Data Item ids are 1 based in the MOF
  1594. Index--;
  1595. if (Index < DataItemCount)
  1596. {
  1597. //
  1598. // Valid data item id, make sure it already isn't
  1599. // in use.
  1600. MofDataItem = &MofClassInfo->DataItems[Index];
  1601. if (MofDataItem->Name != NULL)
  1602. {
  1603. WmipDebugPrint(("WMI: Mof Class %ws has duplicate data item id %d for %ws and %ws\n",
  1604. MofClassInfo->Name, Index,
  1605. MofDataItem->Name, PropertyName));
  1606. Status = ERROR_WMIMOF_DUPLICATE_ID;
  1607. ErrorMessage(TRUE,
  1608. ERROR_WMIMOF_DUPLICATE_ID,
  1609. MofDataItem->Name,
  1610. MofClassInfo->Name,
  1611. Index+1,
  1612. PropertyName);
  1613. goto done;
  1614. }
  1615. MofDataItem->Name = PropertyName;
  1616. PropertyName = NULL;
  1617. Status = WmipParsePropertyObject(
  1618. MofDataItem,
  1619. &MofPropertyData,
  1620. PropQualifierList);
  1621. if (Status != ERROR_SUCCESS)
  1622. {
  1623. WmipDebugPrint(("WMI: MofClass %ws Property %ws not parsed properly %x\n",
  1624. MofClassInfo->Name, MofDataItem->Name, Status));
  1625. ErrorMessage(TRUE,
  1626. Status,
  1627. MofDataItem->Name,
  1628. MofClassInfo->Name);
  1629. goto done;
  1630. }
  1631. if (MofDataItem->Flags & MOFDI_FLAG_WRITEABLE)
  1632. {
  1633. MofClassInfo->Flags &= ~MOFCI_FLAG_READONLY;
  1634. }
  1635. } else {
  1636. WmipDebugPrint(("WMI: MofClass %ws has DataItem Id for %ws out of range %d\n",
  1637. MofClassInfo->Name, PropertyName, Index));
  1638. Status = ERROR_WMIMOF_BAD_DATAITEM_ID;
  1639. ErrorMessage(TRUE,
  1640. ERROR_WMIMOF_BAD_DATAITEM_ID,
  1641. PropertyName,
  1642. MofClassInfo->Name,
  1643. Index+1);
  1644. goto done;
  1645. }
  1646. } else {
  1647. //
  1648. // This property does not have a WmiDataId qualifier
  1649. // so see if it is Active or InstanceName
  1650. QualifierType = VT_BSTR;
  1651. Status = WmipFindMofQualifier(PropQualifierList,
  1652. L"CIMTYPE",
  1653. &QualifierType,
  1654. NULL,
  1655. (PVOID)&StringPtr);
  1656. if (_wcsicmp(PropertyName, L"InstanceName") == 0)
  1657. {
  1658. if ((Status != ERROR_SUCCESS) ||
  1659. (_wcsicmp(StringPtr, L"string") != 0))
  1660. {
  1661. Status = ERROR_WMIMOF_INSTANCENAME_BAD_TYPE;
  1662. ErrorMessage(TRUE,
  1663. ERROR_WMIMOF_INSTANCENAME_BAD_TYPE,
  1664. MofClassInfo->Name,
  1665. StringPtr);
  1666. BMOFFree(StringPtr);
  1667. goto done;
  1668. } else {
  1669. BMOFFree(StringPtr);
  1670. if ((! ProviderQualifier) && (! AbstractClass))
  1671. {
  1672. //
  1673. // If InstanceName specified, but this is an
  1674. // embedded class then an error
  1675. Status = ERROR_WMIMOF_EMBEDDED_CLASS;
  1676. ErrorMessage(TRUE,
  1677. ERROR_WMIMOF_EMBEDDED_CLASS,
  1678. MofClassInfo->Name);
  1679. goto done;
  1680. }
  1681. QualifierType = VT_BOOL;
  1682. Status = WmipFindMofQualifier(PropQualifierList,
  1683. L"key",
  1684. &QualifierType,
  1685. NULL,
  1686. (PVOID)&BooleanValue);
  1687. if ((Status == ERROR_SUCCESS) && BooleanValue)
  1688. {
  1689. MofClassInfo->Flags |= MOFCI_RESERVED1;
  1690. } else {
  1691. Status = ERROR_WMIMOF_INSTANCENAME_NOT_KEY;
  1692. ErrorMessage(TRUE,
  1693. ERROR_WMIMOF_INSTANCENAME_NOT_KEY,
  1694. MofClassInfo->Name);
  1695. goto done;
  1696. }
  1697. }
  1698. } else if (_wcsicmp(PropertyName, L"Active") == 0)
  1699. {
  1700. if ((Status != ERROR_SUCCESS) ||
  1701. (_wcsicmp(StringPtr, L"boolean") != 0))
  1702. {
  1703. Status = ERROR_WMIMOF_ACTIVE_BAD_TYPE;
  1704. ErrorMessage(TRUE,
  1705. ERROR_WMIMOF_ACTIVE_BAD_TYPE,
  1706. MofClassInfo->Name,
  1707. StringPtr);
  1708. BMOFFree(StringPtr);
  1709. goto done;
  1710. } else {
  1711. BMOFFree(StringPtr);
  1712. if ((! ProviderQualifier) && (! AbstractClass))
  1713. {
  1714. //
  1715. // If Boolean specified, but this is an
  1716. // embedded class then an error
  1717. Status = ERROR_WMIMOF_EMBEDDED_CLASS;
  1718. ErrorMessage(TRUE,
  1719. ERROR_WMIMOF_EMBEDDED_CLASS,
  1720. MofClassInfo->Name);
  1721. goto done;
  1722. }
  1723. MofClassInfo->Flags |= MOFCI_RESERVED2;
  1724. }
  1725. } else {
  1726. Status = ERROR_WMIMOF_NO_WMIDATAID;
  1727. ErrorMessage(TRUE,
  1728. ERROR_WMIMOF_NO_WMIDATAID,
  1729. PropertyName,
  1730. MofClassInfo->Name);
  1731. BMOFFree(StringPtr);
  1732. goto done;
  1733. }
  1734. }
  1735. // Do not free PropQualifierList since it is needed for
  1736. // generating header files
  1737. PropQualifierList= NULL;
  1738. }
  1739. #if DBG
  1740. printf("\n");
  1741. #endif
  1742. if (PropertyName != NULL)
  1743. {
  1744. BMOFFree(PropertyName);
  1745. PropertyName = NULL;
  1746. }
  1747. }
  1748. //
  1749. // Now parse the methods
  1750. #if DBG
  1751. printf("Parsing methods\n");
  1752. #endif
  1753. if (MethodCount > 0)
  1754. {
  1755. MethodList = (PULONG)BMOFAlloc(MethodCount * sizeof(ULONG));
  1756. if (MethodList == NULL)
  1757. {
  1758. WmipDebugPrint(("WMI: Not enough memory for Method List\n"));
  1759. return(ERROR_NOT_ENOUGH_MEMORY);
  1760. }
  1761. }
  1762. MethodCount = 0;
  1763. ResetObj(ClassObject);
  1764. while(NextMeth(ClassObject, &MethodName, &MofMethodData))
  1765. {
  1766. #if DBG
  1767. printf(" %ws - ", MethodName);
  1768. #endif
  1769. MethQualifierList = GetMethQualList(ClassObject, MethodName);
  1770. if (MethQualifierList != NULL)
  1771. {
  1772. Status = WmipParseMethodObject(&MofMethodData,
  1773. MethQualifierList,
  1774. &MethodId);
  1775. if (Status != ERROR_SUCCESS)
  1776. {
  1777. WmipDebugPrint(("WMI: MofClass %ws Method %ws not parsed properly %x\n",
  1778. MofClassInfo->Name, MethodName, Status));
  1779. ErrorMessage(TRUE,
  1780. Status,
  1781. MethodName,
  1782. MofClassInfo->Name);
  1783. goto done;
  1784. }
  1785. for (i = 0; i < MethodCount; i++)
  1786. {
  1787. if (MethodId == MethodList[i])
  1788. {
  1789. ErrorMessage(TRUE,
  1790. ERROR_WMIMOF_DUPLICATE_METHODID,
  1791. MethodName,
  1792. MofClassInfo->Name);
  1793. goto done;
  1794. }
  1795. }
  1796. MofDataItem = &MofClassInfo->DataItems[DataItemCount+MethodCount];
  1797. MethodList[MethodCount++] = MethodId;
  1798. MofDataItem->Flags = MOFDI_FLAG_METHOD;
  1799. MofDataItem->Name = MethodName;
  1800. MofDataItem->MethodId = MethodId;
  1801. //
  1802. // Get the header name string which is not required
  1803. QualifierType = VT_BSTR;
  1804. Status = WmipFindMofQualifier(MethQualifierList,
  1805. L"HeaderName",
  1806. &QualifierType,
  1807. NULL,
  1808. (PVOID)&MofDataItem->HeaderName);
  1809. BMOFFree(MethQualifierList);
  1810. MethQualifierList = NULL;
  1811. //
  1812. // parse the parameters for the method call
  1813. //
  1814. Status = WmipParseMethodParameters(&MofMethodData,
  1815. &MethodClassInfo);
  1816. if (Status == ERROR_SUCCESS)
  1817. {
  1818. MofDataItem->MethodClassInfo = MethodClassInfo;
  1819. } else if (Status == ERROR_WMIMOF_METHOD_RETURN_NOT_VOID) {
  1820. ErrorMessage(TRUE,
  1821. ERROR_WMIMOF_METHOD_RETURN_NOT_VOID,
  1822. MethodName,
  1823. MofClassInfo->Name);
  1824. }
  1825. }
  1826. #if DBG
  1827. printf("\n");
  1828. #endif
  1829. // DOn't free method name, kept in MofDataItem
  1830. MethodName = NULL;
  1831. }
  1832. }
  1833. done:
  1834. //
  1835. // Cleanup any loose pointers
  1836. if (MethodList != NULL)
  1837. {
  1838. BMOFFree(MethodList);
  1839. MethodList = NULL;
  1840. }
  1841. if (PropertyName != NULL)
  1842. {
  1843. BMOFFree(PropertyName);
  1844. PropertyName = NULL;
  1845. }
  1846. if (MethodName != NULL)
  1847. {
  1848. BMOFFree(MethodName);
  1849. MethodName = NULL;
  1850. }
  1851. if (PropQualifierList != NULL)
  1852. {
  1853. BMOFFree(PropQualifierList);
  1854. }
  1855. if (MethQualifierList != NULL)
  1856. {
  1857. BMOFFree(MethQualifierList);
  1858. }
  1859. if (ClassQualifierList != NULL)
  1860. {
  1861. BMOFFree(ClassQualifierList);
  1862. }
  1863. //
  1864. // Validate that we have all data item ids filled in, fixup any
  1865. // property references for variable length arrays and setup
  1866. // the appropriate version number in the data items.
  1867. FailedStatus = Status;
  1868. Version = 1;
  1869. for (Index = 0; Index < MofClassInfo->DataItemCount; Index++)
  1870. {
  1871. MofDataItem = &MofClassInfo->DataItems[Index];
  1872. if (MofDataItem->Flags & MOFDI_FLAG_VARIABLE_ARRAY)
  1873. {
  1874. //
  1875. // Resolve the variable length array
  1876. //
  1877. Status = WmipResolveVLArray(MofClassInfo,
  1878. MofDataItem,
  1879. FailedStatus);
  1880. if (Status != ERROR_SUCCESS)
  1881. {
  1882. if (Status == ERROR_WMIMOF_VL_ARRAY_SIZE_NOT_FOUND)
  1883. {
  1884. WmipDebugPrint(("WMI: Could not resolve vl array size property %ws in class %ws\n",
  1885. PropertyName, MofClassInfo->Name));
  1886. ErrorMessage(TRUE,
  1887. ERROR_WMIMOF_VL_ARRAY_SIZE_NOT_FOUND,
  1888. PropertyName,
  1889. MofDataItem->Name,
  1890. MofClassInfo->Name);
  1891. } else if (Status == ERROR_WMIMOF_BAD_VL_ARRAY_SIZE_TYPE) {
  1892. ErrorMessage(TRUE,
  1893. ERROR_WMIMOF_BAD_VL_ARRAY_SIZE_TYPE,
  1894. MofClassInfo->DataItems[MofDataItem->VariableArraySizeId-1].Name,
  1895. MofDataItem->Name,
  1896. MofClassInfo->Name);
  1897. } else if (Status == ERROR_WMIMOF_DATAITEM_NOT_FOUND) {
  1898. ErrorMessage(TRUE,
  1899. ERROR_WMIMOF_VL_ARRAY_NOT_FOUND,
  1900. MofDataItem->Name,
  1901. MofClassInfo->Name);
  1902. } else {
  1903. ErrorMessage(TRUE,
  1904. ERROR_WMIMOF_VL_ARRAY_NOT_RESOLVED,
  1905. MofDataItem->Name,
  1906. MofClassInfo->Name);
  1907. }
  1908. FailedStatus = Status;
  1909. }
  1910. }
  1911. //
  1912. // Ensure that this data item has got a name, that is the mof
  1913. // writer didn't skip a data item id
  1914. if (MofDataItem->Name == NULL)
  1915. {
  1916. //
  1917. // This data item was not filled in
  1918. Status = ERROR_WMIMOF_MISSING_DATAITEM;
  1919. WmipDebugPrint(("WMI: Missing data item %d in class %ws\n",
  1920. Index, MofClassInfo->Name));
  1921. ErrorMessage(TRUE,
  1922. ERROR_WMIMOF_MISSING_DATAITEM,
  1923. Index+1,
  1924. MofClassInfo->Name);
  1925. FailedStatus = Status;
  1926. }
  1927. if (FailedStatus != ERROR_SUCCESS)
  1928. {
  1929. continue;
  1930. }
  1931. //
  1932. // Establish version for data item
  1933. if (MofDataItem->Version == 0)
  1934. {
  1935. MofDataItem->Version = Version;
  1936. } else if ((MofDataItem->Version == Version) ||
  1937. (MofDataItem->Version == Version+1)) {
  1938. Version = MofDataItem->Version;
  1939. } else {
  1940. Status = ERROR_WMIMOF_INCONSISTENT_VERSIONING;
  1941. WmipDebugPrint(("WMI: Inconsistent versioning in class %ws at data item id %d\n",
  1942. MofClassInfo->Name, Index));
  1943. ErrorMessage(TRUE,
  1944. ERROR_WMIMOF_INCONSISTENT_VERSIONING,
  1945. MofDataItem->Name,
  1946. MofClassInfo->Name);
  1947. FailedStatus = Status;
  1948. // fall thru......
  1949. // continue;
  1950. }
  1951. }
  1952. if (FailedStatus == ERROR_SUCCESS)
  1953. {
  1954. //
  1955. // Mof class parsed OK so we set its version number and link it
  1956. // into the list of classes for the MOF resource
  1957. MofClassInfo->Version = Version;
  1958. //
  1959. // Link this into the list of MofClasses for this mof resource
  1960. InsertTailList(&MofResource->MRMCHead, &MofClass->MCMRList);
  1961. } else {
  1962. WmipUnreferenceMC(MofClass);
  1963. Status = FailedStatus;
  1964. }
  1965. return(Status);
  1966. }
  1967. //
  1968. // The routines below were blantenly stolen without remorse from the ole
  1969. // sources in \nt\private\ole32\com\class\compapi.cxx. They are copied here
  1970. // so that WMI doesn't need to load in ole32 only to convert a guid string
  1971. // into its binary representation.
  1972. //
  1973. //+-------------------------------------------------------------------------
  1974. //
  1975. // Function: HexStringToDword (private)
  1976. //
  1977. // Synopsis: scan lpsz for a number of hex digits (at most 8); update lpsz
  1978. // return value in Value; check for chDelim;
  1979. //
  1980. // Arguments: [lpsz] - the hex string to convert
  1981. // [Value] - the returned value
  1982. // [cDigits] - count of digits
  1983. //
  1984. // Returns: TRUE for success
  1985. //
  1986. //--------------------------------------------------------------------------
  1987. BOOL HexStringToDword(LPCWSTR lpsz, DWORD * RetValue,
  1988. int cDigits, WCHAR chDelim)
  1989. {
  1990. int Count;
  1991. DWORD Value;
  1992. Value = 0;
  1993. for (Count = 0; Count < cDigits; Count++, lpsz++)
  1994. {
  1995. if (*lpsz >= '0' && *lpsz <= '9')
  1996. Value = (Value << 4) + *lpsz - '0';
  1997. else if (*lpsz >= 'A' && *lpsz <= 'F')
  1998. Value = (Value << 4) + *lpsz - 'A' + 10;
  1999. else if (*lpsz >= 'a' && *lpsz <= 'f')
  2000. Value = (Value << 4) + *lpsz - 'a' + 10;
  2001. else
  2002. return(FALSE);
  2003. }
  2004. *RetValue = Value;
  2005. if (chDelim != 0)
  2006. return *lpsz++ == chDelim;
  2007. else
  2008. return TRUE;
  2009. }
  2010. //+-------------------------------------------------------------------------
  2011. //
  2012. // Function: wUUIDFromString (internal)
  2013. //
  2014. // Synopsis: Parse UUID such as 00000000-0000-0000-0000-000000000000
  2015. //
  2016. // Arguments: [lpsz] - Supplies the UUID string to convert
  2017. // [pguid] - Returns the GUID.
  2018. //
  2019. // Returns: TRUE if successful
  2020. //
  2021. //--------------------------------------------------------------------------
  2022. BOOL wUUIDFromString(LPCWSTR lpsz, LPGUID pguid)
  2023. {
  2024. DWORD dw;
  2025. if (!HexStringToDword(lpsz, &pguid->Data1, sizeof(DWORD)*2, '-'))
  2026. return FALSE;
  2027. lpsz += sizeof(DWORD)*2 + 1;
  2028. if (!HexStringToDword(lpsz, &dw, sizeof(WORD)*2, '-'))
  2029. return FALSE;
  2030. lpsz += sizeof(WORD)*2 + 1;
  2031. pguid->Data2 = (WORD)dw;
  2032. if (!HexStringToDword(lpsz, &dw, sizeof(WORD)*2, '-'))
  2033. return FALSE;
  2034. lpsz += sizeof(WORD)*2 + 1;
  2035. pguid->Data3 = (WORD)dw;
  2036. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2037. return FALSE;
  2038. lpsz += sizeof(BYTE)*2;
  2039. pguid->Data4[0] = (BYTE)dw;
  2040. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, '-'))
  2041. return FALSE;
  2042. lpsz += sizeof(BYTE)*2+1;
  2043. pguid->Data4[1] = (BYTE)dw;
  2044. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2045. return FALSE;
  2046. lpsz += sizeof(BYTE)*2;
  2047. pguid->Data4[2] = (BYTE)dw;
  2048. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2049. return FALSE;
  2050. lpsz += sizeof(BYTE)*2;
  2051. pguid->Data4[3] = (BYTE)dw;
  2052. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2053. return FALSE;
  2054. lpsz += sizeof(BYTE)*2;
  2055. pguid->Data4[4] = (BYTE)dw;
  2056. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2057. return FALSE;
  2058. lpsz += sizeof(BYTE)*2;
  2059. pguid->Data4[5] = (BYTE)dw;
  2060. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2061. return FALSE;
  2062. lpsz += sizeof(BYTE)*2;
  2063. pguid->Data4[6] = (BYTE)dw;
  2064. if (!HexStringToDword(lpsz, &dw, sizeof(BYTE)*2, 0))
  2065. return FALSE;
  2066. lpsz += sizeof(BYTE)*2;
  2067. pguid->Data4[7] = (BYTE)dw;
  2068. return TRUE;
  2069. }
  2070. //+-------------------------------------------------------------------------
  2071. //
  2072. // Function: wGUIDFromString (internal)
  2073. //
  2074. // Synopsis: Parse GUID such as {00000000-0000-0000-0000-000000000000}
  2075. //
  2076. // Arguments: [lpsz] - the guid string to convert
  2077. // [pguid] - guid to return
  2078. //
  2079. // Returns: TRUE if successful
  2080. //
  2081. //--------------------------------------------------------------------------
  2082. BOOL wGUIDFromString(LPCWSTR lpsz, LPGUID pguid)
  2083. {
  2084. DWORD dw;
  2085. if (*lpsz == '{' )
  2086. lpsz++;
  2087. if(wUUIDFromString(lpsz, pguid) != TRUE)
  2088. return FALSE;
  2089. lpsz +=36;
  2090. if (*lpsz == '}' )
  2091. lpsz++;
  2092. if (*lpsz != '\0') // check for zero terminated string - test bug #18307
  2093. {
  2094. return FALSE;
  2095. }
  2096. return TRUE;
  2097. }
  2098. ULONG GetRootObjectList(
  2099. char *BMofFile,
  2100. CBMOFObjList **ObjectList
  2101. )
  2102. {
  2103. ULONG Status;
  2104. ULONG CompressedSizeHigh;
  2105. BMofFileName = BMofFile;
  2106. FileHandle = CreateFile(BMofFile,
  2107. GENERIC_READ,
  2108. FILE_SHARE_READ,
  2109. NULL,
  2110. OPEN_EXISTING,
  2111. FILE_ATTRIBUTE_NORMAL,
  2112. NULL);
  2113. if (FileHandle == INVALID_HANDLE_VALUE)
  2114. {
  2115. Status = GetLastError();
  2116. ErrorMessage(TRUE, ERROR_WMIMOF_CANT_ACCESS_FILE, BMofFile, Status);
  2117. return(Status);
  2118. }
  2119. CompressedSize = GetFileSize(FileHandle, &CompressedSizeHigh);
  2120. MappingHandle = CreateFileMapping(FileHandle,
  2121. NULL,
  2122. PAGE_READONLY,
  2123. 0, 0,
  2124. NULL);
  2125. if (MappingHandle == NULL)
  2126. {
  2127. CloseHandle(FileHandle);
  2128. Status = GetLastError();
  2129. ErrorMessage(TRUE, ERROR_WMIMOF_CANT_ACCESS_FILE, BMofFile, Status);
  2130. return(Status);
  2131. }
  2132. CompressedFileBuffer = MapViewOfFile(MappingHandle,
  2133. FILE_MAP_READ,
  2134. 0, 0,
  2135. 0);
  2136. if (CompressedFileBuffer == NULL)
  2137. {
  2138. CloseHandle(MappingHandle);
  2139. CloseHandle(FileHandle);
  2140. Status = GetLastError();
  2141. ErrorMessage(TRUE, ERROR_WMIMOF_CANT_ACCESS_FILE, BMofFile, Status);
  2142. return(Status);
  2143. }
  2144. Status = WmipDecompressBuffer(CompressedFileBuffer,
  2145. &FileBuffer,
  2146. &UncompressedSize);
  2147. if (Status != ERROR_SUCCESS)
  2148. {
  2149. UnmapViewOfFile(CompressedFileBuffer);
  2150. CloseHandle(MappingHandle);
  2151. CloseHandle(FileHandle);
  2152. Status = GetLastError();
  2153. ErrorMessage(TRUE, ERROR_WMIMOF_CANT_ACCESS_FILE, BMofFile, Status);
  2154. return(Status);
  2155. }
  2156. fprintf(stderr, "Binary mof file %s expanded to %d bytes\n", BMofFile,
  2157. UncompressedSize);
  2158. //
  2159. // Create an object list structure of all the objects in the MOF
  2160. *ObjectList = CreateObjList(FileBuffer);
  2161. if(*ObjectList == NULL)
  2162. {
  2163. UnmapViewOfFile(CompressedFileBuffer);
  2164. CloseHandle(MappingHandle);
  2165. CloseHandle(FileHandle);
  2166. ErrorMessage(TRUE, ERROR_WMI_INVALID_MOF);
  2167. return(ERROR_WMI_INVALID_MOF);
  2168. }
  2169. return(ERROR_SUCCESS);
  2170. }
  2171. ULONG VerifyClassProperties(
  2172. PMOFRESOURCE MofResource,
  2173. PWCHAR ClassName,
  2174. PWCHAR BaseClassName,
  2175. PMOFCLASS MofClass
  2176. )
  2177. {
  2178. CBMOFObj *ClassObject;
  2179. PMOFCLASSINFOW MofClassInfo;
  2180. CBMOFDataItem MofPropertyData;
  2181. DWORD QualifierType;
  2182. WCHAR *StringPtr;
  2183. ULONG Status = ERROR_SUCCESS;
  2184. PMOFCLASS MofSuperClass;
  2185. ClassObject = (CBMOFObj *)MofClass->ClassObjectHandle;
  2186. MofClassInfo = MofClass->MofClassInfo;
  2187. if ( ((MofClassInfo->Flags & MOFCI_RESERVED1) == 0) &&
  2188. ((MofClassInfo->Flags & MOFCI_RESERVED2) == 0) )
  2189. {
  2190. //
  2191. // This class does not have the instanceName and Active properties
  2192. // so we expect that a superclass should. Look for the superclass
  2193. // and check that
  2194. //
  2195. QualifierType = VT_BSTR;
  2196. Status = WmipFindProperty(ClassObject,
  2197. L"__SUPERCLASS",
  2198. &MofPropertyData,
  2199. &QualifierType,
  2200. (PVOID)&StringPtr);
  2201. if (Status == ERROR_SUCCESS)
  2202. {
  2203. //
  2204. // Find the MofClass for the superclass and see if it has
  2205. // the required properties
  2206. //
  2207. MofSuperClass = WmipFindClassInMofResourceByName(MofResource,
  2208. StringPtr);
  2209. if (MofSuperClass != NULL)
  2210. {
  2211. Status = VerifyClassProperties(MofResource,
  2212. MofSuperClass->MofClassInfo->Name,
  2213. BaseClassName,
  2214. MofSuperClass);
  2215. } else {
  2216. //
  2217. // We could not find the superclass, but we will assume
  2218. // that this is ok
  2219. //
  2220. fprintf(stderr, "%s (0): warning RC2135 : Class %ws and all of its base classes do not have InstanceName and Active properties\n",
  2221. BMofFileName, BaseClassName);
  2222. Status = ERROR_SUCCESS;
  2223. }
  2224. BMOFFree(StringPtr);
  2225. } else {
  2226. Status = ERROR_WMIMOF_NO_INSTANCENAME;
  2227. ErrorMessage(TRUE,
  2228. ERROR_WMIMOF_NO_INSTANCENAME,
  2229. BaseClassName);
  2230. }
  2231. } else {
  2232. //
  2233. // If its got one of the properties make sure that it has
  2234. // both of them.
  2235. if ((MofClassInfo->Flags & MOFCI_RESERVED1) == 0)
  2236. {
  2237. Status = ERROR_WMIMOF_NO_INSTANCENAME;
  2238. ErrorMessage(TRUE,
  2239. ERROR_WMIMOF_NO_INSTANCENAME,
  2240. BaseClassName);
  2241. }
  2242. if ((MofClassInfo->Flags & MOFCI_RESERVED2) == 0)
  2243. {
  2244. Status = ERROR_WMIMOF_NO_ACTIVE;
  2245. ErrorMessage(TRUE,
  2246. ERROR_WMIMOF_NO_ACTIVE,
  2247. BaseClassName);
  2248. }
  2249. }
  2250. return(Status);
  2251. }
  2252. ULONG ParseBinaryMofFile(
  2253. char *BMofFile,
  2254. PMOFRESOURCE *ReturnMofResource
  2255. )
  2256. {
  2257. ULONG Status;
  2258. CBMOFObjList *MofObjList;
  2259. CBMOFObj *ClassObject;
  2260. PMOFRESOURCE MofResource;
  2261. PLIST_ENTRY MofClassList;
  2262. PMOFCLASS MofClass;
  2263. BOOLEAN CleanupOnly;
  2264. PMOFCLASSINFOW MofClassInfo;
  2265. MofResource = WmipAllocMofResource();
  2266. if (MofResource == NULL)
  2267. {
  2268. return(ERROR_NOT_ENOUGH_MEMORY);
  2269. }
  2270. *ReturnMofResource = MofResource;
  2271. InitializeListHead(&MofResource->MRMCHead);
  2272. Status = GetRootObjectList(BMofFile, &MofObjList);
  2273. if (Status == ERROR_SUCCESS)
  2274. {
  2275. ResetObjList (MofObjList);
  2276. while((Status == ERROR_SUCCESS) &&
  2277. (ClassObject = NextObj(MofObjList)) )
  2278. {
  2279. Status = WmipParseClassObject(MofResource, ClassObject);
  2280. if (Status != ERROR_SUCCESS)
  2281. {
  2282. ErrorMessage(TRUE, ERROR_WMIMOF_CLASS_NOT_PARSED);
  2283. }
  2284. }
  2285. //
  2286. // Now that all classes are parsed we need to go back and find
  2287. // the guids for any embedded classes, or if a class failed
  2288. // to parse then we need to free any embedded class names
  2289. CleanupOnly = (Status != ERROR_SUCCESS);
  2290. MofClassList = MofResource->MRMCHead.Flink;
  2291. while (MofClassList != &MofResource->MRMCHead)
  2292. {
  2293. MofClass = CONTAINING_RECORD(MofClassList,
  2294. MOFCLASS,
  2295. MCMRList);
  2296. MofClassInfo = MofClass->MofClassInfo;
  2297. if ((! CleanupOnly) &&
  2298. (MofClassInfo->Flags & MOFCI_RESERVED0))
  2299. {
  2300. //
  2301. // if the class has Provider qualifier, it better have
  2302. // an instancename and Active properties in itself or in a
  2303. // superclass
  2304. //
  2305. Status = VerifyClassProperties(MofResource,
  2306. MofClassInfo->Name,
  2307. MofClassInfo->Name,
  2308. MofClass);
  2309. CleanupOnly = (Status != ERROR_SUCCESS);
  2310. }
  2311. Status = WmipFillEmbeddedClasses(MofResource,
  2312. MofClassInfo,
  2313. CleanupOnly);
  2314. if (Status != ERROR_SUCCESS)
  2315. {
  2316. CleanupOnly = TRUE;
  2317. }
  2318. MofClassList = MofClassList->Flink;
  2319. }
  2320. if (CleanupOnly)
  2321. {
  2322. Status = ERROR_WMI_INVALID_MOF;
  2323. }
  2324. BMOFFree(MofObjList);
  2325. if (Status != ERROR_SUCCESS)
  2326. {
  2327. //
  2328. // Make sure we have a useful Win32 error code and not
  2329. // an internal WMIMOF error code
  2330. Status = ERROR_WMI_INVALID_MOF;
  2331. }
  2332. }
  2333. return(Status);
  2334. }
  2335. ULONG FilePrintVaList(
  2336. HANDLE FileHandle,
  2337. CHAR *Format,
  2338. va_list pArg
  2339. )
  2340. {
  2341. CHAR Buffer[8192];
  2342. ULONG Size, Written;
  2343. ULONG Status;
  2344. Size = _vsnprintf(Buffer, sizeof(Buffer), Format, pArg);
  2345. if (WriteFile(FileHandle,
  2346. Buffer,
  2347. Size,
  2348. &Written,
  2349. NULL))
  2350. {
  2351. Status = ERROR_SUCCESS;
  2352. } else {
  2353. Status = GetLastError();
  2354. }
  2355. return(Status);
  2356. }
  2357. ULONG FilePrint(
  2358. HANDLE FileHandle,
  2359. char *Format,
  2360. ...
  2361. )
  2362. {
  2363. ULONG Status;
  2364. va_list pArg;
  2365. va_start(pArg, Format);
  2366. Status = FilePrintVaList(FileHandle, Format, pArg);
  2367. return(Status);
  2368. }
  2369. ULONG GenerateASLTemplate(
  2370. PCHAR TemplateFile
  2371. )
  2372. {
  2373. return(ERROR_SUCCESS);
  2374. }
  2375. ULONG GenerateBinaryMofData(
  2376. HANDLE FileHandle
  2377. )
  2378. {
  2379. ULONG Lines;
  2380. ULONG LastLine;
  2381. ULONG i;
  2382. ULONG Index;
  2383. PUCHAR BMofBuffer = (PUCHAR)CompressedFileBuffer;
  2384. PCHAR IndentString = " ";
  2385. ULONG Status;
  2386. Lines = CompressedSize / 16;
  2387. LastLine = CompressedSize % 16;
  2388. if (LastLine == 0)
  2389. {
  2390. LastLine = 16;
  2391. Lines--;
  2392. }
  2393. for (i = 0; i < Lines; i++)
  2394. {
  2395. Index = i * 16;
  2396. Status = FilePrint(FileHandle, "%s0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x,\r\n",
  2397. IndentString,
  2398. BMofBuffer[Index],
  2399. BMofBuffer[Index+1],
  2400. BMofBuffer[Index+2],
  2401. BMofBuffer[Index+3],
  2402. BMofBuffer[Index+4],
  2403. BMofBuffer[Index+5],
  2404. BMofBuffer[Index+6],
  2405. BMofBuffer[Index+7],
  2406. BMofBuffer[Index+8],
  2407. BMofBuffer[Index+9],
  2408. BMofBuffer[Index+10],
  2409. BMofBuffer[Index+11],
  2410. BMofBuffer[Index+12],
  2411. BMofBuffer[Index+13],
  2412. BMofBuffer[Index+14],
  2413. BMofBuffer[Index+15]);
  2414. if (Status != ERROR_SUCCESS)
  2415. {
  2416. return(Status);
  2417. }
  2418. }
  2419. LastLine--;
  2420. FilePrint(FileHandle, "%s", IndentString);
  2421. Index = Lines * 16;
  2422. for (i = 0; i < LastLine; i++)
  2423. {
  2424. FilePrint(FileHandle, "0x%02x, ",
  2425. BMofBuffer[Index+i]);
  2426. if (Status != ERROR_SUCCESS)
  2427. {
  2428. return(Status);
  2429. }
  2430. }
  2431. FilePrint(FileHandle, "0x%02x\r\n",
  2432. BMofBuffer[Index+i]);
  2433. return(Status);
  2434. }
  2435. //
  2436. // This will loop over the mof class list and print out once for each class
  2437. // in the list.
  2438. //
  2439. // Handle is the file handle to which to write
  2440. // MR is the MofResource whose classes are being enumerated
  2441. // NamePtr is the variable to place the name of the class
  2442. // Counter is the variable to use as a counter
  2443. // Format is the format template to use to write out
  2444. //
  2445. HANDLE GlobalFilePrintHandle;
  2446. ULONG FilePrintGlobal(CHAR *Format, ...)
  2447. {
  2448. ULONG Status;
  2449. va_list pArg;
  2450. va_start(pArg, Format);
  2451. Status = FilePrintVaList(GlobalFilePrintHandle, Format, pArg);
  2452. return(Status);
  2453. }
  2454. #define FilePrintMofClassLoop( \
  2455. Handle, \
  2456. MR, \
  2457. NamePtr, \
  2458. Counter, \
  2459. DiscardEmbedded, \
  2460. Format \
  2461. ) \
  2462. { \
  2463. PLIST_ENTRY MofClassList; \
  2464. PMOFCLASSINFOW ClassInfo; \
  2465. PMOFCLASS MofClass; \
  2466. GlobalFilePrintHandle = TemplateHandle; \
  2467. (Counter) = 0; \
  2468. MofClassList = (MR)->MRMCHead.Flink; \
  2469. while (MofClassList != &(MR)->MRMCHead) \
  2470. { \
  2471. MofClass = CONTAINING_RECORD(MofClassList, MOFCLASS, MCMRList); \
  2472. ClassInfo = MofClass->MofClassInfo; \
  2473. if (! ((DiscardEmbedded) && \
  2474. (ClassInfo->Flags & MOFCI_FLAG_EMBEDDED_CLASS)) ) \
  2475. { \
  2476. (NamePtr) = ClassInfo->Name; \
  2477. FilePrintGlobal Format; \
  2478. (Counter)++; \
  2479. } \
  2480. MofClassList = MofClassList->Flink; \
  2481. } \
  2482. }
  2483. typedef void (*ENUMMOFCLASSCALLBACK)(
  2484. HANDLE TemplateHandle,
  2485. PMOFCLASS MofClass,
  2486. ULONG Counter,
  2487. PVOID Context
  2488. );
  2489. void EnumerateMofClasses(
  2490. HANDLE TemplateHandle,
  2491. PMOFRESOURCE MR,
  2492. ENUMMOFCLASSCALLBACK Callback,
  2493. PVOID Context
  2494. )
  2495. {
  2496. PLIST_ENTRY MofClassList;
  2497. PMOFCLASS MofClass;
  2498. ULONG Counter;
  2499. Counter = 0;
  2500. MofClassList = MR->MRMCHead.Flink;
  2501. while (MofClassList != &MR->MRMCHead)
  2502. {
  2503. MofClass = CONTAINING_RECORD(MofClassList, MOFCLASS, MCMRList);
  2504. (*Callback)(TemplateHandle,
  2505. MofClass,
  2506. Counter,
  2507. Context);
  2508. MofClassList = MofClassList->Flink;
  2509. Counter++;
  2510. }
  2511. }
  2512. void GenerateGuidListTemplate(
  2513. HANDLE TemplateHandle,
  2514. PMOFCLASS MofClass,
  2515. ULONG Counter,
  2516. PVOID Context
  2517. )
  2518. {
  2519. PMOFCLASSINFOW ClassInfo;
  2520. PWCHAR GuidName1, GuidSuffix1;
  2521. ClassInfo = MofClass->MofClassInfo;
  2522. if ( ! (ClassInfo->Flags & MOFCI_FLAG_EMBEDDED_CLASS))
  2523. {
  2524. //
  2525. // Only generate code for non embedded classes
  2526. //
  2527. if (ClassInfo->GuidName1 != NULL)
  2528. {
  2529. GuidName1 = ClassInfo->GuidName1;
  2530. GuidSuffix1 = L"";
  2531. } else {
  2532. GuidName1 = ClassInfo->Name;
  2533. GuidSuffix1 = L"Guid";
  2534. }
  2535. FilePrint(TemplateHandle,
  2536. "GUID %wsGUID = %ws%ws;\r\n",
  2537. ClassInfo->Name,
  2538. GuidName1, GuidSuffix1);
  2539. }
  2540. }
  2541. void GenerateFunctionControlListTemplate(
  2542. HANDLE TemplateHandle,
  2543. PMOFCLASS MofClass,
  2544. ULONG Counter,
  2545. PVOID Context
  2546. )
  2547. {
  2548. PMOFCLASSINFOW ClassInfo;
  2549. ClassInfo = MofClass->MofClassInfo;
  2550. if (! (ClassInfo->Flags & MOFCI_FLAG_EMBEDDED_CLASS) )
  2551. {
  2552. FilePrint(TemplateHandle,
  2553. " case %wsGuidIndex:\r\n"
  2554. " {\r\n",
  2555. ClassInfo->Name);
  2556. if (ClassInfo->Flags & MOFCI_FLAG_EVENT)
  2557. {
  2558. FilePrint(TemplateHandle,
  2559. " if (Enable)\r\n"
  2560. " {\r\n"
  2561. " //\r\n"
  2562. " // TODO: Event is being enabled, do anything required to\r\n"
  2563. " // allow the event to be fired\r\n"
  2564. " //\r\n"
  2565. " } else {\r\n"
  2566. " //\r\n"
  2567. " // TODO: Event is being disabled, do anything required to\r\n"
  2568. " // keep the event from being fired\r\n"
  2569. " //\r\n"
  2570. " }\r\n");
  2571. } else {
  2572. FilePrint(TemplateHandle,
  2573. " //\r\n"
  2574. " // TODO: Delete this entire case if data block does not have the\r\n"
  2575. " // WMIREG_FLAG_EXPENSIVE flag set\r\n"
  2576. " //\r\n"
  2577. " if (Enable)\r\n"
  2578. " {\r\n"
  2579. " //\r\n"
  2580. " // TODO: Datablock collection is being enabled. If this\r\n"
  2581. " // data block has been marked as expensive in the\r\n"
  2582. " // guid list then this code will be called when the\r\n"
  2583. " // first data consumer opens this data block. If\r\n"
  2584. " // anything needs to be done to allow data to be \r\n"
  2585. " // collected for this data block then it should be\r\n"
  2586. " // done here\r\n"
  2587. " //\r\n"
  2588. " } else {\r\n"
  2589. " //\r\n"
  2590. " // TODO: Datablock collection is being disabled. If this\r\n"
  2591. " // data block has been marked as expensive in the\r\n"
  2592. " // guid list then this code will be called when the\r\n"
  2593. " // last data consumer closes this data block. If\r\n"
  2594. " // anything needs to be done to cleanup after data has \r\n"
  2595. " // been collected for this data block then it should be\r\n"
  2596. " // done here\r\n"
  2597. " //\r\n"
  2598. " }\r\n");
  2599. }
  2600. FilePrint(TemplateHandle,
  2601. " break;\r\n"
  2602. " }\r\n\r\n");
  2603. }
  2604. }
  2605. void GenerateSetList(
  2606. HANDLE TemplateHandle,
  2607. PMOFCLASS MofClass,
  2608. ULONG Counter,
  2609. PVOID Context
  2610. )
  2611. {
  2612. PCHAR Format = (PCHAR)Context;
  2613. PMOFCLASSINFOW ClassInfo;
  2614. ClassInfo = MofClass->MofClassInfo;
  2615. if (! (ClassInfo->Flags & MOFCI_FLAG_EMBEDDED_CLASS) )
  2616. {
  2617. if (! (ClassInfo->Flags & MOFCI_FLAG_READONLY))
  2618. {
  2619. FilePrint(TemplateHandle, Format, ClassInfo->Name);
  2620. } else {
  2621. FilePrint(TemplateHandle,
  2622. " case %wsGuidIndex:\r\n"
  2623. " { \r\n"
  2624. " status = STATUS_WMI_READ_ONLY;\r\n"
  2625. " break;\r\n"
  2626. " }\r\n\r\n",
  2627. ClassInfo->Name);
  2628. }
  2629. }
  2630. }
  2631. void GenerateMethodCTemplate(
  2632. HANDLE TemplateHandle,
  2633. PMOFCLASS MofClass,
  2634. ULONG Counter,
  2635. PVOID Context
  2636. )
  2637. {
  2638. PMOFCLASSINFOW ClassInfo;
  2639. ULONG i;
  2640. PMOFDATAITEMW DataItem;
  2641. ClassInfo = MofClass->MofClassInfo;
  2642. if (ClassInfo->MethodCount > 0)
  2643. {
  2644. FilePrint(TemplateHandle,
  2645. " case %wsGuidIndex:\r\n"
  2646. " {\r\n"
  2647. " switch(MethodId)\r\n"
  2648. " {\r\n",
  2649. ClassInfo->Name);
  2650. for (i = 0; i < ClassInfo->MethodCount; i++)
  2651. {
  2652. DataItem = &ClassInfo->DataItems[i+ClassInfo->DataItemCount];
  2653. FilePrint(TemplateHandle,
  2654. " case %ws:\r\n"
  2655. " { \r\n"
  2656. " //\r\n"
  2657. " // TODO: Validate InstanceIndex, InBufferSize \r\n"
  2658. " // and Buffer contents to ensure that the \r\n"
  2659. " // input buffer is valid and OutBufferSize is\r\n"
  2660. " // large enough to return the output data.\r\n"
  2661. " //\r\n"
  2662. " break;\r\n"
  2663. " }\r\n\r\n",
  2664. DataItem->Name);
  2665. }
  2666. FilePrint(TemplateHandle,
  2667. " default:\r\n"
  2668. " {\r\n"
  2669. " status = STATUS_WMI_ITEMID_NOT_FOUND;\r\n"
  2670. " break;\r\n"
  2671. " }\r\n"
  2672. " }\r\n"
  2673. " break;\r\n"
  2674. " }\r\n"
  2675. "\r\n"
  2676. );
  2677. }
  2678. }
  2679. BOOLEAN DoesSupportMethods(
  2680. PMOFRESOURCE MR
  2681. )
  2682. {
  2683. PLIST_ENTRY MofClassList;
  2684. PMOFCLASS MofClass;
  2685. MofClassList = MR->MRMCHead.Flink;
  2686. while (MofClassList != &MR->MRMCHead)
  2687. {
  2688. MofClass = CONTAINING_RECORD(MofClassList, MOFCLASS, MCMRList);
  2689. if (MofClass->MofClassInfo->MethodCount > 0)
  2690. {
  2691. return(TRUE);
  2692. }
  2693. MofClassList = MofClassList->Flink;
  2694. }
  2695. return(FALSE);
  2696. }
  2697. BOOLEAN DoesReadOnly(
  2698. PMOFRESOURCE MR
  2699. )
  2700. {
  2701. PLIST_ENTRY MofClassList;
  2702. PMOFCLASS MofClass;
  2703. PMOFCLASSINFOW ClassInfo;
  2704. ULONG i;
  2705. MofClassList = MR->MRMCHead.Flink;
  2706. while (MofClassList != &MR->MRMCHead)
  2707. {
  2708. MofClass = CONTAINING_RECORD(MofClassList, MOFCLASS, MCMRList);
  2709. ClassInfo = MofClass->MofClassInfo;
  2710. for (i = 0; i < ClassInfo->DataItemCount; i++)
  2711. {
  2712. if (ClassInfo->DataItems[i].Flags & MOFDI_FLAG_WRITEABLE)
  2713. {
  2714. return(FALSE);
  2715. }
  2716. }
  2717. MofClassList = MofClassList->Flink;
  2718. }
  2719. return(TRUE);
  2720. }
  2721. PCHAR GetBaseNameFromFileName(
  2722. PCHAR FileName,
  2723. PCHAR BaseName
  2724. )
  2725. {
  2726. PCHAR p, p1;
  2727. ULONG Len;
  2728. p = FileName;
  2729. p1 = FileName;
  2730. while ((*p != '.') && (*p != 0))
  2731. {
  2732. if (*p == '\\')
  2733. {
  2734. p1 = p+1;
  2735. }
  2736. p++;
  2737. }
  2738. Len = (ULONG)(p - p1);
  2739. memcpy(BaseName, p1, Len);
  2740. BaseName[Len] = 0;
  2741. return(BaseName);
  2742. }
  2743. ULONG GenerateCTemplate(
  2744. PCHAR TemplateFile,
  2745. PCHAR HFileName,
  2746. PCHAR XFileName,
  2747. PMOFRESOURCE MofResource
  2748. )
  2749. {
  2750. BOOLEAN SupportsMethods, SupportsFunctionControl, IsReadOnly;
  2751. HANDLE TemplateHandle;
  2752. CHAR BaseName[MAX_PATH], BaseXFileName[MAX_PATH], BaseHFileName[MAX_PATH];
  2753. ULONG i;
  2754. PWCHAR ClassName;
  2755. GetBaseNameFromFileName(TemplateFile, BaseName);
  2756. GetBaseNameFromFileName(XFileName, BaseXFileName);
  2757. GetBaseNameFromFileName(XFileName, BaseHFileName);
  2758. BaseName[0] = (CHAR)toupper(BaseName[0]);
  2759. SupportsMethods = DoesSupportMethods(MofResource);
  2760. SupportsFunctionControl = TRUE;
  2761. IsReadOnly = DoesReadOnly(MofResource);
  2762. TemplateHandle = CreateFile(TemplateFile,
  2763. GENERIC_WRITE,
  2764. 0,
  2765. NULL,
  2766. CREATE_ALWAYS,
  2767. FILE_ATTRIBUTE_NORMAL,
  2768. NULL);
  2769. if ((TemplateHandle == NULL) || (TemplateHandle == INVALID_HANDLE_VALUE))
  2770. {
  2771. return(GetLastError());
  2772. }
  2773. FilePrint(TemplateHandle,
  2774. "//\r\n"
  2775. "// %s.c - Code generated by wmimofck tool\r\n"
  2776. "//\r\n"
  2777. "// Finish code by doing all TODO: sections\r\n"
  2778. "//\r\n"
  2779. "\r\n"
  2780. "#include <wdm.h>\r\n"
  2781. "#include <wmistr.h>\r\n"
  2782. "#include <wmiguid.h>\r\n"
  2783. "#include <wmilib.h>\r\n"
  2784. "\r\n"
  2785. "//\r\n"
  2786. "// Include data header for classes\r\n"
  2787. "#include \"%s.h\"\r\n"
  2788. "\r\n"
  2789. "\r\n",
  2790. BaseName,
  2791. BaseHFileName
  2792. );
  2793. FilePrint(TemplateHandle,
  2794. "//\r\n"
  2795. "// TODO: Place the contents in this device extension into the driver's\r\n"
  2796. "// actual device extension. It is only defined here to supply\r\n"
  2797. "// a device extension so that this file can be compiled on its own\r\n"
  2798. "//\r\n"
  2799. "#ifdef MAKE_THIS_COMPILE\r\n"
  2800. "typedef struct DEVICE_EXTENSION\r\n"
  2801. "{\r\n"
  2802. " WMILIB_CONTEXT WmiLib;\r\n"
  2803. " PDEVICE_OBJECT physicalDevObj;\r\n"
  2804. "} DEVICE_EXTENSION, *PDEVICE_EXTENSION;\r\n"
  2805. "#endif\r\n\r\n"
  2806. );
  2807. FilePrint(TemplateHandle,
  2808. "NTSTATUS\r\n"
  2809. "%sInitializeWmilibContext(\r\n"
  2810. " IN PWMILIB_CONTEXT WmilibContext\r\n"
  2811. " );\r\n"
  2812. "\r\n"
  2813. "NTSTATUS\r\n"
  2814. "%sFunctionControl(\r\n"
  2815. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  2816. " IN PIRP Irp,\r\n"
  2817. " IN ULONG GuidIndex,\r\n"
  2818. " IN WMIENABLEDISABLECONTROL Function,\r\n"
  2819. " IN BOOLEAN Enable\r\n"
  2820. " );\r\n"
  2821. "\r\n"
  2822. "NTSTATUS\r\n"
  2823. "%sExecuteWmiMethod(\r\n"
  2824. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  2825. " IN PIRP Irp,\r\n"
  2826. " IN ULONG GuidIndex,\r\n"
  2827. " IN ULONG InstanceIndex,\r\n"
  2828. " IN ULONG MethodId,\r\n"
  2829. " IN ULONG InBufferSize,\r\n"
  2830. " IN ULONG OutBufferSize,\r\n"
  2831. " IN PUCHAR Buffer\r\n"
  2832. " );\r\n"
  2833. "\r\n"
  2834. "NTSTATUS\r\n"
  2835. "%sSetWmiDataItem(\r\n"
  2836. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  2837. " IN PIRP Irp,\r\n"
  2838. " IN ULONG InstanceIndex,\r\n"
  2839. " IN ULONG GuidIndex,\r\n"
  2840. " IN ULONG DataItemId,\r\n"
  2841. " IN ULONG BufferSize,\r\n"
  2842. " IN PUCHAR Buffer\r\n"
  2843. " );\r\n"
  2844. "\r\n"
  2845. "NTSTATUS\r\n"
  2846. "%sSetWmiDataBlock(\r\n"
  2847. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  2848. " IN PIRP Irp,\r\n"
  2849. " IN ULONG GuidIndex,\r\n"
  2850. " IN ULONG InstanceIndex,\r\n"
  2851. " IN ULONG BufferSize,\r\n"
  2852. " IN PUCHAR Buffer\r\n"
  2853. " );\r\n"
  2854. "\r\n"
  2855. "NTSTATUS\r\n"
  2856. "%sQueryWmiDataBlock(\r\n"
  2857. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  2858. " IN PIRP Irp,\r\n"
  2859. " IN ULONG GuidIndex,\r\n"
  2860. " IN ULONG InstanceIndex,\r\n"
  2861. " IN ULONG InstanceCount,\r\n"
  2862. " IN OUT PULONG InstanceLengthArray,\r\n"
  2863. " IN ULONG BufferAvail,\r\n"
  2864. " OUT PUCHAR Buffer\r\n"
  2865. " );\r\n"
  2866. "\r\n"
  2867. "NTSTATUS\r\n"
  2868. "%sQueryWmiRegInfo(\r\n"
  2869. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  2870. " OUT ULONG *RegFlags,\r\n"
  2871. " OUT PUNICODE_STRING InstanceName,\r\n"
  2872. " OUT PUNICODE_STRING *RegistryPath,\r\n"
  2873. " OUT PUNICODE_STRING MofResourceName,\r\n"
  2874. " OUT PDEVICE_OBJECT *Pdo\r\n"
  2875. " );\r\n"
  2876. "\r\n"
  2877. "#ifdef ALLOC_PRAGMA\r\n"
  2878. "#pragma alloc_text(PAGE,%sQueryWmiRegInfo)\r\n"
  2879. "#pragma alloc_text(PAGE,%sQueryWmiDataBlock)\r\n"
  2880. "#pragma alloc_text(PAGE,%sSetWmiDataBlock)\r\n"
  2881. "#pragma alloc_text(PAGE,%sSetWmiDataItem)\r\n"
  2882. "#pragma alloc_text(PAGE,%sExecuteWmiMethod)\r\n"
  2883. "#pragma alloc_text(PAGE,%sFunctionControl)\r\n"
  2884. "#pragma alloc_text(PAGE,%sInitializeWmilibContext)\r\n"
  2885. "#endif\r\n"
  2886. "\r\n",
  2887. BaseName,
  2888. BaseName,
  2889. BaseName,
  2890. BaseName,
  2891. BaseName,
  2892. BaseName,
  2893. BaseName,
  2894. BaseName,
  2895. BaseName,
  2896. BaseName,
  2897. BaseName,
  2898. BaseName,
  2899. BaseName,
  2900. BaseName
  2901. );
  2902. FilePrint(TemplateHandle,
  2903. "//\r\n"
  2904. "// TODO: Decide if your MOF is going to be part of your driver as a resource\r\n"
  2905. "// attached to it. If this is done then all MOF in the resource will be\r\n"
  2906. "// added to the schema. If this is the case be sure that \r\n"
  2907. "// USE_BINARY_MOF_RESOURCE is defined. MOF can also be reported at \r\n"
  2908. "// runtime via a query to the driver. This can be useful if you want\r\n"
  2909. "// the MOF reported to the schema to be dynamic. If MOF is reported via\r\n"
  2910. "// a query then USE_BINARY_MOF_QUERY should be defined.\r\n"
  2911. "\r\n"
  2912. "#define USE_BINARY_MOF_QUERY\r\n"
  2913. "#define USE_BINARY_MOF_RESOURCE\r\n"
  2914. "\r\n"
  2915. "#ifdef USE_BINARY_MOF_QUERY\r\n"
  2916. "//\r\n"
  2917. "// MOF data can be reported by a device driver via a resource attached to\r\n"
  2918. "// the device drivers image file or in response to a query on the binary\r\n"
  2919. "// mof data guid. Here we define global variables containing the binary mof\r\n"
  2920. "// data to return in response to a binary mof guid query. Note that this\r\n"
  2921. "// data is defined to be in a PAGED data segment since it does not need to\r\n"
  2922. "// be in nonpaged memory. Note that instead of a single large mof file\r\n"
  2923. "// we could have broken it into multiple individual files. Each file would\r\n"
  2924. "// have its own binary mof data buffer and get reported via a different\r\n"
  2925. "// instance of the binary mof guid. By mixing and matching the different\r\n"
  2926. "// sets of binary mof data buffers a \"dynamic\" composite mof would be created.\r\n"
  2927. "\r\n"
  2928. "#ifdef ALLOC_DATA_PRAGMA\r\n"
  2929. " #pragma data_seg(\"PAGED\")\r\n"
  2930. "#endif\r\n"
  2931. "\r\n"
  2932. "UCHAR %sBinaryMofData[] =\r\n"
  2933. "{\r\n"
  2934. " #include \"%s.x\"\r\n"
  2935. "};\r\n"
  2936. "#ifdef ALLOC_DATA_PRAGMA\r\n"
  2937. " #pragma data_seg()\r\n"
  2938. "#endif\r\n"
  2939. "#endif\r\n"
  2940. "\r\n",
  2941. BaseName,
  2942. BaseXFileName
  2943. );
  2944. FilePrint(TemplateHandle,
  2945. "//\r\n"
  2946. "// Define symbolic names for the guid indexes\r\n"
  2947. );
  2948. FilePrintMofClassLoop(TemplateHandle, MofResource, ClassName, i, TRUE,
  2949. ("#define %wsGuidIndex %d\r\n",
  2950. ClassName, i));
  2951. FilePrint(TemplateHandle,
  2952. "#ifdef USE_BINARY_MOF_QUERY\r\n"
  2953. "#define BinaryMofGuidIndex %d\r\n"
  2954. "#endif\r\n",
  2955. i
  2956. );
  2957. FilePrint(TemplateHandle,
  2958. "//\r\n"
  2959. "// List of guids supported\r\n\r\n"
  2960. );
  2961. EnumerateMofClasses(TemplateHandle,
  2962. MofResource,
  2963. GenerateGuidListTemplate,
  2964. NULL);
  2965. FilePrint(TemplateHandle,
  2966. "#ifdef USE_BINARY_MOF_QUERY\r\n"
  2967. "GUID %sBinaryMofGUID = BINARY_MOF_GUID;\r\n"
  2968. "#endif\r\n"
  2969. "\r\n"
  2970. "//\r\n"
  2971. "// TODO: Make sure the instance count and flags are set properly for each\r\n"
  2972. "// guid\r\n"
  2973. "WMIGUIDREGINFO %sGuidList[] =\r\n"
  2974. "{\r\n",
  2975. BaseName, BaseName);
  2976. FilePrintMofClassLoop(TemplateHandle, MofResource, ClassName, i, TRUE,
  2977. (" {\r\n"
  2978. " &%wsGUID, // Guid\r\n"
  2979. " 1, // # of instances in each device\r\n"
  2980. " 0 // Flags\r\n"
  2981. " },\r\n",
  2982. ClassName));
  2983. FilePrint(TemplateHandle,
  2984. "#ifdef USE_BINARY_MOF_QUERY\r\n"
  2985. " {\r\n"
  2986. " &%sBinaryMofGUID,\r\n"
  2987. " 1,\r\n"
  2988. " 0\r\n"
  2989. " }\r\n"
  2990. "#endif\r\n"
  2991. "};\r\n\r\n"
  2992. "#define %sGuidCount (sizeof(%sGuidList) / sizeof(WMIGUIDREGINFO))\r\n"
  2993. "\r\n",
  2994. BaseName, BaseName, BaseName);
  2995. FilePrint(TemplateHandle,
  2996. "//\r\n"
  2997. "// We need to hang onto the registry path passed to our driver entry so that\r\n"
  2998. "// we can return it in the QueryWmiRegInfo callback. Be sure to store a copy\r\n"
  2999. "// of it into %sRegistryPath in the DriverEntry routine\r\n"
  3000. "//\r\n"
  3001. "extern UNICODE_STRING %sRegistryPath;\r\n\r\n",
  3002. BaseName, BaseName);
  3003. FilePrint(TemplateHandle,
  3004. "NTSTATUS %sSystemControl(\r\n"
  3005. " PDEVICE_OBJECT DeviceObject,\r\n"
  3006. " PIRP Irp\r\n"
  3007. " )\r\n"
  3008. "/*++\r\n"
  3009. "\r\n"
  3010. "Routine Description:\r\n"
  3011. "\r\n"
  3012. " Dispatch routine for System Control IRPs (MajorFunction == IRP_MJ_SYSTEM_CONTROL)\r\n"
  3013. "\r\n"
  3014. "Arguments:\r\n"
  3015. "\r\n"
  3016. " DeviceObject \r\n"
  3017. " Irp\r\n"
  3018. "\r\n"
  3019. "Return Value:\r\n"
  3020. "\r\n"
  3021. " NT status code\r\n"
  3022. "\r\n"
  3023. "--*/\r\n"
  3024. "{\r\n"
  3025. " PWMILIB_CONTEXT wmilibContext;\r\n"
  3026. " NTSTATUS status;\r\n"
  3027. " SYSCTL_IRP_DISPOSITION disposition;\r\n"
  3028. " PDEVICE_EXTENSION devExt = DeviceObject->DeviceExtension;\r\n"
  3029. "\r\n"
  3030. " //\r\n"
  3031. " // TODO: Point at the WMILIB context within the device extension\r\n"
  3032. " wmilibContext = &devExt->WmiLib;\r\n"
  3033. "\r\n"
  3034. " //\r\n"
  3035. " // Call Wmilib helper function to crack the irp. If this is a wmi irp\r\n"
  3036. " // that is targetted for this device then WmiSystemControl will callback\r\n"
  3037. " // at the appropriate callback routine.\r\n"
  3038. " //\r\n"
  3039. " status = WmiSystemControl(wmilibContext,\r\n"
  3040. " DeviceObject,\r\n"
  3041. " Irp,\r\n"
  3042. " &disposition);\r\n"
  3043. "\r\n"
  3044. " switch(disposition)\r\n"
  3045. " {\r\n"
  3046. " case IrpProcessed:\r\n"
  3047. " {\r\n"
  3048. " //\r\n"
  3049. " // This irp has been processed and may be completed or pending.\r\n"
  3050. " break;\r\n"
  3051. " }\r\n"
  3052. "\r\n"
  3053. " case IrpNotCompleted:\r\n"
  3054. " {\r\n"
  3055. " //\r\n"
  3056. " // This irp has not been completed, but has been fully processed.\r\n"
  3057. " // we will complete it now.\r\n"
  3058. " IoCompleteRequest(Irp, IO_NO_INCREMENT);\r\n"
  3059. " break;\r\n"
  3060. " }\r\n"
  3061. "\r\n"
  3062. " case IrpForward:\r\n"
  3063. " case IrpNotWmi:\r\n"
  3064. " default:\r\n"
  3065. " {\r\n"
  3066. " //\r\n"
  3067. " // This irp is either not a WMI irp or is a WMI irp targetted\r\n"
  3068. " // at a device lower in the stack.\r\n"
  3069. "\r\n"
  3070. " // TODO: Forward IRP down the device stack to the next device\r\n"
  3071. " // Or if this is a PDO then just complete the irp without\r\n"
  3072. " // touching it.\r\n"
  3073. " break;\r\n"
  3074. " }\r\n"
  3075. "\r\n"
  3076. " }\r\n"
  3077. "\r\n"
  3078. " return(status);\r\n"
  3079. "}\r\n",
  3080. BaseName);
  3081. FilePrint(TemplateHandle,
  3082. "NTSTATUS\r\n"
  3083. "%sQueryWmiRegInfo(\r\n"
  3084. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  3085. " OUT ULONG *RegFlags,\r\n"
  3086. " OUT PUNICODE_STRING InstanceName,\r\n"
  3087. " OUT PUNICODE_STRING *RegistryPath,\r\n"
  3088. " OUT PUNICODE_STRING MofResourceName,\r\n"
  3089. " OUT PDEVICE_OBJECT *Pdo\r\n"
  3090. " )\r\n"
  3091. "/*++\r\n"
  3092. "\r\n"
  3093. "Routine Description:\r\n"
  3094. "\r\n"
  3095. " This routine is a callback into the driver to retrieve the list of\r\n"
  3096. " guids or data blocks that the driver wants to register with WMI. This\r\n"
  3097. " routine may not pend or block. Driver should NOT call\r\n"
  3098. " WmiCompleteRequest.\r\n"
  3099. "\r\n"
  3100. "Arguments:\r\n"
  3101. "\r\n"
  3102. " DeviceObject is the device whose registration info is being queried\r\n"
  3103. "\r\n"
  3104. " *RegFlags returns with a set of flags that describe the guids being\r\n"
  3105. " registered for this device. If the device wants enable and disable\r\n"
  3106. " collection callbacks before receiving queries for the registered\r\n"
  3107. " guids then it should return the WMIREG_FLAG_EXPENSIVE flag. Also the\r\n"
  3108. " returned flags may specify WMIREG_FLAG_INSTANCE_PDO in which case\r\n"
  3109. " the instance name is determined from the PDO associated with the\r\n"
  3110. " device object. Note that the PDO must have an associated devnode. If\r\n"
  3111. " WMIREG_FLAG_INSTANCE_PDO is not set then Name must return a unique\r\n"
  3112. " name for the device.\r\n"
  3113. "\r\n"
  3114. " InstanceName returns with the instance name for the guids if\r\n"
  3115. " WMIREG_FLAG_INSTANCE_PDO is not set in the returned *RegFlags. The\r\n"
  3116. " caller will call ExFreePool with the buffer returned.\r\n"
  3117. "\r\n"
  3118. " *RegistryPath returns with the registry path of the driver. The caller\r\n"
  3119. " does NOT free this buffer.\r\n"
  3120. "\r\n"
  3121. " *MofResourceName returns with the name of the MOF resource attached to\r\n"
  3122. " the binary file. If the driver does not have a mof resource attached\r\n"
  3123. " then this can be returned as NULL. The caller does NOT free this\r\n"
  3124. " buffer.\r\n"
  3125. "\r\n"
  3126. " *Pdo returns with the device object for the PDO associated with this\r\n"
  3127. " device if the WMIREG_FLAG_INSTANCE_PDO flag is retured in\r\n"
  3128. " *RegFlags.\r\n"
  3129. "\r\n"
  3130. "Return Value:\r\n"
  3131. "\r\n"
  3132. " status\r\n"
  3133. "\r\n"
  3134. "--*/\r\n"
  3135. "{\r\n"
  3136. " struct DEVICE_EXTENSION * devExt = DeviceObject->DeviceExtension;\r\n"
  3137. "\r\n"
  3138. " //\r\n"
  3139. " // Return the registry path for this driver. This is required so WMI\r\n"
  3140. " // can find your driver image and can attribute any eventlog messages to\r\n"
  3141. " // your driver.\r\n"
  3142. " *RegistryPath = &%sRegistryPath;\r\n"
  3143. " \r\n"
  3144. "#ifndef USE_BINARY_MOF_RESOURCE\r\n"
  3145. " //\r\n"
  3146. " // Return the name specified in the .rc file of the resource which\r\n"
  3147. " // contains the bianry mof data. By default WMI will look for this\r\n"
  3148. " // resource in the driver image (.sys) file, however if the value\r\n"
  3149. " // MofImagePath is specified in the driver's registry key\r\n"
  3150. " // then WMI will look for the resource in the file specified there.\r\n"
  3151. " RtlInitUnicodeString(MofResourceName, L\"MofResourceName\");\r\n"
  3152. "#endif\r\n"
  3153. "\r\n"
  3154. " //\r\n"
  3155. " // Specify that the driver wants WMI to automatically generate instance\r\n"
  3156. " // names for all of the data blocks based upon the device stack's\r\n"
  3157. " // device instance id. Doing this is STRONGLY recommended since additional\r\n"
  3158. " // information about the device would then be available to callers.\r\n"
  3159. " *RegFlags = WMIREG_FLAG_INSTANCE_PDO;\r\n"
  3160. "\r\n"
  3161. " //\r\n"
  3162. " // TODO: Assign the physical device object for the device stack to *Pdo\r\n"
  3163. " *Pdo = devExt->physicalDevObj;\r\n"
  3164. "\r\n"
  3165. " return(STATUS_SUCCESS);\r\n"
  3166. "}\r\n"
  3167. "\r\n",
  3168. BaseName, BaseName);
  3169. FilePrint(TemplateHandle,
  3170. "NTSTATUS\r\n"
  3171. "%sQueryWmiDataBlock(\r\n"
  3172. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  3173. " IN PIRP Irp,\r\n"
  3174. " IN ULONG GuidIndex,\r\n"
  3175. " IN ULONG InstanceIndex,\r\n"
  3176. " IN ULONG InstanceCount,\r\n"
  3177. " IN OUT PULONG InstanceLengthArray,\r\n"
  3178. " IN ULONG BufferAvail,\r\n"
  3179. " OUT PUCHAR Buffer\r\n"
  3180. " )\r\n"
  3181. "/*++\r\n"
  3182. "\r\n"
  3183. "Routine Description:\r\n"
  3184. "\r\n"
  3185. " This routine is a callback into the driver to query for the contents of\r\n"
  3186. " all instances of a data block. If the driver can satisfy the query within\r\n"
  3187. " the callback it should call WmiCompleteRequest to complete the irp before\r\n"
  3188. " returning to the caller. Or the driver can return STATUS_PENDING if the\r\n"
  3189. " irp cannot be completed immediately and must then call WmiCompleteRequest\r\n"
  3190. " once the query is satisfied.\r\n"
  3191. "\r\n"
  3192. "Arguments:\r\n"
  3193. "\r\n"
  3194. " DeviceObject is the device whose data block is being queried\r\n"
  3195. "\r\n"
  3196. " Irp is the Irp that makes this request\r\n"
  3197. "\r\n"
  3198. " GuidIndex is the index into the list of guids provided when the\r\n"
  3199. " device registered\r\n"
  3200. "\r\n"
  3201. " InstanceCount is the number of instnaces expected to be returned for\r\n"
  3202. " the data block.\r\n"
  3203. "\r\n"
  3204. " InstanceLengthArray is a pointer to an array of ULONG that returns the\r\n"
  3205. " lengths of each instance of the data block. If this is NULL then\r\n"
  3206. " there was not enough space in the output buffer to fufill the request\r\n"
  3207. " so the irp should be completed with the buffer needed.\r\n"
  3208. "\r\n"
  3209. " BufferAvail on entry has the maximum size available to write the data\r\n"
  3210. " blocks.\r\n"
  3211. "\r\n"
  3212. " Buffer on return is filled with the returned data blocks. Note that each\r\n"
  3213. " instance of the data block must be aligned on a 8 byte boundry.\r\n"
  3214. "\r\n"
  3215. "\r\n"
  3216. "Return Value:\r\n"
  3217. "\r\n"
  3218. " status\r\n"
  3219. "\r\n"
  3220. "--*/\r\n"
  3221. "{\r\n"
  3222. " NTSTATUS status = STATUS_UNSUCCESSFUL;\r\n"
  3223. " struct DEVICE_EXTENSION * devExt = DeviceObject->DeviceExtension;\r\n"
  3224. " ULONG sizeNeeded;\r\n"
  3225. "\r\n"
  3226. " switch(GuidIndex)\r\n"
  3227. " {\r\n"
  3228. ,BaseName );
  3229. FilePrintMofClassLoop(TemplateHandle, MofResource, ClassName, i, TRUE, (
  3230. " case %wsGuidIndex:\r\n"
  3231. " {\r\n"
  3232. " //\r\n"
  3233. " // TODO: Check that the size of the buffer passed is large enough\r\n"
  3234. " // for all of the instances requested and if so fill Buffer\r\n"
  3235. " // with the data. Make sure that each instance begins on an\r\n"
  3236. " // 8 byte boundry.\r\n"
  3237. " //\r\n"
  3238. " break;\r\n"
  3239. " }\r\n\r\n",
  3240. ClassName));
  3241. FilePrint(TemplateHandle,
  3242. "#ifdef USE_BINARY_MOF_QUERY\r\n"
  3243. " case BinaryMofGuidIndex:\r\n"
  3244. " {\r\n"
  3245. " //\r\n"
  3246. " // TODO: If the driver supports reporting MOF dynamically, \r\n"
  3247. " // change this code to handle multiple instances of the\r\n"
  3248. " // binary mof guid and return only those instances that\r\n"
  3249. " // should be reported to the schema\r\n"
  3250. " //\r\n"
  3251. " sizeNeeded = sizeof(%sBinaryMofData);\r\n"
  3252. "\r\n"
  3253. " if (BufferAvail < sizeNeeded)\r\n"
  3254. " {\r\n"
  3255. " status = STATUS_BUFFER_TOO_SMALL;\r\n"
  3256. " } else {\r\n"
  3257. " RtlCopyMemory(Buffer, %sBinaryMofData, sizeNeeded);\r\n"
  3258. " *InstanceLengthArray = sizeNeeded;\r\n"
  3259. " status = STATUS_SUCCESS;\r\n"
  3260. " }\r\n"
  3261. " break;\r\n"
  3262. " }\r\n"
  3263. "#endif\r\n"
  3264. "\r\n"
  3265. " default:\r\n"
  3266. " {\r\n"
  3267. " status = STATUS_WMI_GUID_NOT_FOUND;\r\n"
  3268. " break;\r\n"
  3269. " }\r\n"
  3270. " }\r\n"
  3271. "\r\n"
  3272. " //\r\n"
  3273. " // Complete the irp. If there was not enough room in the output buffer\r\n"
  3274. " // then status is STATUS_BUFFER_TOO_SMALL and sizeNeeded has the size\r\n"
  3275. " // needed to return all of the data. If there was enough room then\r\n"
  3276. " // status is STATUS_SUCCESS and sizeNeeded is the actual number of bytes\r\n"
  3277. " // being returned.\r\n"
  3278. " status = WmiCompleteRequest(\r\n"
  3279. " DeviceObject,\r\n"
  3280. " Irp,\r\n"
  3281. " status,\r\n"
  3282. " sizeNeeded,\r\n"
  3283. " IO_NO_INCREMENT);\r\n"
  3284. "\r\n"
  3285. " return(status);\r\n"
  3286. "}\r\n"
  3287. , BaseName, BaseName);
  3288. if (! IsReadOnly)
  3289. {
  3290. FilePrint(TemplateHandle,
  3291. "\r\n"
  3292. "NTSTATUS\r\n"
  3293. "%sSetWmiDataBlock(\r\n"
  3294. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  3295. " IN PIRP Irp,\r\n"
  3296. " IN ULONG GuidIndex,\r\n"
  3297. " IN ULONG InstanceIndex,\r\n"
  3298. " IN ULONG BufferSize,\r\n"
  3299. " IN PUCHAR Buffer\r\n"
  3300. " )\r\n"
  3301. "/*++\r\n"
  3302. "\r\n"
  3303. "Routine Description:\r\n"
  3304. "\r\n"
  3305. " This routine is a callback into the driver to change the contents of\r\n"
  3306. " a data block. If the driver can change the data block within\r\n"
  3307. " the callback it should call WmiCompleteRequest to complete the irp before\r\n"
  3308. " returning to the caller. Or the driver can return STATUS_PENDING if the\r\n"
  3309. " irp cannot be completed immediately and must then call WmiCompleteRequest\r\n"
  3310. " once the data is changed.\r\n"
  3311. "\r\n"
  3312. "Arguments:\r\n"
  3313. "\r\n"
  3314. " DeviceObject is the device whose data block is being queried\r\n"
  3315. "\r\n"
  3316. " Irp is the Irp that makes this request\r\n"
  3317. "\r\n"
  3318. " GuidIndex is the index into the list of guids provided when the\r\n"
  3319. " device registered\r\n"
  3320. "\r\n"
  3321. " BufferSize has the size of the data block passed\r\n"
  3322. "\r\n"
  3323. " Buffer has the new values for the data block\r\n"
  3324. "\r\n"
  3325. "\r\n"
  3326. "Return Value:\r\n"
  3327. "\r\n"
  3328. " status\r\n"
  3329. "\r\n"
  3330. "--*/\r\n"
  3331. "{\r\n"
  3332. " NTSTATUS status;\r\n"
  3333. " struct DEVICE_EXTENSION * devExt = DeviceObject->DeviceExtension;\r\n"
  3334. "\r\n"
  3335. "\r\n"
  3336. " switch(GuidIndex)\r\n"
  3337. " {\r\n"
  3338. , BaseName);
  3339. EnumerateMofClasses(TemplateHandle,
  3340. MofResource,
  3341. GenerateSetList,
  3342. "\r\n"
  3343. " case %wsGuidIndex:\r\n"
  3344. " { \r\n"
  3345. " //\r\n"
  3346. " // TODO: Validate InstanceIndex, BufferSize and Buffer contents\r\n"
  3347. " // and if valid then set the underlying data block, write\r\n"
  3348. " // to the hardware, etc.\r\n"
  3349. " break;\r\n"
  3350. " }\r\n"
  3351. "\r\n"
  3352. );
  3353. FilePrint(TemplateHandle,
  3354. " default:\r\n"
  3355. " {\r\n"
  3356. " status = STATUS_WMI_GUID_NOT_FOUND;\r\n"
  3357. " break;\r\n"
  3358. " }\r\n"
  3359. " }\r\n"
  3360. "\r\n"
  3361. " status = WmiCompleteRequest(\r\n"
  3362. " DeviceObject,\r\n"
  3363. " Irp,\r\n"
  3364. " status,\r\n"
  3365. " 0,\r\n"
  3366. " IO_NO_INCREMENT);\r\n"
  3367. "\r\n"
  3368. " return(status);\r\n"
  3369. "\r\n"
  3370. "\r\n"
  3371. "}\r\n"
  3372. );
  3373. FilePrint(TemplateHandle,
  3374. " \r\n"
  3375. "NTSTATUS\r\n"
  3376. "%sSetWmiDataItem(\r\n"
  3377. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  3378. " IN PIRP Irp,\r\n"
  3379. " IN ULONG GuidIndex,\r\n"
  3380. " IN ULONG InstanceIndex,\r\n"
  3381. " IN ULONG DataItemId,\r\n"
  3382. " IN ULONG BufferSize,\r\n"
  3383. " IN PUCHAR Buffer\r\n"
  3384. " )\r\n"
  3385. "/*++\r\n"
  3386. "\r\n"
  3387. "Routine Description:\r\n"
  3388. "\r\n"
  3389. " This routine is a callback into the driver to change the contents of\r\n"
  3390. " a data block. If the driver can change the data block within\r\n"
  3391. " the callback it should call WmiCompleteRequest to complete the irp before\r\n"
  3392. " returning to the caller. Or the driver can return STATUS_PENDING if the\r\n"
  3393. " irp cannot be completed immediately and must then call WmiCompleteRequest\r\n"
  3394. " once the data is changed.\r\n"
  3395. "\r\n"
  3396. "Arguments:\r\n"
  3397. "\r\n"
  3398. " DeviceObject is the device whose data block is being changed\r\n"
  3399. "\r\n"
  3400. " Irp is the Irp that makes this request\r\n"
  3401. "\r\n"
  3402. " GuidIndex is the index into the list of guids provided when the\r\n"
  3403. " device registered\r\n"
  3404. "\r\n"
  3405. " DataItemId has the id of the data item being set\r\n"
  3406. "\r\n"
  3407. " BufferSize has the size of the data item passed\r\n"
  3408. "\r\n"
  3409. " Buffer has the new values for the data item\r\n"
  3410. "\r\n"
  3411. "\r\n"
  3412. "Return Value:\r\n"
  3413. "\r\n"
  3414. " status\r\n"
  3415. "\r\n"
  3416. "--*/\r\n"
  3417. "{\r\n"
  3418. " NTSTATUS status;\r\n"
  3419. "\r\n"
  3420. " switch(GuidIndex)\r\n"
  3421. " {\r\n"
  3422. , BaseName);
  3423. EnumerateMofClasses(TemplateHandle,
  3424. MofResource,
  3425. GenerateSetList,
  3426. " case %wsGuidIndex:\r\n"
  3427. " { \r\n"
  3428. " //\r\n"
  3429. " // TODO: Validate InstanceIndex, DataItemId, BufferSize \r\n"
  3430. " // and Buffer contents\r\n"
  3431. " // and if valid then set the underlying data item, write\r\n"
  3432. " // to the hardware, etc.\r\n"
  3433. " break;\r\n"
  3434. " }\r\n");
  3435. FilePrint(TemplateHandle,
  3436. " default:\r\n"
  3437. " {\r\n"
  3438. " status = STATUS_WMI_GUID_NOT_FOUND;\r\n"
  3439. " break;\r\n"
  3440. " }\r\n"
  3441. " }\r\n"
  3442. "\r\n"
  3443. " status = WmiCompleteRequest(\r\n"
  3444. " DeviceObject,\r\n"
  3445. " Irp,\r\n"
  3446. " status,\r\n"
  3447. " 0,\r\n"
  3448. " IO_NO_INCREMENT);\r\n"
  3449. "\r\n"
  3450. " return(status);\r\n"
  3451. "}\r\n"
  3452. );
  3453. }
  3454. if (SupportsMethods)
  3455. {
  3456. FilePrint(TemplateHandle,
  3457. "NTSTATUS\r\n"
  3458. "%sExecuteWmiMethod(\r\n"
  3459. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  3460. " IN PIRP Irp,\r\n"
  3461. " IN ULONG GuidIndex,\r\n"
  3462. " IN ULONG InstanceIndex,\r\n"
  3463. " IN ULONG MethodId,\r\n"
  3464. " IN ULONG InBufferSize,\r\n"
  3465. " IN ULONG OutBufferSize,\r\n"
  3466. " IN PUCHAR Buffer\r\n"
  3467. " )\r\n"
  3468. "/*++\r\n"
  3469. "\r\n"
  3470. "Routine Description:\r\n"
  3471. "\r\n"
  3472. " This routine is a callback into the driver to execute a method. If\r\n"
  3473. " the driver can complete the method within the callback it should\r\n"
  3474. " call WmiCompleteRequest to complete the irp before returning to the\r\n"
  3475. " caller. Or the driver can return STATUS_PENDING if the irp cannot be\r\n"
  3476. " completed immediately and must then call WmiCompleteRequest once the\r\n"
  3477. " data is changed.\r\n"
  3478. "\r\n"
  3479. "Arguments:\r\n"
  3480. "\r\n"
  3481. " DeviceObject is the device whose method is being executed\r\n"
  3482. "\r\n"
  3483. " Irp is the Irp that makes this request\r\n"
  3484. "\r\n"
  3485. " GuidIndex is the index into the list of guids provided when the\r\n"
  3486. " device registered\r\n"
  3487. "\r\n"
  3488. " MethodId has the id of the method being called\r\n"
  3489. "\r\n"
  3490. " InBufferSize has the size of the data block passed in as the input to\r\n"
  3491. " the method.\r\n"
  3492. "\r\n"
  3493. " OutBufferSize on entry has the maximum size available to write the\r\n"
  3494. " returned data block.\r\n"
  3495. "\r\n"
  3496. " Buffer is filled with the input buffer on entry and returns with\r\n"
  3497. " the output data block\r\n"
  3498. "\r\n"
  3499. "Return Value:\r\n"
  3500. "\r\n"
  3501. " status\r\n"
  3502. "\r\n"
  3503. "--*/\r\n"
  3504. "{\r\n"
  3505. " ULONG sizeNeeded = 0;\r\n"
  3506. " NTSTATUS status;\r\n"
  3507. "\r\n"
  3508. " switch(GuidIndex)\r\n"
  3509. " {\r\n"
  3510. " \r\n"
  3511. , BaseName);
  3512. EnumerateMofClasses(TemplateHandle,
  3513. MofResource,
  3514. GenerateMethodCTemplate,
  3515. NULL);
  3516. FilePrint(TemplateHandle,
  3517. " default:\r\n"
  3518. " {\r\n"
  3519. " status = STATUS_WMI_GUID_NOT_FOUND;\r\n"
  3520. " }\r\n"
  3521. " }\r\n"
  3522. "\r\n"
  3523. " status = WmiCompleteRequest(\r\n"
  3524. " DeviceObject,\r\n"
  3525. " Irp,\r\n"
  3526. " status,\r\n"
  3527. " sizeNeeded,\r\n"
  3528. " IO_NO_INCREMENT);\r\n"
  3529. "\r\n"
  3530. " return(status);\r\n"
  3531. "}\r\n"
  3532. );
  3533. }
  3534. if (SupportsFunctionControl)
  3535. {
  3536. FilePrint(TemplateHandle,
  3537. "NTSTATUS\r\n"
  3538. "%sFunctionControl(\r\n"
  3539. " IN PDEVICE_OBJECT DeviceObject,\r\n"
  3540. " IN PIRP Irp,\r\n"
  3541. " IN ULONG GuidIndex,\r\n"
  3542. " IN WMIENABLEDISABLECONTROL Function,\r\n"
  3543. " IN BOOLEAN Enable\r\n"
  3544. " )\r\n"
  3545. "/*++\r\n"
  3546. "\r\n"
  3547. "Routine Description:\r\n"
  3548. "\r\n"
  3549. " This routine is a callback into the driver to enabled or disable event\r\n"
  3550. " generation or data block collection. A device should only expect a\r\n"
  3551. " single enable when the first event or data consumer enables events or\r\n"
  3552. " data collection and a single disable when the last event or data\r\n"
  3553. " consumer disables events or data collection. Data blocks will only\r\n"
  3554. " receive collection enable/disable if they were registered as requiring\r\n"
  3555. " it. If the driver can complete enabling/disabling within the callback it\r\n"
  3556. " should call WmiCompleteRequest to complete the irp before returning to\r\n"
  3557. " the caller. Or the driver can return STATUS_PENDING if the irp cannot be\r\n"
  3558. " completed immediately and must then call WmiCompleteRequest once the\r\n"
  3559. " data is changed.\r\n"
  3560. "\r\n"
  3561. "Arguments:\r\n"
  3562. "\r\n"
  3563. " DeviceObject is the device object\r\n"
  3564. "\r\n"
  3565. " GuidIndex is the index into the list of guids provided when the\r\n"
  3566. " device registered\r\n"
  3567. "\r\n"
  3568. " Function specifies which functionality is being enabled or disabled\r\n"
  3569. "\r\n"
  3570. " Enable is TRUE then the function is being enabled else disabled\r\n"
  3571. "\r\n"
  3572. "Return Value:\r\n"
  3573. "\r\n"
  3574. " status\r\n"
  3575. "\r\n"
  3576. "--*/\r\n"
  3577. "{\r\n"
  3578. " NTSTATUS status;\r\n"
  3579. "\r\n"
  3580. " switch(GuidIndex)\r\n"
  3581. " {\r\n",
  3582. BaseName);
  3583. EnumerateMofClasses(TemplateHandle,
  3584. MofResource,
  3585. GenerateFunctionControlListTemplate,
  3586. NULL);
  3587. FilePrint(TemplateHandle,
  3588. " \r\n"
  3589. " default:\r\n"
  3590. " {\r\n"
  3591. " status = STATUS_WMI_GUID_NOT_FOUND;\r\n"
  3592. " break;\r\n"
  3593. " }\r\n"
  3594. " }\r\n"
  3595. " \r\n"
  3596. " status = WmiCompleteRequest(\r\n"
  3597. " DeviceObject,\r\n"
  3598. " Irp,\r\n"
  3599. " STATUS_SUCCESS,\r\n"
  3600. " 0,\r\n"
  3601. " IO_NO_INCREMENT);\r\n"
  3602. " return(status);\r\n"
  3603. "}\r\n"
  3604. );
  3605. }
  3606. FilePrint(TemplateHandle,
  3607. "NTSTATUS\r\n"
  3608. "%sInitializeWmilibContext(\r\n"
  3609. " IN PWMILIB_CONTEXT WmilibContext\r\n"
  3610. " )\r\n"
  3611. "/*++\r\n"
  3612. "\r\n"
  3613. "Routine Description:\r\n"
  3614. "\r\n"
  3615. " This routine will initialize the wmilib context structure with the\r\n"
  3616. " guid list and the pointers to the wmilib callback functions. This routine\r\n"
  3617. " should be called before calling IoWmiRegistrationControl to register\r\n"
  3618. " your device object.\r\n"
  3619. "\r\n"
  3620. "Arguments:\r\n"
  3621. "\r\n"
  3622. " WmilibContext is pointer to the wmilib context.\r\n"
  3623. "\r\n"
  3624. "Return Value:\r\n"
  3625. "\r\n"
  3626. " status\r\n"
  3627. "\r\n"
  3628. "--*/\r\n"
  3629. "{\r\n"
  3630. " RtlZeroMemory(WmilibContext, sizeof(WMILIB_CONTEXT));\r\n"
  3631. " \r\n"
  3632. " WmilibContext->GuidCount = %sGuidCount;\r\n"
  3633. " WmilibContext->GuidList = %sGuidList; \r\n"
  3634. " \r\n"
  3635. " WmilibContext->QueryWmiRegInfo = %sQueryWmiRegInfo;\r\n"
  3636. " WmilibContext->QueryWmiDataBlock = %sQueryWmiDataBlock;\r\n",
  3637. BaseName,
  3638. BaseName,
  3639. BaseName,
  3640. BaseName,
  3641. BaseName);
  3642. if (! IsReadOnly)
  3643. {
  3644. FilePrint(TemplateHandle,
  3645. " WmilibContext->SetWmiDataBlock = %sSetWmiDataBlock;\r\n"
  3646. " WmilibContext->SetWmiDataItem = %sSetWmiDataItem;\r\n",
  3647. BaseName, BaseName);
  3648. }
  3649. if (SupportsMethods)
  3650. {
  3651. FilePrint(TemplateHandle,
  3652. " WmilibContext->ExecuteWmiMethod = %sExecuteWmiMethod;\r\n",
  3653. BaseName);
  3654. }
  3655. if (SupportsFunctionControl)
  3656. {
  3657. FilePrint(TemplateHandle,
  3658. " WmilibContext->WmiFunctionControl = %sFunctionControl;\r\n",
  3659. BaseName);
  3660. }
  3661. FilePrint(TemplateHandle,
  3662. "\r\n"
  3663. " return(STATUS_SUCCESS);\r\n"
  3664. "}"
  3665. );
  3666. CloseHandle(TemplateHandle);
  3667. return(ERROR_SUCCESS);
  3668. }
  3669. //
  3670. // A data item is variable length if it is a variable length array or a
  3671. // string that does not have a maxiumum length specified
  3672. //
  3673. #define WmipIsDataitemVariableLen(DataItem) \
  3674. ( (DataItem->Flags & MOFDI_FLAG_VARIABLE_ARRAY) || \
  3675. ((DataItem->DataType == MOFString) && \
  3676. (DataItem->MaxLen == 0)) || \
  3677. (DataItem->DataType == MOFZTString) || \
  3678. (DataItem->DataType == MOFAnsiString) )
  3679. BOOLEAN ClassCanCreateHeader(
  3680. PMOFCLASSINFOW ClassInfo,
  3681. ULONG RequiredFlags,
  3682. PULONG ItemCount
  3683. )
  3684. {
  3685. ULONG i;
  3686. BOOLEAN HasVariableLength = FALSE;
  3687. PMOFDATAITEMW DataItem;
  3688. ULONG Count;
  3689. Count = 0;
  3690. for (i = 0; i < ClassInfo->DataItemCount; i++)
  3691. {
  3692. DataItem = &ClassInfo->DataItems[i];
  3693. if ((RequiredFlags == 0xffffffff) ||
  3694. (DataItem->Flags & RequiredFlags))
  3695. {
  3696. if (HasVariableLength)
  3697. {
  3698. *ItemCount = Count;
  3699. return(FALSE);
  3700. }
  3701. Count++;
  3702. HasVariableLength = (! ForceHeaderGeneration) &&
  3703. WmipIsDataitemVariableLen(DataItem);
  3704. }
  3705. }
  3706. *ItemCount = Count;
  3707. return(TRUE);
  3708. }
  3709. PWCHAR MofDataTypeText[15] =
  3710. {
  3711. L"LONG", // 32bit integer
  3712. L"ULONG", // 32bit unsigned integer
  3713. L"LONGLONG", // 64bit integer
  3714. L"ULONGLONG", // 32bit unsigned integer
  3715. L"SHORT", // 16bit integer
  3716. L"USHORT", // 16bit unsigned integer
  3717. L"CHAR", // 8bit integer
  3718. L"UCHAR", // 8bit unsigned integer
  3719. L"WCHAR", // Wide (16bit) character
  3720. L"DATETIME", // Date field
  3721. L"BOOLEAN", // 8bit Boolean value
  3722. L"MOFEmbedded", // Embedded class
  3723. L"MOFString", // Counted String type
  3724. L"MOFZTString", // NULL terminated unicode string
  3725. L"MOFAnsiString" // NULL terminated ansi string
  3726. };
  3727. ULONG GenerateClassHeader(
  3728. HANDLE TemplateHandle,
  3729. PMOFRESOURCE MofResource,
  3730. PWCHAR ClassName,
  3731. PMOFCLASSINFOW ClassInfo,
  3732. ULONG RequiredFlags
  3733. )
  3734. {
  3735. ULONG Status;
  3736. CBMOFDataItem *PropertyObject;
  3737. CBMOFQualList *PropertyQualifier;
  3738. ULONG Status2, QualifierType;
  3739. PVOID ptr;
  3740. ULONG ValueMapCount, DefineValuesCount, ValuesCount;
  3741. PWCHAR *ValueMapPtr, *DefineValuesPtr, *ValuesPtr;
  3742. ULONG BitMapCount, DefineBitMapCount, BitValuesCount, BitMapValue;
  3743. PWCHAR *BitMapPtr, *DefineBitMapPtr, *BitValuesPtr;
  3744. PWCHAR DefineDataId;
  3745. WCHAR DefineDataIdText[MAX_PATH];
  3746. PMOFDATAITEMW DataItem;
  3747. PWCHAR Description;
  3748. PMOFCLASS EmbeddedClass;
  3749. ULONG i, j;
  3750. PWCHAR DataTypeText;
  3751. ULONG ItemCount;
  3752. PWCHAR VLCommentText = L" ";
  3753. WmipDebugPrint(("Generate class header for %ws\n", ClassName));
  3754. if ((ClassCanCreateHeader(ClassInfo, RequiredFlags, &ItemCount)) &&
  3755. (ItemCount != 0))
  3756. {
  3757. Status = FilePrint(TemplateHandle,
  3758. "typedef struct _%ws\r\n{\r\n",
  3759. ClassName);
  3760. for (i = 0; i < ClassInfo->DataItemCount; i++)
  3761. {
  3762. DataItem = &ClassInfo->DataItems[i];
  3763. if ((RequiredFlags == 0xffffffff) ||
  3764. (DataItem->Flags & RequiredFlags))
  3765. {
  3766. PropertyQualifier = (CBMOFQualList *)DataItem->PropertyQualifierHandle;
  3767. //
  3768. // Handle any bit maps via the DefineBitMap qualifier
  3769. //
  3770. QualifierType = VT_ARRAY | VT_BSTR;
  3771. if (WmipFindMofQualifier(PropertyQualifier,
  3772. L"DefineBitMap",
  3773. &QualifierType,
  3774. &DefineBitMapCount,
  3775. &DefineBitMapPtr) == ERROR_SUCCESS)
  3776. {
  3777. QualifierType = VT_ARRAY | VT_BSTR;
  3778. if (WmipFindMofQualifier(PropertyQualifier,
  3779. L"BitValues",
  3780. &QualifierType,
  3781. &BitValuesCount,
  3782. &BitValuesPtr) == ERROR_SUCCESS)
  3783. {
  3784. if (DefineBitMapCount == BitValuesCount)
  3785. {
  3786. QualifierType = VT_ARRAY | VT_BSTR;
  3787. if (WmipFindMofQualifier(PropertyQualifier,
  3788. L"BitMap",
  3789. &QualifierType,
  3790. &BitMapCount,
  3791. &BitMapPtr) != ERROR_SUCCESS)
  3792. {
  3793. BitMapPtr = NULL;
  3794. }
  3795. FilePrint(TemplateHandle,
  3796. "\r\n");
  3797. for (j = 0; j < DefineBitMapCount; j++)
  3798. {
  3799. if ((BitMapPtr != NULL) &&
  3800. (j < BitMapCount) &&
  3801. (BitMapPtr[j] != NULL))
  3802. {
  3803. FilePrint(TemplateHandle,
  3804. "// %ws\r\n",
  3805. BitMapPtr[j]);
  3806. }
  3807. BitMapValue = 1 << _wtoi(BitValuesPtr[j]);
  3808. FilePrint(TemplateHandle,
  3809. "#define %ws 0x%x\r\n",
  3810. DefineBitMapPtr[j],
  3811. BitMapValue);
  3812. }
  3813. FilePrint(TemplateHandle,
  3814. "\r\n");
  3815. } else {
  3816. FilePrint(TemplateHandle, "// Warning: Cannot create Bitmap definitions\r\n// Requires DefineBitMap and BitValues qualifier with same number of elements\r\n\r\n");
  3817. }
  3818. for (j = 0; j < BitValuesCount; j++)
  3819. {
  3820. BMOFFree(BitValuesPtr[j]);
  3821. }
  3822. BMOFFree(BitValuesPtr);
  3823. if (BitMapPtr != NULL)
  3824. {
  3825. for (j = 0; j < BitMapCount; j++)
  3826. {
  3827. BMOFFree(BitMapPtr[j]);
  3828. }
  3829. BMOFFree(BitMapPtr);
  3830. }
  3831. } else {
  3832. FilePrint(TemplateHandle, "// Warning: Cannot create Bitmap definitions\r\n// Requires DefineBitMap and BitValues qualifier with same number of elements\r\n\r\n");
  3833. }
  3834. for (j = 0; j < DefineBitMapCount; j++)
  3835. {
  3836. BMOFFree(DefineBitMapPtr[j]);
  3837. }
  3838. BMOFFree(DefineBitMapPtr);
  3839. }
  3840. //
  3841. // Handle any enumerations via the DefineValueMap qualifier
  3842. //
  3843. QualifierType = VT_ARRAY | VT_BSTR;
  3844. if (WmipFindMofQualifier(PropertyQualifier,
  3845. L"DefineValues",
  3846. &QualifierType,
  3847. &DefineValuesCount,
  3848. &DefineValuesPtr) == ERROR_SUCCESS)
  3849. {
  3850. QualifierType = VT_ARRAY | VT_BSTR;
  3851. if (WmipFindMofQualifier(PropertyQualifier,
  3852. L"ValueMap",
  3853. &QualifierType,
  3854. &ValueMapCount,
  3855. &ValueMapPtr) == ERROR_SUCCESS)
  3856. {
  3857. if (DefineValuesCount == ValueMapCount)
  3858. {
  3859. QualifierType = VT_ARRAY | VT_BSTR;
  3860. if (WmipFindMofQualifier(PropertyQualifier,
  3861. L"Values",
  3862. &QualifierType,
  3863. &ValuesCount,
  3864. &ValuesPtr) != ERROR_SUCCESS)
  3865. {
  3866. ValuesPtr = NULL;
  3867. }
  3868. FilePrint(TemplateHandle,
  3869. "\r\n");
  3870. for (j = 0; j < DefineValuesCount; j++)
  3871. {
  3872. if ((ValuesPtr != NULL) &&
  3873. (j < ValuesCount) &&
  3874. (ValuesPtr[j] != NULL))
  3875. {
  3876. FilePrint(TemplateHandle,
  3877. "// %ws\r\n",
  3878. ValuesPtr[j]);
  3879. }
  3880. FilePrint(TemplateHandle,
  3881. "#define %ws %ws\r\n",
  3882. DefineValuesPtr[j],
  3883. ValueMapPtr[j]);
  3884. }
  3885. FilePrint(TemplateHandle,
  3886. "\r\n");
  3887. } else {
  3888. FilePrint(TemplateHandle, "// Warning: Cannot create ValueMap enumeration definitions\r\n// Requires DefineValues and ValueMap qualifier with same number of elements\r\n\r\n");
  3889. }
  3890. for (j = 0; j < ValueMapCount; j++)
  3891. {
  3892. BMOFFree(ValueMapPtr[j]);
  3893. }
  3894. BMOFFree(ValueMapPtr);
  3895. if (ValuesPtr != NULL)
  3896. {
  3897. for (j = 0; j < ValuesCount; j++)
  3898. {
  3899. BMOFFree(ValuesPtr[j]);
  3900. }
  3901. BMOFFree(ValuesPtr);
  3902. }
  3903. } else {
  3904. FilePrint(TemplateHandle, "// Warning: Cannot create ValueMap enumeration definitions\r\n// Requires DefineValues and ValueMap qualifier with same number of elements\r\n\r\n");
  3905. }
  3906. for (j = 0; j < DefineValuesCount; j++)
  3907. {
  3908. BMOFFree(DefineValuesPtr[j]);
  3909. }
  3910. BMOFFree(DefineValuesPtr);
  3911. }
  3912. //
  3913. // Generate structure element from property information
  3914. //
  3915. if (DataItem->Description != NULL)
  3916. {
  3917. Description = DataItem->Description;
  3918. } else {
  3919. Description = L"";
  3920. }
  3921. //
  3922. // Produce a #define for the data id of the property
  3923. //
  3924. QualifierType = VT_BSTR;
  3925. if (WmipFindMofQualifier(PropertyQualifier,
  3926. L"DefineDataId",
  3927. &QualifierType,
  3928. NULL,
  3929. &DefineDataId) != ERROR_SUCCESS)
  3930. {
  3931. swprintf(DefineDataIdText,
  3932. L"%ws_%ws",
  3933. ClassName,
  3934. DataItem->Name);
  3935. DefineDataId = DefineDataIdText;
  3936. }
  3937. if (DataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  3938. {
  3939. // Get Embedded Class name
  3940. EmbeddedClass = WmipFindClassInMofResourceByGuid(
  3941. MofResource,
  3942. &DataItem->EmbeddedClassGuid);
  3943. if (EmbeddedClass != NULL)
  3944. {
  3945. if (EmbeddedClass->MofClassInfo->HeaderName != NULL)
  3946. {
  3947. DataTypeText = EmbeddedClass->MofClassInfo->HeaderName;
  3948. } else {
  3949. DataTypeText = EmbeddedClass->MofClassInfo->Name;
  3950. }
  3951. } else {
  3952. DataTypeText = L"UNKNOWN";
  3953. }
  3954. } else {
  3955. // Standard data type
  3956. if ((DataItem->DataType == MOFString) ||
  3957. (DataItem->DataType == MOFZTString) ||
  3958. (DataItem->DataType == MOFAnsiString) ||
  3959. (DataItem->DataType == MOFDate))
  3960. {
  3961. DataTypeText = L"WCHAR";
  3962. } else {
  3963. DataTypeText = MofDataTypeText[DataItem->DataType];
  3964. }
  3965. }
  3966. if (DataItem->Flags & MOFDI_FLAG_FIXED_ARRAY)
  3967. {
  3968. Status = FilePrint(TemplateHandle,
  3969. " // %ws\r\n%ws %ws %ws[%d];\r\n"
  3970. " #define %ws_SIZE sizeof(%ws[%d])\r\n",
  3971. Description,
  3972. VLCommentText,
  3973. DataTypeText,
  3974. DataItem->Name,
  3975. DataItem->FixedArrayElements,
  3976. DefineDataId,
  3977. DataTypeText,
  3978. DataItem->FixedArrayElements);
  3979. } else if (DataItem->Flags & MOFDI_FLAG_VARIABLE_ARRAY) {
  3980. Status = FilePrint(TemplateHandle,
  3981. " // %ws\r\n%ws %ws %ws[1];\r\n",
  3982. Description,
  3983. VLCommentText,
  3984. DataTypeText,
  3985. DataItem->Name);
  3986. } else if (DataItem->DataType == MOFDate) {
  3987. Status = FilePrint(TemplateHandle,
  3988. " // %ws\r\n%ws WCHAR %ws[25];\r\n"
  3989. " #define %ws_SIZE sizeof(WCHAR[25])\r\n",
  3990. Description,
  3991. VLCommentText,
  3992. DataItem->Name,
  3993. DefineDataId);
  3994. } else if ((DataItem->DataType == MOFString) ||
  3995. (DataItem->DataType == MOFZTString) ||
  3996. (DataItem->DataType == MOFAnsiString)) {
  3997. if (DataItem->MaxLen == 0)
  3998. {
  3999. Status = FilePrint(TemplateHandle,
  4000. " // %ws\r\n%ws CHAR VariableData[1];\r\n",
  4001. Description,
  4002. VLCommentText);
  4003. } else {
  4004. Status = FilePrint(TemplateHandle,
  4005. " // %ws\r\n%ws WCHAR %ws[%d + 1];\r\n",
  4006. Description,
  4007. VLCommentText,
  4008. DataItem->Name,
  4009. DataItem->MaxLen
  4010. );
  4011. }
  4012. } else {
  4013. Status = FilePrint(TemplateHandle,
  4014. " // %ws\r\n%ws %ws %ws;\r\n"
  4015. " #define %ws_SIZE sizeof(%ws)\r\n",
  4016. Description,
  4017. VLCommentText,
  4018. DataTypeText,
  4019. DataItem->Name,
  4020. DefineDataId,
  4021. DataTypeText);
  4022. }
  4023. if (WmipIsDataitemVariableLen(DataItem))
  4024. {
  4025. VLCommentText = L"//";
  4026. }
  4027. Status = FilePrint(TemplateHandle,
  4028. " #define %ws_ID %d\r\n\r\n",
  4029. DefineDataId,
  4030. i+1
  4031. );
  4032. if (DefineDataId != DefineDataIdText)
  4033. {
  4034. BMOFFree(DefineDataId);
  4035. }
  4036. }
  4037. }
  4038. Status = FilePrint(TemplateHandle,
  4039. "} %ws, *P%ws;\r\n\r\n",
  4040. ClassName,
  4041. ClassName);
  4042. } else {
  4043. #if DBG
  4044. printf("Warning: Header for class %ws cannot be created\n",
  4045. ClassName);
  4046. #endif
  4047. if (ItemCount != 0)
  4048. {
  4049. Status = FilePrint(TemplateHandle,
  4050. "// Warning: Header for class %ws cannot be created\r\n"
  4051. "typedef struct _%ws\r\n{\r\n char VariableData[1];\r\n\r\n",
  4052. ClassName,
  4053. ClassName);
  4054. Status = FilePrint(TemplateHandle,
  4055. "} %ws, *P%ws;\r\n\r\n",
  4056. ClassName,
  4057. ClassName);
  4058. } else {
  4059. Status = ERROR_SUCCESS;
  4060. }
  4061. }
  4062. return(Status);
  4063. }
  4064. ULONG GenerateHTemplate(
  4065. PCHAR TemplateFile,
  4066. PMOFRESOURCE MofResource
  4067. )
  4068. {
  4069. HANDLE TemplateHandle;
  4070. ULONG Status;
  4071. ULONG i,j;
  4072. PWCHAR DataTypeText, ClassName;
  4073. WCHAR MethodClassName[MAX_PATH];
  4074. PWCHAR GuidName1, GuidName2;
  4075. PWCHAR GuidSuffix1, GuidSuffix2;
  4076. PMOFDATAITEMW DataItem;
  4077. PMOFCLASSINFOW ClassInfo, MethodClassInfo;
  4078. PLIST_ENTRY MofClassList;
  4079. PMOFCLASS MofClass;
  4080. PCHAR p;
  4081. ULONG Len;
  4082. CBMOFObj *ClassObject;
  4083. PWCHAR MethodBaseClassName;
  4084. TemplateHandle = CreateFile(TemplateFile,
  4085. GENERIC_WRITE,
  4086. 0,
  4087. NULL,
  4088. CREATE_ALWAYS,
  4089. FILE_ATTRIBUTE_NORMAL,
  4090. NULL);
  4091. if ((TemplateHandle == NULL) || (TemplateHandle == INVALID_HANDLE_VALUE))
  4092. {
  4093. return(GetLastError());
  4094. }
  4095. //
  4096. // move back to only get the last part of the path and convert any
  4097. // . into _
  4098. //
  4099. Len = strlen(TemplateFile);
  4100. p = TemplateFile + Len;
  4101. while ((p >= TemplateFile) && (*p != '\\'))
  4102. {
  4103. if (*p == '.')
  4104. {
  4105. *p = '_';
  4106. }
  4107. p--;
  4108. }
  4109. p++;
  4110. Status = FilePrint(TemplateHandle,
  4111. "#ifndef _%s_\r\n#define _%s_\r\n\r\n",
  4112. p, p);
  4113. //
  4114. // Loop over all mof classes
  4115. MofClassList = MofResource->MRMCHead.Flink;
  4116. while (MofClassList != &MofResource->MRMCHead)
  4117. {
  4118. MofClass = CONTAINING_RECORD(MofClassList,
  4119. MOFCLASS,
  4120. MCMRList);
  4121. ClassInfo = MofClass->MofClassInfo;
  4122. ClassObject = (CBMOFObj *)MofClass->ClassObjectHandle;
  4123. if (ClassInfo->HeaderName != NULL)
  4124. {
  4125. ClassName = ClassInfo->HeaderName;
  4126. } else {
  4127. ClassName = ClassInfo->Name;
  4128. }
  4129. if (ClassInfo->GuidName1 != NULL)
  4130. {
  4131. GuidName1 = ClassInfo->GuidName1;
  4132. GuidSuffix1 = L"";
  4133. } else {
  4134. GuidName1 = ClassInfo->Name;
  4135. GuidSuffix1 = L"Guid";
  4136. }
  4137. if (ClassInfo->GuidName2 != NULL)
  4138. {
  4139. GuidName2 = ClassInfo->GuidName2;
  4140. GuidSuffix2 = L"";
  4141. } else {
  4142. GuidName2 = ClassInfo->Name;
  4143. GuidSuffix2 = L"_GUID";
  4144. }
  4145. Status = FilePrint(TemplateHandle,
  4146. "// %ws - %ws\r\n",
  4147. ClassInfo->Name,
  4148. ClassName);
  4149. if (ClassInfo->Description != NULL)
  4150. {
  4151. Status = FilePrint(TemplateHandle,
  4152. "// %ws\r\n",
  4153. ClassInfo->Description);
  4154. }
  4155. Status = FilePrint(TemplateHandle,
  4156. "#define %ws%ws \\\r\n"
  4157. " { 0x%08x,0x%04x,0x%04x, { 0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x } }\r\n\r\n",
  4158. GuidName1, GuidSuffix1,
  4159. ClassInfo->Guid.Data1, ClassInfo->Guid.Data2,
  4160. ClassInfo->Guid.Data3,
  4161. ClassInfo->Guid.Data4[0], ClassInfo->Guid.Data4[1],
  4162. ClassInfo->Guid.Data4[2], ClassInfo->Guid.Data4[3],
  4163. ClassInfo->Guid.Data4[4], ClassInfo->Guid.Data4[5],
  4164. ClassInfo->Guid.Data4[6], ClassInfo->Guid.Data4[7]);
  4165. Status = FilePrint(TemplateHandle,
  4166. "DEFINE_GUID(%ws%ws, \\\r\n"
  4167. " 0x%08x,0x%04x,0x%04x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x);\r\n\r\n",
  4168. GuidName2, GuidSuffix2,
  4169. ClassInfo->Guid.Data1, ClassInfo->Guid.Data2,
  4170. ClassInfo->Guid.Data3,
  4171. ClassInfo->Guid.Data4[0], ClassInfo->Guid.Data4[1],
  4172. ClassInfo->Guid.Data4[2], ClassInfo->Guid.Data4[3],
  4173. ClassInfo->Guid.Data4[4], ClassInfo->Guid.Data4[5],
  4174. ClassInfo->Guid.Data4[6], ClassInfo->Guid.Data4[7]);
  4175. if (ClassInfo->MethodCount > 0)
  4176. {
  4177. Status = FilePrint(TemplateHandle,
  4178. "//\r\n// Method id definitions for %ws\r\n",
  4179. ClassInfo->Name);
  4180. }
  4181. for (i = 0; i < ClassInfo->MethodCount; i++)
  4182. {
  4183. DataItem = &ClassInfo->DataItems[i+ClassInfo->DataItemCount];
  4184. Status = FilePrint(TemplateHandle,
  4185. "#define %ws %d\r\n",
  4186. DataItem->Name,
  4187. DataItem->MethodId);
  4188. MethodClassInfo = DataItem->MethodClassInfo;
  4189. if (DataItem->HeaderName != NULL)
  4190. {
  4191. MethodBaseClassName = DataItem->HeaderName;
  4192. } else {
  4193. MethodBaseClassName = DataItem->Name;
  4194. }
  4195. if (DoMethodHeaderGeneration)
  4196. {
  4197. swprintf(MethodClassName, L"%ws_IN", MethodBaseClassName);
  4198. Status = GenerateClassHeader(TemplateHandle,
  4199. MofResource,
  4200. MethodClassName,
  4201. MethodClassInfo,
  4202. MOFDI_FLAG_INPUT_METHOD);
  4203. swprintf(MethodClassName, L"%ws_OUT", MethodBaseClassName);
  4204. Status = GenerateClassHeader(TemplateHandle,
  4205. MofResource,
  4206. MethodClassName,
  4207. MethodClassInfo,
  4208. MOFDI_FLAG_OUTPUT_METHOD);
  4209. }
  4210. }
  4211. Status = FilePrint(TemplateHandle,
  4212. "\r\n");
  4213. Status = GenerateClassHeader(TemplateHandle,
  4214. MofResource,
  4215. ClassName,
  4216. ClassInfo,
  4217. 0xffffffff);
  4218. MofClassList = MofClassList->Flink;
  4219. }
  4220. Status = FilePrint(TemplateHandle,
  4221. "#endif\r\n");
  4222. CloseHandle(TemplateHandle);
  4223. if (Status != ERROR_SUCCESS)
  4224. {
  4225. DeleteFile(TemplateFile);
  4226. }
  4227. return(Status);
  4228. }
  4229. ULONG FilePrintDataItem(
  4230. HANDLE TemplateHandle,
  4231. PMOFRESOURCE MofResource,
  4232. ULONG Level,
  4233. PCHAR Prefix,
  4234. PCHAR DisplayPrefix,
  4235. PMOFCLASSINFOW ClassInfo,
  4236. PMOFDATAITEMW DataItem
  4237. )
  4238. {
  4239. ULONG Status = ERROR_SUCCESS;
  4240. CHAR NewPrefix[MAX_PATH];
  4241. CHAR NewDisplayPrefix[MAX_PATH];
  4242. CHAR ArrayLenBuffer[MAX_PATH];
  4243. PCHAR ArrayLen;
  4244. PMOFCLASSINFOW EmbeddedClassInfo;
  4245. PMOFCLASS EmbeddedClass;
  4246. PMOFDATAITEMW NewDataItem;
  4247. ULONG j;
  4248. if (DataItem->Flags & MOFDI_FLAG_FIXED_ARRAY)
  4249. {
  4250. sprintf(ArrayLenBuffer, "%d",
  4251. DataItem->FixedArrayElements);
  4252. ArrayLen = ArrayLenBuffer;
  4253. } else if (DataItem->Flags & MOFDI_FLAG_VARIABLE_ARRAY) {
  4254. sprintf(ArrayLenBuffer, "%s%ws",
  4255. Prefix,
  4256. ClassInfo->DataItems[DataItem->VariableArraySizeId-1].Name);
  4257. ArrayLen = ArrayLenBuffer;
  4258. } else {
  4259. ArrayLen = NULL;
  4260. }
  4261. if (ArrayLen != NULL)
  4262. {
  4263. Status = FilePrint(TemplateHandle,
  4264. " for i%d = 0 to (%s-1)\r\n",
  4265. Level,
  4266. ArrayLen);
  4267. sprintf(NewPrefix, "%s%ws(i%d)",
  4268. Prefix,
  4269. DataItem->Name,
  4270. Level);
  4271. sprintf(NewDisplayPrefix, "%s%ws(\"&i%d&\")",
  4272. DisplayPrefix,
  4273. DataItem->Name,
  4274. Level);
  4275. } else {
  4276. sprintf(NewPrefix, "%s%ws",
  4277. Prefix,
  4278. DataItem->Name);
  4279. sprintf(NewDisplayPrefix, "%s%ws",
  4280. DisplayPrefix,
  4281. DataItem->Name);
  4282. }
  4283. if (DataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  4284. {
  4285. EmbeddedClass = WmipFindClassInMofResourceByGuid(
  4286. MofResource,
  4287. &DataItem->EmbeddedClassGuid);
  4288. if (EmbeddedClass != NULL)
  4289. {
  4290. strcat(NewPrefix, ".");
  4291. strcat(NewDisplayPrefix, ".");
  4292. EmbeddedClassInfo = EmbeddedClass->MofClassInfo;
  4293. for (j = 0; j < EmbeddedClassInfo->DataItemCount; j++)
  4294. {
  4295. NewDataItem = &EmbeddedClassInfo->DataItems[j];
  4296. Status = FilePrintDataItem(TemplateHandle,
  4297. MofResource,
  4298. Level+1,
  4299. NewPrefix,
  4300. NewDisplayPrefix,
  4301. EmbeddedClassInfo,
  4302. NewDataItem);
  4303. }
  4304. } else {
  4305. #if DBG
  4306. printf("WARNING - Cannot create test for %s, cannot find embedded class\n",
  4307. NewPrefix);
  4308. #endif
  4309. FilePrint(TemplateHandle, "REM WARNING - Cannot create test for %s, cannot find embedded class\r\n",
  4310. NewPrefix);
  4311. }
  4312. } else {
  4313. Status = FilePrint(TemplateHandle,
  4314. " a.WriteLine(\" %s=\" & %s)\r\n",
  4315. NewDisplayPrefix,
  4316. NewPrefix);
  4317. }
  4318. if (ArrayLen != NULL)
  4319. {
  4320. Status = FilePrint(TemplateHandle,
  4321. " next 'i%d\r\n",
  4322. Level);
  4323. }
  4324. return(Status);
  4325. }
  4326. BOOLEAN CanCreateTest(
  4327. PMOFCLASSINFOW ClassInfo
  4328. )
  4329. {
  4330. //
  4331. // Cannot create tests for embedded classes or events
  4332. if (((ClassInfo->Flags & MOFCI_RESERVED1) == 0) ||
  4333. (ClassInfo->Flags & MOFCI_FLAG_EVENT))
  4334. {
  4335. return(FALSE);
  4336. }
  4337. return(TRUE);
  4338. }
  4339. ULONG GenerateTTemplate(
  4340. PCHAR TemplateFile,
  4341. PMOFRESOURCE MofResource
  4342. )
  4343. {
  4344. HANDLE TemplateHandle;
  4345. ULONG Status;
  4346. ULONG i;
  4347. PMOFDATAITEMW DataItem;
  4348. PMOFCLASSINFOW ClassInfo;
  4349. PLIST_ENTRY MofClassList;
  4350. PMOFCLASS MofClass;
  4351. PCHAR p, p1;
  4352. TemplateHandle = CreateFile(TemplateFile,
  4353. GENERIC_WRITE,
  4354. 0,
  4355. NULL,
  4356. CREATE_ALWAYS,
  4357. FILE_ATTRIBUTE_NORMAL,
  4358. NULL);
  4359. if ((TemplateHandle != NULL) && (TemplateHandle != INVALID_HANDLE_VALUE))
  4360. {
  4361. p = TemplateFile;
  4362. p1 = TemplateFile;
  4363. while ((*p != '.') && (*p != 0))
  4364. {
  4365. if (*p == '\\')
  4366. {
  4367. p1 = p+1;
  4368. }
  4369. p++;
  4370. }
  4371. if (*p == '.')
  4372. {
  4373. *p = 0;
  4374. }
  4375. Status = FilePrint(TemplateHandle,
  4376. "REM Note that missing classes in log file mean tthe hat WMI cannot access them.\r\n"
  4377. "REM Most likely this indicates a problem with the driver.\r\n"
  4378. "REM See %%windir%%\\system32\\wbem\\wmiprov.log and nt eventlog for more details.\r\n"
  4379. "REM You could also delete the line On Error Resume Next and examine the\r\n"
  4380. "REM specific VBScript error\r\n\r\n\r\n");
  4381. Status = FilePrint(TemplateHandle,
  4382. "On Error Resume Next\r\n\r\n");
  4383. Status = FilePrint(TemplateHandle,
  4384. "Set fso = CreateObject(\"Scripting.FileSystemObject\")\r\n");
  4385. Status = FilePrint(TemplateHandle,
  4386. "Set a = fso.CreateTextFile(\"%s.log\", True)\r\n",
  4387. p1);
  4388. Status = FilePrint(TemplateHandle,
  4389. "Set Service = GetObject(\"winmgmts:{impersonationLevel=impersonate}!root/wmi\")\r\n");
  4390. //
  4391. // Loop over all mof classes
  4392. MofClassList = MofResource->MRMCHead.Flink;
  4393. while (MofClassList != &MofResource->MRMCHead)
  4394. {
  4395. MofClass = CONTAINING_RECORD(MofClassList,
  4396. MOFCLASS,
  4397. MCMRList);
  4398. ClassInfo = MofClass->MofClassInfo;
  4399. if (CanCreateTest(ClassInfo))
  4400. {
  4401. Status = FilePrint(TemplateHandle,
  4402. "Rem %ws - %ws\r\n",
  4403. ClassInfo->Name,
  4404. ClassInfo->Description ? ClassInfo->Description : L"");
  4405. Status = FilePrint(TemplateHandle,
  4406. "Set enumSet = Service.InstancesOf (\"%ws\")\r\n"
  4407. "a.WriteLine(\"%ws\")\r\n",
  4408. ClassInfo->Name,
  4409. ClassInfo->Name);
  4410. Status = FilePrint(TemplateHandle,
  4411. "for each instance in enumSet\r\n");
  4412. Status = FilePrint(TemplateHandle,
  4413. " a.WriteLine(\" InstanceName=\" & instance.InstanceName)\r\n");
  4414. for (i = 0; i < ClassInfo->DataItemCount; i++)
  4415. {
  4416. DataItem = &ClassInfo->DataItems[i];
  4417. FilePrintDataItem(TemplateHandle,
  4418. MofResource,
  4419. 1,
  4420. "instance.",
  4421. "instance.",
  4422. ClassInfo,
  4423. DataItem);
  4424. }
  4425. Status = FilePrint(TemplateHandle,
  4426. "next 'instance\r\n\r\n");
  4427. }
  4428. MofClassList = MofClassList->Flink;
  4429. }
  4430. Status = FilePrint(TemplateHandle,
  4431. "a.Close\r\n"
  4432. "Wscript.Echo \"%s Test Completed, see %s.log for details\"\r\n",
  4433. p1, p1);
  4434. CloseHandle(TemplateHandle);
  4435. if (Status != ERROR_SUCCESS)
  4436. {
  4437. DeleteFile(TemplateFile);
  4438. }
  4439. } else {
  4440. Status = GetLastError();
  4441. }
  4442. return(Status);
  4443. }
  4444. ULONG GenerateXTemplate(
  4445. PCHAR TemplateFile
  4446. )
  4447. {
  4448. HANDLE TemplateHandle;
  4449. ULONG Status;
  4450. TemplateHandle = CreateFile(TemplateFile,
  4451. GENERIC_WRITE,
  4452. 0,
  4453. NULL,
  4454. CREATE_ALWAYS,
  4455. FILE_ATTRIBUTE_NORMAL,
  4456. NULL);
  4457. if ((TemplateHandle == NULL) || (TemplateHandle == INVALID_HANDLE_VALUE))
  4458. {
  4459. return(GetLastError());
  4460. }
  4461. Status = GenerateBinaryMofData(TemplateHandle);
  4462. CloseHandle(TemplateHandle);
  4463. if (Status != ERROR_SUCCESS)
  4464. {
  4465. DeleteFile(TemplateFile);
  4466. }
  4467. return(Status);
  4468. }
  4469. typedef void (*PROPERTYCALLBACK)(
  4470. HANDLE TemplateHandle,
  4471. PWCHAR FormName,
  4472. PWCHAR InstanceName,
  4473. PWCHAR InstancePrefix,
  4474. PMOFRESOURCE MofResource,
  4475. PMOFCLASSINFOW ClassInfo,
  4476. PMOFDATAITEMW DataItem,
  4477. ULONG Counter,
  4478. PVOID Context
  4479. );
  4480. void EnumerateClassProperties(
  4481. HANDLE TemplateHandle,
  4482. PWCHAR FormName,
  4483. PWCHAR InstanceName,
  4484. PMOFRESOURCE MofResource,
  4485. PMOFCLASSINFOW ClassInfo,
  4486. PROPERTYCALLBACK Callback,
  4487. BOOLEAN Recurse,
  4488. PVOID Context
  4489. )
  4490. {
  4491. ULONG i;
  4492. WCHAR I[1];
  4493. WCHAR F[1];
  4494. if (InstanceName == NULL)
  4495. {
  4496. I[0] = UNICODE_NULL;
  4497. InstanceName = I;
  4498. }
  4499. if (FormName == NULL)
  4500. {
  4501. F[0] = UNICODE_NULL;
  4502. FormName = F;
  4503. }
  4504. for (i = 0; i < ClassInfo->DataItemCount; i++)
  4505. {
  4506. WCHAR FName[MAX_PATH];
  4507. WCHAR IName[MAX_PATH];
  4508. PMOFCLASS EmbeddedClass;
  4509. PMOFDATAITEMW DataItem;
  4510. DataItem = &ClassInfo->DataItems[i];
  4511. wcscpy(IName, InstanceName);
  4512. wcscat(IName, L".");
  4513. wcscat(IName, DataItem->Name);
  4514. wcscpy(FName, FormName);
  4515. wcscat(FName, DataItem->Name);
  4516. if (DataItem->Flags & (MOFDI_FLAG_FIXED_ARRAY |
  4517. MOFDI_FLAG_VARIABLE_ARRAY))
  4518. {
  4519. wcscat(IName, L"(");
  4520. wcscat(IName, FName);
  4521. wcscat(IName, L"Index)");
  4522. }
  4523. (*Callback)(TemplateHandle,
  4524. FName,
  4525. IName,
  4526. InstanceName,
  4527. MofResource,
  4528. ClassInfo,
  4529. DataItem,
  4530. i,
  4531. Context);
  4532. if (Recurse && (DataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS))
  4533. {
  4534. EmbeddedClass = WmipFindClassInMofResourceByGuid(
  4535. MofResource,
  4536. &DataItem->EmbeddedClassGuid);
  4537. if (EmbeddedClass != NULL)
  4538. {
  4539. EnumerateClassProperties(TemplateHandle,
  4540. FName,
  4541. IName,
  4542. MofResource,
  4543. EmbeddedClass->MofClassInfo,
  4544. Callback,
  4545. Recurse,
  4546. Context);
  4547. }
  4548. }
  4549. }
  4550. }
  4551. void GenerateChangeText(
  4552. HANDLE TemplateHandle,
  4553. PWCHAR FormName,
  4554. PWCHAR InstanceName,
  4555. PWCHAR InstancePrefix,
  4556. PMOFRESOURCE MofResource,
  4557. PMOFCLASSINFOW ClassInfo,
  4558. PMOFDATAITEMW DataItem,
  4559. ULONG Counter,
  4560. PVOID Context
  4561. )
  4562. {
  4563. //
  4564. // Generate code to change the contents of a property
  4565. //
  4566. if (! (DataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS))
  4567. {
  4568. FilePrint(TemplateHandle,
  4569. " Instance%ws = TheForm.%wsText.Value\r\n",
  4570. InstanceName,
  4571. FormName);
  4572. }
  4573. }
  4574. void GenerateReloadText(
  4575. HANDLE TemplateHandle,
  4576. PWCHAR FormName,
  4577. PWCHAR InstanceName,
  4578. PWCHAR InstancePrefix,
  4579. PMOFRESOURCE MofResource,
  4580. PMOFCLASSINFOW ClassInfo,
  4581. PMOFDATAITEMW DataItem,
  4582. ULONG Counter,
  4583. PVOID Context
  4584. )
  4585. {
  4586. //
  4587. // Generate code to redisplay the contents of the property
  4588. //
  4589. if (! (DataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS))
  4590. {
  4591. FilePrint(TemplateHandle,
  4592. " TheForm.%wsText.Value = Instance%ws\r\n",
  4593. FormName,
  4594. InstanceName);
  4595. }
  4596. if (DataItem->Flags & (MOFDI_FLAG_FIXED_ARRAY | MOFDI_FLAG_VARIABLE_ARRAY))
  4597. {
  4598. FilePrint(TemplateHandle,
  4599. " TheForm.%wsIndexText.Value = %wsIndex\r\n",
  4600. FormName, FormName);
  4601. if (DataItem->Flags & MOFDI_FLAG_FIXED_ARRAY)
  4602. {
  4603. FilePrint(TemplateHandle,
  4604. " %wsMaxIndex = %d\r\n",
  4605. FormName, DataItem->FixedArrayElements);
  4606. } else {
  4607. FilePrint(TemplateHandle,
  4608. " %wsMaxIndex = Instance%ws.%ws\r\n",
  4609. FormName, InstancePrefix,
  4610. ClassInfo->DataItems[DataItem->VariableArraySizeId-1].Name);
  4611. }
  4612. }
  4613. }
  4614. void GenerateTextFormText(
  4615. HANDLE TemplateHandle,
  4616. PWCHAR FormName,
  4617. PWCHAR InstanceName,
  4618. PWCHAR InstancePrefix,
  4619. PMOFRESOURCE MofResource,
  4620. PMOFCLASSINFOW ClassInfo,
  4621. PMOFDATAITEMW DataItem,
  4622. ULONG Counter,
  4623. PVOID Context
  4624. )
  4625. {
  4626. if (! (DataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)) {
  4627. FilePrint(TemplateHandle,
  4628. "<p class=MsoNormal>%ws: <span style='mso-tab-count:2'>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><INPUT TYPE=\"TEXT\" SIZE=\"96\" NAME=\"%wsText\"></p>\r\n"
  4629. "\r\n",
  4630. FormName, FormName);
  4631. }
  4632. if (DataItem->Flags &
  4633. (MOFDI_FLAG_FIXED_ARRAY | MOFDI_FLAG_VARIABLE_ARRAY))
  4634. {
  4635. FilePrint(TemplateHandle,
  4636. "\r\n"
  4637. "<input name=Next%wsButton type=BUTTON value=Next OnClick=\"NextIndexButton_OnClick %wsIndex, %wsMaxIndex\">\r\n"
  4638. "\r\n"
  4639. "<input name=Prev%wsButton type=BUTTON value=Previous OnClick=\"PrevIndexButton_OnClick %wsIndex, %wsMaxIndex\">\r\n"
  4640. "\r\n"
  4641. "%wsArrayIndex: <INPUT TYPE=\"TEXT\" SIZE=\"5\" NAME=\"%wsIndexText\">\r\n"
  4642. "\r\n"
  4643. "<input name=GoTo%wsButton type=BUTTON value=GoTo OnClick=\"GoToIndexButton_OnClick %wsIndex, %wsMaxIndex, Document.ClassForm.%wsIndexText.Value\">\r\n"
  4644. "\r\n",
  4645. FormName,
  4646. FormName,
  4647. FormName,
  4648. FormName,
  4649. FormName,
  4650. FormName,
  4651. FormName,
  4652. FormName,
  4653. FormName,
  4654. FormName,
  4655. FormName,
  4656. FormName);
  4657. }
  4658. }
  4659. void GenerateArrayDimsText(
  4660. HANDLE TemplateHandle,
  4661. PWCHAR FormName,
  4662. PWCHAR InstanceName,
  4663. PWCHAR InstancePrefix,
  4664. PMOFRESOURCE MofResource,
  4665. PMOFCLASSINFOW ClassInfo,
  4666. PMOFDATAITEMW MofDataItem,
  4667. ULONG Counter,
  4668. PVOID Context
  4669. )
  4670. {
  4671. //
  4672. // Declare an index variable that tracks the current index of an array
  4673. //
  4674. if (MofDataItem->Flags &
  4675. (MOFDI_FLAG_FIXED_ARRAY | MOFDI_FLAG_VARIABLE_ARRAY))
  4676. {
  4677. FilePrint(TemplateHandle,
  4678. "Dim %wsIndex\r\n"
  4679. "%wsIndex = 0\r\n"
  4680. "Dim %wsMaxIndex\r\n"
  4681. "%wsMaxIndex = 1\r\n",
  4682. FormName,
  4683. FormName,
  4684. FormName,
  4685. FormName);
  4686. }
  4687. }
  4688. void GenerateMethodInL2Text(
  4689. HANDLE TemplateHandle,
  4690. PWCHAR FormName,
  4691. PWCHAR InstanceName,
  4692. PWCHAR InstancePrefix,
  4693. PMOFRESOURCE MofResource,
  4694. PMOFCLASSINFOW ClassInfo,
  4695. PMOFDATAITEMW MofDataItem,
  4696. ULONG Counter,
  4697. PVOID Context
  4698. )
  4699. {
  4700. if (! (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS))
  4701. {
  4702. FilePrint(TemplateHandle,
  4703. " %ws = TheForm.%ws%wsText.Value\r\n",
  4704. InstanceName,
  4705. Context,
  4706. FormName);
  4707. }
  4708. }
  4709. void GenerateMethodInText(
  4710. HANDLE TemplateHandle,
  4711. PWCHAR FormName,
  4712. PWCHAR InstanceName,
  4713. PWCHAR InstancePrefix,
  4714. PMOFRESOURCE MofResource,
  4715. PMOFCLASSINFOW ClassInfo,
  4716. PMOFDATAITEMW MofDataItem,
  4717. ULONG Counter,
  4718. PVOID Context
  4719. )
  4720. {
  4721. //
  4722. // Declare classes for all IN and OUT embedded classes
  4723. //
  4724. if (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  4725. {
  4726. PWCHAR EmbeddedClassName = (PWCHAR)MofDataItem->EcTempPtr +
  4727. (sizeof(L"object") / sizeof(WCHAR));
  4728. if (MofDataItem->Flags & (MOFDI_FLAG_INPUT_METHOD))
  4729. {
  4730. FilePrint(TemplateHandle,
  4731. " Set %ws = Service.Get(\"%ws\").SpawnInstance_\r\n",
  4732. FormName,
  4733. EmbeddedClassName);
  4734. } else {
  4735. FilePrint(TemplateHandle,
  4736. " Dim %ws\r\n",
  4737. FormName);
  4738. }
  4739. }
  4740. if (MofDataItem->Flags & (MOFDI_FLAG_INPUT_METHOD))
  4741. {
  4742. if (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  4743. {
  4744. PMOFCLASS EmbeddedClass;
  4745. EmbeddedClass = WmipFindClassInMofResourceByGuid(
  4746. MofResource,
  4747. &MofDataItem->EmbeddedClassGuid);
  4748. if (EmbeddedClass != NULL)
  4749. {
  4750. EnumerateClassProperties(TemplateHandle,
  4751. FormName,
  4752. FormName,
  4753. MofResource,
  4754. EmbeddedClass->MofClassInfo,
  4755. GenerateMethodInL2Text,
  4756. TRUE,
  4757. Context);
  4758. }
  4759. } else {
  4760. FilePrint(TemplateHandle,
  4761. " %ws = TheForm.%ws%wsText.Value\r\n",
  4762. FormName,
  4763. Context,
  4764. FormName);
  4765. }
  4766. }
  4767. }
  4768. void GenerateMethodOutL2Text(
  4769. HANDLE TemplateHandle,
  4770. PWCHAR FormName,
  4771. PWCHAR InstanceName,
  4772. PWCHAR InstancePrefix,
  4773. PMOFRESOURCE MofResource,
  4774. PMOFCLASSINFOW ClassInfo,
  4775. PMOFDATAITEMW MofDataItem,
  4776. ULONG Counter,
  4777. PVOID Context
  4778. )
  4779. {
  4780. if (! (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS))
  4781. {
  4782. FilePrint(TemplateHandle,
  4783. " TheForm.%ws%wsText.Value = %ws\r\n",
  4784. Context,
  4785. FormName,
  4786. InstanceName);
  4787. }
  4788. }
  4789. void GenerateMethodOutText(
  4790. HANDLE TemplateHandle,
  4791. PWCHAR FormName,
  4792. PWCHAR InstanceName,
  4793. PWCHAR InstancePrefix,
  4794. PMOFRESOURCE MofResource,
  4795. PMOFCLASSINFOW ClassInfo,
  4796. PMOFDATAITEMW MofDataItem,
  4797. ULONG Counter,
  4798. PVOID Context
  4799. )
  4800. {
  4801. if (MofDataItem->Flags & (MOFDI_FLAG_OUTPUT_METHOD))
  4802. {
  4803. if (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  4804. {
  4805. PMOFCLASS EmbeddedClass;
  4806. PWCHAR EmbeddedClassName = (PWCHAR)MofDataItem->EcTempPtr +
  4807. (sizeof(L"object") / sizeof(WCHAR));
  4808. EmbeddedClass = WmipFindClassInMofResourceByGuid(
  4809. MofResource,
  4810. &MofDataItem->EmbeddedClassGuid);
  4811. if (EmbeddedClass != NULL)
  4812. {
  4813. EnumerateClassProperties(TemplateHandle,
  4814. FormName,
  4815. FormName,
  4816. MofResource,
  4817. EmbeddedClass->MofClassInfo,
  4818. GenerateMethodOutL2Text,
  4819. TRUE,
  4820. Context);
  4821. }
  4822. } else {
  4823. FilePrint(TemplateHandle,
  4824. " TheForm.%ws%wsText.Value = %ws\r\n",
  4825. Context,
  4826. FormName,
  4827. FormName);
  4828. }
  4829. }
  4830. }
  4831. void GenerateMethodCallText(
  4832. HANDLE TemplateHandle,
  4833. PWCHAR FormName,
  4834. PWCHAR InstanceName,
  4835. PWCHAR InstancePrefix,
  4836. PMOFRESOURCE MofResource,
  4837. PMOFCLASSINFOW ClassInfo,
  4838. PMOFDATAITEMW MofDataItem,
  4839. ULONG Counter,
  4840. PVOID Context
  4841. )
  4842. {
  4843. //
  4844. // Declare an index variable that tracks the current index of an array
  4845. //
  4846. FilePrint(TemplateHandle,
  4847. " %ws",
  4848. FormName);
  4849. if (Counter != PtrToUlong(Context))
  4850. {
  4851. FilePrint(TemplateHandle, ", ");
  4852. }
  4853. }
  4854. void GenerateMethodControlText(
  4855. HANDLE TemplateHandle,
  4856. PWCHAR FormName,
  4857. PWCHAR InstanceName,
  4858. PWCHAR InstancePrefix,
  4859. PMOFRESOURCE MofResource,
  4860. PMOFCLASSINFOW ClassInfo,
  4861. PMOFDATAITEMW MofDataItem,
  4862. ULONG Counter,
  4863. PVOID Context
  4864. )
  4865. {
  4866. PWCHAR MethodName = (PWCHAR)Context;
  4867. FilePrint(TemplateHandle,
  4868. "<p> "
  4869. );
  4870. if (MofDataItem->Flags & (MOFDI_FLAG_INPUT_METHOD))
  4871. {
  4872. FilePrint(TemplateHandle,
  4873. " [in] ");
  4874. }
  4875. if (MofDataItem->Flags & (MOFDI_FLAG_OUTPUT_METHOD))
  4876. {
  4877. FilePrint(TemplateHandle,
  4878. " [out] ");
  4879. }
  4880. if (MofDataItem->Flags & MOFDI_FLAG_EMBEDDED_CLASS)
  4881. {
  4882. FilePrint(TemplateHandle, "%ws </p>\r\n", FormName);
  4883. } else {
  4884. FilePrint(TemplateHandle,
  4885. " %ws <INPUT TYPE=\"TEXT\" SIZE=\"70\" NAME=\"%ws%wsText\"></p>\r\n",
  4886. FormName,
  4887. MethodName,
  4888. FormName);
  4889. }
  4890. if (MofDataItem->Flags &
  4891. (MOFDI_FLAG_FIXED_ARRAY | MOFDI_FLAG_VARIABLE_ARRAY))
  4892. {
  4893. FilePrint(TemplateHandle,
  4894. "\r\n"
  4895. "<input name=Next%wsButton type=BUTTON value=Next OnClick=\"NextIndexButton_OnClick %ws%wsIndex, %ws%wsMaxIndex\">\r\n"
  4896. "\r\n"
  4897. "<input name=Prev%wsButton type=BUTTON value=Previous OnClick=\"PrevIndexButton_OnClick %ws%wsIndex, %ws%wsMaxIndex\">\r\n"
  4898. "\r\n"
  4899. "%ws%wsArrayIndex: <INPUT TYPE=\"TEXT\" SIZE=\"5\" NAME=\"%ws%wsIndexText\">\r\n"
  4900. "\r\n"
  4901. "<input name=GoTo%ws%wsButton type=BUTTON value=GoTo OnClick=\"GoToIndexButton_OnClick %ws%wsIndex, %ws%wsMaxIndex, Document.ClassForm.%ws%wsIndexText.Value\">\r\n"
  4902. "\r\n",
  4903. FormName,
  4904. MethodName,
  4905. FormName,
  4906. MethodName,
  4907. FormName,
  4908. MethodName,
  4909. FormName,
  4910. MethodName,
  4911. FormName,
  4912. MethodName,
  4913. FormName,
  4914. MethodName,
  4915. FormName,
  4916. MethodName,
  4917. FormName,
  4918. MethodName,
  4919. FormName,
  4920. MethodName,
  4921. FormName,
  4922. MethodName,
  4923. FormName,
  4924. MethodName,
  4925. FormName,
  4926. MethodName);
  4927. }
  4928. }
  4929. typedef void (*METHODCALLBACK)(
  4930. HANDLE TemplateHandle,
  4931. PWCHAR MethodName,
  4932. PMOFRESOURCE MofResource,
  4933. PMOFCLASSINFOW ClassInfo,
  4934. PMOFDATAITEMW MethodDataItem,
  4935. ULONG Counter,
  4936. PVOID Context
  4937. );
  4938. void EnumerateClassMethods(
  4939. HANDLE TemplateHandle,
  4940. PMOFRESOURCE MofResource,
  4941. PMOFCLASSINFOW ClassInfo,
  4942. METHODCALLBACK Callback,
  4943. PVOID Context
  4944. )
  4945. {
  4946. PMOFDATAITEMW DataItem;
  4947. ULONG i;
  4948. for (i = 0; i < ClassInfo->MethodCount; i++)
  4949. {
  4950. DataItem = &ClassInfo->DataItems[i+ClassInfo->DataItemCount];
  4951. WmipAssert(DataItem->Flags & MOFDI_FLAG_METHOD);
  4952. (*Callback)(TemplateHandle,
  4953. DataItem->Name,
  4954. MofResource,
  4955. ClassInfo,
  4956. DataItem,
  4957. i,
  4958. Context);
  4959. }
  4960. }
  4961. void GenerateMethodButtonsText(
  4962. HANDLE TemplateHandle,
  4963. PWCHAR MethodName,
  4964. PMOFRESOURCE MofResource,
  4965. PMOFCLASSINFOW ClassInfo,
  4966. PMOFDATAITEMW MethodDataItem,
  4967. ULONG Counter,
  4968. PVOID Context
  4969. )
  4970. {
  4971. PMOFCLASSINFOW MethodClassInfo;
  4972. FilePrint(TemplateHandle,
  4973. "<p class=MsoNormal>Method %ws: <input name=%wsButton type=BUTTON value=Execute> </p>\r\n",
  4974. MethodName, MethodName);
  4975. MethodClassInfo = MethodDataItem->MethodClassInfo;
  4976. EnumerateClassProperties(TemplateHandle,
  4977. NULL,
  4978. NULL,
  4979. MofResource,
  4980. MethodClassInfo,
  4981. GenerateMethodControlText,
  4982. TRUE,
  4983. MethodName);
  4984. }
  4985. void GenerateMethodSubsText(
  4986. HANDLE TemplateHandle,
  4987. PWCHAR MethodName,
  4988. PMOFRESOURCE MofResource,
  4989. PMOFCLASSINFOW ClassInfo,
  4990. PMOFDATAITEMW MethodDataItem,
  4991. ULONG Counter,
  4992. PVOID Context
  4993. )
  4994. {
  4995. PMOFCLASSINFOW MethodClassInfo;
  4996. FilePrint(TemplateHandle,
  4997. "Sub %wsButton_OnClick\r\n"
  4998. " if InstanceCount <> 0 Then\r\n"
  4999. " On Error Resume Next\r\n"
  5000. " Err.Clear\r\n"
  5001. " Set Instance = Service.Get(InstancePaths(CurrentInstanceIndex))\r\n"
  5002. " if Err.Number = 0 Then\r\n"
  5003. " Set TheForm = Document.ClassForm\r\n"
  5004. " Err.Clear\r\n",
  5005. MethodName);
  5006. MethodClassInfo = MethodDataItem->MethodClassInfo;
  5007. EnumerateClassProperties(TemplateHandle,
  5008. NULL,
  5009. NULL,
  5010. MofResource,
  5011. MethodClassInfo,
  5012. GenerateMethodInText,
  5013. FALSE,
  5014. MethodName);
  5015. FilePrint(TemplateHandle,
  5016. " Instance.%ws ",
  5017. MethodName);
  5018. EnumerateClassProperties(TemplateHandle,
  5019. NULL,
  5020. NULL,
  5021. MofResource,
  5022. MethodClassInfo,
  5023. GenerateMethodCallText,
  5024. FALSE,
  5025. UlongToPtr(MethodClassInfo->DataItemCount-1));
  5026. FilePrint(TemplateHandle,
  5027. "\r\n if Err.Number = 0 Then\r\n"
  5028. );
  5029. EnumerateClassProperties(TemplateHandle,
  5030. NULL,
  5031. NULL,
  5032. MofResource,
  5033. MethodClassInfo,
  5034. GenerateMethodOutText,
  5035. FALSE,
  5036. MethodName);
  5037. FilePrint(TemplateHandle,
  5038. " MsgBox \"Method Execution Succeeded\"\r\n"
  5039. " Else\r\n"
  5040. " MsgBox Err.Description,, \"Method Execution Failed\"\r\n"
  5041. " End if\r\n"
  5042. " End if\r\n"
  5043. " End if\r\n"
  5044. "End Sub\r\n\r\n"
  5045. );
  5046. }
  5047. ULONG GenerateClassWebPage(
  5048. HANDLE TemplateHandle,
  5049. PMOFCLASS MofClass,
  5050. PMOFRESOURCE MofResource
  5051. )
  5052. {
  5053. PMOFCLASSINFOW ClassInfo = MofClass->MofClassInfo;
  5054. BOOLEAN IsEvent=(ClassInfo->Flags & MOFCI_FLAG_EVENT) == MOFCI_FLAG_EVENT;
  5055. FilePrint(TemplateHandle,
  5056. "<html xmlns:v=\"urn:schemas-microsoft-com:vml\"\r\n"
  5057. "xmlns:o=\"urn:schemas-microsoft-com:office:office\"\r\n"
  5058. "xmlns:w=\"urn:schemas-microsoft-com:office:word\"\r\n"
  5059. "xmlns=\"http://www.w3.org/TR/REC-html40\">\r\n"
  5060. "\r\n"
  5061. "<head>\r\n"
  5062. "<meta http-equiv=Content-Type content=\"text/html; charset=us-ascii\">\r\n"
  5063. "<meta name=ProgId content=Word.Document>\r\n"
  5064. "<meta name=Generator content=\"Microsoft Word 9\">\r\n"
  5065. "<meta name=Originator content=\"Microsoft Word 9\">\r\n"
  5066. "<link rel=File-List href=\"./valid_files/filelist.xml\">\r\n"
  5067. "<link rel=Edit-Time-Data href=\"./valid_files/editdata.mso\">\r\n"
  5068. "<!--[if !mso]>\r\n"
  5069. "<style>\r\n"
  5070. "v\\:* {behavior:url(#default#VML);}\r\n"
  5071. "o\\:* {behavior:url(#default#VML);}\r\n"
  5072. "w\\:* {behavior:url(#default#VML);}\r\n"
  5073. ".shape {behavior:url(#default#VML);}\r\n"
  5074. "</style>\r\n"
  5075. "<![endif]-->\r\n"
  5076. "<title>Class %ws</title>\r\n"
  5077. "<!--[if gte mso 9]><xml>\r\n"
  5078. " <o:DocumentProperties>\r\n"
  5079. " <o:Author>Wmi Mof Checking Tool</o:Author>\r\n"
  5080. " <o:Template>Normal</o:Template>\r\n"
  5081. " <o:LastAuthor>Wmi Mof Checking Tool</o:LastAuthor>\r\n"
  5082. " <o:Revision>2</o:Revision>\r\n"
  5083. " <o:TotalTime>3</o:TotalTime>\r\n"
  5084. " <o:Created>1999-09-10T01:09:00Z</o:Created>\r\n"
  5085. " <o:LastSaved>1999-09-10T01:12:00Z</o:LastSaved>\r\n"
  5086. " <o:Pages>1</o:Pages>\r\n"
  5087. " <o:Words>51</o:Words>\r\n"
  5088. " <o:Characters>292</o:Characters>\r\n"
  5089. " <o:Company>Microsoft</o:Company>\r\n"
  5090. " <o:Lines>2</o:Lines>\r\n"
  5091. " <o:Paragraphs>1</o:Paragraphs>\r\n"
  5092. " <o:CharactersWithSpaces>358</o:CharactersWithSpaces>\r\n"
  5093. " <o:Version>9.2720</o:Version>\r\n"
  5094. " </o:DocumentProperties>\r\n"
  5095. "</xml><![endif]--><!--[if gte mso 9]><xml>\r\n"
  5096. " <w:WordDocument>\r\n"
  5097. " <w:Compatibility>\r\n"
  5098. " <w:UseFELayout/>\r\n"
  5099. " </w:Compatibility>\r\n"
  5100. " </w:WordDocument>\r\n"
  5101. "</xml><![endif]-->\r\n"
  5102. "<style>\r\n"
  5103. "<!--\r\n"
  5104. " /* Font Definitions */\r\n"
  5105. "@font-face\r\n"
  5106. " {font-family:\"MS Mincho\";\r\n"
  5107. " panose-1:2 2 6 9 4 2 5 8 3 4;\r\n"
  5108. " mso-font-alt:\"\\FF2D\\FF33 \\660E\\671D\";\r\n"
  5109. " mso-font-charset:128;\r\n"
  5110. " mso-generic-font-family:roman;\r\n"
  5111. " mso-font-format:other;\r\n"
  5112. " mso-font-pitch:fixed;\r\n"
  5113. " mso-font-signature:1 134676480 16 0 131072 0;}\r\n"
  5114. "@font-face\r\n"
  5115. " {font-family:\"\\@MS Mincho\";\r\n"
  5116. " panose-1:2 2 6 9 4 2 5 8 3 4;\r\n"
  5117. " mso-font-charset:128;\r\n"
  5118. " mso-generic-font-family:modern;\r\n"
  5119. " mso-font-pitch:fixed;\r\n"
  5120. " mso-font-signature:-1610612033 1757936891 16 0 131231 0;}\r\n"
  5121. " /* Style Definitions */\r\n"
  5122. "p.MsoNormal, li.MsoNormal, div.MsoNormal\r\n"
  5123. " {mso-style-parent:\"\";\r\n"
  5124. " margin:0in;\r\n"
  5125. " margin-bottom:.0001pt;\r\n"
  5126. " mso-pagination:widow-orphan;\r\n"
  5127. " font-size:12.0pt;\r\n"
  5128. " font-family:\"Times New Roman\";\r\n"
  5129. " mso-fareast-font-family:\"MS Mincho\";}\r\n"
  5130. "@page Section1\r\n"
  5131. " {size:8.5in 11.0in;\r\n"
  5132. " margin:1.0in 1.25in 1.0in 1.25in;\r\n"
  5133. " mso-header-margin:.5in;\r\n"
  5134. " mso-footer-margin:.5in;\r\n"
  5135. " mso-paper-source:0;}\r\n"
  5136. "div.Section1\r\n"
  5137. " {page:Section1;}\r\n"
  5138. "-->\r\n"
  5139. "</style>\r\n"
  5140. "<!--[if gte mso 9]><xml>\r\n"
  5141. " <o:shapedefaults v:ext=\"edit\" spidmax=\"1026\"/>\r\n"
  5142. "</xml><![endif]--><!--[if gte mso 9]><xml>\r\n"
  5143. " <o:shapelayout v:ext=\"edit\">\r\n"
  5144. " <o:idmap v:ext=\"edit\" data=\"1\"/>\r\n"
  5145. " </o:shapelayout></xml><![endif]-->\r\n"
  5146. "</head>\r\n"
  5147. "\r\n"
  5148. "<body lang=EN-US style='tab-interval:.5in'>\r\n"
  5149. "\r\n"
  5150. "<div class=Section1>\r\n"
  5151. "\r\n"
  5152. "<h3>Class %ws</h3>\r\n"
  5153. "\r\n"
  5154. "\r\n"
  5155. "<div class=MsoNormal align=center style='text-align:center'>\r\n"
  5156. "\r\n"
  5157. "<hr size=2 width=\"100%\" align=center>\r\n"
  5158. "\r\n"
  5159. "</div>\r\n"
  5160. "\r\n"
  5161. "\r\n"
  5162. "<form NAME=ClassForm>\r\n"
  5163. "\r\n"
  5164. "<p class=MsoNormal><span style='display:none;mso-hide:all'><script language=\"VBScript\">\r\n"
  5165. "<!--\r\n"
  5166. "On Error Resume Next\r\n"
  5167. "Dim Locator\r\n"
  5168. "Dim Service\r\n"
  5169. "Dim Collection\r\n"
  5170. "Dim InstancePaths()\r\n"
  5171. "Dim InstanceCount\r\n"
  5172. "Dim CurrentInstanceIndex\r\n"
  5173. "\r\n",
  5174. ClassInfo->Name, ClassInfo->Name, ClassInfo->Name);
  5175. EnumerateClassProperties(TemplateHandle,
  5176. NULL,
  5177. NULL,
  5178. MofResource,
  5179. ClassInfo,
  5180. GenerateArrayDimsText,
  5181. TRUE,
  5182. NULL);
  5183. FilePrint(TemplateHandle, "\r\n");
  5184. if (IsEvent)
  5185. {
  5186. FilePrint(TemplateHandle,
  5187. "Dim LastEventObject\r\n"
  5188. "Dim ReceivedEvent\r\n"
  5189. "ReceivedEvent = FALSE\r\n"
  5190. "InstanceCount = 1\r\n"
  5191. "\r\n"
  5192. "Sub window_onLoad \r\n"
  5193. " Set Locator = CreateObject(\"WbemScripting.SWbemLocator\")\r\n"
  5194. " Locator.Security_.Privileges.AddAsString \"SeSecurityPrivilege\"\r\n"
  5195. " Set Service = Locator.ConnectServer(, \"root\\wmi\")\r\n"
  5196. " Service.Security_.ImpersonationLevel=3\r\n"
  5197. " On Error Resume Next\r\n"
  5198. " Err.Clear\r\n"
  5199. " Service.ExecNotificationQueryAsync mysink, _\r\n"
  5200. " \"select * from %ws\"\r\n"
  5201. "\r\n"
  5202. " if Err.Number <> 0 Then\r\n"
  5203. " MsgBox Err.Description,, \"Error Registering for event\"\r\n"
  5204. " End If\r\n"
  5205. "End Sub\r\n"
  5206. "\r\n",
  5207. ClassInfo->Name);
  5208. FilePrint(TemplateHandle,
  5209. "Sub ReloadInstance\r\n"
  5210. " Set TheForm = Document.ClassForm\r\n"
  5211. " if ReceivedEvent Then\r\n"
  5212. " Set Instance = LastEventObject\r\n"
  5213. " TheForm.InstanceNameText.Value = Instance.InstanceName\r\n"
  5214. );
  5215. EnumerateClassProperties(TemplateHandle,
  5216. NULL,
  5217. NULL,
  5218. MofResource,
  5219. ClassInfo,
  5220. GenerateReloadText,
  5221. TRUE,
  5222. NULL);
  5223. FilePrint(TemplateHandle,
  5224. " End If\r\n"
  5225. "\r\n"
  5226. "End Sub\r\n"
  5227. "\r\n"
  5228. );
  5229. } else {
  5230. FilePrint(TemplateHandle,
  5231. "Set Locator = CreateObject(\"WbemScripting.SWbemLocator\")\r\n"
  5232. "' Note that Locator.ConnectServer can be used to connect to remote computers\r\n"
  5233. "Set Service = Locator.ConnectServer(, \"root\\wmi\")\r\n"
  5234. "Service.Security_.ImpersonationLevel=3\r\n"
  5235. );
  5236. FilePrint(TemplateHandle,
  5237. "Set Collection = Service.InstancesOf (\"%ws\")\r\n"
  5238. "\r\n"
  5239. "InstanceCount = 0\r\n"
  5240. "Err.Clear\r\n"
  5241. "for each Instance in Collection\r\n"
  5242. " if Err.Number = 0 Then\r\n"
  5243. " InstanceCount = InstanceCount + 1\r\n"
  5244. "\r\n"
  5245. " ReDim Preserve InstancePaths(InstanceCount)\r\n"
  5246. "\r\n"
  5247. " Set ObjectPath = Instance.Path_\r\n"
  5248. " InstancePaths(InstanceCount) = ObjectPath.Path\r\n"
  5249. " End If\r\n"
  5250. "next 'Instance\r\n"
  5251. "\r\n"
  5252. "if InstanceCount = 0 Then\r\n"
  5253. " MsgBox \"No instances available for this class\"\r\n"
  5254. "Else\r\n"
  5255. " CurrentInstanceIndex = 1\r\n"
  5256. "End if\r\n"
  5257. "\r\n",
  5258. ClassInfo->Name
  5259. );
  5260. FilePrint(TemplateHandle,
  5261. "Sub ChangeButton_OnClick\r\n"
  5262. " Set TheForm = Document.ClassForm\r\n"
  5263. " if InstanceCount = 0 Then\r\n"
  5264. " MsgBox \"No instances available for this class\"\r\n"
  5265. " Else\r\n"
  5266. " On Error Resume Next\r\n"
  5267. " Err.Clear\r\n"
  5268. " Set Instance = Service.Get(InstancePaths(CurrentInstanceIndex))\r\n"
  5269. " if Err.Number = 0 Then\r\n"
  5270. );
  5271. EnumerateClassProperties(TemplateHandle,
  5272. NULL,
  5273. NULL,
  5274. MofResource,
  5275. ClassInfo,
  5276. GenerateChangeText,
  5277. TRUE,
  5278. NULL);
  5279. FilePrint(TemplateHandle,
  5280. "\r\n"
  5281. " Err.Clear\r\n"
  5282. " Instance.Put_()\r\n"
  5283. " if Err.Number <> 0 Then\r\n"
  5284. " MsgBox Err.Description, ,CurrentObjectPath\r\n"
  5285. " End If\r\n"
  5286. " Else\r\n"
  5287. " MsgBox Err.Description, ,CurrentObjectPath\r\n"
  5288. " End If\r\n"
  5289. " End If\r\n"
  5290. "End Sub\r\n"
  5291. "\r\n"
  5292. "Sub ReloadInstance\r\n"
  5293. " Set TheForm = Document.ClassForm\r\n"
  5294. " if InstanceCount = 0 Then\r\n"
  5295. " TheForm.InstanceNameText.Value = \"No Instances Available\"\r\n"
  5296. " Else\r\n"
  5297. " On Error Resume Next\r\n"
  5298. " Err.Clear\r\n"
  5299. " Set Instance = Service.Get(InstancePaths(CurrentInstanceIndex))\r\n"
  5300. " if Err.Number = 0 Then\r\n"
  5301. "\r\n"
  5302. " TheForm.InstanceNameText.Value = InstancePaths(CurrentInstanceIndex)\r\n"
  5303. );
  5304. EnumerateClassProperties(TemplateHandle,
  5305. NULL,
  5306. NULL,
  5307. MofResource,
  5308. ClassInfo,
  5309. GenerateReloadText,
  5310. TRUE,
  5311. NULL);
  5312. FilePrint(TemplateHandle,
  5313. " Else\r\n"
  5314. " MsgBox Err.Description, ,CurrentObjectPath\r\n"
  5315. " End If\r\n"
  5316. " End If\r\n"
  5317. "\r\n"
  5318. "End Sub\r\n"
  5319. "\r\n"
  5320. "Sub RefreshButton_OnClick\r\n"
  5321. " if InstanceCount = 0 Then\r\n"
  5322. " MsgBox \"No instances available for this class\"\r\n"
  5323. " Else\r\n"
  5324. " call ReloadInstance\r\n"
  5325. " End If\r\n"
  5326. "End Sub\r\n"
  5327. "\r\n"
  5328. "Sub NextButton_OnClick\r\n"
  5329. "\r\n"
  5330. " if InstanceCount = 0 Then\r\n"
  5331. " MsgBox \"No instances available for this class\"\r\n"
  5332. " Else\r\n"
  5333. " if CurrentInstanceIndex = InstanceCount Then\r\n"
  5334. " CurrentInstanceIndex = 1\r\n"
  5335. " Else \r\n"
  5336. " CurrentInstanceIndex = CurrentInstanceIndex + 1\r\n"
  5337. " End If\r\n"
  5338. " call ReloadInstance\r\n"
  5339. " End if\r\n"
  5340. "\r\n"
  5341. "\r\n"
  5342. "End Sub\r\n"
  5343. "\r\n"
  5344. "Sub PrevButton_OnClick\r\n"
  5345. "\r\n"
  5346. " if InstanceCount = 0 Then\r\n"
  5347. " MsgBox \"No instances available for this class\"\r\n"
  5348. " Else\r\n"
  5349. " if CurrentInstanceIndex = 1 Then\r\n"
  5350. " CurrentInstanceIndex = InstanceCount\r\n"
  5351. " Else\r\n"
  5352. " CurrentInstanceIndex = CurrentInstanceIndex - 1\r\n"
  5353. " End if\r\n"
  5354. " call ReloadInstance\r\n"
  5355. " End if\r\n"
  5356. "\r\n"
  5357. "\r\n"
  5358. "End Sub\r\n"
  5359. "\r\n");
  5360. }
  5361. FilePrint(TemplateHandle,
  5362. "Sub NextIndexButton_OnClick(ByRef Index, MaxIndex)\r\n"
  5363. " if InstanceCount <> 0 Then\r\n"
  5364. " Index = Index + 1\r\n"
  5365. " if Index = MaxIndex Then\r\n"
  5366. " Index = 0\r\n"
  5367. " End If\r\n"
  5368. " Call ReloadInstance\r\n"
  5369. " End If\r\n"
  5370. "End Sub\r\n"
  5371. "\r\n"
  5372. "Sub PrevIndexButton_OnClick(ByRef Index, MaxIndex)\r\n"
  5373. " if InstanceCount <> 0 Then\r\n"
  5374. " if Index = 0 Then\r\n"
  5375. " Index = MaxIndex - 1\r\n"
  5376. " Else\r\n"
  5377. " Index = Index - 1\r\n"
  5378. " End If\r\n"
  5379. " Call ReloadInstance\r\n"
  5380. " End If\r\n"
  5381. "End Sub\r\n"
  5382. "\r\n"
  5383. "Sub GotoIndexButton_OnClick(ByRef Index, MaxIndex, NewIndex)\r\n"
  5384. " if InstanceCount <> 0 Then\r\n"
  5385. " DestIndex = NewIndex + 0\r\n"
  5386. " if DestIndex >= 0 And DestIndex < MaxIndex Then\r\n"
  5387. " Index = DestIndex\r\n"
  5388. " Call ReloadInstance\r\n"
  5389. " Else\r\n"
  5390. " MsgBox \"Enter an index between 0 and \" & MaxIndex-1, ,\"Index out of range\"\r\n"
  5391. " End If\r\n"
  5392. " End If\r\n"
  5393. "End Sub\r\n"
  5394. "\r\n");
  5395. EnumerateClassMethods(TemplateHandle,
  5396. MofResource,
  5397. ClassInfo,
  5398. GenerateMethodSubsText,
  5399. NULL);
  5400. FilePrint(TemplateHandle,
  5401. "-->\r\n"
  5402. "</script></span>"
  5403. "<INPUT TYPE=\"TEXT\" SIZE=\"128\" NAME=\"InstanceNameText\" VALUE=\"\"></p>\r\n"
  5404. );
  5405. if (IsEvent)
  5406. {
  5407. FilePrint(TemplateHandle,
  5408. "<SCRIPT FOR=\"mysink\" EVENT=\"OnObjectReady(Instance, objAsyncContext)\" LANGUAGE=\"VBScript\">\r\n"
  5409. );
  5410. FilePrint(TemplateHandle,
  5411. " Set LastEventObject = Instance\r\n"
  5412. " ReceivedEvent = TRUE\r\n"
  5413. " Call ReloadInstance\r\n"
  5414. );
  5415. FilePrint(TemplateHandle,
  5416. "</SCRIPT>\r\n"
  5417. );
  5418. } else {
  5419. FilePrint(TemplateHandle,
  5420. "\r\n"
  5421. "<input name=NextButton type=BUTTON value=Next>\r\n"
  5422. "\r\n"
  5423. "<input name=PrevButton type=BUTTON value=Previous>\r\n"
  5424. "\r\n"
  5425. "<input name=ChangeButton type=BUTTON value=Change>\r\n"
  5426. "\r\n"
  5427. "<input name=RefreshButton type=BUTTON value=Refresh>\r\n"
  5428. "\r\n"
  5429. "<p class=MsoNormal><![if !supportEmptyParas]>&nbsp;<![endif]><o:p></o:p></p>\r\n"
  5430. "\r\n"
  5431. );
  5432. }
  5433. EnumerateClassProperties(TemplateHandle,
  5434. NULL,
  5435. NULL,
  5436. MofResource,
  5437. ClassInfo,
  5438. GenerateTextFormText,
  5439. TRUE,
  5440. NULL);
  5441. EnumerateClassMethods(TemplateHandle,
  5442. MofResource,
  5443. ClassInfo,
  5444. GenerateMethodButtonsText,
  5445. NULL);
  5446. FilePrint(TemplateHandle,
  5447. "<p class=MsoNormal><![if !supportEmptyParas]>&nbsp;<![endif]><o:p></o:p></p>\r\n"
  5448. "\r\n"
  5449. "<p class=MsoNormal><![if !supportEmptyParas]>&nbsp;<![endif]><o:p></o:p></p>\r\n"
  5450. "\r\n"
  5451. "<p class=MsoNormal><a href=\"index.htm\"\r\n"
  5452. "title=\"Goes back to list of classes in this MOF\">Back to List</a></p>\r\n"
  5453. "\r\n"
  5454. "</form>\r\n"
  5455. "\r\n"
  5456. );
  5457. if (! IsEvent)
  5458. {
  5459. FilePrint(TemplateHandle,
  5460. "<p class=MsoNormal><span style='display:none;mso-hide:all'><script language=\"VBScript\">\r\n"
  5461. "<!--\r\n"
  5462. " call ReloadInstance\r\n"
  5463. "-->\r\n"
  5464. "</script></span></p>\r\n"
  5465. "\r\n"
  5466. );
  5467. }
  5468. FilePrint(TemplateHandle,
  5469. "</div>\r\n"
  5470. "\r\n"
  5471. );
  5472. if (IsEvent)
  5473. {
  5474. FilePrint(TemplateHandle,
  5475. "<OBJECT ID=\"mysink\" CLASSID=\"CLSID:75718C9A-F029-11d1-A1AC-00C04FB6C223\"></OBJECT>\r\n"
  5476. );
  5477. }
  5478. FilePrint(TemplateHandle,
  5479. "</body>\r\n"
  5480. "\r\n"
  5481. "</html>\r\n"
  5482. );
  5483. return(ERROR_SUCCESS);
  5484. }
  5485. ULONG GenerateWebFiles(
  5486. PCHAR WebDir,
  5487. PMOFRESOURCE MofResource
  5488. )
  5489. {
  5490. ULONG Status;
  5491. HANDLE IndexHandle, TemplateHandle;
  5492. CHAR PathName[MAX_PATH];
  5493. PCHAR FileName;
  5494. ULONG Len, Index;
  5495. PLIST_ENTRY MofClassList;
  5496. PMOFCLASS MofClass;
  5497. PMOFCLASSINFOW ClassInfo;
  5498. CBMOFObj *ClassObject;
  5499. if (! CreateDirectory(WebDir, NULL))
  5500. {
  5501. Status = GetLastError();
  5502. if (Status != ERROR_ALREADY_EXISTS)
  5503. {
  5504. return(Status);
  5505. }
  5506. }
  5507. strcpy(PathName, WebDir);
  5508. Len = strlen(PathName)-1;
  5509. if (PathName[Len] != '\\')
  5510. {
  5511. PathName[++Len] = '\\';
  5512. PathName[++Len] = 0;
  5513. } else {
  5514. Len++;
  5515. }
  5516. FileName = &PathName[Len];
  5517. strcpy(FileName, "index.htm");
  5518. IndexHandle = CreateFile(PathName,
  5519. GENERIC_WRITE,
  5520. 0,
  5521. NULL,
  5522. CREATE_ALWAYS,
  5523. FILE_ATTRIBUTE_NORMAL,
  5524. NULL);
  5525. if ((IndexHandle == NULL) || (IndexHandle == INVALID_HANDLE_VALUE))
  5526. {
  5527. return(GetLastError());
  5528. }
  5529. FilePrint(IndexHandle,
  5530. "<HTML>\r\n<HEAD><TITLE>Class List</TITLE></HEAD><BODY>\r\n");
  5531. //
  5532. // Loop over all mof classes
  5533. Index = 0;
  5534. MofClassList = MofResource->MRMCHead.Flink;
  5535. while (MofClassList != &MofResource->MRMCHead)
  5536. {
  5537. MofClass = CONTAINING_RECORD(MofClassList,
  5538. MOFCLASS,
  5539. MCMRList);
  5540. ClassInfo = MofClass->MofClassInfo;
  5541. ClassObject = (CBMOFObj *)MofClass->ClassObjectHandle;
  5542. if (! (ClassInfo->Flags & MOFCI_FLAG_EMBEDDED_CLASS))
  5543. {
  5544. //
  5545. // don't create pages for embedded classes or events
  5546. sprintf(FileName, "%ws.htm", ClassInfo->Name);
  5547. TemplateHandle = CreateFile(PathName,
  5548. GENERIC_WRITE,
  5549. 0,
  5550. NULL,
  5551. CREATE_ALWAYS,
  5552. FILE_ATTRIBUTE_NORMAL,
  5553. NULL);
  5554. if ((TemplateHandle == NULL) || (TemplateHandle == INVALID_HANDLE_VALUE))
  5555. {
  5556. CloseHandle(IndexHandle);
  5557. return(GetLastError());
  5558. }
  5559. FilePrint(IndexHandle,
  5560. "<H3><A href=\"%ws.htm\">%ws</A></H3><HR>",
  5561. ClassInfo->Name,
  5562. ClassInfo->Name);
  5563. Status = GenerateClassWebPage(TemplateHandle,
  5564. MofClass,
  5565. MofResource);
  5566. CloseHandle(TemplateHandle);
  5567. }
  5568. MofClassList = MofClassList->Flink;
  5569. }
  5570. FilePrint(IndexHandle, "</BODY>\r\n</HTML>\r\n");
  5571. CloseHandle(IndexHandle);
  5572. return(ERROR_SUCCESS);
  5573. }
  5574. ULONG AppendUnicodeTextFiles(
  5575. char *DestFile,
  5576. char *SrcFile1,
  5577. char *SrcFile2
  5578. )
  5579. {
  5580. #define READ_BLOCK_SIZE 0x8000
  5581. HANDLE DestHandle, SrcHandle;
  5582. ULONG BytesRead, BytesWritten;
  5583. PUCHAR Buffer, p;
  5584. BOOL b;
  5585. ULONG Status = ERROR_SUCCESS;
  5586. BOOLEAN FirstTime;
  5587. ULONG TotalBytesRead = 0;
  5588. ULONG TotalBytesWritten = 0;
  5589. ULONG ReadSize;
  5590. CHAR c;
  5591. //
  5592. // This is a very simple procedure. We append the second file onto
  5593. // the end of the first file, however we always skip the first 2
  5594. // bytes of the second file if they are a 0xFFFE. This signature
  5595. // denotes that the file is a unicode text file, but if it gets
  5596. // appended in the middle of the file then mofcomp will get really
  5597. // pissed off and barf.
  5598. //
  5599. Buffer = (PUCHAR)WmipAlloc(READ_BLOCK_SIZE);
  5600. if (Buffer != NULL)
  5601. {
  5602. DestHandle = CreateFile(DestFile,
  5603. GENERIC_READ | GENERIC_WRITE,
  5604. 0,
  5605. NULL,
  5606. CREATE_ALWAYS,
  5607. FILE_ATTRIBUTE_NORMAL,
  5608. NULL);
  5609. if (DestHandle != INVALID_HANDLE_VALUE)
  5610. {
  5611. SrcHandle = CreateFile(SrcFile1,
  5612. GENERIC_READ,
  5613. 0,
  5614. NULL,
  5615. OPEN_EXISTING,
  5616. FILE_ATTRIBUTE_NORMAL,
  5617. NULL);
  5618. if (SrcHandle != INVALID_HANDLE_VALUE)
  5619. {
  5620. //
  5621. // Just copy over all data from first file into
  5622. // destination
  5623. //
  5624. do
  5625. {
  5626. b = ReadFile(SrcHandle,
  5627. Buffer,
  5628. READ_BLOCK_SIZE,
  5629. &BytesRead,
  5630. NULL);
  5631. if (b)
  5632. {
  5633. TotalBytesRead += BytesRead;
  5634. WmipDebugPrint(("Read 0x%x/0x%x from Source 1\n",
  5635. BytesRead, TotalBytesRead));
  5636. b = WriteFile(DestHandle,
  5637. Buffer,
  5638. BytesRead,
  5639. &BytesWritten,
  5640. NULL);
  5641. if (!b)
  5642. {
  5643. Status = GetLastError();
  5644. break;
  5645. } else if (BytesWritten != BytesRead) {
  5646. Status = ERROR_BAD_LENGTH;
  5647. break;
  5648. }
  5649. TotalBytesWritten += BytesWritten;
  5650. WmipDebugPrint(("Wrote 0x%x/0x%x to Dest\n",
  5651. BytesWritten, TotalBytesWritten));
  5652. } else {
  5653. Status = GetLastError();
  5654. break;
  5655. }
  5656. } while (BytesRead == READ_BLOCK_SIZE);
  5657. CloseHandle(SrcHandle);
  5658. //
  5659. // Now copy the data from the second file, but make
  5660. // sure we skip any 0xFFFE at the beginning of the
  5661. // second file
  5662. //
  5663. TotalBytesRead = 0;
  5664. SrcHandle = CreateFile(SrcFile2,
  5665. GENERIC_READ,
  5666. 0,
  5667. NULL,
  5668. OPEN_EXISTING,
  5669. FILE_ATTRIBUTE_NORMAL,
  5670. NULL);
  5671. if (SrcHandle != INVALID_HANDLE_VALUE)
  5672. {
  5673. FirstTime = TRUE;
  5674. do
  5675. {
  5676. b = ReadFile(SrcHandle,
  5677. Buffer,
  5678. READ_BLOCK_SIZE,
  5679. &BytesRead,
  5680. NULL);
  5681. if (b)
  5682. {
  5683. ReadSize = READ_BLOCK_SIZE;
  5684. TotalBytesRead += BytesRead;
  5685. WmipDebugPrint(("Read 0x%x/0x%x from Source 2\n",
  5686. BytesRead, TotalBytesRead));
  5687. if (FirstTime)
  5688. {
  5689. FirstTime = FALSE;
  5690. if ( *((PWCHAR)Buffer) = 0xFFFE )
  5691. {
  5692. WmipDebugPrint(("First Time and need to skip 2 bytes\n"));
  5693. p = Buffer + 2;
  5694. BytesRead -= 2;
  5695. ReadSize -= 2;
  5696. }
  5697. } else {
  5698. p = Buffer;
  5699. }
  5700. b = WriteFile(DestHandle,
  5701. p,
  5702. BytesRead,
  5703. &BytesWritten,
  5704. NULL);
  5705. if (!b)
  5706. {
  5707. Status = GetLastError();
  5708. break;
  5709. } else if (BytesWritten != BytesRead) {
  5710. Status = ERROR_BAD_LENGTH;
  5711. break;
  5712. }
  5713. TotalBytesWritten += BytesWritten;
  5714. WmipDebugPrint(("Wrote 0x%x/0x%x to Dest\n",
  5715. BytesWritten, TotalBytesWritten));
  5716. } else {
  5717. Status = GetLastError();
  5718. break;
  5719. }
  5720. } while (BytesRead == ReadSize);
  5721. if (Status == ERROR_SUCCESS)
  5722. {
  5723. //
  5724. // Copy put a ^Z at the end so well do that too
  5725. //
  5726. c = 0x1a;
  5727. b = WriteFile(DestHandle,
  5728. &c,
  5729. 1,
  5730. &BytesWritten,
  5731. NULL);
  5732. if (!b)
  5733. {
  5734. Status = GetLastError();
  5735. } else if (BytesWritten != 1) {
  5736. Status = ERROR_BAD_LENGTH;
  5737. }
  5738. TotalBytesWritten += BytesWritten;
  5739. WmipDebugPrint(("Wrote 0x%x/0x%x to Dest\n",
  5740. BytesWritten, TotalBytesWritten));
  5741. }
  5742. CloseHandle(SrcHandle);
  5743. }
  5744. } else {
  5745. Status = GetLastError();
  5746. }
  5747. CloseHandle(DestHandle);
  5748. } else {
  5749. Status = GetLastError();
  5750. }
  5751. WmipFree(Buffer);
  5752. } else {
  5753. Status = ERROR_NOT_ENOUGH_MEMORY;
  5754. }
  5755. return(Status);
  5756. }
  5757. void Usage(
  5758. void
  5759. )
  5760. {
  5761. printf("WMI Mof Check Tool - Revision 19\n");
  5762. printf("\n");
  5763. printf("WmiMofCk validates that the classes, properties, methods and events specified \n");
  5764. printf("in a binary mof file (.bmf) are valid for use with WMI. It also generates \n");
  5765. printf("useful output files needed to build and test the WMI data provider.\n");
  5766. printf("\n");
  5767. printf("If the -h parameter is specified then a C language header file is created\n");
  5768. printf("that defines the guids, data structures and method indicies specified in the\n");
  5769. printf("MOF file.\n");
  5770. printf("\n");
  5771. printf("If the -t parameter is specified then a VBScript applet is created that will\n");
  5772. printf("query all data blocks and properties specified in the MOF file. This can be\n");
  5773. printf("useful for testing WMI data providers.\n");
  5774. printf("\n");
  5775. printf("If the -x parameter is specified then a text file is created that contains\n");
  5776. printf("the text representation of the binary mof data. This can be included in \n");
  5777. printf("the source of the driver if the driver supports reporting the binary mof \n");
  5778. printf("via a WMI query rather than a resource on the driver image file.\n\n");
  5779. printf("If the -c parameter is specified then a C language source file is\n");
  5780. printf("generated that contains a template for implementing WMI code in\n");
  5781. printf("a device driver\n\n");
  5782. printf("if the -w parameter is specified then a set of HTML files are\n");
  5783. printf("generated that create a rudimentary UI that can be used to access\n");
  5784. printf("the wmi data blocks\n\n");
  5785. printf("if the -m parameter is specified then structure definitions for\n");
  5786. printf("method parameter lists are generated in the generated header file.\n\n");
  5787. printf("if the -u parameter is specified then structure definitions for all\n");
  5788. printf("data blocks are generated unconditionally\n\n");
  5789. printf("\n");
  5790. printf("Usage:\n");
  5791. printf(" wmimofck -h<C Source Language Header output file>\n");
  5792. printf(" -c<C Source Language Code output file>\n");
  5793. printf(" -x<Hexdump output file>\n");
  5794. printf(" -t<Command line VBScript test output file>\n");
  5795. printf(" -w<HTML UI output file directory>\n");
  5796. printf(" -y<MofFile> -z<MflFile>\n");
  5797. printf(" -m\n");
  5798. printf(" -u\n");
  5799. printf(" <binary mof input file>\n\n");
  5800. }
  5801. #define IsWhiteSpace(c) ( (c == ' ') || (c == '\t') )
  5802. ULONG GetParameter(
  5803. char *Parameter,
  5804. ULONG ParameterSize,
  5805. char *CommandLine
  5806. )
  5807. {
  5808. ULONG i;
  5809. i = 0;
  5810. ParameterSize--;
  5811. while ( (! IsWhiteSpace(*CommandLine)) &&
  5812. ( *CommandLine != 0) &&
  5813. (i < ParameterSize) )
  5814. {
  5815. *Parameter++ = *CommandLine++;
  5816. i++;
  5817. }
  5818. *Parameter = 0;
  5819. return(i);
  5820. }
  5821. int _cdecl main(int argc, char *argv[])
  5822. {
  5823. char BMofFile[MAX_PATH];
  5824. char *Parameter;
  5825. int i;
  5826. ULONG Status;
  5827. char ASLFile[MAX_PATH];
  5828. char CFile[MAX_PATH];
  5829. char HFile[MAX_PATH];
  5830. char XFile[MAX_PATH];
  5831. char TFile[MAX_PATH];
  5832. char WebDir[MAX_PATH];
  5833. char MofFile[MAX_PATH];
  5834. char MflFile[MAX_PATH];
  5835. char c;
  5836. PMOFRESOURCE MofResource;
  5837. *ASLFile = 0;
  5838. *CFile = 0;
  5839. *HFile = 0;
  5840. *XFile = 0;
  5841. *TFile = 0;
  5842. *WebDir = 0;
  5843. *MofFile = 0;
  5844. *MflFile = 0;
  5845. *BMofFile = 0;
  5846. printf("Microsoft (R) WDM Extensions To WMI MOF Checking Utility Version 1.50.0000\n");
  5847. printf("Copyright (c) Microsoft Corp. 1997-2000. All rights reserved.\n\n");
  5848. SetErrorMode(SEM_NOALIGNMENTFAULTEXCEPT); // BUGBUG: Remove when MOF format maintains alignment correctly
  5849. if (argc == 1)
  5850. {
  5851. Usage();
  5852. return(1);
  5853. }
  5854. i = 1;
  5855. while (i < argc)
  5856. {
  5857. Parameter = argv[i++];
  5858. if (IsWhiteSpace(*Parameter))
  5859. {
  5860. Parameter++;
  5861. continue;
  5862. }
  5863. if (*Parameter != '-')
  5864. {
  5865. //
  5866. // Parameter does not start with -, must be bmof filename
  5867. if (*BMofFile != 0)
  5868. {
  5869. //
  5870. // Only one filename allowed
  5871. Usage();
  5872. }
  5873. GetParameter(BMofFile, sizeof(BMofFile), Parameter);
  5874. } else {
  5875. Parameter++;
  5876. // Check for - parameters here
  5877. c = (CHAR)toupper(*Parameter);
  5878. Parameter++;
  5879. switch (c)
  5880. {
  5881. case 'A' :
  5882. {
  5883. GetParameter(ASLFile, sizeof(ASLFile), Parameter);
  5884. break;
  5885. }
  5886. case 'C':
  5887. {
  5888. GetParameter(CFile, sizeof(CFile), Parameter);
  5889. break;
  5890. }
  5891. case 'H':
  5892. {
  5893. GetParameter(HFile, sizeof(HFile), Parameter);
  5894. break;
  5895. }
  5896. case 'U':
  5897. {
  5898. ForceHeaderGeneration = TRUE;
  5899. }
  5900. case 'M':
  5901. {
  5902. DoMethodHeaderGeneration = TRUE;
  5903. break;
  5904. }
  5905. case 'X':
  5906. {
  5907. GetParameter(XFile, sizeof(XFile), Parameter);
  5908. break;
  5909. }
  5910. case 'T':
  5911. {
  5912. GetParameter(TFile, sizeof(TFile), Parameter);
  5913. break;
  5914. }
  5915. case 'W':
  5916. {
  5917. GetParameter(WebDir, sizeof(WebDir), Parameter);
  5918. break;
  5919. }
  5920. case 'Y':
  5921. {
  5922. GetParameter(MofFile, sizeof(MofFile), Parameter);
  5923. break;
  5924. }
  5925. case 'Z':
  5926. {
  5927. GetParameter(MflFile, sizeof(MflFile), Parameter);
  5928. break;
  5929. }
  5930. default: {
  5931. Usage();
  5932. return(1);
  5933. }
  5934. }
  5935. }
  5936. }
  5937. if (*BMofFile == 0)
  5938. {
  5939. //
  5940. // We must have a filename
  5941. Usage();
  5942. return(1);
  5943. }
  5944. if (*MofFile != 0)
  5945. {
  5946. if (*MflFile != 0)
  5947. {
  5948. Status = AppendUnicodeTextFiles(BMofFile, MofFile, MflFile);
  5949. } else {
  5950. Usage();
  5951. return(1);
  5952. }
  5953. return(Status);
  5954. }
  5955. Status = ParseBinaryMofFile(BMofFile, &MofResource);
  5956. if (Status == ERROR_SUCCESS)
  5957. {
  5958. if (*HFile != 0)
  5959. {
  5960. //
  5961. // Generate C Header file
  5962. Status = GenerateHTemplate(HFile, MofResource);
  5963. if (Status != ERROR_SUCCESS)
  5964. {
  5965. //
  5966. // TODO: Better message
  5967. printf("Error %d creating C Header Template file \n", Status);
  5968. }
  5969. }
  5970. if (*XFile != 0)
  5971. {
  5972. //
  5973. // Generate X Header file
  5974. Status = GenerateXTemplate(XFile);
  5975. if (Status != ERROR_SUCCESS)
  5976. {
  5977. //
  5978. // TODO: Better message
  5979. printf("Error %d creating X Header Template file \n", Status);
  5980. }
  5981. }
  5982. if (*TFile != 0)
  5983. {
  5984. //
  5985. // Generate C output template
  5986. Status = GenerateTTemplate(TFile, MofResource);
  5987. if (Status != ERROR_SUCCESS)
  5988. {
  5989. //
  5990. // TODO: Better message
  5991. printf("Error %d creating C Template file \n", Status);
  5992. }
  5993. }
  5994. if (*CFile != 0)
  5995. {
  5996. //
  5997. // Generate C output template
  5998. Status = GenerateCTemplate(CFile,
  5999. HFile,
  6000. *XFile == 0 ? CFile : XFile,
  6001. MofResource);
  6002. if (Status != ERROR_SUCCESS)
  6003. {
  6004. //
  6005. // TODO: Better message
  6006. printf("Error %d creating C Template file \n", Status);
  6007. }
  6008. }
  6009. if (*WebDir != 0)
  6010. {
  6011. //
  6012. // Generate HTML UI for classes
  6013. //
  6014. Status = GenerateWebFiles(WebDir,
  6015. MofResource);
  6016. }
  6017. if (*ASLFile != 0)
  6018. {
  6019. //
  6020. // Generate ASL output template
  6021. Status = GenerateASLTemplate(ASLFile);
  6022. }
  6023. }
  6024. return(Status);
  6025. }