diff --git a/README.md b/README.md index d079bf89..9751a588 100644 --- a/README.md +++ b/README.md @@ -49,6 +49,8 @@ OPTIONS: --uefi Print UEFI boot information. --shares List network mapped drives. --audio List audio devices. + --public-ip Print public IP address. + --product-policy Print ProductPolicy values. ``` ## Credits diff --git a/libnw/libnw.c b/libnw/libnw.c index 35c6e6fa..c824e209 100644 --- a/libnw/libnw.c +++ b/libnw/libnw.c @@ -86,6 +86,8 @@ VOID NW_Print(LPCSTR lpFileName) NW_Audio(); if (NWLC->PublicIpInfo) NW_PublicIp(); + if (NWLC->ProductPolicyInfo) + NW_ProductPolicy(); NW_Libinfo(); NW_Export(NWLC->NwRoot, NWLC->NwFile); } diff --git a/libnw/libnw.h b/libnw/libnw.h index ae3dd330..6517e090 100644 --- a/libnw/libnw.h +++ b/libnw/libnw.h @@ -42,6 +42,7 @@ typedef struct _NWLIB_CONTEXT BOOL ShareInfo; BOOL AudioInfo; BOOL PublicIpInfo; + BOOL ProductPolicyInfo; BOOL Debug; BOOL HideSensitive; @@ -117,6 +118,7 @@ PNODE NW_Uefi(VOID); PNODE NW_NetShare(VOID); PNODE NW_Audio(VOID); PNODE NW_PublicIp(VOID); +PNODE NW_ProductPolicy(VOID); VOID NWL_GetUptime(CHAR* szUptime, DWORD dwSize); VOID NWL_GetHostname(CHAR* szHostname); diff --git a/libnw/libnw.vcxproj b/libnw/libnw.vcxproj index 28739e7d..e426eb6f 100644 --- a/libnw/libnw.vcxproj +++ b/libnw/libnw.vcxproj @@ -166,6 +166,7 @@ copy "$(SolutionDir)winring0\HwRwDrvx64.sys" "$(SolutionDir)$(Platform)\$(Config + diff --git a/libnw/libnw.vcxproj.filters b/libnw/libnw.vcxproj.filters index 210ad610..546fc29e 100644 --- a/libnw/libnw.vcxproj.filters +++ b/libnw/libnw.vcxproj.filters @@ -128,6 +128,9 @@ 源文件 + + 源文件 + diff --git a/libnw/productpolicy.c b/libnw/productpolicy.c new file mode 100644 index 00000000..23b47e34 --- /dev/null +++ b/libnw/productpolicy.c @@ -0,0 +1,96 @@ +// SPDX-License-Identifier: Unlicense + +#include +#include +#include + +#include "libnw.h" +#include "utils.h" + + +#pragma pack(1) +typedef struct _PRODUCT_POLICY_HEADER +{ + DWORD dwSize; // total size, including this header + DWORD dwArraySize; // size of values array that follows this header + DWORD dwEndSize; // size of end marker that follows the values array + DWORD unused; + DWORD dwVersion; // must be 1 +} PRODUCT_POLICY_HEADER, * PPRODUCT_POLICY_HEADER; + +typedef struct _PRODUCT_POLICY_VALUE +{ + UINT16 wSize; // total size, including this header + UINT16 wNameSize; // size of name + // type of data + // 1 - REG_SZ, 2- REG_EXPAND_SZ, 3 - REG_BINARY, 4 - REG_BINARY + UINT16 wType; + UINT16 wDataSize; // size of data + DWORD dwFlags; + DWORD unused; +} PRODUCT_POLICY_VALUE, * PPRODUCT_POLICY_VALUE; + +#define PRODUCT_POLICY_END_MARK 0x45ul // dword +#pragma pack() + +PNODE NW_ProductPolicy(VOID) +{ + CHAR hex[] = "0123456789ABCDEF"; + PNODE pp = NWL_NodeAlloc("Product Policy", 0); + if (NWLC->ProductPolicyInfo) + NWL_NodeAppendChild(NWLC->NwRoot, pp); + DWORD dwType; + DWORD ppSize; + PPRODUCT_POLICY_HEADER ppHeader = NULL; + PPRODUCT_POLICY_VALUE ppValue; + PUINT8 ppData = NWL_NtGetRegValue(HKEY_LOCAL_MACHINE, + L"SYSTEM\\CurrentControlSet\\Control\\ProductOptions", + L"ProductPolicy", &ppSize, &dwType); + if (ppData == NULL || dwType != REG_BINARY || ppSize <= sizeof(PRODUCT_POLICY_HEADER)) + goto end; + ppHeader = (PPRODUCT_POLICY_HEADER)ppData; + if (ppSize != ppHeader->dwSize || ppHeader->dwVersion != 1) + goto end; + if (ppHeader->dwArraySize + ppHeader->dwEndSize + sizeof(PRODUCT_POLICY_HEADER) > ppSize) + goto end; + for (ppValue = (PPRODUCT_POLICY_VALUE)(ppData + sizeof(PRODUCT_POLICY_HEADER)); + (PUINT8)ppValue < ppData + sizeof(PRODUCT_POLICY_HEADER) + ppHeader->dwArraySize; + ppValue = (PPRODUCT_POLICY_VALUE)((PUINT8)ppValue + ppValue->wSize)) + { + CHAR* pName = _strdup(NWL_Ucs2ToUtf8((WCHAR*)((PUINT8)ppValue + sizeof(PRODUCT_POLICY_VALUE)))); + PUINT8 pValueData = (PUINT8)ppValue + sizeof(PRODUCT_POLICY_VALUE) + ppValue->wNameSize; + pName[ppValue->wNameSize / sizeof(WCHAR)] = '\0'; + switch (ppValue->wType) + { + case REG_SZ: + case REG_EXPAND_SZ: + case REG_MULTI_SZ: //FIXME + NWL_NodeAttrSet(pp, pName, NWL_Ucs2ToUtf8((LPCWSTR)pValueData), 0); + break; + case REG_DWORD: + NWL_NodeAttrSetf(pp, pName, NAFLG_FMT_NUMERIC, "%lu", *(PDWORD)pValueData); + break; + case REG_QWORD: + NWL_NodeAttrSetf(pp, pName, NAFLG_FMT_NUMERIC, "%llu", *(PUINT64)pValueData); + break; + default: + { + CHAR* tmp = calloc(ppValue->wDataSize, 3); + for (UINT i = 0; i < ppValue->wDataSize; i++) + { + tmp[i * 3] = hex[(pValueData[i] & 0xF0) >> 4]; + tmp[i * 3 + 1] = hex[pValueData[i] & 0x0F]; + if (i < ppValue->wDataSize - 1U) + tmp[i * 3 + 2] = ' '; + } + NWL_NodeAttrSet(pp, pName, tmp, 0); + free(tmp); + } + break; + } + free(pName); + } +end: + free(ppData); + return pp; +} diff --git a/nwinfo.c b/nwinfo.c index f5cb5376..fc9baf6d 100644 --- a/nwinfo.c +++ b/nwinfo.c @@ -45,6 +45,7 @@ static void nwinfo_help(void) " --shares Print network mapped drives.\n" " --audio Print audio devices.\n" " --public-ip Print public IP address.\n" + " --product-policy Print ProductPolicy.\n"); } int main(int argc, char* argv[]) @@ -142,6 +143,8 @@ int main(int argc, char* argv[]) nwContext.AudioInfo = TRUE; else if (_stricmp(argv[i], "--public-ip") == 0) nwContext.PublicIpInfo = TRUE; + else if (_stricmp(argv[i], "--product-policy") == 0) + nwContext.ProductPolicyInfo = TRUE; else if (_stricmp(argv[i], "--debug") == 0) nwContext.Debug = TRUE; else if (_stricmp(argv[i], "--hide-sensitive") == 0)