Windows2000/private/shell/shlwapi/machinfo.cpp
2020-09-30 17:12:32 +02:00

547 lines
17 KiB
C++

// machinfo.cpp - SHGetMachineInfo and related functions
#include "priv.h"
#include <dbt.h>
#include <cfgmgr32.h>
#include <apithk.h>
#ifndef UNIX
#include <batclass.h>
const GUID GUID_DEVICE_BATTERY = { 0x72631e54L, 0x78A4, 0x11d0,
{ 0xbc, 0xf7, 0x00, 0xaa, 0x00, 0xb7, 0xb3, 0x2a } };
#include <hydra\winsta.h>
#endif
// Win95 does not decorate BroadcastSystemMessage, so we can't either.
#undef BroadcastSystemMessage
extern "C" {
WINUSERAPI long WINAPI
BroadcastSystemMessage(DWORD, LPDWORD, UINT, WPARAM, LPARAM);
};
/**
* DOCK STATE - Win95, Win98, and WinNT all do this differently (yuck)
**/
C_ASSERT(GMID_NOTDOCKABLE == CM_HWPI_NOT_DOCKABLE);
C_ASSERT(GMID_UNDOCKED == CM_HWPI_UNDOCKED);
C_ASSERT(GMID_DOCKED == CM_HWPI_DOCKED);
#if defined(_X86_) && !defined(UNIX)
typedef struct CMHDR {
LPVOID pArgs;
DWORD dwService;
DWORD dwRet;
} CMHDR, *PCMHDR;
#define CONFIGMG_Get_Hardware_Profile_Info 0x00330052
// GetDockedState95
DWORD GetDockedState95()
{
struct GHWPI95 { // Get_Hardware_Profile_Info parameter blk
CMHDR cmhdr;
ULONG ulIndex;
PHWPROFILEINFO_A pHWProfileInfo;
ULONG ulFlags;
HWPROFILEINFO_A HWProfileInfo;
} *pghwpi;
HANDLE hheap;
DWORD Result = GMID_NOTDOCKABLE; // assume the worst
#define HEAP_SHARED 0x04000000 /* put heap in shared memory--undoc'd */
// Win95 Configmg requires the parameter block to reside in the shared
// heap since we're going to do a cross-process SendMessage.
hheap = HeapCreate(HEAP_SHARED, 1, 4096);
if (hheap) {
// Allocate parameter block in shared memory
pghwpi = (struct GHWPI95 *)HeapAlloc(hheap, HEAP_ZERO_MEMORY,
sizeof(*pghwpi));
if (pghwpi) {
DWORD dwRecipients = BSM_VXDS;
pghwpi->cmhdr.dwRet = 0;
pghwpi->cmhdr.dwService = CONFIGMG_Get_Hardware_Profile_Info;
pghwpi->cmhdr.pArgs = &pghwpi->ulIndex;
pghwpi->ulIndex = 0xFFFFFFFF;
pghwpi->pHWProfileInfo = &pghwpi->HWProfileInfo;
pghwpi->ulFlags = 0;
// "Call" the service
BroadcastSystemMessage(0, &dwRecipients, WM_DEVICECHANGE,
DBT_CONFIGMGAPI32, (LPARAM)pghwpi);
if (pghwpi->cmhdr.dwRet == CR_SUCCESS) {
Result = pghwpi->HWProfileInfo.HWPI_dwFlags;
} else {
TraceMsg(DM_WARNING, "GetDockedState95: CONFIGMG did not respond");
}
}
HeapDestroy(hheap);
} else {
TraceMsg(DM_WARNING, "GetDockedState95: Unable to create shared heap");
}
return Result;
}
// On Win98, use the 32-bit interface to configmg.
CONFIGRET __cdecl
CallConfigmg98(DWORD dwServiceNumber, ...)
{
CONFIGRET cr;
HANDLE hCM;
hCM = CreateFileA("\\\\.\\CONFIGMG",
GENERIC_READ|GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL, OPEN_EXISTING, 0, NULL);
if (hCM != INVALID_HANDLE_VALUE) {
DWORD dwRet;
// Evil hack that works only on x86. Fortunately, this code is
// inside an #ifdef _X86_ block, so we're safe.
LPVOID pvArg = 1 + &dwServiceNumber;
if (DeviceIoControl(hCM, dwServiceNumber, &pvArg, sizeof(pvArg),
&cr, sizeof(cr), &dwRet, 0) &&
dwRet == sizeof(cr)) {
} else {
TraceMsg(DM_WARNING, "CallConfigmg98: CONFIGMG did not respond");
cr = CR_FAILURE;
}
CloseHandle(hCM);
} else {
TraceMsg(DM_WARNING, "CallConfigmg98: Couldn't connect to CONFIGMG");
cr = CR_FAILURE;
}
return cr;
}
DWORD GetDockedState98()
{
CONFIGRET cr;
DWORD Result = GMID_NOTDOCKABLE; // assume the worst
HWPROFILEINFO_A HWProfileInfo;
cr = CallConfigmg98(
0x80000000 + LOWORD(CONFIGMG_Get_Hardware_Profile_Info),
-1, // ulIndex, -1 means "current profile"
&HWProfileInfo, // PHWPROFILEINFO
0); // ulFlags
if (cr == CR_SUCCESS) {
Result = HWProfileInfo.HWPI_dwFlags;
}
return Result;
}
#endif
typedef BOOL (WINAPI *GETCURRENTHWPROFILEA)(LPHW_PROFILE_INFOA);
DWORD GetDockedStateNT()
{
HW_PROFILE_INFOA hpi;
GETCURRENTHWPROFILEA GetCurrentHwProfileA;
DWORD Result = GMID_NOTDOCKABLE; // assume the worst
GetCurrentHwProfileA = (GETCURRENTHWPROFILEA)
GetProcAddress(GetModuleHandle("ADVAPI32"),
"GetCurrentHwProfileA");
if (GetCurrentHwProfileA && GetCurrentHwProfileA(&hpi)) {
Result = hpi.dwDockInfo & (DOCKINFO_UNDOCKED | DOCKINFO_DOCKED);
// Wackiness: If the machine does not support docking, then
// NT returns >both< flags set. Go figure.
if (Result == (DOCKINFO_UNDOCKED | DOCKINFO_DOCKED)) {
Result = GMID_NOTDOCKABLE;
}
} else {
TraceMsg(DM_WARNING, "GetDockedStateNT: GetCurrentHwProfile failed");
}
return Result;
}
#if defined(_X86_) && !defined(UNIX)
// Platforms that support Win95/Win98 need to do version switching
DWORD GetDockedState()
{
if (g_bRunningOnNT) {
return GetDockedStateNT();
} else if (g_bRunningOnMemphis) {
return GetDockedState98();
} else {
return GetDockedState95();
}
}
#else
// Platforms that do not support Win95/Win98 can just call the NT version.
#define GetDockedState() GetDockedStateNT()
#endif
#ifndef UNIX
/**
* BATTERY STATE - Once again, Win95 and Win98 and NT all do it differently
**/
// Values for SYSTEM_POWER_STATUS.ACLineStatus
#define SPSAC_OFFLINE 0
#define SPSAC_ONLINE 1
// Values for SYSTEM_POWER_STATUS.BatteryFlag
#define SPSBF_NOBATTERY 128
// So many ways to detect batteries, so little time...
DWORD GetBatteryState()
{
// Since GMIB_HASBATTERY is cumulative (any battery turns it on)
// and GMIB_ONBATTERY is subtractive (any AC turns it off), the
// state you have to start in before you find a battery is
// GMIB_HASBATTERY off and GMIB_ONBATTERY on.
// dwResult & GMIB_ONBATTERY means we have yet to find AC power.
// dwResult & GMIB_HASBATTERY means we have found a non-UPS battery.
DWORD dwResult = GMIB_ONBATTERY;
// First try - IOCTL_BATTERY_QUERY_INFORMATION
// Windows 98 and Windows 2000 support IOCTL_BATTERY_QUERY_INFORMATION,
// which lets us enumerate the batteries and ask each one for information.
// Except that on Windows 98, we can enumerate only ACPI batteries.
// We still have to use VPOWERD to enumerate APM batteries.
// BUGBUG -- deal with Win98 APM batteries
HDEVINFO hdev = SetupDiGetClassDevs(&GUID_DEVICE_BATTERY, 0, 0,
DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
if (hdev != INVALID_HANDLE_VALUE) {
SP_DEVICE_INTERFACE_DATA did;
did.cbSize = sizeof(did);
// Stop at 100 batteries so we don't go haywire
for (int idev = 0; idev < 100; idev++) {
// Pre-set the error code because our DLLLOAD wrapper doesn't
// and Windows NT 4 supports SetupDiGetClassDevs but not
// SetupDiEnumDeviceInterfaces (go figure).
SetLastError(ERROR_NO_MORE_ITEMS);
if (SetupDiEnumDeviceInterfaces(hdev, 0, &GUID_DEVICE_BATTERY, idev, &did)) {
DWORD cbRequired = 0;
/*
* Ask for the required size then allocate it then fill it.
* Sigh. Windows NT and Windows 98 implement
* SetupDiGetDeviceInterfaceDetail differently if you are
* querying for the buffer size.
* Windows 98 returns FALSE, and GetLastError() returns
* ERROR_INSUFFICIENT_BUFFER.
* Windows NT returns TRUE.
* So we allow the cases either where the call succeeds or
* the call fails with ERROR_INSUFFICIENT_BUFFER.
*/
if (SetupDiGetDeviceInterfaceDetail(hdev, &did, 0, 0, &cbRequired, 0) ||
GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
PSP_DEVICE_INTERFACE_DETAIL_DATA pdidd;
pdidd = (PSP_DEVICE_INTERFACE_DETAIL_DATA)LocalAlloc(LPTR, cbRequired);
if (pdidd) {
pdidd->cbSize = sizeof(*pdidd);
if (SetupDiGetDeviceInterfaceDetail(hdev, &did, pdidd, cbRequired, &cbRequired, 0)) {
/*
* Finally enumerated a battery. Ask it for information.
*/
HANDLE hBattery = CreateFile(pdidd->DevicePath,
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL, OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL, NULL);
if (hBattery != INVALID_HANDLE_VALUE) {
/*
* Now you have to ask the battery for its tag.
*/
BATTERY_QUERY_INFORMATION bqi;
DWORD dwWait = 0;
DWORD dwOut;
bqi.BatteryTag = 0;
if (DeviceIoControl(hBattery, IOCTL_BATTERY_QUERY_TAG,
&dwWait, sizeof(dwWait),
&bqi.BatteryTag, sizeof(bqi.BatteryTag),
&dwOut, NULL) && bqi.BatteryTag) {
/*
* With the tag, you can query the battery info.
*/
BATTERY_INFORMATION bi;
bqi.InformationLevel = BatteryInformation;
bqi.AtRate = 0;
if (DeviceIoControl(hBattery, IOCTL_BATTERY_QUERY_INFORMATION,
&bqi, sizeof(bqi),
&bi, sizeof(bi),
&dwOut, NULL)) {
// Only system batteries count
if (bi.Capabilities & BATTERY_SYSTEM_BATTERY) {
if (!(bi.Capabilities & BATTERY_IS_SHORT_TERM)) {
dwResult |= GMIB_HASBATTERY;
}
/*
* And then query the battery status.
*/
BATTERY_WAIT_STATUS bws;
BATTERY_STATUS bs;
ZeroMemory(&bws, sizeof(bws));
bws.BatteryTag = bqi.BatteryTag;
if (DeviceIoControl(hBattery, IOCTL_BATTERY_QUERY_STATUS,
&bws, sizeof(bws),
&bs, sizeof(bs),
&dwOut, NULL)) {
if (bs.PowerState & BATTERY_POWER_ON_LINE) {
dwResult &= ~GMIB_ONBATTERY;
}
}
}
}
}
CloseHandle(hBattery);
}
}
LocalFree(pdidd);
}
}
} else {
// Enumeration failed - perhaps we're out of items
if (GetLastError() == ERROR_NO_MORE_ITEMS)
break;
}
}
SetupDiDestroyDeviceInfoList(hdev);
}
// On Windows NT, SetupDi tells us everything there is to know.
// So once you get this far, you're finished.
if (g_bRunningOnNT) {
goto finish;
}
// Second try - GetSystemPowerStatus
// On Windows 9x, GetSystemPowerStatus enumerates a disjoint set of
// batteries from SetupDi, so it's worth calling to find out.
SYSTEM_POWER_STATUS status;
if (GetSystemPowerStatus(&status)) {
// HACKHACK: Some APM BIOS implementations set BatteryFlag = 0
// instead of 128 or 255 when they don't have a
// battery, so we have to check both.
if (status.BatteryFlag != 0 &&
!(status.BatteryFlag & SPSBF_NOBATTERY)) {
// Found an APM battery.
dwResult |= GMIB_HASBATTERY;
}
if (status.ACLineStatus == SPSAC_ONLINE) {
dwResult &= ~GMIB_ONBATTERY;
}
}
#ifdef TRY_NtPowerInformation // Hopefully the Third Try won't be necessary
SYSTEM_POWER_CAPABILITIES caps;
// Third try - NtPowerInformation
// NtPowerInformation is supported on Win98 and Windows 2000, but not
// Windows 95 or NT4.
if (SUCCEEDED(NtPowerInformation(SystemPowerCapabilities, NULL, 0,
&caps, sizeof(caps)))) {
if (caps.BatteriesAreShortTerm) {
#error futz futz
}
if (caps.SystemBatteriesPresent && !fFoundUPS) {
#error futz futz
}
}
#endif // TRY_NtPowerInformation
finish:
// Final cleanup: If we didn't find a battery, then presume that we
// are on AC power.
if (!(dwResult & GMIB_HASBATTERY))
dwResult &= ~GMIB_ONBATTERY;
return dwResult;
}
/**
* TERMINAL SERVER CLIENT
* This is particularly gruesome because Terminal Server for NT4 SP3 goes
* to extraordinary lengths to prevent you from detecting it. Even the
* semi-documented NtCurrentPeb()->SessionId trick doesn't work on NT4 SP3.
* So we have to go to the totally undocumented winsta.dll to find out.
**/
BOOL g_fTSClient = -1; // Tri-state, 0 = no, 1 = yes, -1 = don't know
BOOL IsTSClientNT4(void)
{
BOOL fTS = FALSE; // Assume not
HINSTANCE hinstWinSta = LoadLibrary("winsta.dll");
if (hinstWinSta) {
PWINSTATIONQUERYINFORMATIONW WinStationQueryInformationW;
WINSTATIONINFORMATIONW wi;
WinStationQueryInformationW = (PWINSTATIONQUERYINFORMATIONW) GetProcAddress(hinstWinSta, "WinStationQueryInformationW");
if (WinStationQueryInformationW &&
WinStationQueryInformationW(SERVERNAME_CURRENT, LOGONID_CURRENT, WinStationInformation, &wi, sizeof(wi), NULL) &&
wi.LogonId != 0) {
fTS = TRUE;
}
FreeLibrary(hinstWinSta);
}
return fTS;
}
BOOL IsTSClient(void)
{
if (!g_bRunningOnNT) {
// Windows 9x doesn't support Terminal Server
return FALSE;
} else if (g_bRunningOnNT5OrHigher) {
// NT5 has a new system metric to detect this
return GetSystemMetrics(SM_REMOTESESSION);
} else {
// NT4 is gross and evil. This is slow, so cache the result.
if (g_fTSClient < 0)
g_fTSClient = IsTSClientNT4();
return g_fTSClient;
}
}
/**
* SHGetMachineInfo
**/
// SHGetMachineInfo
// Given an index, returns some info about that index. See shlwapi.w
// for documentation on the flags available.
STDAPI_(DWORD_PTR) SHGetMachineInfo(UINT gmi)
{
switch (gmi) {
case GMI_DOCKSTATE:
return GetDockedState();
case GMI_BATTERYSTATE:
return GetBatteryState();
// It smell like a laptop if it has a battery or if it can be docked.
case GMI_LAPTOP:
return (GetBatteryState() & GMIB_HASBATTERY) ||
(GetDockedState() != GMID_NOTDOCKABLE);
case GMI_TSCLIENT:
return IsTSClient();
}
TraceMsg(DM_WARNING, "SHGetMachineInfo: Unknown info query %d", gmi);
return 0;
}
#else
STDAPI_(DWORD_PTR) SHGetMachineInfo(UINT gmi)
{
// IEUNIX : Stubbed out this api to resolve undefind symbol in linking.
TraceMsg(DM_WARNING, "SHGetMachineInfo: Unknown info query %d", gmi);
return 0;
}
#endif