/[rdesktop]/sourceforge.net/trunk/seamlessrdp/ServerExe/HookDll/hookdll.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Annotation of /sourceforge.net/trunk/seamlessrdp/ServerExe/HookDll/hookdll.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1177 - (hide annotations)
Tue Mar 21 14:42:25 2006 UTC (18 years, 3 months ago) by ossman_
File MIME type: text/plain
File size: 12784 byte(s)
Support for window groups and modal windows.

1 ossman_ 1071 /* -*- c-basic-offset: 8 -*-
2     rdesktop: A Remote Desktop Protocol client.
3     Seamless windows - Remote server hook DLL
4 ossman_ 1069
5 ossman_ 1071 Based on code copyright (C) 2004-2005 Martin Wickett
6    
7     Copyright (C) Peter Åstrand <astrand@cendio.se> 2005-2006
8     Copyright (C) Pierre Ossman <ossman@cendio.se> 2006
9    
10     This program is free software; you can redistribute it and/or modify
11     it under the terms of the GNU General Public License as published by
12     the Free Software Foundation; either version 2 of the License, or
13     (at your option) any later version.
14    
15     This program is distributed in the hope that it will be useful,
16     but WITHOUT ANY WARRANTY; without even the implied warranty of
17     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18     GNU General Public License for more details.
19    
20     You should have received a copy of the GNU General Public License
21     along with this program; if not, write to the Free Software
22     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
23     */
24    
25 ossman_ 1069 #include <stdio.h>
26     #include <stdarg.h>
27    
28 ossman_ 1071 #include <windows.h>
29     #include <winuser.h>
30 ossman_ 1069
31 ossman_ 1073 #include "../vchannel.h"
32 ossman_ 1071
33 ossman_ 1069 #define DLL_EXPORT __declspec(dllexport)
34    
35 ossman_ 1145 #ifdef __GNUC__
36     #define SHARED __attribute__((section ("SHAREDDATA"), shared))
37     #else
38     #define SHARED
39     #endif
40    
41 ossman_ 1069 // Shared DATA
42     #pragma data_seg ( "SHAREDDATA" )
43    
44     // this is the total number of processes this dll is currently attached to
45 ossman_ 1145 int g_instance_count SHARED = 0;
46 ossman_ 1069
47 ossman_ 1145 // blocks for locally generated events
48 ossman_ 1161 HWND g_block_move_hwnd SHARED = NULL;
49 ossman_ 1165 unsigned int g_block_move_serial SHARED = 0;
50 ossman_ 1145 RECT g_block_move SHARED = { 0, 0, 0, 0 };
51 ossman_ 1165
52     unsigned int g_blocked_zchange_serial SHARED = 0;
53 ossman_ 1158 HWND g_blocked_zchange[2] SHARED = { NULL, NULL };
54 ossman_ 1165
55     unsigned int g_blocked_focus_serial SHARED = 0;
56 ossman_ 1158 HWND g_blocked_focus SHARED = NULL;
57 ossman_ 1165
58     unsigned int g_blocked_state_serial SHARED = 0;
59 ossman_ 1161 HWND g_blocked_state_hwnd SHARED = NULL;
60 ossman_ 1160 int g_blocked_state SHARED = -1;
61 ossman_ 1145
62 ossman_ 1069 #pragma data_seg ()
63    
64     #pragma comment(linker, "/section:SHAREDDATA,rws")
65    
66 ossman_ 1153 #define FOCUS_MSG_NAME "WM_SEAMLESS_FOCUS"
67     static UINT g_wm_seamless_focus;
68    
69 ossman_ 1071 static HHOOK g_cbt_hook = NULL;
70     static HHOOK g_wndproc_hook = NULL;
71 ossman_ 1134 static HHOOK g_wndprocret_hook = NULL;
72 ossman_ 1069
73 ossman_ 1071 static HINSTANCE g_instance = NULL;
74 ossman_ 1069
75 ossman_ 1071 static HANDLE g_mutex = NULL;
76 ossman_ 1069
77 ossman_ 1081 static void
78     update_position(HWND hwnd)
79     {
80 ossman_ 1145 RECT rect, blocked;
81 ossman_ 1161 HWND blocked_hwnd;
82 ossman_ 1165 unsigned int serial;
83 ossman_ 1081
84 ossman_ 1145 WaitForSingleObject(g_mutex, INFINITE);
85 ossman_ 1165 blocked_hwnd = g_block_move_hwnd;
86     serial = g_block_move_serial;
87 ossman_ 1145 memcpy(&blocked, &g_block_move, sizeof(RECT));
88     ReleaseMutex(g_mutex);
89    
90 ossman_ 1167 vchannel_block();
91    
92     if (!GetWindowRect(hwnd, &rect))
93     {
94     debug("GetWindowRect failed!\n");
95     goto end;
96     }
97    
98 ossman_ 1161 if ((hwnd == blocked_hwnd) && (rect.left == blocked.left) && (rect.top == blocked.top)
99 ossman_ 1145 && (rect.right == blocked.right) && (rect.bottom == blocked.bottom))
100 ossman_ 1167 goto end;
101    
102     vchannel_write("POSITION", "0x%p,%d,%d,%d,%d,0x%x",
103     hwnd,
104     rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, 0);
105    
106     end:
107     vchannel_unblock();
108 ossman_ 1081 }
109    
110 ossman_ 1162 static void
111     update_zorder(HWND hwnd)
112     {
113     HWND behind;
114     HWND block_hwnd, block_behind;
115 ossman_ 1165 unsigned int serial;
116 ossman_ 1162
117     WaitForSingleObject(g_mutex, INFINITE);
118 ossman_ 1165 serial = g_blocked_zchange_serial;
119 ossman_ 1162 block_hwnd = g_blocked_zchange[0];
120     block_behind = g_blocked_zchange[1];
121     ReleaseMutex(g_mutex);
122    
123 ossman_ 1167 vchannel_block();
124    
125 ossman_ 1162 behind = GetNextWindow(hwnd, GW_HWNDPREV);
126     while (behind)
127     {
128     LONG style;
129    
130     style = GetWindowLong(behind, GWL_STYLE);
131    
132     if ((!(style & WS_CHILD) || (style & WS_POPUP)) && (style & WS_VISIBLE))
133     break;
134    
135     behind = GetNextWindow(behind, GW_HWNDPREV);
136     }
137    
138     if ((hwnd == block_hwnd) && (behind == block_behind))
139 ossman_ 1165 vchannel_write("ACK", "%u", serial);
140     else
141     vchannel_write("ZCHANGE", "0x%p,0x%p,0x%x", hwnd, behind, 0);
142 ossman_ 1167
143     vchannel_unblock();
144 ossman_ 1162 }
145    
146 ossman_ 1071 static LRESULT CALLBACK
147     wndproc_hook_proc(int code, WPARAM cur_thread, LPARAM details)
148 ossman_ 1069 {
149 ossman_ 1097 HWND hwnd, parent;
150 ossman_ 1079 UINT msg;
151     WPARAM wparam;
152     LPARAM lparam;
153 ossman_ 1069
154 ossman_ 1079 LONG style;
155 ossman_ 1069
156 ossman_ 1071 if (code < 0)
157     goto end;
158 ossman_ 1069
159 ossman_ 1079 hwnd = ((CWPSTRUCT *) details)->hwnd;
160     msg = ((CWPSTRUCT *) details)->message;
161     wparam = ((CWPSTRUCT *) details)->wParam;
162     lparam = ((CWPSTRUCT *) details)->lParam;
163    
164     style = GetWindowLong(hwnd, GWL_STYLE);
165    
166 ossman_ 1082 /* Docs say that WS_CHILD and WS_POPUP is an illegal combination,
167     but they exist nonetheless. */
168     if ((style & WS_CHILD) && !(style & WS_POPUP))
169 ossman_ 1079 goto end;
170    
171 ossman_ 1097 if (style & WS_POPUP)
172 ossman_ 1109 {
173 ossman_ 1097 parent = (HWND) GetWindowLong(hwnd, GWL_HWNDPARENT);
174 ossman_ 1109 if (!parent)
175     parent = (HWND) - 1;
176     }
177 ossman_ 1097 else
178     parent = NULL;
179    
180 ossman_ 1071 switch (msg)
181     {
182     case WM_WINDOWPOSCHANGED:
183 ossman_ 1079 {
184     WINDOWPOS *wp = (WINDOWPOS *) lparam;
185 ossman_ 1069
186 ossman_ 1079 if (wp->flags & SWP_SHOWWINDOW)
187     {
188 ossman_ 1134 unsigned short title[150];
189 ossman_ 1099 int state;
190 ossman_ 1177 DWORD pid;
191     int flags;
192 ossman_ 1099
193 ossman_ 1177 GetWindowThreadProcessId(hwnd, &pid);
194 ossman_ 1069
195 ossman_ 1177 flags = 0;
196     if (style & DS_MODALFRAME)
197     flags |= SEAMLESS_CREATE_MODAL;
198    
199     vchannel_write("CREATE", "0x%08lx,0x%08lx,0x%08lx,0x%08x",
200     (long) hwnd, (long) pid, (long) parent,
201     flags);
202    
203 ossman_ 1134 GetWindowTextW(hwnd, title, sizeof(title) / sizeof(*title));
204 ossman_ 1069
205 ossman_ 1163 vchannel_write("TITLE", "0x%x,%s,0x%x", hwnd,
206 ossman_ 1134 vchannel_strfilter_unicode(title), 0);
207 ossman_ 1099
208     if (style & WS_MAXIMIZE)
209     state = 2;
210     else if (style & WS_MINIMIZE)
211     state = 1;
212     else
213     state = 0;
214    
215 ossman_ 1081 update_position(hwnd);
216 ossman_ 1099
217 ossman_ 1163 vchannel_write("STATE", "0x%p,0x%x,0x%x", hwnd, state, 0);
218 ossman_ 1079 }
219 ossman_ 1069
220 ossman_ 1079 if (wp->flags & SWP_HIDEWINDOW)
221 ossman_ 1163 vchannel_write("DESTROY", "0x%p,0x%x", hwnd, 0);
222 ossman_ 1079
223 ossman_ 1108 if (!(style & WS_VISIBLE) || (style & WS_MINIMIZE))
224 ossman_ 1079 break;
225    
226 ossman_ 1080 if (!(wp->flags & SWP_NOMOVE && wp->flags & SWP_NOSIZE))
227 ossman_ 1081 update_position(hwnd);
228 ossman_ 1079
229 ossman_ 1071 break;
230 ossman_ 1079 }
231 ossman_ 1069
232 ossman_ 1081 case WM_SIZE:
233 ossman_ 1108 if (!(style & WS_VISIBLE) || (style & WS_MINIMIZE))
234 ossman_ 1083 break;
235 ossman_ 1081 update_position(hwnd);
236     break;
237    
238     case WM_MOVE:
239 ossman_ 1108 if (!(style & WS_VISIBLE) || (style & WS_MINIMIZE))
240 ossman_ 1083 break;
241 ossman_ 1081 update_position(hwnd);
242     break;
243    
244 ossman_ 1134 case WM_DESTROY:
245     if (!(style & WS_VISIBLE))
246     break;
247 ossman_ 1163 vchannel_write("DESTROY", "0x%p,0x%x", hwnd, 0);
248 ossman_ 1134 break;
249    
250     default:
251     break;
252     }
253    
254     end:
255     return CallNextHookEx(g_wndproc_hook, code, cur_thread, details);
256     }
257    
258     static LRESULT CALLBACK
259     wndprocret_hook_proc(int code, WPARAM cur_thread, LPARAM details)
260     {
261     HWND hwnd, parent;
262     UINT msg;
263     WPARAM wparam;
264     LPARAM lparam;
265    
266     LONG style;
267    
268     if (code < 0)
269     goto end;
270    
271     hwnd = ((CWPRETSTRUCT *) details)->hwnd;
272     msg = ((CWPRETSTRUCT *) details)->message;
273     wparam = ((CWPRETSTRUCT *) details)->wParam;
274     lparam = ((CWPRETSTRUCT *) details)->lParam;
275    
276     style = GetWindowLong(hwnd, GWL_STYLE);
277    
278     /* Docs say that WS_CHILD and WS_POPUP is an illegal combination,
279     but they exist nonetheless. */
280     if ((style & WS_CHILD) && !(style & WS_POPUP))
281     goto end;
282    
283 ossman_ 1158 if (style & WS_POPUP)
284     {
285     parent = (HWND) GetWindowLong(hwnd, GWL_HWNDPARENT);
286     if (!parent)
287     parent = (HWND) - 1;
288     }
289     else
290     parent = NULL;
291    
292 ossman_ 1134 switch (msg)
293     {
294 ossman_ 1162 case WM_WINDOWPOSCHANGED:
295     {
296     WINDOWPOS *wp = (WINDOWPOS *) lparam;
297    
298     if (!(style & WS_VISIBLE) || (style & WS_MINIMIZE))
299     break;
300    
301     if (!(wp->flags & SWP_NOZORDER))
302     update_zorder(hwnd);
303    
304     break;
305     }
306    
307    
308 ossman_ 1102 case WM_SETTEXT:
309 ossman_ 1113 {
310 ossman_ 1134 unsigned short title[150];
311 ossman_ 1113 if (!(style & WS_VISIBLE))
312     break;
313 ossman_ 1134 /* We cannot use the string in lparam because
314     we need unicode. */
315     GetWindowTextW(hwnd, title, sizeof(title) / sizeof(*title));
316 ossman_ 1163 vchannel_write("TITLE", "0x%p,%s,0x%x", hwnd,
317 ossman_ 1134 vchannel_strfilter_unicode(title), 0);
318 ossman_ 1102 break;
319 ossman_ 1113 }
320 ossman_ 1102
321 ossman_ 1071 default:
322     break;
323     }
324    
325 ossman_ 1153 if (msg == g_wm_seamless_focus)
326 ossman_ 1158 {
327     /* FIXME: SetActiveWindow() kills menus. Need to find a clean
328     way to solve this. */
329     if ((GetActiveWindow() != hwnd) && !parent)
330     SetActiveWindow(hwnd);
331 ossman_ 1165
332     vchannel_write("ACK", "%u", g_blocked_focus_serial);
333 ossman_ 1158 }
334 ossman_ 1153
335 ossman_ 1071 end:
336 ossman_ 1134 return CallNextHookEx(g_wndprocret_hook, code, cur_thread, details);
337 ossman_ 1069 }
338    
339 ossman_ 1071 static LRESULT CALLBACK
340     cbt_hook_proc(int code, WPARAM wparam, LPARAM lparam)
341 ossman_ 1069 {
342 ossman_ 1071 if (code < 0)
343     goto end;
344 ossman_ 1069
345 ossman_ 1071 switch (code)
346     {
347     case HCBT_MINMAX:
348     {
349 ossman_ 1160 int show, state, blocked;
350 ossman_ 1161 HWND blocked_hwnd;
351 ossman_ 1165 unsigned int serial;
352 ossman_ 1069
353 ossman_ 1160 WaitForSingleObject(g_mutex, INFINITE);
354 ossman_ 1161 blocked_hwnd = g_blocked_state_hwnd;
355 ossman_ 1165 serial = g_blocked_state_serial;
356 ossman_ 1160 blocked = g_blocked_state;
357     ReleaseMutex(g_mutex);
358    
359 ossman_ 1071 show = LOWORD(lparam);
360 ossman_ 1069
361 ossman_ 1128 if ((show == SW_NORMAL) || (show == SW_SHOWNORMAL)
362     || (show == SW_RESTORE))
363 ossman_ 1079 state = 0;
364 ossman_ 1106 else if ((show == SW_MINIMIZE) || (show == SW_SHOWMINIMIZED))
365 ossman_ 1079 state = 1;
366 ossman_ 1106 else if ((show == SW_MAXIMIZE) || (show == SW_SHOWMAXIMIZED))
367 ossman_ 1079 state = 2;
368 ossman_ 1100 else
369 ossman_ 1106 {
370     debug("Unexpected show: %d", show);
371 ossman_ 1100 break;
372 ossman_ 1106 }
373 ossman_ 1160
374 ossman_ 1165 if ((blocked_hwnd == (HWND) wparam) && (blocked == state))
375     vchannel_write("ACK", "%u", serial);
376     else
377 ossman_ 1163 vchannel_write("STATE", "0x%p,0x%x,0x%x", (HWND) wparam,
378     state, 0);
379 ossman_ 1160
380 ossman_ 1071 break;
381     }
382 ossman_ 1069
383 ossman_ 1071 default:
384     break;
385     }
386 ossman_ 1069
387 ossman_ 1071 end:
388     return CallNextHookEx(g_cbt_hook, code, wparam, lparam);
389 ossman_ 1069 }
390    
391 ossman_ 1071 DLL_EXPORT void
392     SetHooks(void)
393 ossman_ 1069 {
394 ossman_ 1071 if (!g_cbt_hook)
395     g_cbt_hook = SetWindowsHookEx(WH_CBT, cbt_hook_proc, g_instance, 0);
396 ossman_ 1069
397 ossman_ 1071 if (!g_wndproc_hook)
398     g_wndproc_hook = SetWindowsHookEx(WH_CALLWNDPROC, wndproc_hook_proc, g_instance, 0);
399 ossman_ 1134
400     if (!g_wndprocret_hook)
401     g_wndprocret_hook =
402     SetWindowsHookEx(WH_CALLWNDPROCRET, wndprocret_hook_proc, g_instance, 0);
403 ossman_ 1069 }
404    
405 ossman_ 1071 DLL_EXPORT void
406     RemoveHooks(void)
407 ossman_ 1069 {
408 ossman_ 1071 if (g_cbt_hook)
409     UnhookWindowsHookEx(g_cbt_hook);
410 ossman_ 1069
411 ossman_ 1071 if (g_wndproc_hook)
412     UnhookWindowsHookEx(g_wndproc_hook);
413 ossman_ 1134
414     if (g_wndprocret_hook)
415     UnhookWindowsHookEx(g_wndprocret_hook);
416 ossman_ 1069 }
417    
418 ossman_ 1145 DLL_EXPORT void
419 ossman_ 1165 SafeMoveWindow(unsigned int serial, HWND hwnd, int x, int y, int width, int height)
420 ossman_ 1145 {
421 ossman_ 1165 RECT rect;
422    
423 ossman_ 1145 WaitForSingleObject(g_mutex, INFINITE);
424 ossman_ 1161 g_block_move_hwnd = hwnd;
425 ossman_ 1165 g_block_move_serial = serial;
426 ossman_ 1145 g_block_move.left = x;
427     g_block_move.top = y;
428     g_block_move.right = x + width;
429     g_block_move.bottom = y + height;
430     ReleaseMutex(g_mutex);
431    
432     SetWindowPos(hwnd, NULL, x, y, width, height, SWP_NOACTIVATE | SWP_NOZORDER);
433    
434 ossman_ 1167 vchannel_write("ACK", "%u", serial);
435    
436     if (!GetWindowRect(hwnd, &rect))
437     debug("GetWindowRect failed!\n");
438     else if ((rect.left != x) || (rect.top != y) || (rect.right != x + width)
439     || (rect.bottom != y + height))
440     update_position(hwnd);
441    
442 ossman_ 1145 WaitForSingleObject(g_mutex, INFINITE);
443 ossman_ 1161 g_block_move_hwnd = NULL;
444 ossman_ 1145 memset(&g_block_move, 0, sizeof(RECT));
445     ReleaseMutex(g_mutex);
446     }
447    
448 ossman_ 1158 DLL_EXPORT void
449 ossman_ 1165 SafeZChange(unsigned int serial, HWND hwnd, HWND behind)
450 ossman_ 1158 {
451     WaitForSingleObject(g_mutex, INFINITE);
452 ossman_ 1165 g_blocked_zchange_serial = serial;
453 ossman_ 1158 g_blocked_zchange[0] = hwnd;
454     g_blocked_zchange[1] = behind;
455     ReleaseMutex(g_mutex);
456    
457 ossman_ 1162 if (behind == NULL)
458     behind = HWND_TOP;
459    
460 ossman_ 1158 SetWindowPos(hwnd, behind, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE);
461    
462     WaitForSingleObject(g_mutex, INFINITE);
463     g_blocked_zchange[0] = NULL;
464     g_blocked_zchange[1] = NULL;
465     ReleaseMutex(g_mutex);
466     }
467    
468     DLL_EXPORT void
469 ossman_ 1165 SafeFocus(unsigned int serial, HWND hwnd)
470 ossman_ 1158 {
471     WaitForSingleObject(g_mutex, INFINITE);
472 ossman_ 1165 g_blocked_focus_serial = serial;
473 ossman_ 1158 g_blocked_focus = hwnd;
474     ReleaseMutex(g_mutex);
475    
476     SendMessage(hwnd, g_wm_seamless_focus, 0, 0);
477    
478     WaitForSingleObject(g_mutex, INFINITE);
479     g_blocked_focus = NULL;
480     ReleaseMutex(g_mutex);
481     }
482    
483 ossman_ 1160 DLL_EXPORT void
484 ossman_ 1165 SafeSetState(unsigned int serial, HWND hwnd, int state)
485 ossman_ 1160 {
486 ossman_ 1165 LONG style;
487     int curstate;
488    
489     vchannel_block();
490    
491     style = GetWindowLong(hwnd, GWL_STYLE);
492    
493     if (style & WS_MAXIMIZE)
494     curstate = 2;
495     else if (style & WS_MINIMIZE)
496     curstate = 1;
497     else
498     curstate = 0;
499    
500     if (state == curstate)
501     {
502     vchannel_write("ACK", "%u", serial);
503     vchannel_unblock();
504 ossman_ 1167 return;
505 ossman_ 1165 }
506    
507 ossman_ 1160 WaitForSingleObject(g_mutex, INFINITE);
508 ossman_ 1161 g_blocked_state_hwnd = hwnd;
509 ossman_ 1165 g_blocked_state_serial = serial;
510 ossman_ 1160 g_blocked_state = state;
511     ReleaseMutex(g_mutex);
512    
513 ossman_ 1165 vchannel_unblock();
514    
515 ossman_ 1160 if (state == 0)
516     ShowWindow(hwnd, SW_RESTORE);
517     else if (state == 1)
518     ShowWindow(hwnd, SW_MINIMIZE);
519     else if (state == 2)
520     ShowWindow(hwnd, SW_MAXIMIZE);
521     else
522     debug("Invalid state %d sent.", state);
523    
524     WaitForSingleObject(g_mutex, INFINITE);
525 ossman_ 1161 g_blocked_state_hwnd = NULL;
526 ossman_ 1160 g_blocked_state = -1;
527     ReleaseMutex(g_mutex);
528     }
529    
530 ossman_ 1071 DLL_EXPORT int
531     GetInstanceCount()
532 ossman_ 1069 {
533 ossman_ 1071 return g_instance_count;
534 ossman_ 1069 }
535    
536 ossman_ 1071 BOOL APIENTRY
537     DllMain(HINSTANCE hinstDLL, DWORD ul_reason_for_call, LPVOID lpReserved)
538 ossman_ 1069 {
539 ossman_ 1071 switch (ul_reason_for_call)
540     {
541     case DLL_PROCESS_ATTACH:
542     // remember our instance handle
543     g_instance = hinstDLL;
544 ossman_ 1069
545 ossman_ 1073 g_mutex = CreateMutex(NULL, FALSE, "Local\\SeamlessDLL");
546 ossman_ 1071 if (!g_mutex)
547     return FALSE;
548 ossman_ 1069
549 ossman_ 1071 WaitForSingleObject(g_mutex, INFINITE);
550     ++g_instance_count;
551     ReleaseMutex(g_mutex);
552 ossman_ 1069
553 ossman_ 1153 g_wm_seamless_focus = RegisterWindowMessage(FOCUS_MSG_NAME);
554    
555 ossman_ 1071 vchannel_open();
556 ossman_ 1069
557 ossman_ 1071 break;
558 ossman_ 1069
559 ossman_ 1071 case DLL_THREAD_ATTACH:
560     break;
561 ossman_ 1069
562 ossman_ 1071 case DLL_THREAD_DETACH:
563     break;
564 ossman_ 1069
565 ossman_ 1071 case DLL_PROCESS_DETACH:
566     WaitForSingleObject(g_mutex, INFINITE);
567     --g_instance_count;
568     ReleaseMutex(g_mutex);
569 ossman_ 1069
570 ossman_ 1071 vchannel_close();
571 ossman_ 1069
572 ossman_ 1071 CloseHandle(g_mutex);
573 ossman_ 1069
574 ossman_ 1071 break;
575     }
576 ossman_ 1069
577 ossman_ 1071 return TRUE;
578 ossman_ 1069 }

Properties

Name Value
svn:executable *

  ViewVC Help
Powered by ViewVC 1.1.26