VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxGINA/Helper.cpp@ 28800

Last change on this file since 28800 was 28800, checked in by vboxsync, 15 years ago

Automated rebranding to Oracle copyright/license strings via filemuncher

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.1 KB
Line 
1/** @file
2 *
3 * VBoxGINA -- Windows Logon DLL for VirtualBox Helper Functions
4 *
5 * Copyright (C) 2006-2007 Oracle Corporation
6 *
7 * This file is part of VirtualBox Open Source Edition (OSE), as
8 * available from http://www.virtualbox.org. This file is free software;
9 * you can redistribute it and/or modify it under the terms of the GNU
10 * General Public License (GPL) as published by the Free Software
11 * Foundation, in version 2 as it comes in the "COPYING" file of the
12 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
13 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
14 */
15
16#include <windows.h>
17#include "winwlx.h"
18#include "Helper.h"
19#include "VBoxGINA.h"
20
21#include <VBox/VBoxGuest.h>
22#include <VBox/VMMDev.h>
23#include <iprt/string.h>
24
25/* the credentials */
26wchar_t g_Username[VMMDEV_CREDENTIALS_STRLEN];
27wchar_t g_Password[VMMDEV_CREDENTIALS_STRLEN];
28wchar_t g_Domain[VMMDEV_CREDENTIALS_STRLEN];
29
30
31HANDLE getVBoxDriver(void)
32{
33 static HANDLE sVBoxDriver = INVALID_HANDLE_VALUE;
34 if (sVBoxDriver == INVALID_HANDLE_VALUE)
35 {
36 sVBoxDriver = CreateFile(L"\\\\.\\VBoxGuest", /** @todo use define */
37 GENERIC_READ | GENERIC_WRITE,
38 FILE_SHARE_READ | FILE_SHARE_WRITE,
39 NULL,
40 OPEN_EXISTING,
41 FILE_ATTRIBUTE_NORMAL,
42 NULL);
43 if (sVBoxDriver == INVALID_HANDLE_VALUE)
44 Log(("VBoxGINA::sVBoxDriver: failed to open VBoxGuest driver, last error = %d\n", GetLastError()));
45 }
46 return sVBoxDriver;
47}
48
49void credentialsReset(void)
50{
51 RT_ZERO(g_Username);
52 RT_ZERO(g_Password);
53 RT_ZERO(g_Domain);
54}
55
56bool credentialsAvailable(void)
57{
58 HANDLE vboxDriver = getVBoxDriver();
59 if (vboxDriver == INVALID_HANDLE_VALUE)
60 return false;
61
62 /* query the VMMDev whether there are credentials */
63 VMMDevCredentials vmmreqCredentials = {0};
64 vmmdevInitRequest((VMMDevRequestHeader*)&vmmreqCredentials, VMMDevReq_QueryCredentials);
65 vmmreqCredentials.u32Flags |= VMMDEV_CREDENTIALS_QUERYPRESENCE;
66 DWORD cbReturned;
67 if (!DeviceIoControl(vboxDriver, VBOXGUEST_IOCTL_VMMREQUEST(sizeof(vmmreqCredentials)), &vmmreqCredentials, sizeof(vmmreqCredentials),
68 &vmmreqCredentials, sizeof(vmmreqCredentials), &cbReturned, NULL))
69 {
70 Log(("VBoxGINA::credentialsAvailable: error doing IOCTL, last error: %d\n", GetLastError()));
71 return false;
72 }
73 bool fAvailable = ((vmmreqCredentials.u32Flags & VMMDEV_CREDENTIALS_PRESENT) != 0);
74 /*Log(("VBoxGINA::credentialsAvailable: fAvailable: %d\n", fAvailable));*/
75 return fAvailable;
76}
77
78bool credentialsRetrieve(void)
79{
80 Log(("VBoxGINA::credentialsRetrieve\n"));
81
82 HANDLE vboxDriver = getVBoxDriver();
83 if (vboxDriver == INVALID_HANDLE_VALUE)
84 return false;
85
86 /* to be safe, reset the credentials */
87 credentialsReset();
88
89 /* query the credentials */
90 VMMDevCredentials vmmreqCredentials = {0};
91 vmmdevInitRequest((VMMDevRequestHeader*)&vmmreqCredentials, VMMDevReq_QueryCredentials);
92 vmmreqCredentials.u32Flags |= VMMDEV_CREDENTIALS_READ;
93 vmmreqCredentials.u32Flags |= VMMDEV_CREDENTIALS_CLEAR;
94 DWORD cbReturned;
95 if (!DeviceIoControl(vboxDriver, VBOXGUEST_IOCTL_VMMREQUEST(sizeof(vmmreqCredentials)), &vmmreqCredentials, sizeof(vmmreqCredentials),
96 &vmmreqCredentials, sizeof(vmmreqCredentials), &cbReturned, NULL))
97 {
98 Log(("VBoxGINA::credentialsRetrieve: error doing IOCTL, last error: %d\n", GetLastError()));
99 return false;
100 }
101 /* convert from UTF-8 to UTF-16 and store in global variables */
102 PRTUTF16 ptr = NULL;
103 if (RT_SUCCESS(RTStrToUtf16(vmmreqCredentials.szUserName, &ptr)) && ptr)
104 {
105 wcscpy(g_Username, ptr);
106 RTUtf16Free(ptr);
107 }
108 ptr = NULL;
109 if (RT_SUCCESS(RTStrToUtf16(vmmreqCredentials.szPassword, &ptr)) && ptr)
110 {
111 wcscpy(g_Password, ptr);
112 RTUtf16Free(ptr);
113 }
114 ptr = NULL;
115 if (RT_SUCCESS(RTStrToUtf16(vmmreqCredentials.szDomain, &ptr)) && ptr)
116 {
117 wcscpy(g_Domain, ptr);
118 RTUtf16Free(ptr);
119 }
120 Log(("VBoxGINA::credentialsRetrieve: returning user '%s', password '%s', domain '%s'\n",
121 vmmreqCredentials.szUserName, vmmreqCredentials.szPassword, vmmreqCredentials.szDomain));
122 return true;
123}
124
125/* handle of the poller thread */
126RTTHREAD gThreadPoller = NIL_RTTHREAD;
127
128
129/**
130 * Poller thread. Checks periodically whether there are credentials.
131 */
132static DECLCALLBACK(int) credentialsPoller(RTTHREAD ThreadSelf, void *pvUser)
133{
134 Log(("VBoxGINA::credentialsPoller\n"));
135
136 do
137 {
138 if (credentialsAvailable())
139 {
140 Log(("VBoxGINA::credentialsPoller: got credentials, simulating C-A-D\n"));
141 /* tell WinLogon to start the attestation process */
142 pWlxFuncs->WlxSasNotify(hGinaWlx, WLX_SAS_TYPE_CTRL_ALT_DEL);
143 /* time to say goodbye */
144 return 0;
145 }
146 /* wait a bit */
147 if (RTThreadUserWait(ThreadSelf, 500) == VINF_SUCCESS)
148 {
149 Log(("VBoxGINA::credentialsPoller: we were asked to terminate\n"));
150 /* we were asked to terminate, do that instantly! */
151 return 0;
152 }
153 }
154 while (1);
155
156 return 0;
157}
158
159bool credentialsPollerCreate(void)
160{
161 Log(("VBoxGINA::credentialsPollerCreate\n"));
162
163 /* don't create more than one of them */
164 if (gThreadPoller != NIL_RTTHREAD)
165 {
166 Log(("VBoxGINA::credentialsPollerCreate: thread already running, returning!\n"));
167 return false;
168 }
169
170 /* create the poller thread */
171 int rc = RTThreadCreate(&gThreadPoller, credentialsPoller, NULL, 0, RTTHREADTYPE_INFREQUENT_POLLER,
172 RTTHREADFLAGS_WAITABLE, "creds");
173 if (RT_FAILURE(rc))
174 {
175 Log(("VBoxGINA::credentialsPollerCreate: failed to create thread, rc = %Rrc\n", rc));
176 return false;
177 }
178 return true;
179}
180
181bool credentialsPollerTerminate(void)
182{
183 Log(("VBoxGINA::credentialsPollerTerminate\n"));
184
185 if (gThreadPoller == NIL_RTTHREAD)
186 {
187 Log(("VBoxGINA::credentialsPollerTerminate: either thread or exit sem is NULL!\n"));
188 return false;
189 }
190 /* post termination event semaphore */
191 int rc = RTThreadUserSignal(gThreadPoller);
192 if (RT_SUCCESS(rc))
193 {
194 Log(("VBoxGINA::credentialsPollerTerminate: waiting for thread to terminate\n"));
195 /* wait until the thread has terminated */
196 rc = RTThreadWait(gThreadPoller, RT_INDEFINITE_WAIT, NULL);
197 Log(("VBoxGINA::credentialsPollerTermiante: thread has (probably) terminated (rc = %Rrc)\n", rc));
198 }
199 else
200 {
201 /* failed to signal the thread - very unlikely - so no point in waiting long. */
202 Log(("VBoxGINA::credentialsPollerTermiante: failed to signal semaphore, rc = %Rrc\n", rc));
203 rc = RTThreadWait(gThreadPoller, 100, NULL);
204 Log(("VBoxGINA::credentialsPollerTermiante: thread has terminated? wait rc = %Rrc\n", rc));
205 }
206 /* now cleanup */
207 gThreadPoller = NIL_RTTHREAD;
208 return true;
209}
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