// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This source file has been modified by the following contributors: // Copyright (C) 2014-2020 Jan BOON (Kaetemi) // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdmisc.h" #include "nel/misc/types_nl.h" #include "nel/misc/mem_displayer.h" #include "nel/misc/path.h" #include "nel/misc/command.h" #include "nel/misc/debug.h" #ifdef NL_OS_WINDOWS # include # pragma comment(lib, "imagehlp.lib") # ifdef NL_OS_WIN64 # define DWORD_TYPE DWORD64 # else # define DWORD_TYPE DWORD # endif // NL_OS_WIN64 #endif // NL_OS_WINDOWS using namespace std; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NLMISC { // UNTIL we found who to walk in the call stack wihtout error, we disactive this feature #ifdef NL_OS_WINDOWS static const uint32 stringSize = 1024; static string getFuncInfo (DWORD_TYPE funcAddr, DWORD_TYPE stackAddr) { string str ("NoSymbol"); DWORD symSize = 10000; PIMAGEHLP_SYMBOL sym = (PIMAGEHLP_SYMBOL) GlobalAlloc (GMEM_FIXED, symSize); if (!sym) return str; ::ZeroMemory (sym, symSize); sym->SizeOfStruct = symSize; sym->MaxNameLength = symSize - sizeof(IMAGEHLP_SYMBOL); DWORD_TYPE disp = 0; if (SymGetSymFromAddr (GetCurrentProcess(), funcAddr, &disp, sym) == FALSE) { return str; } CHAR undecSymbol[stringSize]; if (UnDecorateSymbolName (sym->Name, undecSymbol, stringSize, UNDNAME_COMPLETE | UNDNAME_NO_THISTYPE | UNDNAME_NO_SPECIAL_SYMS | UNDNAME_NO_MEMBER_TYPE | UNDNAME_NO_MS_KEYWORDS | UNDNAME_NO_ACCESS_SPECIFIERS ) > 0) { str = undecSymbol; } else if (SymUnDName (sym, undecSymbol, stringSize) == TRUE) { str = undecSymbol; } if (disp != 0) { str += " + "; str += toString ((uint32)disp); str += " bytes"; } // replace param with the value of the stack for this param string parse = str; str.clear(); uint pos = 0; sint stop = 0; for (uint i = 0; i < parse.size (); i++) { if (parse[i] == '<') stop++; if (parse[i] == '>') stop--; if (stop==0 && (parse[i] == ',' || parse[i] == ')')) { char tmp[32]; sprintf(tmp, "=0x%p", (void *)(*((DWORD_TYPE*)(stackAddr) + 2 + pos++))); str += tmp; } str += parse[i]; } GlobalFree (sym); return str; } static string getSourceInfo (DWORD_TYPE addr) { string str; IMAGEHLP_LINE line; ::ZeroMemory (&line, sizeof (line)); line.SizeOfStruct = sizeof(line); // It doesn't work in windows 98 /* DWORD disp; if (SymGetLineFromAddr (GetCurrentProcess(), addr, &disp, &line)) { str = line.FileName; str += "(" + toString (line.LineNumber) + ")"; } else */ { IMAGEHLP_MODULE module; ::ZeroMemory (&module, sizeof(module)); module.SizeOfStruct = sizeof(module); if (SymGetModuleInfo (GetCurrentProcess(), addr, &module)) { str = module.ModuleName; } else { str = ""; } str += toString("!0x%p", (void*)addr); } return str; } #ifdef NL_OS_WIN64 static DWORD64 __stdcall GetModuleBase(HANDLE hProcess, DWORD64 dwReturnAddress) #else static DWORD __stdcall GetModuleBase(HANDLE hProcess, DWORD dwReturnAddress) #endif { IMAGEHLP_MODULE moduleInfo; if (SymGetModuleInfo(hProcess, dwReturnAddress, &moduleInfo)) return moduleInfo.BaseOfImage; else { MEMORY_BASIC_INFORMATION memoryBasicInfo; if (::VirtualQueryEx(hProcess, (LPVOID) dwReturnAddress, &memoryBasicInfo, sizeof(memoryBasicInfo))) { DWORD cch = 0; char szFile[MAX_PATH] = { 0 }; cch = GetModuleFileNameA((HINSTANCE)memoryBasicInfo.AllocationBase, szFile, MAX_PATH); if (cch && (lstrcmpA(szFile, "DBFN")== 0)) { char mn[] = { 'M', 'N', 0x00 }; #ifdef NL_OS_WIN64 if (!SymLoadModule64( #else if (!SymLoadModule( #endif hProcess, NULL, mn, NULL, (uintptr_t)memoryBasicInfo.AllocationBase, 0)) { // DWORD dwError = GetLastError(); // nlinfo("Error: %d", dwError); } } else { #ifdef NL_OS_WIN64 if (!SymLoadModule64( #else if (!SymLoadModule( #endif hProcess, NULL, ((cch) ? szFile : NULL), NULL, (uintptr_t)memoryBasicInfo.AllocationBase, 0)) { // DWORD dwError = GetLastError(); // nlinfo("Error: %d", dwError); } } return (uintptr_t)memoryBasicInfo.AllocationBase; } // else // nlinfo("Error is %d", GetLastError()); } return 0; } static void displayCallStack (CLog *log) { static string symbolPath; DWORD symOptions = SymGetOptions(); symOptions |= SYMOPT_LOAD_LINES; symOptions &= ~SYMOPT_UNDNAME; SymSetOptions (symOptions); // // Create the path where to find the symbol // if (symbolPath.empty()) { wchar_t tmpPath[stringSize]; symbolPath = "."; if (GetEnvironmentVariableW (L"_NT_SYMBOL_PATH", tmpPath, stringSize)) { symbolPath += ";"; symbolPath += wideToUtf8(tmpPath); } if (GetEnvironmentVariableW (L"_NT_ALTERNATE_SYMBOL_PATH", tmpPath, stringSize)) { symbolPath += ";"; symbolPath += wideToUtf8(tmpPath); } if (GetEnvironmentVariableW (L"SYSTEMROOT", tmpPath, stringSize)) { symbolPath += ";"; symbolPath += wideToUtf8(tmpPath); symbolPath += ";"; symbolPath += wideToUtf8(tmpPath); symbolPath += "\\system32"; } } // // Initialize // if (SymInitialize (GetCurrentProcess(), NULL, FALSE) == FALSE) { nlwarning ("DISP: SymInitialize(%p, '%s') failed", GetCurrentProcess(), symbolPath.c_str()); return; } // FIXME: Implement this for MinGW #ifndef NL_COMP_MINGW CONTEXT context; ::ZeroMemory (&context, sizeof(context)); context.ContextFlags = CONTEXT_FULL; if (GetThreadContext (GetCurrentThread(), &context) == FALSE) { nlwarning ("DISP: GetThreadContext(%p) failed", GetCurrentThread()); return; } STACKFRAME callStack; ::ZeroMemory (&callStack, sizeof(callStack)); #ifdef NL_OS_WIN64 callStack.AddrPC.Offset = context.Rip; callStack.AddrStack.Offset = context.Rsp; callStack.AddrFrame.Offset = context.Rbp; #else callStack.AddrPC.Offset = context.Eip; callStack.AddrStack.Offset = context.Esp; callStack.AddrFrame.Offset = context.Ebp; #endif callStack.AddrPC.Mode = AddrModeFlat; callStack.AddrStack.Mode = AddrModeFlat; callStack.AddrFrame.Mode = AddrModeFlat; for (uint32 i = 0; ; i++) { DWORD MachineType; #ifdef NL_OS_WIN64 MachineType = IMAGE_FILE_MACHINE_AMD64; BOOL res = StackWalk64(MachineType, GetCurrentProcess(), GetCurrentThread(), &callStack, NULL, NULL, SymFunctionTableAccess, GetModuleBase, NULL); #else MachineType = IMAGE_FILE_MACHINE_I386; BOOL res = StackWalk(MachineType, GetCurrentProcess(), GetCurrentThread(), &callStack, NULL, NULL, SymFunctionTableAccess, GetModuleBase, NULL); #endif /* if (res == FALSE) { DWORD r = GetLastError (); nlinfo ("%d",r); } */ if (i == 0) continue; if (res == FALSE || callStack.AddrFrame.Offset == 0) break; string symInfo, srcInfo; symInfo = getFuncInfo (callStack.AddrPC.Offset, callStack.AddrFrame.Offset); srcInfo = getSourceInfo (callStack.AddrPC.Offset); log->displayNL (" %s : %s", srcInfo.c_str(), symInfo.c_str()); } #endif } #else // NL_OS_WINDOWS static void displayCallStack (CLog *log) { log->displayNL ("no call stack info available"); } #endif // NL_OS_WINDOWS /* * Constructor */ CMemDisplayer::CMemDisplayer (const char *displayerName) : IDisplayer (displayerName), _NeedHeader(true), _MaxStrings(50), _CanUseStrings(true) { setParam (50); } void CMemDisplayer::setParam (uint32 maxStrings) { _MaxStrings = maxStrings; } // Log format: "2000/01/15 12:05:30 : " void CMemDisplayer::doDisplay ( const CLog::TDisplayInfo& args, const char *message ) { // stringstream ss; string str; bool needSpace = false; if (!_CanUseStrings) return; if (_NeedHeader) { str += HeaderString(); _NeedHeader = false; } if (args.Date != 0) { str += dateToHumanString(args.Date); needSpace = true; } if (!args.ProcessName.empty()) { if (needSpace) { str += " "; needSpace = false; } str += args.ProcessName; needSpace = true; } if (args.LogType != CLog::LOG_NO) { if (needSpace) { str += " "; needSpace = false; } str += logTypeToString(args.LogType); needSpace = true; } // Write thread identifier if ( args.ThreadId != 0 ) { if (needSpace) { str += " "; needSpace = false; } #ifdef NL_OS_WINDOWS str += NLMISC::toString("%5x", args.ThreadId); #else str += NLMISC::toString("%08x", args.ThreadId); #endif needSpace = true; } if (args.FileName != NULL) { if (needSpace) { str += " "; needSpace = false; } str += CFile::getFilename(args.FileName); needSpace = true; } if (args.Line != -1) { if (needSpace) { str += " "; needSpace = false; } str += NLMISC::toString(args.Line); needSpace = true; } if (needSpace) { str += " : "; needSpace = false; } str += message; // clear old line while (_Strings.size () > _MaxStrings) { _Strings.pop_front (); } _Strings.push_back (str); } void CMemDisplayer::write (CLog *log, bool quiet) { if (log == NULL) log = InfoLog; if ( ! quiet ) { log->forceDisplayRaw ("------------------------------------------------------------------------------\n"); log->forceDisplayRaw ("----------------------------------------- display MemDisplayer history -------\n"); log->forceDisplayRaw ("------------------------------------------------------------------------------\n"); } for (deque::iterator it = _Strings.begin(); it != _Strings.end(); it++) { log->forceDisplayRaw ((*it).c_str()); } if ( ! quiet ) { log->forceDisplayRaw ("------------------------------------------------------------------------------\n"); log->forceDisplayRaw ("----------------------------------------- display MemDisplayer callstack -----\n"); log->forceDisplayRaw ("------------------------------------------------------------------------------\n"); displayCallStack(log); log->forceDisplayRaw ("------------------------------------------------------------------------------\n"); log->forceDisplayRaw ("----------------------------------------- end of MemDisplayer display --------\n"); log->forceDisplayRaw ("------------------------------------------------------------------------------\n"); } } void CMemDisplayer::write (string &str, bool crLf) { for (deque::iterator it = _Strings.begin(); it != _Strings.end(); it++) { str += (*it); if ( crLf ) { if ( (!str.empty()) && (str[str.size()-1] == '\n') ) { str[str.size()-1] = '\r'; str += '\n'; } } } } void CLightMemDisplayer::doDisplay ( const CLog::TDisplayInfo& /* args */, const char *message ) { //stringstream ss; string str; //bool needSpace = false; if (!_CanUseStrings) return; str += message; // clear old line while (_Strings.size () >= _MaxStrings) { _Strings.pop_front (); } _Strings.push_back (str); } } // NLMISC