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.

551 lines
17 KiB

  1. // extbase.h: implementation of link and embed foreign objects in SCE store
  2. // Copyright (c)1997-2001 Microsoft Corporation
  3. //
  4. // this is the extension model base (hence the name)
  5. //////////////////////////////////////////////////////////////////////
  6. #if !defined(AFX_EXTBASE_H__BD7570F7_9F0E_4C6B_B525_E078691B6D0E__INCLUDED_)
  7. #define AFX_EXTBASE_H__BD7570F7_9F0E_4C6B_B525_E078691B6D0E__INCLUDED_
  8. #if _MSC_VER >= 1000
  9. #pragma once
  10. #endif // _MSC_VER >= 1000
  11. #include "GenericClass.h"
  12. //
  13. // forward declaration
  14. //
  15. class CPropValuePair;
  16. //
  17. // For quick lookup of embedding class's foreign class information
  18. // by the embedding class's name we use a map.
  19. //
  20. typedef std::map<BSTR, CForeignClassInfo*, strLessThan<BSTR> > MapExtClasses;
  21. typedef MapExtClasses::iterator ExtClassIterator;
  22. //=============================================================================
  23. /*
  24. Class description
  25. Naming:
  26. CExtClasses stands Extension Classes.
  27. Base class:
  28. None.
  29. Purpose of class:
  30. (1) This class is to provide, for each embedding class, the foreign class' information.
  31. A foreign class is one that is provided by other WMI providers. Normally, we don't
  32. have any control over it. For example, we can't force it to be saved in a store for
  33. later use.
  34. We invented the notation called Embedding so that we can store foreign class instances
  35. in our SCE store and later knows how to use it.
  36. For each foreign class that SCE wants to use, SCE will have a embedding class derived
  37. from WMI class Sce_EmbedFO (stands for Embedding Foreign Object). See sceprov.mof for
  38. its schema.
  39. We need a global map that maps our embedding class' name to the foreign class'
  40. information. This is the task of this class.
  41. Design:
  42. (1) For quick name to Foreign Class Info lookup, we use a map: m_mapExtClasses.
  43. (2) Populating all such embedding class's foreign information is costly. We delay that loading
  44. until the need for embedding classes arrives. For that we use m_bPopulated. Once this
  45. is populated, we no longer populate again. For that reasons, if you register more embedding
  46. classes, you need to make sure that our dll is unloaded so that we can populate again.
  47. $consider: should we change this behavior?
  48. (3) GetForeignClassInfo is all it takes to find the embedding class' foreign class information.
  49. Use:
  50. (1) Create an instance of this class.
  51. (2) Call GetForeignClassInfo when you need a foreign class information.
  52. */
  53. class CExtClasses
  54. {
  55. public:
  56. CExtClasses();
  57. ~CExtClasses();
  58. const CForeignClassInfo* GetForeignClassInfo (
  59. IWbemServices* pNamespace,
  60. IWbemContext* pCtx,
  61. BSTR pszEmbedClassName
  62. );
  63. private:
  64. HRESULT PopulateExtensionClasses (
  65. IWbemServices* pNamespace,
  66. IWbemContext* pCtx
  67. );
  68. HRESULT PutExtendedClass (
  69. BSTR pszSubClassName,
  70. CForeignClassInfo* pFCI
  71. );
  72. HRESULT GetSubclasses (
  73. IWbemServices* pNamespace,
  74. IWbemContext* pCtx,
  75. IEnumWbemClassObject* pEnumObj,
  76. EnumExtClassType dwClassType
  77. );
  78. HRESULT PopulateKeyPropertyNames (
  79. IWbemServices* pNamespace,
  80. IWbemContext* pCtx,
  81. BSTR bstrClassName,
  82. CForeignClassInfo* pNewSubclass
  83. );
  84. MapExtClasses m_mapExtClasses;
  85. bool m_bPopulated;
  86. };
  87. extern CExtClasses g_ExtClasses;
  88. //=============================================================================
  89. /*
  90. Class description
  91. Naming:
  92. CSceExtBaseObject stands Sce Extension Base Object.
  93. Base class:
  94. (1) CComObjectRootEx for threading model and IUnknown.
  95. (2) ISceClassObject our interface for auto persistence (that IScePersistMgr uses)
  96. Purpose of class:
  97. (1) This is our implementation of ISceClassObject. Our embedding classes uses this
  98. for IScePersistMgr's persistence. We no longer needs to write persistence functionality
  99. like what we do for all the core object. For embedding classes, this class together with
  100. CScePersistMgr takes care of support for persistence.
  101. Design:
  102. (1) This is not a directly instantiatable class. See the constructor and destructor, they are
  103. both protected. See Use section for creation steps.
  104. (2) This is not an externally createable class. It's for internal use. No class factory support
  105. is given.
  106. Use:
  107. (1) Create an instance of this class. Since it's not a directly instantiatable class, you need
  108. to use CComObject<CSceExtBaseObject> for creation:
  109. CComObject<CSceExtBaseObject> *pSceObject = NULL;
  110. hr = CComObject<CSceExtBaseObject>::CreateInstance(&pSceObject);
  111. (2) Call PopulateProperties. This populate the ourselves.
  112. (3) Since what this class is good at is to provide ISceClassObject. So you normally
  113. QueryInterface for ISceClassObject.
  114. (4) After you get the ISceClassObject interface pointer, you attach (Attach) the Wbem object
  115. to this object. Since we managed WMI object's persistence in sce store, there must be an
  116. wbem object you need to persist. That is how you do it.
  117. (5) Since the sole purpose of this class is for IScePersistMgr to use it for retrieving data,
  118. you normally has a IScePersistMgr object waiting for this object. Once we have done the
  119. three steps above, you can finally satisfy IScePersistMgr by attaching this object to the
  120. IScePersistMgr.
  121. Everything happens automatically.
  122. See CEmbedForeignObj::CreateScePersistMgr for sample code.
  123. */
  124. class ATL_NO_VTABLE CSceExtBaseObject :
  125. public CComObjectRootEx<CComMultiThreadModel>,
  126. public ISceClassObject
  127. {
  128. protected:
  129. CSceExtBaseObject();
  130. virtual ~CSceExtBaseObject();
  131. public:
  132. //
  133. // ISceClassObject is the only interface we support, well besides IUnknown.
  134. //
  135. BEGIN_COM_MAP(CSceExtBaseObject)
  136. COM_INTERFACE_ENTRY(ISceClassObject)
  137. END_COM_MAP()
  138. //
  139. // we allow this to be aggregated
  140. //
  141. DECLARE_NOT_AGGREGATABLE( CSceExtBaseObject )
  142. //
  143. // though we don't have registry resource, ATL requires this macro. No harm.
  144. //
  145. DECLARE_REGISTRY_RESOURCEID(IDR_SceProv)
  146. //
  147. // ISceClassObject methods
  148. //
  149. STDMETHOD(GetPersistPath) (
  150. BSTR* pbstrPath // [out]
  151. );
  152. STDMETHOD(GetClassName) (
  153. BSTR* pbstrClassName // [out]
  154. );
  155. STDMETHOD(GetLogPath) (
  156. BSTR* pbstrPath // [out]
  157. );
  158. STDMETHOD(Validate)();
  159. STDMETHOD(GetProperty) (
  160. LPCWSTR pszPropName, // [in, string]
  161. VARIANT* pValue // [in]
  162. );
  163. STDMETHOD(GetPropertyCount) (
  164. SceObjectPropertyType type, // [in]
  165. DWORD* pCount // [out]
  166. );
  167. STDMETHOD(GetPropertyValue) (
  168. SceObjectPropertyType type, // [in]
  169. DWORD dwIndex, // [in]
  170. BSTR* pbstrPropName, // [out]
  171. VARIANT* pValue // [out]
  172. );
  173. STDMETHOD(Attach) (
  174. IWbemClassObject* pInst // [in]
  175. );
  176. STDMETHOD(GetClassObject) (
  177. IWbemClassObject** ppInst //[out]
  178. );
  179. void CleanUp();
  180. HRESULT PopulateProperties (
  181. ISceKeyChain *pKeyChain,
  182. IWbemServices *pNamespace,
  183. IWbemContext *pCtx,
  184. const CForeignClassInfo* pClsInfo
  185. );
  186. private:
  187. enum GetIndexFlags
  188. {
  189. GIF_Keys = 0x00000001,
  190. GIF_NonKeys = 0x00000002,
  191. GIF_Both = 0x00000003,
  192. };
  193. int GetIndex(LPCWSTR pszName, GetIndexFlags fKey);
  194. const CForeignClassInfo* m_pClsInfo;
  195. std::vector<BSTR> m_vecKeyProps;
  196. std::vector<BSTR> m_vecNonKeyProps;
  197. std::vector<VARIANT*> m_vecKeyValues;
  198. std::vector<VARIANT*> m_vecPropValues;
  199. CComPtr<ISceKeyChain> m_srpKeyChain;
  200. CComPtr<IWbemServices> m_srpNamespace;
  201. CComPtr<IWbemContext> m_srpCtx;
  202. CComPtr<IWbemClassObject> m_srpWbemObject;
  203. CComBSTR m_bstrLogPath;
  204. CComBSTR m_bstrClassName;
  205. };
  206. //=============================================================================
  207. /*
  208. Class description
  209. Naming:
  210. CEmbedForeignObj stands Sce Embed Foreign Object.
  211. Base class:
  212. CGenericClass since this class will be persisted. It is representing any sub-classes of Sce_EmbedFO.
  213. Purpose of class:
  214. (1) This is our implementation of the embedding model for open extension architecture.
  215. Design:
  216. (1) It knows how to PutInst, how to CreateObject (for querying, enumerating, deleting, getting
  217. single instance, etc), and most importantly, it knows how to execute a method, abeit it
  218. uses CExtClassMethodCaller.
  219. (2) It knows how to create a persistence manager for its persistence needs.
  220. (3) In order to use the persistence manager, it must knows how to create a ISceClassObject
  221. representing this object.
  222. Use:
  223. (1) You will not create this your own. Everything is handled by CRequestObject via the
  224. interface of CGenericClass. This is just a CGenericClass.
  225. */
  226. class CEmbedForeignObj : public CGenericClass
  227. {
  228. public:
  229. CEmbedForeignObj (
  230. ISceKeyChain *pKeyChain,
  231. IWbemServices *pNamespace,
  232. IWbemContext *pCtx,
  233. const CForeignClassInfo* pClsInfo
  234. );
  235. virtual ~CEmbedForeignObj();
  236. public:
  237. virtual HRESULT PutInst (
  238. IWbemClassObject *pInst,
  239. IWbemObjectSink *pHandler,
  240. IWbemContext *pCtx
  241. );
  242. virtual HRESULT CreateObject (
  243. IWbemObjectSink *pHandler,
  244. ACTIONTYPE atAction
  245. );
  246. virtual HRESULT ExecMethod (
  247. BSTR bstrPath,
  248. BSTR bstrMethod,
  249. bool bIsInstance,
  250. IWbemClassObject *pInParams,
  251. IWbemObjectSink *pHandler,
  252. IWbemContext *pCtx
  253. );
  254. private:
  255. HRESULT CreateBaseObject (
  256. ISceClassObject** ppObj
  257. );
  258. HRESULT CreateScePersistMgr (
  259. IWbemClassObject *pInst,
  260. IScePersistMgr** ppPersistMgr
  261. );
  262. const CForeignClassInfo* m_pClsInfo;
  263. };
  264. //=============================================================================
  265. /*
  266. Class description
  267. Naming:
  268. CMethodResultRecorder stands Method call Results Recorder.
  269. Base class:
  270. None.
  271. Purpose of class:
  272. (1) Make logging method call results easy. Logging a result is a compliated and repeatitive work.
  273. This class hides all the details of creating those WMI objects for logging, etc.
  274. Design:
  275. (1) Just two functions to make it extremely simple to use.
  276. Use:
  277. (1) Create an instance.
  278. (2) Call Initialize. You can actually call this multiple times to switch the context.
  279. (3) Call LogResult when you need to push the information to the log file.
  280. */
  281. class CMethodResultRecorder
  282. {
  283. public:
  284. CMethodResultRecorder ();
  285. HRESULT Initialize (
  286. LPCWSTR pszLogFilePath,
  287. LPCWSTR pszClassName,
  288. IWbemServices *pNativeNS,
  289. IWbemContext *pCtx
  290. );
  291. HRESULT LogResult (
  292. HRESULT hrResult, // [in]
  293. IWbemClassObject *pObj, // [in]
  294. IWbemClassObject *pParam, // [in]
  295. IWbemClassObject *pOutParam, // [in]
  296. LPCWSTR pszMethod, // [in]
  297. LPCWSTR pszForeignAction, // [in]
  298. UINT uMsgResID, // [in]
  299. LPCWSTR pszExtraInfo // [in]
  300. )const;
  301. private:
  302. HRESULT FormatVerboseMsg (
  303. IWbemClassObject *pObject, // [in]
  304. BSTR* pbstrMsg // [out]
  305. )const;
  306. CComBSTR m_bstrLogFilePath;
  307. CComBSTR m_bstrClassName;
  308. CComPtr<IWbemContext> m_srpCtx;
  309. CComPtr<IWbemServices> m_srpNativeNS;
  310. };
  311. //=============================================================================
  312. /*
  313. Class description
  314. Naming:
  315. CExtClassMethodCaller stands for Extension Class Method Caller.
  316. Base class:
  317. None.
  318. Purpose of class:
  319. Help to make executing a method on foreign objects easy. It works with CMethodResultRecorder.
  320. Design:
  321. (1) Just two functions to make it extremely simple to use.
  322. Use:
  323. (1) Create an instance.
  324. (2) Call Initialize. You can actually call this multiple times to switch the context.
  325. (3) Call LogResult when you need to push the information to the log file.
  326. */
  327. class CExtClassMethodCaller
  328. {
  329. public:
  330. CExtClassMethodCaller (
  331. ISceClassObject* pSceClassObj,
  332. const CForeignClassInfo* pClsInfo
  333. );
  334. ~CExtClassMethodCaller ();
  335. HRESULT Initialize (
  336. CMethodResultRecorder* pLog
  337. );
  338. HRESULT ExecuteForeignMethod (
  339. LPCWSTR pszMethod,
  340. IWbemClassObject * pInParams,
  341. IWbemObjectSink * pHandler,
  342. IWbemContext * pCtx,
  343. IWbemClassObject ** ppOut
  344. );
  345. private:
  346. HRESULT ParseMethodEncodingString (
  347. LPCWSTR pszEncodeString,
  348. DWORD* pdwContext,
  349. BSTR* pbstrError
  350. );
  351. HRESULT BuildMethodContext (
  352. LPCWSTR szMethodName,
  353. LPCWSTR szParameter,
  354. BSTR* pbstrError
  355. );
  356. HRESULT PopulateForeignObject (
  357. IWbemClassObject* pForeignObj,
  358. ISceClassObject* pSceObject,
  359. CMethodResultRecorder* LogRecord
  360. )const;
  361. bool IsMemberParameter (
  362. LPCWSTR szName
  363. )const;
  364. bool IsInComingParameter (
  365. LPCWSTR szName
  366. )const;
  367. bool IsStaticMethod (
  368. LPCWSTR szName
  369. )const;
  370. void FormatSyntaxError (
  371. WCHAR wchMissChar,
  372. DWORD dwMissCharIndex,
  373. LPCWSTR pszEncodeString, // [in]
  374. BSTR* pbstrError // [out]
  375. );
  376. CComPtr<ISceClassObject> m_srpSceObject;
  377. CComPtr<IWbemServices> m_srpForeignNamespace;
  378. CComPtr<IWbemClassObject> m_srpClass;
  379. const CForeignClassInfo* m_pClsInfo;
  380. //
  381. // CMethodContext is internal class just to make resource management easy.
  382. // It manages the method call context - its parameters and the method name.
  383. //
  384. class CMethodContext
  385. {
  386. public:
  387. CMethodContext();
  388. ~CMethodContext();
  389. LPWSTR m_pszMethodName;
  390. std::vector<VARIANT*> m_vecParamValues;
  391. std::vector<LPWSTR> m_vecParamNames;
  392. };
  393. std::vector<CMethodContext*> m_vecMethodContext;
  394. typedef std::vector<CMethodContext*>::iterator MCIterator;
  395. CMethodResultRecorder* m_pLogRecord;
  396. bool m_bStaticCall;
  397. };
  398. #endif