Показываем диалоговое окошко завершения работы системы
Следующий завершает работу системы при помощи функции InitiateSystemShutdown на компьютере, на котором залогинен пользователь. Сначало необходимо получить привелегию SE_SHUTDOWN_NAME.
BOOL MySystemShutdown( LPTSTR lpMsg )
{
HANDLE hToken; // дескриптор маркера процесса
TOKEN_PRIVILEGES tkp; // указатель на структуру маркера
BOOL fResult; // флаг завершения работы системы
// Чтобы получить привелегию завершения работы системы, получаем
// дескриптор маркера текущего процесса.
if (!OpenProcessToken(GetCurrentProcess(),
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken))
return FALSE;
// Получаем LUID для привелегии завершения работы.
LookupPrivilegeValue(NULL, SE_SHUTDOWN_NAME,
&tkp.Privileges[0].Luid);
tkp.PrivilegeCount = 1; // будет установлена одна привелегия
tkp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
// Получаем привелегию завершения работы для этого процесса.
AdjustTokenPrivileges(hToken, FALSE, &tkp, 0,
(PTOKEN_PRIVILEGES) NULL, 0);
// Если не удалось получить привелегию:
if (GetLastError() != ERROR_SUCCESS)
return FALSE;
// Отображаем диалоговое окошко завершения работы и начинаем
// отсчёт.
fResult = InitiateSystemShutdown(
NULL, // локальный компьютер
lpMsg, // сообщение для пользователя
20, // таймаут
FALSE, // просить пользователя закрыть приложения ?
TRUE); // Перезагрузить после завершения работы ?
if (!fResult)
return FALSE;
// Запрещаем привелегию завершения работы для нашего процесса.
tkp.Privileges[0].Attributes = 0;
AdjustTokenPrivileges(hToken, FALSE, &tkp, 0,
(PTOKEN_PRIVILEGES) NULL, 0);
return TRUE;
}
Если выполнить функцию AbortSystemShutdown в период таймаута, указанного в InitiateSystemShutdown, то завершения работы системы не произойдёт.
BOOL PreventSystemShutdown()
{
HANDLE hToken; // дескриптор маркера процесса
TOKEN_PRIVILEGES tkp; // указатель на структуру маркера
// Чтобы получить привелегию завершения работы системы, получаем
// дескриптор маркера текущего процесса.
if (!OpenProcessToken(GetCurrentProcess(),
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &hToken))
return FALSE;
// Получаем LUID для привелегии завершения работы.
LookupPrivilegeValue(NULL, SE_SHUTDOWN_NAME,
&tkp.Privileges[0].Luid);
tkp.PrivilegeCount = 1; // one privilege to set
tkp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
// Получаем привелегию завершения работы для этого процесса.
AdjustTokenPrivileges(hToken, FALSE, &tkp, 0,
(PTOKEN_PRIVILEGES)NULL, 0);
if (GetLastError() != ERROR_SUCCESS)
return FALSE;
// Предотвращаем завершение работы системы.
if ( !AbortSystemShutdown(NULL) )
return FALSE;
// Запрещаем привелегию завершения работы для нашего процесса.
tkp.Privileges[0].Attributes = 0;
AdjustTokenPrivileges(hToken, FALSE, &tkp, 0,
(PTOKEN_PRIVILEGES) NULL, 0);
return TRUE;
}