2017-07-27 11:46:31 +02:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2017-07-27 13:57:12 +02:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Concurrent;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.ComponentModel;
|
2017-08-03 15:35:22 +02:00
|
|
|
|
using System.Linq;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
using System.Text;
|
2018-09-25 12:10:34 +02:00
|
|
|
|
using System.Threading;
|
2017-08-03 15:35:22 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Monitoring;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.WindowsApi;
|
2018-09-27 11:24:13 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.WindowsApi.Events;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Constants;
|
2017-08-03 15:35:22 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Monitoring;
|
2017-08-04 08:25:49 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Types;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.WindowsApi
|
|
|
|
|
{
|
|
|
|
|
public class NativeMethods : INativeMethods
|
|
|
|
|
{
|
2017-08-03 15:35:22 +02:00
|
|
|
|
private ConcurrentDictionary<IntPtr, KeyboardHook> KeyboardHooks = new ConcurrentDictionary<IntPtr, KeyboardHook>();
|
2017-08-04 15:20:33 +02:00
|
|
|
|
private ConcurrentDictionary<IntPtr, MouseHook> MouseHooks = new ConcurrentDictionary<IntPtr, MouseHook>();
|
2018-09-27 11:24:13 +02:00
|
|
|
|
private ConcurrentDictionary<IntPtr, SystemHook> SystemHooks = new ConcurrentDictionary<IntPtr, SystemHook>();
|
2017-08-03 15:35:22 +02:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
2017-08-04 12:19:56 +02:00
|
|
|
|
/// Upon finalization, unregister all active system events and hooks...
|
2017-08-03 15:35:22 +02:00
|
|
|
|
/// </summary>
|
|
|
|
|
~NativeMethods()
|
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
foreach (var hook in SystemHooks.Values)
|
2017-08-03 15:35:22 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
hook.Detach();
|
2017-08-03 15:35:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
foreach (var hook in KeyboardHooks.Values)
|
2017-08-03 15:35:22 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
hook.Detach();
|
2017-08-03 15:35:22 +02:00
|
|
|
|
}
|
2017-08-04 15:20:33 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
foreach (var hook in MouseHooks.Values)
|
2017-08-04 15:20:33 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
hook.Detach();
|
2017-08-04 15:20:33 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void DeregisterKeyboardHook(IKeyboardInterceptor interceptor)
|
|
|
|
|
{
|
|
|
|
|
var hook = KeyboardHooks.Values.FirstOrDefault(h => h.Interceptor == interceptor);
|
|
|
|
|
|
|
|
|
|
if (hook != null)
|
|
|
|
|
{
|
|
|
|
|
var success = hook.Detach();
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
KeyboardHooks.TryRemove(hook.Handle, out _);
|
2017-08-04 15:20:33 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void DeregisterMouseHook(IMouseInterceptor interceptor)
|
|
|
|
|
{
|
|
|
|
|
var hook = MouseHooks.Values.FirstOrDefault(h => h.Interceptor == interceptor);
|
|
|
|
|
|
|
|
|
|
if (hook != null)
|
|
|
|
|
{
|
|
|
|
|
var success = hook.Detach();
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
MouseHooks.TryRemove(hook.Handle, out _);
|
2017-08-04 15:20:33 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
public void DeregisterSystemEventHook(Guid hookId)
|
2017-08-04 15:20:33 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
var hook = SystemHooks.Values.FirstOrDefault(h => h.Id == hookId);
|
2017-08-04 15:20:33 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
if (hook != null)
|
2017-08-04 15:20:33 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
var success = hook.Detach();
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
2017-08-04 15:20:33 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
SystemHooks.TryRemove(hook.Handle, out _);
|
|
|
|
|
}
|
2017-08-03 15:35:22 +02:00
|
|
|
|
}
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
2017-08-07 18:08:55 +02:00
|
|
|
|
public void EmptyClipboard()
|
|
|
|
|
{
|
|
|
|
|
var success = true;
|
|
|
|
|
|
|
|
|
|
success &= User32.OpenClipboard(IntPtr.Zero);
|
|
|
|
|
success &= User32.EmptyClipboard();
|
|
|
|
|
success &= User32.CloseClipboard();
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 11:46:31 +02:00
|
|
|
|
public IEnumerable<IntPtr> GetOpenWindows()
|
|
|
|
|
{
|
|
|
|
|
var windows = new List<IntPtr>();
|
2018-08-16 11:23:37 +02:00
|
|
|
|
|
|
|
|
|
bool EnumWindows(IntPtr hWnd, IntPtr lParam)
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
|
|
|
|
if (hWnd != GetShellWindowHandle() && User32.IsWindowVisible(hWnd) && User32.GetWindowTextLength(hWnd) > 0)
|
|
|
|
|
{
|
|
|
|
|
windows.Add(hWnd);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
2018-08-16 11:23:37 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var success = User32.EnumWindows(EnumWindows, IntPtr.Zero);
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return windows;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public uint GetProcessIdFor(IntPtr window)
|
|
|
|
|
{
|
|
|
|
|
User32.GetWindowThreadProcessId(window, out uint processId);
|
|
|
|
|
|
|
|
|
|
return processId;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IntPtr GetShellWindowHandle()
|
|
|
|
|
{
|
|
|
|
|
return User32.FindWindow("Shell_TrayWnd", null);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public uint GetShellProcessId()
|
|
|
|
|
{
|
|
|
|
|
var handle = GetShellWindowHandle();
|
|
|
|
|
var threadId = User32.GetWindowThreadProcessId(handle, out uint processId);
|
|
|
|
|
|
|
|
|
|
return processId;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 08:28:17 +02:00
|
|
|
|
public string GetWallpaperPath()
|
|
|
|
|
{
|
|
|
|
|
const int MAX_PATH = 260;
|
|
|
|
|
var buffer = new String('\0', MAX_PATH);
|
|
|
|
|
var success = User32.SystemParametersInfo(SPI.GETDESKWALLPAPER, buffer.Length, buffer, 0);
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var path = buffer.Substring(0, buffer.IndexOf('\0'));
|
|
|
|
|
|
|
|
|
|
return path;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 11:46:31 +02:00
|
|
|
|
public string GetWindowTitle(IntPtr window)
|
|
|
|
|
{
|
|
|
|
|
var length = User32.GetWindowTextLength(window);
|
|
|
|
|
|
|
|
|
|
if (length > 0)
|
|
|
|
|
{
|
|
|
|
|
var builder = new StringBuilder(length);
|
|
|
|
|
|
|
|
|
|
User32.GetWindowText(window, builder, length + 1);
|
|
|
|
|
|
|
|
|
|
return builder.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-04 08:25:49 +02:00
|
|
|
|
public IBounds GetWorkingArea()
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
|
|
|
|
var workingArea = new RECT();
|
|
|
|
|
var success = User32.SystemParametersInfo(SPI.GETWORKAREA, 0, ref workingArea, SPIF.NONE);
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-04 08:25:49 +02:00
|
|
|
|
return workingArea.ToBounds();
|
2017-07-27 11:46:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 13:57:12 +02:00
|
|
|
|
public bool HideWindow(IntPtr window)
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
2017-07-27 13:57:12 +02:00
|
|
|
|
return User32.ShowWindow(window, (int) ShowWindowCommand.Hide);
|
2017-07-27 11:46:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void MinimizeAllOpenWindows()
|
|
|
|
|
{
|
|
|
|
|
var handle = GetShellWindowHandle();
|
|
|
|
|
|
2018-02-14 15:26:05 +01:00
|
|
|
|
User32.SendMessage(handle, Constant.WM_COMMAND, (IntPtr) Constant.MIN_ALL, IntPtr.Zero);
|
2017-07-27 11:46:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void PostCloseMessageToShell()
|
|
|
|
|
{
|
|
|
|
|
// NOTE: The close message 0x5B4 posted to the shell is undocumented and not officially supported:
|
|
|
|
|
// https://stackoverflow.com/questions/5689904/gracefully-exit-explorer-programmatically/5705965#5705965
|
|
|
|
|
|
|
|
|
|
var handle = GetShellWindowHandle();
|
|
|
|
|
var success = User32.PostMessage(handle, 0x5B4, IntPtr.Zero, IntPtr.Zero);
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 11:16:44 +02:00
|
|
|
|
public void PreventSleepMode()
|
|
|
|
|
{
|
|
|
|
|
Kernel32.SetThreadExecutionState(EXECUTION_STATE.CONTINUOUS | EXECUTION_STATE.DISPLAY_REQUIRED | EXECUTION_STATE.SYSTEM_REQUIRED);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-03 15:35:22 +02:00
|
|
|
|
public void RegisterKeyboardHook(IKeyboardInterceptor interceptor)
|
|
|
|
|
{
|
2018-09-25 12:10:34 +02:00
|
|
|
|
var hookReadyEvent = new AutoResetEvent(false);
|
|
|
|
|
var hookThread = new Thread(() =>
|
|
|
|
|
{
|
|
|
|
|
var hook = new KeyboardHook(interceptor);
|
2018-09-27 11:24:13 +02:00
|
|
|
|
var sleepEvent = new AutoResetEvent(false);
|
2018-09-25 12:10:34 +02:00
|
|
|
|
|
|
|
|
|
hook.Attach();
|
|
|
|
|
KeyboardHooks[hook.Handle] = hook;
|
|
|
|
|
hookReadyEvent.Set();
|
|
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
sleepEvent.WaitOne();
|
2018-09-25 12:10:34 +02:00
|
|
|
|
}
|
|
|
|
|
});
|
2017-08-03 15:35:22 +02:00
|
|
|
|
|
2018-09-25 12:10:34 +02:00
|
|
|
|
hookThread.SetApartmentState(ApartmentState.STA);
|
|
|
|
|
hookThread.IsBackground = true;
|
|
|
|
|
hookThread.Start();
|
2017-08-03 15:35:22 +02:00
|
|
|
|
|
2018-09-25 12:10:34 +02:00
|
|
|
|
hookReadyEvent.WaitOne();
|
2017-08-03 15:35:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-04 15:20:33 +02:00
|
|
|
|
public void RegisterMouseHook(IMouseInterceptor interceptor)
|
|
|
|
|
{
|
2018-09-25 12:10:34 +02:00
|
|
|
|
var hookReadyEvent = new AutoResetEvent(false);
|
|
|
|
|
var hookThread = new Thread(() =>
|
|
|
|
|
{
|
|
|
|
|
var hook = new MouseHook(interceptor);
|
2018-09-27 11:24:13 +02:00
|
|
|
|
var sleepEvent = new AutoResetEvent(false);
|
2018-09-25 12:10:34 +02:00
|
|
|
|
|
|
|
|
|
hook.Attach();
|
|
|
|
|
MouseHooks[hook.Handle] = hook;
|
|
|
|
|
hookReadyEvent.Set();
|
|
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
sleepEvent.WaitOne();
|
2018-09-25 12:10:34 +02:00
|
|
|
|
}
|
|
|
|
|
});
|
2017-08-04 15:20:33 +02:00
|
|
|
|
|
2018-09-25 12:10:34 +02:00
|
|
|
|
hookThread.SetApartmentState(ApartmentState.STA);
|
|
|
|
|
hookThread.IsBackground = true;
|
|
|
|
|
hookThread.Start();
|
2017-08-04 15:20:33 +02:00
|
|
|
|
|
2018-09-25 12:10:34 +02:00
|
|
|
|
hookReadyEvent.WaitOne();
|
2017-08-04 15:20:33 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
public Guid RegisterSystemCaptureStartEvent(SystemEventCallback callback)
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
return RegisterSystemEvent(callback, Constant.EVENT_SYSTEM_CAPTURESTART);
|
|
|
|
|
}
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
public Guid RegisterSystemForegroundEvent(SystemEventCallback callback)
|
|
|
|
|
{
|
|
|
|
|
return RegisterSystemEvent(callback, Constant.EVENT_SYSTEM_FOREGROUND);
|
2017-07-27 11:46:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
internal Guid RegisterSystemEvent(SystemEventCallback callback, uint eventId)
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
var hookId = default(Guid);
|
|
|
|
|
var hookReadyEvent = new AutoResetEvent(false);
|
|
|
|
|
var hookThread = new Thread(() =>
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
2018-09-27 11:24:13 +02:00
|
|
|
|
var hook = new SystemHook(callback, eventId);
|
|
|
|
|
|
|
|
|
|
hook.Attach();
|
|
|
|
|
hookId = hook.Id;
|
|
|
|
|
SystemHooks[hook.Handle] = hook;
|
|
|
|
|
hookReadyEvent.Set();
|
|
|
|
|
hook.AwaitDetach();
|
|
|
|
|
});
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
hookThread.SetApartmentState(ApartmentState.STA);
|
|
|
|
|
hookThread.IsBackground = true;
|
|
|
|
|
hookThread.Start();
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
hookReadyEvent.WaitOne();
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
2018-09-27 11:24:13 +02:00
|
|
|
|
return hookId;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 08:28:17 +02:00
|
|
|
|
public void RemoveWallpaper()
|
|
|
|
|
{
|
|
|
|
|
SetWallpaper(string.Empty);
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 11:46:31 +02:00
|
|
|
|
public void RestoreWindow(IntPtr window)
|
|
|
|
|
{
|
|
|
|
|
User32.ShowWindow(window, (int)ShowWindowCommand.Restore);
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-21 11:33:32 +02:00
|
|
|
|
public bool ResumeThread(int threadId)
|
|
|
|
|
{
|
|
|
|
|
const int FAILURE = -1;
|
|
|
|
|
var handle = Kernel32.OpenThread(ThreadAccess.SUSPEND_RESUME, false, (uint) threadId);
|
|
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var result = Kernel32.ResumeThread(handle);
|
|
|
|
|
var success = result != FAILURE;
|
|
|
|
|
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
Kernel32.CloseHandle(handle);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 13:57:12 +02:00
|
|
|
|
public void SendCloseMessageTo(IntPtr window)
|
|
|
|
|
{
|
|
|
|
|
User32.SendMessage(window, Constant.WM_SYSCOMMAND, (IntPtr) SystemCommand.CLOSE, IntPtr.Zero);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 08:28:17 +02:00
|
|
|
|
public void SetWallpaper(string filePath)
|
|
|
|
|
{
|
|
|
|
|
var success = User32.SystemParametersInfo(SPI.SETDESKWALLPAPER, 0, filePath, SPIF.UPDATEANDCHANGE);
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-04 08:25:49 +02:00
|
|
|
|
public void SetWorkingArea(IBounds bounds)
|
2017-07-27 11:46:31 +02:00
|
|
|
|
{
|
2017-08-04 08:25:49 +02:00
|
|
|
|
var workingArea = new RECT { Left = bounds.Left, Top = bounds.Top, Right = bounds.Right, Bottom = bounds.Bottom };
|
2019-03-19 12:26:03 +01:00
|
|
|
|
var success = User32.SystemParametersInfo(SPI.SETWORKAREA, 0, ref workingArea, SPIF.UPDATEINIFILE);
|
2017-07-27 11:46:31 +02:00
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
throw new Win32Exception(Marshal.GetLastWin32Error());
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-09-21 11:33:32 +02:00
|
|
|
|
|
|
|
|
|
public bool SuspendThread(int threadId)
|
|
|
|
|
{
|
|
|
|
|
const int FAILURE = -1;
|
|
|
|
|
var handle = Kernel32.OpenThread(ThreadAccess.SUSPEND_RESUME, false, (uint) threadId);
|
|
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var result = Kernel32.SuspendThread(handle);
|
|
|
|
|
var success = result != FAILURE;
|
|
|
|
|
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
Kernel32.CloseHandle(handle);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-07-27 11:46:31 +02:00
|
|
|
|
}
|
|
|
|
|
}
|