VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/ClientTokenHolder.cpp@ 47848

Last change on this file since 47848 was 47561, checked in by vboxsync, 11 years ago

Main/VirtualBox+Machine+Session: separate out the client death detection functionality into separate objects

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.9 KB
Line 
1/** @file
2 *
3 * VirtualBox API client token holder (in the client process)
4 */
5
6/*
7 * Copyright (C) 2006-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#include <iprt/asm.h>
19#include <iprt/assert.h>
20#include <iprt/log.h>
21#include <iprt/semaphore.h>
22#include <iprt/process.h>
23
24#ifdef VBOX_WITH_SYS_V_IPC_SESSION_WATCHER
25# include <errno.h>
26# include <sys/types.h>
27# include <sys/stat.h>
28# include <sys/ipc.h>
29# include <sys/sem.h>
30#endif
31
32#include <VBox/com/defs.h>
33
34#include "ClientTokenHolder.h"
35#include "SessionImpl.h"
36
37
38#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
39/** client token holder thread */
40static DECLCALLBACK(int) ClientTokenHolderThread(RTTHREAD Thread, void *pvUser);
41#endif
42
43
44Session::ClientTokenHolder::ClientTokenHolder()
45{
46 AssertReleaseFailed();
47}
48
49Session::ClientTokenHolder::~ClientTokenHolder()
50{
51 /* release the client token */
52#if defined(RT_OS_WINDOWS)
53
54 if (mSem && mThreadSem)
55 {
56 /*
57 * tell the thread holding the token to release it;
58 * it will close mSem handle
59 */
60 ::SetEvent(mSem);
61 /* wait for the thread to finish */
62 ::WaitForSingleObject(mThreadSem, INFINITE);
63 ::CloseHandle(mThreadSem);
64
65 mThreadSem = NULL;
66 mSem = NULL;
67 mThread = NIL_RTTHREAD;
68 }
69
70#elif defined(RT_OS_OS2)
71
72 if (mThread != NIL_RTTHREAD)
73 {
74 Assert(mSem != NIL_RTSEMEVENT);
75
76 /* tell the thread holding the token to release it */
77 int vrc = RTSemEventSignal(mSem);
78 AssertRC(vrc == NO_ERROR);
79
80 /* wait for the thread to finish */
81 vrc = RTThreadUserWait(mThread, RT_INDEFINITE_WAIT);
82 Assert(RT_SUCCESS(vrc) || vrc == VERR_INTERRUPTED);
83
84 mThread = NIL_RTTHREAD;
85 }
86
87 if (mSem != NIL_RTSEMEVENT)
88 {
89 RTSemEventDestroy(mSem);
90 mSem = NIL_RTSEMEVENT;
91 }
92
93#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
94
95 if (mSem >= 0)
96 {
97 ::sembuf sop = { 0, 1, SEM_UNDO };
98 ::semop(mSem, &sop, 1);
99
100 mSem = -1;
101 }
102
103#else
104# error "Port me!"
105#endif
106}
107
108Session::ClientTokenHolder::ClientTokenHolder(const Utf8Str &strTokenId) :
109 mClientTokenId(strTokenId)
110{
111 mSem = CTHSEMARG;
112 mThread = NIL_RTTHREAD;
113
114#if defined(RT_OS_WINDOWS)
115 mThreadSem = CTHTHREADSEMARG;
116
117 Bstr bstrTokenId(strTokenId);
118
119 /*
120 * Since there is no guarantee that the constructor and destructor will be
121 * called in the same thread, we need a separate thread to hold the token.
122 */
123
124 mThreadSem = ::CreateEvent(NULL, FALSE, FALSE, NULL);
125 AssertMsgReturnVoid(mThreadSem,
126 ("Cannot create an event sem, err=%d", ::GetLastError()));
127
128 void *data[3];
129 data[0] = (void*)(BSTR)bstrTokenId.raw();
130 data[1] = (void*)mThreadSem;
131 data[2] = 0; /* will get an output from the thread */
132
133 /* create a thread to hold the token until signalled to release it */
134 int vrc = RTThreadCreate(&mThread, ClientTokenHolderThread, (void*)data, 0, RTTHREADTYPE_MAIN_WORKER, 0, "IPCHolder");
135 AssertRCReturnVoid(vrc);
136
137 /* wait until thread init is completed */
138 DWORD wrc = ::WaitForSingleObject(mThreadSem, INFINITE);
139 AssertMsg(wrc == WAIT_OBJECT_0, ("Wait failed, err=%d\n", ::GetLastError()));
140 Assert(data[2]);
141
142 if (wrc == WAIT_OBJECT_0 && data[2])
143 {
144 /* memorize the event sem we should signal in close() */
145 mSem = (HANDLE)data[2];
146 }
147 else
148 {
149 ::CloseHandle(mThreadSem);
150 mThreadSem = NULL;
151 }
152#elif defined(RT_OS_OS2)
153 Bstr bstrTokenId(strTokenId);
154
155 /*
156 * Since there is no guarantee that the constructor and destructor will be
157 * called in the same thread, we need a separate thread to hold the token.
158 */
159
160 int vrc = RTSemEventCreate(&mSem);
161 AssertRCReturnVoid(vrc);
162
163 void *data[3];
164 data[0] = (void*)bstrTokenId.raw();
165 data[1] = (void*)mSem;
166 data[2] = (void*)false; /* will get the thread result here */
167
168 /* create a thread to hold the token until signalled to release it */
169 vrc = RTThreadCreate(&mThread, ClientTokenHolderThread, (void *) data,
170 0, RTTHREADTYPE_MAIN_WORKER, 0, "IPCHolder");
171 AssertRCReturnVoid(vrc);
172 /* wait until thread init is completed */
173 vrc = RTThreadUserWait(mThread, RT_INDEFINITE_WAIT);
174 AssertReturnVoid(RT_SUCCESS(vrc) || vrc == VERR_INTERRUPTED);
175
176 /* the thread must succeed */
177 AssertReturnVoid((bool)data[2]);
178
179#elif defined(VBOX_WITH_SYS_V_IPC_SESSION_WATCHER)
180
181# ifdef VBOX_WITH_NEW_SYS_V_KEYGEN
182 key_t key = RTStrToUInt32(strTokenId.c_str());
183 AssertMsgReturnVoid(key != 0,
184 ("Key value of 0 is not valid for client token"));
185# else /* !VBOX_WITH_NEW_SYS_V_KEYGEN */
186 char *pszSemName = NULL;
187 RTStrUtf8ToCurrentCP(&pszSemName, strTokenId);
188 key_t key = ::ftok(pszSemName, 'V');
189 RTStrFree(pszSemName);
190# endif /* !VBOX_WITH_NEW_SYS_V_KEYGEN */
191 int s = ::semget(key, 0, 0);
192 AssertMsgReturnVoid(s >= 0,
193 ("Cannot open semaphore, errno=%d", errno));
194
195 /* grab the semaphore */
196 ::sembuf sop = { 0, -1, SEM_UNDO };
197 int rv = ::semop(s, &sop, 1);
198 AssertMsgReturnVoid(rv == 0,
199 ("Cannot grab semaphore, errno=%d", errno));
200 mSem = s;
201
202#else
203# error "Port me!"
204#endif
205}
206
207bool Session::ClientTokenHolder::isReady()
208{
209 return mSem != CTHSEMARG;
210}
211
212#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
213/** client token holder thread */
214DECLCALLBACK(int) ClientTokenHolderThread(RTTHREAD Thread, void *pvUser)
215{
216 LogFlowFuncEnter();
217
218 Assert(pvUser);
219
220 void **data = (void **)pvUser;
221
222# if defined(RT_OS_WINDOWS)
223 BSTR sessionId = (BSTR)data[0];
224 HANDLE initDoneSem = (HANDLE)data[1];
225
226 HANDLE mutex = ::OpenMutex(MUTEX_ALL_ACCESS, FALSE, sessionId);
227 AssertMsg(mutex, ("cannot open token, err=%d\n", ::GetLastError()));
228
229 if (mutex)
230 {
231 /* grab the token */
232 DWORD wrc = ::WaitForSingleObject(mutex, 0);
233 AssertMsg(wrc == WAIT_OBJECT_0, ("cannot grab token, err=%d\n", wrc));
234 if (wrc == WAIT_OBJECT_0)
235 {
236 HANDLE finishSem = ::CreateEvent(NULL, FALSE, FALSE, NULL);
237 AssertMsg(finishSem, ("cannot create event sem, err=%d\n", ::GetLastError()));
238 if (finishSem)
239 {
240 data[2] = (void*)finishSem;
241 /* signal we're done with init */
242 ::SetEvent(initDoneSem);
243 /* wait until we're signaled to release the token */
244 ::WaitForSingleObject(finishSem, INFINITE);
245 /* release the token */
246 LogFlow(("ClientTokenHolderThread(): releasing token...\n"));
247 BOOL success = ::ReleaseMutex(mutex);
248 AssertMsg(success, ("cannot release token, err=%d\n", ::GetLastError()));
249 ::CloseHandle(mutex);
250 ::CloseHandle(finishSem);
251 }
252 }
253 }
254
255 /* signal we're done */
256 ::SetEvent(initDoneSem);
257# elif defined(RT_OS_OS2)
258 Utf8Str sessionId = (BSTR)data[0];
259 RTSEMEVENT finishSem = (RTSEMEVENT)data[1];
260
261 LogFlowFunc(("sessionId='%s', finishSem=%p\n", sessionId.raw(), finishSem));
262
263 HMTX mutex = NULLHANDLE;
264 APIRET arc = ::DosOpenMutexSem((PSZ)sessionId.raw(), &mutex);
265 AssertMsg(arc == NO_ERROR, ("cannot open token, arc=%ld\n", arc));
266
267 if (arc == NO_ERROR)
268 {
269 /* grab the token */
270 LogFlowFunc(("grabbing token...\n"));
271 arc = ::DosRequestMutexSem(mutex, SEM_IMMEDIATE_RETURN);
272 AssertMsg(arc == NO_ERROR, ("cannot grab token, arc=%ld\n", arc));
273 if (arc == NO_ERROR)
274 {
275 /* store the answer */
276 data[2] = (void*)true;
277 /* signal we're done */
278 int vrc = RTThreadUserSignal(Thread);
279 AssertRC(vrc);
280
281 /* wait until we're signaled to release the token */
282 LogFlowFunc(("waiting for termination signal..\n"));
283 vrc = RTSemEventWait(finishSem, RT_INDEFINITE_WAIT);
284 Assert(arc == ERROR_INTERRUPT || ERROR_TIMEOUT);
285
286 /* release the token */
287 LogFlowFunc(("releasing token...\n"));
288 arc = ::DosReleaseMutexSem(mutex);
289 AssertMsg(arc == NO_ERROR, ("cannot release token, arc=%ld\n", arc));
290 }
291 ::DosCloseMutexSem(mutex);
292 }
293
294 /* store the answer */
295 data[1] = (void*)false;
296 /* signal we're done */
297 int vrc = RTThreadUserSignal(Thread);
298 AssertRC(vrc);
299# else
300# error "Port me!"
301# endif
302
303 LogFlowFuncLeave();
304
305 return 0;
306}
307#endif
308
309/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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