VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxCredProv/VBoxCredProvProvider.cpp@ 50035

Last change on this file since 50035 was 46382, checked in by vboxsync, 12 years ago

VBoxCredProv: Handle bad_alloc.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.6 KB
Line 
1/* $Id: VBoxCredProvProvider.cpp 46382 2013-06-04 13:32:38Z vboxsync $ */
2/** @file
3 * VBoxCredProvProvider - The actual credential provider class.
4 */
5
6/*
7 * Copyright (C) 2012-2013 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 <new> /* For bad_alloc. */
22
23#include <credentialprovider.h>
24
25#include <iprt/err.h>
26#include <VBox/VBoxGuestLib.h>
27
28#include "VBoxCredentialProvider.h"
29#include "VBoxCredProvProvider.h"
30#include "VBoxCredProvCredential.h"
31
32
33
34VBoxCredProvProvider::VBoxCredProvProvider(void) :
35 m_cRefs(1),
36 m_pPoller(NULL),
37 m_pCred(NULL),
38 m_pEvents(NULL),
39 m_fHandleRemoteSessions(false)
40{
41 VBoxCredentialProviderAcquire();
42
43 VBoxCredProvReportStatus(VBoxGuestFacilityStatus_Init);
44}
45
46
47VBoxCredProvProvider::~VBoxCredProvProvider(void)
48{
49 VBoxCredProvVerbose(0, "VBoxCredProv: Destroying\n");
50
51 if (m_pCred)
52 {
53 m_pCred->Release();
54 m_pCred = NULL;
55 }
56
57 if (m_pPoller)
58 {
59 m_pPoller->Shutdown();
60 delete m_pPoller;
61 m_pPoller = NULL;
62 }
63
64 VBoxCredProvReportStatus(VBoxGuestFacilityStatus_Terminated);
65
66 VBoxCredentialProviderRelease();
67}
68
69
70/* IUnknown overrides. */
71ULONG
72VBoxCredProvProvider::AddRef(void)
73{
74 LONG cRefs = InterlockedIncrement(&m_cRefs);
75 VBoxCredProvVerbose(0, "VBoxCredProv: AddRef: Returning refcount=%ld\n",
76 cRefs);
77 return cRefs;
78}
79
80
81ULONG
82VBoxCredProvProvider::Release(void)
83{
84 LONG cRefs = InterlockedDecrement(&m_cRefs);
85 VBoxCredProvVerbose(0, "VBoxCredProv: Release: Returning refcount=%ld\n",
86 cRefs);
87 if (!cRefs)
88 {
89 VBoxCredProvVerbose(0, "VBoxCredProv: Calling destructor\n");
90 delete this;
91 }
92 return cRefs;
93}
94
95
96HRESULT
97VBoxCredProvProvider::QueryInterface(REFIID interfaceID, void **ppvInterface)
98{
99 HRESULT hr = S_OK;
100 if (ppvInterface)
101 {
102 if ( IID_IUnknown == interfaceID
103 || IID_ICredentialProvider == interfaceID)
104 {
105 *ppvInterface = static_cast<IUnknown*>(this);
106 reinterpret_cast<IUnknown*>(*ppvInterface)->AddRef();
107 }
108 else
109 {
110 *ppvInterface = NULL;
111 hr = E_NOINTERFACE;
112 }
113 }
114 else
115 hr = E_INVALIDARG;
116
117 return hr;
118}
119
120
121/**
122 * Loads the global configuration from registry.
123 *
124 * @return DWORD Windows error code.
125 */
126DWORD
127VBoxCredProvProvider::LoadConfiguration(void)
128{
129 HKEY hKey;
130 /** @todo Add some registry wrapper function(s) as soon as we got more values to retrieve. */
131 DWORD dwRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Oracle\\VirtualBox Guest Additions\\AutoLogon",
132 0L, KEY_QUERY_VALUE, &hKey);
133 if (dwRet == ERROR_SUCCESS)
134 {
135 DWORD dwValue;
136 DWORD dwType = REG_DWORD;
137 DWORD dwSize = sizeof(DWORD);
138
139 dwRet = RegQueryValueEx(hKey, L"HandleRemoteSessions", NULL, &dwType, (LPBYTE)&dwValue, &dwSize);
140 if ( dwRet == ERROR_SUCCESS
141 && dwType == REG_DWORD
142 && dwSize == sizeof(DWORD))
143 {
144 m_fHandleRemoteSessions = true;
145 }
146
147 dwRet = RegQueryValueEx(hKey, L"LoggingEnabled", NULL, &dwType, (LPBYTE)&dwValue, &dwSize);
148 if ( dwRet == ERROR_SUCCESS
149 && dwType == REG_DWORD
150 && dwSize == sizeof(DWORD))
151 {
152 g_dwVerbosity = 1; /* Default logging level. */
153 }
154
155 if (g_dwVerbosity) /* Do we want logging at all? */
156 {
157 dwRet = RegQueryValueEx(hKey, L"LoggingLevel", NULL, &dwType, (LPBYTE)&dwValue, &dwSize);
158 if ( dwRet == ERROR_SUCCESS
159 && dwType == REG_DWORD
160 && dwSize == sizeof(DWORD))
161 {
162 g_dwVerbosity = dwValue;
163 }
164 }
165
166 RegCloseKey(hKey);
167 }
168 /* Do not report back an error here yet. */
169 return ERROR_SUCCESS;
170}
171
172
173/**
174 * Determines whether we should handle the current session or not.
175 *
176 * @return bool true if we should handle this session, false if not.
177 */
178bool
179VBoxCredProvProvider::HandleCurrentSession(void)
180{
181 /* Load global configuration from registry. */
182 int rc = LoadConfiguration();
183 if (RT_FAILURE(rc))
184 VBoxCredProvVerbose(0, "VBoxCredProv: Error loading global configuration, rc=%Rrc\n",
185 rc);
186
187 bool fHandle = false;
188 if (VbglR3AutoLogonIsRemoteSession())
189 {
190 if (m_fHandleRemoteSessions) /* Force remote session handling. */
191 fHandle = true;
192 }
193 else /* No remote session. */
194 fHandle = true;
195
196 VBoxCredProvVerbose(3, "VBoxCredProv: Handling current session=%RTbool\n", fHandle);
197 return fHandle;
198}
199
200
201/**
202 * Tells this provider the current usage scenario.
203 *
204 * @return HRESULT
205 * @param enmUsageScenario Current usage scenario this provider will be
206 * used in.
207 * @param dwFlags Optional flags for the usage scenario.
208 */
209HRESULT
210VBoxCredProvProvider::SetUsageScenario(CREDENTIAL_PROVIDER_USAGE_SCENARIO enmUsageScenario, DWORD dwFlags)
211{
212 HRESULT hr = S_OK;
213 DWORD dwErr;
214
215 VBoxCredProvVerbose(0, "VBoxCredProv::SetUsageScenario: enmUsageScenario=%d, dwFlags=%ld\n",
216 enmUsageScenario, dwFlags);
217
218 m_enmUsageScenario = enmUsageScenario;
219
220 switch (m_enmUsageScenario)
221 {
222 case CPUS_LOGON:
223 case CPUS_UNLOCK_WORKSTATION:
224 {
225 VBoxCredProvReportStatus(VBoxGuestFacilityStatus_Active);
226
227 dwErr = LoadConfiguration();
228 if (dwErr != ERROR_SUCCESS)
229 VBoxCredProvVerbose(0, "VBoxCredProv: Error while loading configuration, error=%ld\n", dwErr);
230 /* Do not stop running on a misconfigured system. */
231
232 /*
233 * If we're told to not handle the current session just bail out and let the
234 * user know.
235 */
236 if (!HandleCurrentSession())
237 break;
238
239 if (!m_pPoller)
240 {
241 try
242 {
243 m_pPoller = new VBoxCredProvPoller();
244 AssertPtr(m_pPoller);
245 int rc = m_pPoller->Initialize(this);
246 if (RT_FAILURE(rc))
247 VBoxCredProvVerbose(0, "VBoxCredProv::SetUsageScenario: Error initializing poller thread, rc=%Rrc\n", rc);
248 }
249 catch (std::bad_alloc &ex)
250 {
251 NOREF(ex);
252 hr = E_OUTOFMEMORY;
253 }
254 }
255
256 if ( SUCCEEDED(hr)
257 && !m_pCred)
258 {
259 try
260 {
261 m_pCred = new VBoxCredProvCredential();
262 AssertPtr(m_pPoller);
263 hr = m_pCred->Initialize(m_enmUsageScenario);
264 }
265 catch (std::bad_alloc &ex)
266 {
267 NOREF(ex);
268 hr = E_OUTOFMEMORY;
269 }
270 }
271 else
272 {
273 /* All set up already! Nothing to do here right now. */
274 }
275
276 /* If we failed, do some cleanup. */
277 if (FAILED(hr))
278 {
279 if (m_pCred != NULL)
280 {
281 m_pCred->Release();
282 m_pCred = NULL;
283 }
284 }
285 break;
286 }
287
288 case CPUS_CHANGE_PASSWORD:
289 case CPUS_CREDUI:
290 case CPUS_PLAP:
291
292 hr = E_NOTIMPL;
293 break;
294
295 default:
296
297 hr = E_INVALIDARG;
298 break;
299 }
300
301 VBoxCredProvVerbose(0, "VBoxCredProv::SetUsageScenario returned hr=0x%08x\n", hr);
302 return hr;
303}
304
305
306/**
307 * Tells this provider how the serialization will be handled. Currently not used.
308 *
309 * @return STDMETHODIMP
310 * @param pcpCredentialSerialization Credentials serialization.
311 */
312STDMETHODIMP
313VBoxCredProvProvider::SetSerialization(const CREDENTIAL_PROVIDER_CREDENTIAL_SERIALIZATION *pcpCredentialSerialization)
314{
315 NOREF(pcpCredentialSerialization);
316 return E_NOTIMPL;
317}
318
319
320/**
321 * Initializes the communication with LogonUI through callbacks events which we can later
322 * use to start re-enumeration of credentials.
323 *
324 * @return HRESULT
325 * @param pcpEvents Pointer to event interface.
326 * @param upAdviseContext The current advise context.
327 */
328HRESULT
329VBoxCredProvProvider::Advise(ICredentialProviderEvents *pcpEvents, UINT_PTR upAdviseContext)
330{
331 VBoxCredProvVerbose(0, "VBoxCredProv::Advise, pcpEvents=0x%p, upAdviseContext=%u\n",
332 pcpEvents, upAdviseContext);
333 if (m_pEvents)
334 {
335 m_pEvents->Release();
336 m_pEvents = NULL;
337 }
338
339 m_pEvents = pcpEvents;
340 if (m_pEvents)
341 m_pEvents->AddRef();
342
343 /*
344 * Save advice context for later use when binding to
345 * certain ICredentialProviderEvents events.
346 */
347 m_upAdviseContext = upAdviseContext;
348 return S_OK;
349}
350
351
352/**
353 * Uninitializes the callback events so that they're no longer valid.
354 *
355 * @return HRESULT
356 */
357HRESULT
358VBoxCredProvProvider::UnAdvise(void)
359{
360 VBoxCredProvVerbose(0, "VBoxCredProv::UnAdvise: pEvents=0x%p\n",
361 m_pEvents);
362 if (m_pEvents)
363 {
364 m_pEvents->Release();
365 m_pEvents = NULL;
366 }
367
368 return S_OK;
369}
370
371
372/**
373 * Retrieves the total count of fields we're handling (needed for field enumeration
374 * through LogonUI).
375 *
376 * @return HRESULT
377 * @param pdwCount Receives total count of fields.
378 */
379HRESULT
380VBoxCredProvProvider::GetFieldDescriptorCount(DWORD *pdwCount)
381{
382 if (pdwCount)
383 {
384 *pdwCount = VBOXCREDPROV_NUM_FIELDS;
385 VBoxCredProvVerbose(0, "VBoxCredProv::GetFieldDescriptorCount: %ld\n", *pdwCount);
386 }
387 return S_OK;
388}
389
390
391/**
392 * Retrieves a descriptor of a specified field.
393 *
394 * @return HRESULT
395 * @param dwIndex ID of field to retrieve descriptor for.
396 * @param ppFieldDescriptor Pointer which receives the allocated field
397 * descriptor.
398 */
399HRESULT
400VBoxCredProvProvider::GetFieldDescriptorAt(DWORD dwIndex, CREDENTIAL_PROVIDER_FIELD_DESCRIPTOR **ppFieldDescriptor)
401{
402 HRESULT hr = S_OK;
403 if ( dwIndex < VBOXCREDPROV_NUM_FIELDS
404 && ppFieldDescriptor)
405 {
406 PCREDENTIAL_PROVIDER_FIELD_DESCRIPTOR pcpFieldDesc =
407 (PCREDENTIAL_PROVIDER_FIELD_DESCRIPTOR)CoTaskMemAlloc(sizeof(CREDENTIAL_PROVIDER_FIELD_DESCRIPTOR));
408
409 if (pcpFieldDesc)
410 {
411 const VBOXCREDPROV_FIELD &field = s_VBoxCredProvFields[dwIndex];
412
413 RT_BZERO(pcpFieldDesc, sizeof(CREDENTIAL_PROVIDER_FIELD_DESCRIPTOR));
414
415 pcpFieldDesc->dwFieldID = field.desc.dwFieldID;
416 pcpFieldDesc->cpft = field.desc.cpft;
417 if (field.desc.pszLabel)
418 hr = SHStrDupW(field.desc.pszLabel, &pcpFieldDesc->pszLabel);
419 }
420 else
421 hr = E_OUTOFMEMORY;
422
423 if (SUCCEEDED(hr))
424 *ppFieldDescriptor = pcpFieldDesc;
425 else
426 CoTaskMemFree(pcpFieldDesc);
427 }
428 else
429 hr = E_INVALIDARG;
430
431 VBoxCredProvVerbose(0, "VBoxCredProv::GetFieldDescriptorAt: dwIndex=%ld, ppDesc=0x%p, hr=0x%08x\n",
432 dwIndex, ppFieldDescriptor, hr);
433 return hr;
434}
435
436
437/**
438 * Retrieves the total number of credentials this provider can offer at the current time and
439 * if a logon attempt should be made.
440 *
441 * @return HRESULT
442 * @param pdwCount Receives number of credentials to serve.
443 * @param pdwDefault Receives the credentials index to try
444 * logging on if there is more than one
445 * credential provided. 0 is default.
446 * @param pfAutoLogonWithDefault Receives a flag indicating whether a
447 * logon attempt using the default
448 * credential should be made or not.
449 */
450HRESULT
451VBoxCredProvProvider::GetCredentialCount(DWORD *pdwCount, DWORD *pdwDefault, BOOL *pfAutoLogonWithDefault)
452{
453 AssertPtr(pdwCount);
454 AssertPtr(pdwDefault);
455 AssertPtr(pfAutoLogonWithDefault);
456
457 bool fHasCredentials = false;
458
459 /* Do we have credentials? */
460 if (m_pCred)
461 {
462 int rc = m_pCred->RetrieveCredentials();
463 fHasCredentials = rc == VINF_SUCCESS;
464 }
465
466 if (fHasCredentials)
467 {
468 *pdwCount = 1; /* This provider always has the same number of credentials (1). */
469 *pdwDefault = 0; /* The credential we provide is *always* at index 0! */
470 *pfAutoLogonWithDefault = TRUE; /* We always at least try to auto-login (if password is correct). */
471 }
472 else
473 {
474 *pdwCount = 0;
475 *pdwDefault = CREDENTIAL_PROVIDER_NO_DEFAULT;
476 *pfAutoLogonWithDefault = FALSE;
477 }
478
479 VBoxCredProvVerbose(0, "VBoxCredProv::GetCredentialCount: *pdwCount=%ld, *pdwDefault=%ld, *pfAutoLogonWithDefault=%s\n",
480 *pdwCount, *pdwDefault, *pfAutoLogonWithDefault ? "true" : "false");
481 return S_OK;
482}
483
484
485/**
486 * Called by Winlogon to retrieve the interface of our current ICredentialProviderCredential interface.
487 *
488 * @return HRESULT
489 * @param dwIndex Index of credential (in case there is more than one credential at a time) to
490 * retrieve the interface for.
491 * @param ppCredProvCredential Pointer that receives the credential interface.
492 */
493HRESULT
494VBoxCredProvProvider::GetCredentialAt(DWORD dwIndex, ICredentialProviderCredential **ppCredProvCredential)
495{
496 VBoxCredProvVerbose(0, "VBoxCredProv::GetCredentialAt: Index=%ld, ppCredProvCredential=0x%p\n",
497 dwIndex, ppCredProvCredential);
498 if (!m_pCred)
499 {
500 VBoxCredProvVerbose(0, "VBoxCredProv::GetCredentialAt: No credentials available\n");
501 return E_INVALIDARG;
502 }
503
504 HRESULT hr;
505 if ( dwIndex == 0
506 && ppCredProvCredential)
507 {
508 hr = m_pCred->QueryInterface(IID_ICredentialProviderCredential,
509 reinterpret_cast<void**>(ppCredProvCredential));
510 }
511 else
512 {
513 VBoxCredProvVerbose(0, "VBoxCredProv::GetCredentialAt: More than one credential not supported!\n");
514 hr = E_INVALIDARG;
515 }
516 return hr;
517}
518
519
520/**
521 * Triggers a credential re-enumeration -- will be called by our poller thread. This then invokes
522 * GetCredentialCount() and GetCredentialAt() called by Winlogon.
523 */
524void
525VBoxCredProvProvider::OnCredentialsProvided(void)
526{
527 VBoxCredProvVerbose(0, "VBoxCredProv::OnCredentialsProvided\n");
528
529 if (m_pEvents)
530 m_pEvents->CredentialsChanged(m_upAdviseContext);
531}
532
533
534/**
535 * Creates our provider. This happens *before* CTRL-ALT-DEL was pressed!
536 */
537HRESULT
538VBoxCredProvProviderCreate(REFIID interfaceID, void **ppvInterface)
539{
540 HRESULT hr;
541
542 try
543 {
544 VBoxCredProvProvider *pProvider = new VBoxCredProvProvider();
545 AssertPtr(pProvider);
546 hr = pProvider->QueryInterface(interfaceID, ppvInterface);
547 pProvider->Release();
548 }
549 catch (std::bad_alloc &ex)
550 {
551 NOREF(ex);
552 hr = E_OUTOFMEMORY;
553 }
554
555 return hr;
556}
557
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