/*++ BUILD Version: 0001 // Increment this if a change has global effects Copyright (c) 1992-1994 Microsoft Corporation Module Name: perfname.c Abstract: This file returns the Counter names or help text. Author: HonWah Chan 10/12/93 Revision History: --*/ #define UNICODE #define _UNICODE // // Include files // #pragma warning(disable:4306) #include #include #include #include #include #include #include "ntconreg.h" #include "perflib.h" #pragma warning(default:4306) #define QUERY_GLOBAL 1 #define QUERY_ITEMS 2 #define QUERY_FOREIGN 3 #define QUERY_COSTLY 4 #define QUERY_COUNTER 5 #define QUERY_HELP 6 #define QUERY_ADDCOUNTER 7 #define QUERY_ADDHELP 8 #define tohexdigit(x) ((CHAR) (((x) < 10) ? ((x) + L'0') : ((x) + L'a' - 10))) #define LANG_ID_START 25 const WCHAR FileNameTemplate[] = L"\\SystemRoot\\system32\\perf0000.dat"; const WCHAR SubFileNameTemplate[] = L"\\SystemRoot\\system32\\prf00000.dat"; const WCHAR DefaultLangId[] = L"009"; extern WCHAR COUNTER_STRING[]; extern WCHAR HELP_STRING[]; extern WCHAR ADDCOUNTER_STRING[]; extern WCHAR ADDHELP_STRING[]; DWORD GetLangIdFromSzLang(LPWSTR szLangId); VOID Perflib004Update(LPWSTR pLangIdRequest); LPWSTR PerflibCheckPerfFile(LPWSTR szLangId, LPWSTR szRtnLang, DWORD dwRtnLang); NTSTATUS PerfGetNames( IN DWORD QueryType, IN PUNICODE_STRING lpValueName, OUT LPBYTE lpData, OUT LPDWORD lpcbData, OUT LPDWORD lpcbLen OPTIONAL, IN LPWSTR lpLanguageId OPTIONAL ) /*++ PerfGetCounterName Arguments - Get either counter names or help text for the given language. If there is no language ID specified in the input, the default English version is returned. Inputs - QueryType - Either QUERY_COUNTER or QUERY_HELP or QUERY_ADDCOUNTER or QUERY_ADDHELP lpValueName - Either "Counter ???" or "Explain ???" or "Addcounter ???" or "Addexplain ???" lpData - pointer to a buffer to receive the names lpcbData - pointer to a variable containing the size in bytes of the output buffer; on output, will receive the number of bytes actually returned lpcbLen - Return the number of bytes to transmit to the client (used by RPC) (optional). lpLanguageId - Input string for the language id desired. Return Value - error code indicating status of call or ERROR_SUCCESS if all ok --*/ { UNICODE_STRING NtFileName; NTSTATUS Status; WCHAR Names[50], QueryChar; WCHAR szRtnLang[5]; ULONG NameLen, StartIndex; OBJECT_ATTRIBUTES ObjectAttributes; IO_STATUS_BLOCK IoStatus; FILE_STANDARD_INFORMATION FileInformation; HANDLE File; LPWSTR pLangIdRequest; LPWSTR pTmpLangId; BOOL bAddNames, bSubLang; HRESULT hError; // build the file name hError = StringCchCopyW(Names, 50, FileNameTemplate); TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, ARG_DEF(ARG_TYPE_WSTR, 2), 0, &QueryType, sizeof(QueryType), lpValueName->Buffer, WSTRSIZE(lpValueName->Buffer), NULL)); if (QueryType == QUERY_ADDCOUNTER || QueryType == QUERY_ADDHELP) { bAddNames = TRUE; } else { bAddNames = FALSE; } if (QueryType == QUERY_COUNTER || QueryType == QUERY_ADDCOUNTER) { QueryChar = L'c'; NameLen = (ULONG) wcslen(COUNTER_STRING); } else { NameLen = (ULONG) wcslen(HELP_STRING); QueryChar = L'h'; } if (lpLanguageId) { DWORD dwLangId = PRIMARYLANGID(GetLangIdFromSzLang(lpLanguageId)); if (dwLangId == 0x004) { pLangIdRequest = lpLanguageId; } else { pLangIdRequest = PerflibCheckPerfFile(lpLanguageId, szRtnLang, 5); if (pLangIdRequest == NULL) { // It is possible that no PERFCxx.DAT and PERFHxxx.DAT files are present. // Restore pLangIdRequest to original value if PerflibCheckPerfFile() returns // NULL. // pLangIdRequest = lpLanguageId; } } } else { // get the lang id from the input lpValueName pLangIdRequest = lpValueName->Buffer + NameLen; do { if (lpValueName->Length < (NameLen + 3) * sizeof(WCHAR)) { // lpValueName is too small to contain the lang id, use default pLangIdRequest = (LPWSTR) DefaultLangId; break; } if (*pLangIdRequest >= L'0' && *pLangIdRequest <= L'9') { // found the first digit break; } pLangIdRequest++; NameLen++; } while (NameLen > 0); // always TRUE // Specially for 004 (CHT and CHS) if this is a Whistler upgrade. // Need to copy perfc004.dat/perfh004.dat to prfc0?04.dat/prfh0?04.dat // then rename perfc004.dat/perfh004.dat so that PERFLIB will not find // them in the future. // Currently this is a hack. // Perflib004Update(pLangIdRequest); pTmpLangId = pLangIdRequest; pLangIdRequest = PerflibCheckPerfFile(pTmpLangId, szRtnLang, 5); if (pLangIdRequest == NULL) { pLangIdRequest = pTmpLangId; } } bSubLang = ((pLangIdRequest[3] >= L'0') && (pLangIdRequest[3] <= L'9')); StartIndex = LANG_ID_START; if (bSubLang) { StartIndex = LANG_ID_START - 1; hError = StringCchCopyW(Names, 50, SubFileNameTemplate); } Names[StartIndex] = QueryChar; Names[StartIndex + 1] = *pLangIdRequest++; Names[StartIndex + 2] = *pLangIdRequest++; Names[StartIndex + 3] = *pLangIdRequest++; if (bSubLang) { Names[StartIndex + 4] = *pLangIdRequest; } TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, ARG_DEF(ARG_TYPE_WSTR, 1), 0, Names, WSTRSIZE(Names), NULL)); RtlInitUnicodeString(& NtFileName, Names); // open the file for info InitializeObjectAttributes( &ObjectAttributes, &NtFileName, OBJ_CASE_INSENSITIVE, (HANDLE)NULL, NULL ); if (bAddNames) { // writing name to data file LARGE_INTEGER ByteOffset; ByteOffset.LowPart = ByteOffset.HighPart = 0; Status = NtCreateFile( &File, SYNCHRONIZE | GENERIC_WRITE, &ObjectAttributes, &IoStatus, NULL, // no initial size FILE_ATTRIBUTE_NORMAL, FILE_SHARE_READ, FILE_SUPERSEDE, // always create FILE_SYNCHRONOUS_IO_NONALERT, NULL, // no ea buffer 0 // no ea buffer ); if (!NT_SUCCESS( Status )) { TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); return( Status ); } if (ARGUMENT_PRESENT(lpData) && ARGUMENT_PRESENT(lpcbData)) { Status = NtWriteFile( File, NULL, NULL, NULL, &IoStatus, lpData, *lpcbData, &ByteOffset, NULL ); } else { Status = ERROR_INVALID_PARAMETER; } if (!NT_SUCCESS( Status )) { TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); NtClose( File ); return( Status ); } } else { // reading name from data file Status = NtOpenFile( &File, SYNCHRONIZE | GENERIC_READ, &ObjectAttributes, &IoStatus, FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE, FILE_SYNCHRONOUS_IO_NONALERT | FILE_NON_DIRECTORY_FILE ); if (!NT_SUCCESS( Status )) { TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); return( Status ); } Status = NtQueryInformationFile( File, &IoStatus, (PVOID)&FileInformation, sizeof( FileInformation ), FileStandardInformation ); if (NT_SUCCESS( Status )) { if (FileInformation.EndOfFile.HighPart) { Status = STATUS_BUFFER_OVERFLOW; TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); } } if (!NT_SUCCESS( Status )) { TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); NtClose( File ); return( Status ); } if (! ARGUMENT_PRESENT(lpData) || ! ARGUMENT_PRESENT(lpcbData) || * lpcbData < FileInformation.EndOfFile.LowPart) { NtClose( File ); TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); if (ARGUMENT_PRESENT (lpcbLen)) { // no data yet for the rpc *lpcbLen = 0; } if (ARGUMENT_PRESENT(lpcbData)) { *lpcbData = FileInformation.EndOfFile.LowPart; } if (ARGUMENT_PRESENT (lpData)) { TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); return (STATUS_BUFFER_OVERFLOW); } return(STATUS_SUCCESS); } Status = NtReadFile( File, NULL, NULL, NULL, &IoStatus, lpData, FileInformation.EndOfFile.LowPart, NULL, NULL ); if (NT_SUCCESS( Status )) { Status = IoStatus.Status; if (NT_SUCCESS( Status )) { if (IoStatus.Information != FileInformation.EndOfFile.LowPart) { Status = STATUS_END_OF_FILE; } } } else { TRACE((WINPERF_DBG_TRACE_INFO), (&PerflibGuid, __LINE__, PERF_GET_NAMES, 0, Status, NULL)); } if (NT_SUCCESS( Status )) { if (ARGUMENT_PRESENT(lpcbData)) { *lpcbData = FileInformation.EndOfFile.LowPart; } if (ARGUMENT_PRESENT(lpcbLen)) { *lpcbLen = FileInformation.EndOfFile.LowPart; } } } // end of reading names NtClose (File); return (Status); } VOID PerfGetPrimaryLangId( DWORD dwLangId, WCHAR * PrimaryLangId ) { // build the native language id WCHAR LangId; WCHAR nDigit; LangId = (WCHAR) PRIMARYLANGID(dwLangId); nDigit = (WCHAR) (LangId >> 8); PrimaryLangId[0] = tohexdigit(nDigit); nDigit = (WCHAR) (LangId & 0XF0) >> 4; PrimaryLangId[1] = tohexdigit(nDigit); nDigit = (WCHAR) (LangId & 0xF); PrimaryLangId[2] = tohexdigit(nDigit); PrimaryLangId[3] = L'\0'; } VOID PerfGetLangId( WCHAR * FullLangId ) { WCHAR LangId; WCHAR nDigit; LangId = (WCHAR) GetUserDefaultUILanguage(); nDigit = (LangId & 0xF000) >> 12; FullLangId[0] = tohexdigit(nDigit); nDigit = (LangId & 0x0F00) >> 8; FullLangId[1] = tohexdigit(nDigit); nDigit = (LangId & 0x00F0) >> 4; FullLangId[2] = tohexdigit(nDigit); nDigit = LangId & 0xF; FullLangId[3] = tohexdigit(nDigit); FullLangId[4] = L'\0'; } DWORD GetLangIdFromSzLang( LPWSTR szLangId ) { DWORD dwLangId = 0; DWORD dwLangLen = lstrlenW(szLangId); DWORD i; WCHAR wszDigit; for (i = 0; i < dwLangLen; i ++) { dwLangId <<= 4; wszDigit = szLangId[i]; if (wszDigit >= L'0' && wszDigit <= L'9') { dwLangId += (wszDigit - L'0'); } else if (wszDigit >= L'a' && wszDigit <= 'f') { dwLangId += (10 + wszDigit - L'a'); } else if (wszDigit >= L'A' && wszDigit <= 'F') { dwLangId += (10 + wszDigit - L'A'); } else { dwLangId = 0; break; } } return dwLangId; } LPCWSTR szCtrLangFile = L"perfc"; LPCWSTR szCtrSubLangFile = L"prfc"; LPCWSTR szHlpLangFile = L"perfh"; LPCWSTR szHlpSubLangFile = L"prfh"; LPCWSTR sz004CtrLangFile = L"perfc004.dat"; LPCWSTR sz004HlpLangFile = L"perfh004.dat"; LPCWSTR szFileExt = L".dat"; LPWSTR PerflibFindCounterFile( IN LPWSTR szFilePath, IN BOOLEAN bCheckSubLang, IN LPWSTR szLangId, OUT LPWSTR szRtnLang, IN DWORD dwRtnLang ) { WCHAR szThisLang[5]; WCHAR szPrimaryLang[5]; DWORD dwThisLang; DWORD dwFileLen; WIN32_FIND_DATAW FindFileData; DWORD bResult = FALSE; HANDLE hFile = NULL; HRESULT hError; hFile = FindFirstFileExW(szFilePath, FindExInfoStandard, & FindFileData, FindExSearchNameMatch, NULL, 0); RtlZeroMemory(szRtnLang, dwRtnLang * sizeof(WCHAR)); if (hFile != INVALID_HANDLE_VALUE) { if (bCheckSubLang) { do { dwFileLen = lstrlenW(FindFileData.cFileName); if (dwFileLen == 12) { ZeroMemory(szThisLang, 5 * sizeof(WCHAR)); ZeroMemory(szPrimaryLang, 5 * sizeof(WCHAR)); hError = StringCchCopyW(szThisLang, 5, (LPWSTR) (FindFileData.cFileName + (dwFileLen - 8))); dwThisLang = GetLangIdFromSzLang(szThisLang); if (dwThisLang != 0) { PerfGetPrimaryLangId(dwThisLang, szPrimaryLang); bResult = (lstrcmpiW(szPrimaryLang, szLangId) == 0); if (bResult == TRUE) { hError = StringCchCopyW(szRtnLang, dwRtnLang, szThisLang); } } } } while (FindNextFileW(hFile, & FindFileData)); } else { bResult = TRUE; hError = StringCchCopyW(szRtnLang, dwRtnLang, szLangId); } FindClose(hFile); } return bResult ? szRtnLang : NULL; } LPWSTR PerflibCheckPerfFile( IN LPWSTR LangId, OUT LPWSTR szRtnLang, IN DWORD dwRtnLang ) { DWORD dwSysDir = GetSystemDirectoryW(NULL, 0); DWORD dwSrchDir; DWORD dwLangLen = lstrlenW(LangId); DWORD dwLangId; LPWSTR szLangId = NULL; WCHAR szLang[5]; LPWSTR szSysDir = NULL; LPWSTR szSrchDir = NULL; HRESULT hError; if (dwSysDir == 0) goto Cleanup; dwSrchDir = lstrlenW(szCtrSubLangFile) + lstrlenW(LangId) + lstrlenW(szFileExt) + 1; if (dwSrchDir < 13) dwSrchDir = 13; // "8.3" filename format with NULL dwSrchDir += dwSysDir + 1; szSysDir = (LPWSTR) ALLOCMEM(sizeof(WCHAR) * (dwSysDir + 1 + dwSrchDir)); if (szSysDir == NULL) goto Cleanup; else if (GetSystemDirectoryW(szSysDir, dwSysDir + 1) == 0) goto Cleanup; szSrchDir = szSysDir + dwSysDir + 1; if (dwLangLen == 4) { hError = StringCchPrintfW(szSrchDir, dwSrchDir, L"%ws\\%ws%ws%ws", szSysDir, szCtrSubLangFile, LangId, szFileExt); szLangId = PerflibFindCounterFile(szSrchDir, FALSE, LangId, szRtnLang, dwRtnLang); if (szLangId == NULL) { dwLangId = GetLangIdFromSzLang(LangId); if (dwLangId != 0) { ZeroMemory(szLang, sizeof(WCHAR) * 5); PerfGetPrimaryLangId(dwLangId, szLang); ZeroMemory(szSrchDir, sizeof(WCHAR) * dwSrchDir); hError = StringCchPrintfW(szSrchDir, dwSrchDir, L"%ws\\%ws%ws%ws", szSysDir, szCtrLangFile, szLang, szFileExt); szLangId = PerflibFindCounterFile(szSrchDir, FALSE, szLang, szRtnLang, dwRtnLang); } } } else { // dwLangId should be 3, this is primary UserDefaultUILanguage. // hError = StringCchPrintfW(szSrchDir, dwSrchDir, L"%ws\\%ws%ws%ws", szSysDir, szCtrLangFile, LangId, szFileExt); szLangId = PerflibFindCounterFile(szSrchDir, FALSE, LangId, szRtnLang, dwRtnLang); if (szLangId == NULL) { ZeroMemory(szSrchDir, sizeof(WCHAR) * dwSrchDir); hError = StringCchPrintfW(szSrchDir, dwSrchDir, L"%ws\\%ws??%ws%ws", szSysDir, szCtrSubLangFile, (LPWSTR) (LangId + 1), szFileExt); szLangId = PerflibFindCounterFile(szSrchDir, TRUE, LangId, szRtnLang, dwRtnLang); } } Cleanup: FREEMEM(szSysDir); return szLangId; } VOID PerflibRename004File( IN LPWSTR szSysDir, IN LPWSTR szLangId, IN BOOLEAN bCounterFile ) { DWORD dwSrchDir = lstrlenW(szSysDir) + 2 + 13 + 4; // 13 is for "8.3" filename with NULL; 4 is for ".tmp" LPWSTR szTmpFile = NULL; LPWSTR szSrchDir = NULL; LPWSTR szName; HRESULT hError; szSrchDir = ALLOCMEM(2 * sizeof(WCHAR) * dwSrchDir); if (szSrchDir == NULL) goto Cleanup; szTmpFile = szSrchDir + dwSrchDir; szName = (LPWSTR) ((bCounterFile) ? (sz004CtrLangFile) : (sz004HlpLangFile)); hError = StringCchPrintfW(szSrchDir, dwSrchDir, L"%ws\\%ws", szSysDir, szName); if (szLangId) { szName = (LPWSTR) ((bCounterFile) ? (szCtrSubLangFile) : (szHlpSubLangFile)); hError = StringCchPrintfW(szTmpFile, dwSrchDir, L"%ws\\%ws%ws%ws", szSysDir, szName, szLangId, szFileExt); CopyFileW(szSrchDir, szTmpFile, FALSE); } else { szName = (LPWSTR) ((bCounterFile) ? (sz004CtrLangFile) : (sz004HlpLangFile)); hError = StringCchPrintfW(szTmpFile, dwSrchDir, L"%ws\\%ws.tmp", szSysDir, szName); DeleteFileW(szTmpFile); MoveFileW(szSrchDir, szTmpFile); } Cleanup: FREEMEM(szSrchDir); } VOID Perflib004Update( IN LPWSTR pLangIdRequest ) { LPWSTR szSysDir = NULL; LPWSTR szTmpFile = NULL; LPWSTR szRtnLang = NULL; DWORD dwSysDir; DWORD dwTmpDir; HRESULT hError; if (GetLangIdFromSzLang(pLangIdRequest) != LANG_CHINESE) goto Cleanup; dwSysDir = GetSystemDirectoryW(NULL, 0); dwTmpDir = lstrlenW(szCtrSubLangFile) + 5 + lstrlenW(szFileExt) + 1; // 5 is for LangId if (dwSysDir == 0) goto Cleanup; szSysDir = (LPWSTR) ALLOCMEM(sizeof(WCHAR) * (dwSysDir + 1 + dwTmpDir + 5)); if (szSysDir == NULL) goto Cleanup; szRtnLang = szSysDir + dwSysDir + 1; szTmpFile = szRtnLang + 5; // Search whether PERFC004.DAT and PRFC0?04.DAT are in System32 directory // if (GetSystemDirectoryW(szSysDir, dwSysDir + 1) == 0) goto Cleanup; if (SearchPathW(szSysDir, sz004CtrLangFile, NULL, 0, NULL, NULL) == 0) goto Cleanup; PerfGetLangId(szRtnLang); hError = StringCchPrintfW(szTmpFile, dwTmpDir, L"%ws%ws%ws", szCtrSubLangFile, szRtnLang, szFileExt); if (SearchPathW(szSysDir, szTmpFile, NULL, 0, NULL, NULL) == 0) goto Cleanup; // Found PERFC004.DAT, assume that PERFH004.DAT is also there. // Rename to PRFC0?04.DAT/PRFH0?04.DAT // PerflibRename004File(szSysDir, szRtnLang, TRUE); PerflibRename004File(szSysDir, szRtnLang, FALSE); PerflibRename004File(szSysDir, NULL, TRUE); PerflibRename004File(szSysDir, NULL, FALSE); Cleanup: FREEMEM(szSysDir); return; }