VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxCredProv/VBoxCredentialProvider.cpp@ 53100

Last change on this file since 53100 was 50076, checked in by vboxsync, 11 years ago

VBoxCredProv: Added some more interface checks.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 16.8 KB
Line 
1/* $Id: VBoxCredentialProvider.cpp 50076 2014-01-14 08:49:41Z vboxsync $ */
2/** @file
3 * VBoxCredentialProvider - Main file of the VirtualBox Credential Provider.
4 */
5
6/*
7 * Copyright (C) 2012-2014 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21#include <windows.h>
22#include <initguid.h>
23#include <new> /* For bad_alloc. */
24
25#ifdef VBOX_WITH_SENS
26# include <eventsys.h>
27# include <sens.h>
28# include <Sensevts.h>
29#endif
30
31#include <iprt/buildconfig.h>
32#include <iprt/initterm.h>
33#ifdef VBOX_WITH_SENS
34# include <iprt/string.h>
35#endif
36#include <VBox/VBoxGuestLib.h>
37
38#include "VBoxCredentialProvider.h"
39#include "VBoxCredProvFactory.h"
40
41/*******************************************************************************
42* Global Variables *
43*******************************************************************************/
44static LONG g_cDllRefs = 0; /**< Global DLL reference count. */
45static HINSTANCE g_hDllInst = NULL; /**< Global DLL hInstance. */
46
47#ifdef VBOX_WITH_SENS
48static bool g_fSENSEnabled = false;
49static IEventSystem *g_pIEventSystem = NULL; /**< Pointer to IEventSystem interface. */
50
51/**
52 * Subscribed SENS events.
53 */
54static struct VBOXCREDPROVSENSEVENTS
55{
56 /** The actual method name the subscription is for. */
57 char *pszMethod;
58 /** A friendly name for the subscription. */
59 char *pszSubscriptionName;
60 /** The actual subscription UUID.
61 * Should not be changed. */
62 char *pszSubscriptionUUID;
63} g_aSENSEvents[] = {
64 { "Logon", "VBoxCredProv SENS Logon", "{561D0791-47C0-4BC3-87C0-CDC2621EA653}" },
65 { "Logoff", "VBoxCredProv SENS Logoff", "{12B618B1-F2E0-4390-BADA-7EB1DC31A70A}" },
66 { "StartShell", "VBoxCredProv SENS StartShell", "{5941931D-015A-4F91-98DA-81AAE262D090}" },
67 { "DisplayLock", "VBoxCredProv SENS DisplayLock", "{B7E2C510-501A-4961-938F-A458970930D7}" },
68 { "DisplayUnlock", "VBoxCredProv SENS DisplayUnlock", "{11305987-8FFC-41AD-A264-991BD5B7488A}" },
69 { "StartScreenSaver", "VBoxCredProv SENS StartScreenSaver", "{6E2D26DF-0095-4EC4-AE00-2395F09AF7F2}" },
70 { "StopScreenSaver", "VBoxCredProv SENS StopScreenSaver", "{F53426BC-412F-41E8-9A5F-E5FA8A164BD6}" }
71};
72
73/**
74 * Implementation of the ISensLogon interface for getting
75 * SENS (System Event Notification Service) events. SENS must be up
76 * and running on this OS!
77 */
78interface VBoxCredProvSensLogon : public ISensLogon
79{
80public:
81
82 VBoxCredProvSensLogon(void) :
83 m_cRefs(1)
84 {
85 }
86
87 STDMETHODIMP QueryInterface(REFIID interfaceID, void **ppvInterface)
88 {
89 if ( IsEqualIID(interfaceID, IID_IUnknown)
90 || IsEqualIID(interfaceID, IID_IDispatch)
91 || IsEqualIID(interfaceID, IID_ISensLogon))
92 {
93 *ppvInterface = this;
94 AddRef();
95 return S_OK;
96 }
97
98 *ppvInterface = NULL;
99 return E_NOINTERFACE;
100 }
101
102 ULONG STDMETHODCALLTYPE AddRef(void)
103 {
104 return InterlockedIncrement(&m_cRefs);
105 }
106
107 ULONG STDMETHODCALLTYPE Release(void)
108 {
109 ULONG ulTemp = InterlockedDecrement(&m_cRefs);
110 return ulTemp;
111 }
112
113 HRESULT STDMETHODCALLTYPE GetTypeInfoCount(unsigned int FAR* pctinfo)
114 {
115 return E_NOTIMPL;
116 }
117
118 HRESULT STDMETHODCALLTYPE GetTypeInfo(unsigned int iTInfo, LCID lcid, ITypeInfo FAR* FAR* ppTInfo)
119 {
120 return E_NOTIMPL;
121 }
122
123 HRESULT STDMETHODCALLTYPE GetIDsOfNames(REFIID riid,
124 OLECHAR FAR* FAR* rgszNames, unsigned int cNames,
125 LCID lcid, DISPID FAR* rgDispId)
126 {
127 return E_NOTIMPL;
128 }
129
130 HRESULT STDMETHODCALLTYPE Invoke(DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags,
131 DISPPARAMS FAR* pDispParams, VARIANT FAR* parResult, EXCEPINFO FAR* pExcepInfo,
132 unsigned int FAR* puArgErr)
133 {
134 return E_NOTIMPL;
135 }
136
137 /* ISensLogon methods */
138 STDMETHODIMP Logon(BSTR bstrUserName)
139 {
140 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: Logon\n");
141 return S_OK;
142 }
143
144 STDMETHODIMP Logoff(BSTR bstrUserName)
145 {
146 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: Logoff\n");
147 return S_OK;
148 }
149
150 STDMETHODIMP StartShell(BSTR bstrUserName)
151 {
152 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: Logon\n");
153 return S_OK;
154 }
155
156 STDMETHODIMP DisplayLock(BSTR bstrUserName)
157 {
158 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: DisplayLock\n");
159 return S_OK;
160 }
161
162 STDMETHODIMP DisplayUnlock(BSTR bstrUserName)
163 {
164 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: DisplayUnlock\n");
165 return S_OK;
166 }
167
168 STDMETHODIMP StartScreenSaver(BSTR bstrUserName)
169 {
170 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: StartScreenSaver\n");
171 return S_OK;
172 }
173
174 STDMETHODIMP StopScreenSaver(BSTR bstrUserName)
175 {
176 VBoxCredProvVerbose(0, "VBoxCredProvSensLogon: StopScreenSaver\n");
177 return S_OK;
178 }
179
180protected:
181
182 LONG m_cRefs;
183};
184static VBoxCredProvSensLogon *g_pISensLogon = NULL;
185
186
187/**
188 * Register events to be called by SENS.
189 *
190 * @return HRESULT
191 */
192static HRESULT VBoxCredentialProviderRegisterSENS(void)
193{
194 VBoxCredProvVerbose(0, "VBoxCredentialProviderRegisterSENS\n");
195
196 HRESULT hr = CoCreateInstance(CLSID_CEventSystem, 0, CLSCTX_SERVER, IID_IEventSystem, (void**)&g_pIEventSystem);
197 if (FAILED(hr))
198 {
199 VBoxCredProvVerbose(0, "VBoxCredentialProviderRegisterSENS: Could not connect to CEventSystem, hr=%Rhrc\n",
200 hr);
201 return hr;
202 }
203
204 try
205 {
206 g_pISensLogon = new VBoxCredProvSensLogon();
207 AssertPtr(g_pISensLogon);
208 }
209 catch (std::bad_alloc &ex)
210 {
211 NOREF(ex);
212 hr = E_OUTOFMEMORY;
213 }
214
215 if ( SUCCEEDED(hr)
216 && g_pIEventSystem)
217 {
218 IEventSubscription *pIEventSubscription;
219 int i;
220 for (i = 0; i < RT_ELEMENTS(g_aSENSEvents); i++)
221 {
222 VBoxCredProvVerbose(0, "VBoxCredProv: Registering \"%s\" (%s) ...\n",
223 g_aSENSEvents[i].pszMethod, g_aSENSEvents[i].pszSubscriptionName);
224
225 hr = CoCreateInstance(CLSID_CEventSubscription, 0, CLSCTX_SERVER, IID_IEventSubscription, (LPVOID*)&pIEventSubscription);
226 if (FAILED(hr))
227 continue;
228
229 hr = pIEventSubscription->put_EventClassID(L"{d5978630-5b9f-11d1-8dd2-00aa004abd5e}" /* SENSGUID_EVENTCLASS_LOGON */);
230 if (FAILED(hr))
231 break;
232
233 hr = pIEventSubscription->put_SubscriberInterface((IUnknown*)g_pISensLogon);
234 if (FAILED(hr))
235 break;
236
237 PRTUTF16 pwszTemp;
238 int rc = RTStrToUtf16(g_aSENSEvents[i].pszMethod, &pwszTemp);
239 if (RT_SUCCESS(rc))
240 {
241 hr = pIEventSubscription->put_MethodName(pwszTemp);
242 RTUtf16Free(pwszTemp);
243 }
244 else
245 hr = ERROR_OUTOFMEMORY;
246 if (FAILED(hr))
247 break;
248
249 rc = RTStrToUtf16(g_aSENSEvents[i].pszSubscriptionName, &pwszTemp);
250 if (RT_SUCCESS(rc))
251 {
252 hr = pIEventSubscription->put_SubscriptionName(pwszTemp);
253 RTUtf16Free(pwszTemp);
254 }
255 else
256 hr = ERROR_OUTOFMEMORY;
257 if (FAILED(hr))
258 break;
259
260 rc = RTStrToUtf16(g_aSENSEvents[i].pszSubscriptionUUID, &pwszTemp);
261 if (RT_SUCCESS(rc))
262 {
263 hr = pIEventSubscription->put_SubscriptionID(pwszTemp);
264 RTUtf16Free(pwszTemp);
265 }
266 else
267 hr = ERROR_OUTOFMEMORY;
268 if (FAILED(hr))
269 break;
270
271 hr = pIEventSubscription->put_PerUser(TRUE);
272 if (FAILED(hr))
273 break;
274
275 hr = g_pIEventSystem->Store(PROGID_EventSubscription, (IUnknown*)pIEventSubscription);
276 if (FAILED(hr))
277 break;
278
279 pIEventSubscription->Release();
280 pIEventSubscription = NULL;
281 }
282
283 if (FAILED(hr))
284 VBoxCredProvVerbose(0, "VBoxCredentialProviderRegisterSENS: Could not register \"%s\" (%s), hr=%Rhrc\n",
285 g_aSENSEvents[i].pszMethod, g_aSENSEvents[i].pszSubscriptionName, hr);
286
287 if (pIEventSubscription != NULL)
288 pIEventSubscription->Release();
289 }
290
291 if (FAILED(hr))
292 {
293 VBoxCredProvVerbose(0, "VBoxCredentialProviderRegisterSENS: Error registering SENS provider, hr=%Rhrc\n", hr);
294 if (g_pIEventSystem)
295 {
296 g_pIEventSystem->Release();
297 g_pIEventSystem = NULL;
298 }
299 }
300
301 VBoxCredProvVerbose(0, "VBoxCredentialProviderRegisterSENS: Returning hr=%Rhrc\n", hr);
302 return hr;
303}
304
305/**
306 * Unregisters registered SENS events.
307 */
308static void VBoxCredentialProviderUnregisterSENS(void)
309{
310 if (g_pIEventSystem)
311 {
312 g_pIEventSystem->Release();
313 g_pIEventSystem = NULL;
314 }
315
316 /* We need to reconnecto to the event system because we can be called
317 * in a different context COM can't handle. */
318 HRESULT hr = CoCreateInstance(CLSID_CEventSystem, 0,
319 CLSCTX_SERVER, IID_IEventSystem, (void**)&g_pIEventSystem);
320 if ( SUCCEEDED(hr)
321 && g_pIEventSystem)
322 {
323 VBoxCredProvVerbose(0, "VBoxCredentialProviderUnregisterSENS\n");
324
325 HRESULT hr;
326
327 for (int i = 0; i < RT_ELEMENTS(g_aSENSEvents); i++)
328 {
329 int iErrorIdX;
330
331 char *pszSubToRemove;
332 if (!RTStrAPrintf(&pszSubToRemove, "SubscriptionID=%s",
333 g_aSENSEvents[i].pszSubscriptionUUID))
334 {
335 continue; /* Keep going. */
336 }
337
338 PRTUTF16 pwszTemp;
339 int rc2 = RTStrToUtf16(pszSubToRemove, &pwszTemp);
340 if (RT_SUCCESS(rc2))
341 {
342 hr = g_pIEventSystem->Remove(PROGID_EventSubscription, pwszTemp,
343 &iErrorIdX);
344 RTUtf16Free(pwszTemp);
345 }
346 else
347 hr = ERROR_OUTOFMEMORY;
348
349 if (FAILED(hr))
350 VBoxCredProvVerbose(0, "VBoxCredentialProviderUnregisterSENS: Could not unregister \"%s\" (query: %s), hr=%Rhrc (index: %d)\n",
351 g_aSENSEvents[i].pszMethod, pszSubToRemove, hr, iErrorIdX);
352 /* Keep going. */
353
354 RTStrFree(pszSubToRemove);
355 }
356
357 g_pIEventSystem->Release();
358 g_pIEventSystem = NULL;
359 }
360
361 if (g_pISensLogon)
362 {
363 delete g_pISensLogon;
364 g_pISensLogon = NULL;
365 }
366
367 VBoxCredProvVerbose(0, "VBoxCredentialProviderUnregisterSENS: Returning hr=%Rhrc\n", hr);
368}
369#endif /* VBOX_WITH_SENS */
370
371
372BOOL WINAPI DllMain(HINSTANCE hInst, DWORD dwReason, LPVOID pReserved)
373{
374 NOREF(pReserved);
375
376 g_hDllInst = hInst;
377
378 switch (dwReason)
379 {
380 case DLL_PROCESS_ATTACH:
381 {
382 int rc = RTR3InitDll(RTR3INIT_FLAGS_UNOBTRUSIVE);
383 if (RT_SUCCESS(rc))
384 rc = VbglR3Init();
385
386 if (RT_SUCCESS(rc))
387 {
388 VBoxCredProvVerbose(0, "VBoxCredProv: v%s r%s (%s %s) loaded (refs=%ld)\n",
389 RTBldCfgVersion(), RTBldCfgRevisionStr(),
390 __DATE__, __TIME__, g_cDllRefs);
391 }
392
393 DisableThreadLibraryCalls(hInst);
394 break;
395 }
396
397 case DLL_PROCESS_DETACH:
398
399 VBoxCredProvVerbose(0, "VBoxCredProv: Unloaded (refs=%ld)\n", g_cDllRefs);
400 if (!g_cDllRefs)
401 VbglR3Term();
402 break;
403
404 case DLL_THREAD_ATTACH:
405 case DLL_THREAD_DETACH:
406 break;
407
408 default:
409 break;
410 }
411
412 return TRUE;
413}
414
415
416/**
417 * Increments the reference count by one. Must be released
418 * with VBoxCredentialProviderRelease() when finished.
419 */
420void VBoxCredentialProviderAcquire(void)
421{
422 LONG cRefCount = InterlockedIncrement(&g_cDllRefs);
423 VBoxCredProvVerbose(0, "VBoxCredentialProviderAcquire: Increasing global refcount to %ld\n",
424 cRefCount);
425}
426
427
428/**
429 * Decrements the reference count by one.
430 */
431void VBoxCredentialProviderRelease(void)
432{
433 LONG cRefCount = InterlockedDecrement(&g_cDllRefs);
434 VBoxCredProvVerbose(0, "VBoxCredentialProviderRelease: Decreasing global refcount to %ld\n",
435 cRefCount);
436}
437
438
439/**
440 * Returns the current DLL reference count.
441 *
442 * @return LONG The current reference count.
443 */
444LONG VBoxCredentialProviderRefCount(void)
445{
446 return g_cDllRefs;
447}
448
449
450/**
451 * Entry point for determining whether the credential
452 * provider DLL can be unloaded or not.
453 *
454 * @return HRESULT
455 */
456HRESULT __stdcall DllCanUnloadNow(void)
457{
458 VBoxCredProvVerbose(0, "DllCanUnloadNow (refs=%ld)\n",
459 g_cDllRefs);
460
461#ifdef VBOX_WITH_SENS
462 if (!g_cDllRefs)
463 {
464 if (g_fSENSEnabled)
465 VBoxCredentialProviderUnregisterSENS();
466
467 CoUninitialize();
468 }
469#endif
470 return (g_cDllRefs > 0) ? S_FALSE : S_OK;
471}
472
473
474/**
475 * Create the VirtualBox credential provider by creating
476 * its factory which then in turn can create instances of the
477 * provider itself.
478 *
479 * @return HRESULT
480 * @param classID The class ID.
481 * @param interfaceID The interface ID.
482 * @param ppvInterface Receives the interface pointer on successful
483 * object creation.
484 */
485HRESULT VBoxCredentialProviderCreate(REFCLSID classID, REFIID interfaceID,
486 void **ppvInterface)
487{
488 HRESULT hr;
489 if (classID == CLSID_VBoxCredProvider)
490 {
491 try
492 {
493 VBoxCredProvFactory* pFactory = new VBoxCredProvFactory();
494 AssertPtr(pFactory);
495 hr = pFactory->QueryInterface(interfaceID,
496 ppvInterface);
497 pFactory->Release();
498
499#ifdef VBOX_WITH_SENS
500 g_fSENSEnabled = true; /* By default SENS support is enabled. */
501
502 HKEY hKey;
503 /** @todo Add some registry wrapper function(s) as soon as we got more values to retrieve. */
504 DWORD dwRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Oracle\\VirtualBox Guest Additions\\AutoLogon",
505 0L, KEY_QUERY_VALUE, &hKey);
506 if (dwRet == ERROR_SUCCESS)
507 {
508 DWORD dwValue;
509 DWORD dwType = REG_DWORD;
510 DWORD dwSize = sizeof(DWORD);
511
512 dwRet = RegQueryValueEx(hKey, L"HandleSENS", NULL, &dwType, (LPBYTE)&dwValue, &dwSize);
513 if ( dwRet == ERROR_SUCCESS
514 && dwType == REG_DWORD
515 && dwSize == sizeof(DWORD))
516 {
517 g_fSENSEnabled = RT_BOOL(dwValue);
518 }
519
520 RegCloseKey(hKey);
521 }
522
523 VBoxCredProvVerbose(0, "VBoxCredentialProviderCreate: g_fSENSEnabled=%RTbool\n",
524 g_fSENSEnabled);
525 if ( SUCCEEDED(hr)
526 && g_fSENSEnabled)
527 {
528 HRESULT hRes = CoInitializeEx(NULL, COINIT_MULTITHREADED);
529 VBoxCredentialProviderRegisterSENS();
530 }
531#else
532 VBoxCredProvVerbose(0, "VBoxCredentialProviderCreate: SENS support is disabled\n");
533#endif
534 }
535 catch (std::bad_alloc &ex)
536 {
537 NOREF(ex);
538 hr = E_OUTOFMEMORY;
539 }
540 }
541 else
542 hr = CLASS_E_CLASSNOTAVAILABLE;
543
544 return hr;
545}
546
547
548/**
549 * Entry point for getting the actual credential provider
550 * class object.
551 *
552 * @return HRESULT
553 * @param classID The class ID.
554 * @param interfaceID The interface ID.
555 * @param ppvInterface Receives the interface pointer on successful
556 * object creation.
557 */
558HRESULT __stdcall DllGetClassObject(REFCLSID classID, REFIID interfaceID,
559 void **ppvInterface)
560{
561 VBoxCredProvVerbose(0, "DllGetClassObject (refs=%ld)\n",
562 g_cDllRefs);
563
564 return VBoxCredentialProviderCreate(classID, interfaceID, ppvInterface);
565}
566
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette