VirtualBox

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

Last change on this file since 57358 was 57358, checked in by vboxsync, 9 years ago

*: scm cleanup run.

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