VirtualBox

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

Last change on this file since 1959 was 1, checked in by vboxsync, 55 years ago

import

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