VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/win/RTSystemShutdown-win.cpp@ 95765

Last change on this file since 95765 was 93115, checked in by vboxsync, 3 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.8 KB
Line 
1/* $Id: RTSystemShutdown-win.cpp 93115 2022-01-01 11:31:46Z vboxsync $ */
2/** @file
3 * IPRT - RTSystemShutdown, Windows.
4 */
5
6/*
7 * Copyright (C) 2012-2022 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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/system.h>
32#include "internal/iprt.h"
33
34#include <iprt/assert.h>
35#include <iprt/err.h>
36#include <iprt/string.h>
37#include <iprt/utf16.h>
38
39#include <iprt/win/windows.h>
40
41
42RTDECL(int) RTSystemShutdown(RTMSINTERVAL cMsDelay, uint32_t fFlags, const char *pszLogMsg)
43{
44 AssertPtrReturn(pszLogMsg, VERR_INVALID_POINTER);
45 AssertReturn(!(fFlags & ~RTSYSTEM_SHUTDOWN_VALID_MASK), VERR_INVALID_PARAMETER);
46
47 /*
48 * Before we start, try grant the necessary privileges.
49 */
50 DWORD dwErr;
51 HANDLE hToken = NULL;
52 if (OpenThreadToken(GetCurrentThread(), TOKEN_ADJUST_PRIVILEGES, TRUE /*OpenAsSelf*/, &hToken))
53 dwErr = NO_ERROR;
54 else
55 {
56 dwErr = GetLastError();
57 if (dwErr == ERROR_NO_TOKEN)
58 {
59 if (OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hToken))
60 dwErr = NO_ERROR;
61 else
62 dwErr = GetLastError();
63 }
64 }
65 if (dwErr == NO_ERROR)
66 {
67 union
68 {
69 TOKEN_PRIVILEGES TokenPriv;
70 char ab[sizeof(TOKEN_PRIVILEGES) + sizeof(LUID_AND_ATTRIBUTES)];
71 } u;
72 u.TokenPriv.PrivilegeCount = 1;
73 u.TokenPriv.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
74 if (LookupPrivilegeValue(NULL /*localhost*/, SE_SHUTDOWN_NAME, &u.TokenPriv.Privileges[0].Luid))
75 {
76 if (!AdjustTokenPrivileges(hToken,
77 FALSE /*DisableAllPrivileges*/,
78 &u.TokenPriv,
79 RT_UOFFSETOF(TOKEN_PRIVILEGES, Privileges[1]),
80 NULL,
81 NULL) )
82 dwErr = GetLastError();
83 }
84 else
85 dwErr = GetLastError();
86 CloseHandle(hToken);
87 }
88
89 /*
90 * Do some parameter conversion.
91 */
92 PRTUTF16 pwszLogMsg;
93 int rc = RTStrToUtf16(pszLogMsg, &pwszLogMsg);
94 if (RT_FAILURE(rc))
95 return rc;
96 DWORD cSecsTimeout = (cMsDelay + 499) / 1000;
97
98 /*
99 * If we're told to power off the system, we should try use InitiateShutdownW (6.0+)
100 * or ExitWindowsEx (3.50) rather than InitiateSystemShutdownW, because these other
101 * APIs allows us to explicitly specify that we want to power off.
102 *
103 * Note! For NT version 4, 3.51, and 3.50 the system may instaed reboot since the
104 * x86 HALs typically didn't know how to perform a power off.
105 */
106 bool fDone = false;
107 if ( (fFlags & RTSYSTEM_SHUTDOWN_ACTION_MASK) == RTSYSTEM_SHUTDOWN_POWER_OFF
108 || (fFlags & RTSYSTEM_SHUTDOWN_ACTION_MASK) == RTSYSTEM_SHUTDOWN_POWER_OFF_HALT)
109 {
110 /* This API has the grace period thing. */
111 decltype(InitiateShutdownW) *pfnInitiateShutdownW;
112 pfnInitiateShutdownW = (decltype(InitiateShutdownW) *)GetProcAddress(GetModuleHandleW(L"ADVAPI32.DLL"), "InitiateShutdownW");
113 if (pfnInitiateShutdownW)
114 {
115 DWORD fShutdownFlags = SHUTDOWN_POWEROFF;
116 if (fFlags & RTSYSTEM_SHUTDOWN_FORCE)
117 fShutdownFlags |= SHUTDOWN_FORCE_OTHERS | SHUTDOWN_FORCE_SELF;
118 DWORD fReason = SHTDN_REASON_MAJOR_OTHER | (fFlags & RTSYSTEM_SHUTDOWN_PLANNED ? SHTDN_REASON_FLAG_PLANNED : 0);
119 dwErr = pfnInitiateShutdownW(NULL /*pwszMachineName*/, pwszLogMsg, cSecsTimeout, fShutdownFlags, fReason);
120 if (dwErr == ERROR_INVALID_PARAMETER)
121 {
122 fReason &= ~SHTDN_REASON_FLAG_PLANNED; /* just in case... */
123 dwErr = pfnInitiateShutdownW(NULL /*pwszMachineName*/, pwszLogMsg, cSecsTimeout, fShutdownFlags, fReason);
124 }
125 if (dwErr == ERROR_SUCCESS)
126 {
127 rc = VINF_SUCCESS;
128 fDone = true;
129 }
130 }
131
132 if (!fDone)
133 {
134 /* No grace period here, too bad. */
135 decltype(ExitWindowsEx) *pfnExitWindowsEx;
136 pfnExitWindowsEx = (decltype(ExitWindowsEx) *)GetProcAddress(GetModuleHandleW(L"USER32.DLL"), "ExitWindowsEx");
137 if (pfnExitWindowsEx)
138 {
139 DWORD fExitWindows = EWX_POWEROFF | EWX_SHUTDOWN;
140 if (fFlags & RTSYSTEM_SHUTDOWN_FORCE)
141 fExitWindows |= EWX_FORCE | EWX_FORCEIFHUNG;
142
143 if (pfnExitWindowsEx(fExitWindows, SHTDN_REASON_MAJOR_OTHER))
144 fDone = true;
145 else if (pfnExitWindowsEx(fExitWindows & ~EWX_FORCEIFHUNG, SHTDN_REASON_MAJOR_OTHER))
146 fDone = true;
147 }
148 }
149 }
150
151 /*
152 * Fall back on the oldest API.
153 */
154 if (!fDone)
155 {
156 BOOL fRebootAfterShutdown = (fFlags & RTSYSTEM_SHUTDOWN_ACTION_MASK) == RTSYSTEM_SHUTDOWN_REBOOT
157 ? TRUE : FALSE;
158 BOOL fForceAppsClosed = fFlags & RTSYSTEM_SHUTDOWN_FORCE ? TRUE : FALSE;
159 if (InitiateSystemShutdownW(NULL /*pwszMachineName = NULL = localhost*/,
160 pwszLogMsg,
161 cSecsTimeout,
162 fForceAppsClosed,
163 fRebootAfterShutdown))
164 rc = (fFlags & RTSYSTEM_SHUTDOWN_ACTION_MASK) == RTSYSTEM_SHUTDOWN_HALT ? VINF_SYS_MAY_POWER_OFF : VINF_SUCCESS;
165 else
166 rc = RTErrConvertFromWin32(dwErr);
167 }
168
169 RTUtf16Free(pwszLogMsg);
170 return rc;
171}
172RT_EXPORT_SYMBOL(RTSystemShutdown);
173
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