Thanks for the heads-up, Chris. Dimitar, we have had a similar problem for elf core file tests, where lldb would get confused if we get a running process with the same pid as the core file. I fixed this in <https://reviews.llvm.org/D18697>, and I suspect you will need a similar fix for your ProcessMinidump. Could you check if this is the case, and fix it?
pl On 31 October 2016 at 23:07, Chris Bieneman via lldb-commits <lldb-commits@lists.llvm.org> wrote: > I think the new mini-dump test case added here might be flaky. > > See: > http://lab.llvm.org:8011/builders/lldb-x86_64-ubuntu-14.04-cmake/builds/381 > > The next build was successful. > > -Chris > >> On Oct 31, 2016, at 8:35 AM, Dimitar Vlahovski via lldb-commits >> <lldb-commits@lists.llvm.org> wrote: >> >> Author: dvlahovski >> Date: Mon Oct 31 10:35:18 2016 >> New Revision: 285587 >> >> URL: http://llvm.org/viewvc/llvm-project?rev=285587&view=rev >> Log: >> Minidump plugin: Adding ProcessMinidump, ThreadMinidump and register the >> plugin in SystemInitializerFull >> >> Summary: >> This plugin resembles the already existing Windows-only Minidump plugin. >> The WinMinidumpPlugin uses the Windows API for parsing Minidumps >> while this plugin is cross-platform because it includes a Minidump >> parser (which is already commited) >> >> It is able to produce a backtrace, to read the general puprose regiters, >> inspect local variables, show image list, do memory reads, etc. >> >> For now the only arches that this supports are x86_32 and x86_64. >> This is because I have only written register contexts for those. >> Others will come in next CLs. >> >> I copied the WinMinidump tests and adapted them a little bit for them to >> work with the new plugin (and they pass) >> I will add more tests, aiming for better code coverage. >> >> There is still functionality to be added, see TODOs in code. >> >> Reviewers: labath, zturner >> >> Subscribers: beanz, mgorny, modocache, lldb-commits, amccarth >> >> Differential Revision: https://reviews.llvm.org/D25905 >> >> Added: >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/TestMiniDumpNew.py >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/install_breakpad.cpp >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64 >> (with props) >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.cpp >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.dmp >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed >> (with props) >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.cpp >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.dmp >> >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/makefile.txt >> lldb/trunk/source/Plugins/Process/minidump/NtStructures.h >> lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.cpp >> lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.h >> lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.cpp >> lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.h >> Modified: >> lldb/trunk/source/API/SystemInitializerFull.cpp >> lldb/trunk/source/Plugins/Process/minidump/CMakeLists.txt >> lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.cpp >> lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.h >> lldb/trunk/source/Plugins/Process/minidump/MinidumpTypes.h >> lldb/trunk/unittests/Process/minidump/MinidumpParserTest.cpp >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/TestMiniDumpNew.py >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/TestMiniDumpNew.py?rev=285587&view=auto >> ============================================================================== >> --- >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/TestMiniDumpNew.py >> (added) >> +++ >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/TestMiniDumpNew.py >> Mon Oct 31 10:35:18 2016 >> @@ -0,0 +1,100 @@ >> +""" >> +Test basics of Minidump debugging. >> +""" >> + >> +from __future__ import print_function >> +from six import iteritems >> + >> + >> +import lldb >> +from lldbsuite.test.decorators import * >> +from lldbsuite.test.lldbtest import * >> +from lldbsuite.test import lldbutil >> + >> + >> +class MiniDumpNewTestCase(TestBase): >> + >> + mydir = TestBase.compute_mydir(__file__) >> + >> + NO_DEBUG_INFO_TESTCASE = True >> + >> + def test_process_info_in_minidump(self): >> + """Test that lldb can read the process information from the >> Minidump.""" >> + # target create -c linux-x86_64.dmp >> + self.dbg.CreateTarget(None) >> + self.target = self.dbg.GetSelectedTarget() >> + self.process = self.target.LoadCore("linux-x86_64.dmp") >> + self.assertTrue(self.process, PROCESS_IS_VALID) >> + self.assertEqual(self.process.GetNumThreads(), 1) >> + self.assertEqual(self.process.GetProcessID(), 29917) >> + >> + def test_thread_info_in_minidump(self): >> + """Test that lldb can read the thread information from the >> Minidump.""" >> + # target create -c linux-x86_64.dmp >> + self.dbg.CreateTarget(None) >> + self.target = self.dbg.GetSelectedTarget() >> + self.process = self.target.LoadCore("linux-x86_64.dmp") >> + # This process crashed due to a segmentation fault in its >> + # one and only thread. >> + self.assertEqual(self.process.GetNumThreads(), 1) >> + thread = self.process.GetThreadAtIndex(0) >> + self.assertEqual(thread.GetStopReason(), lldb.eStopReasonSignal) >> + stop_description = thread.GetStopDescription(256) >> + self.assertTrue("SIGSEGV" in stop_description) >> + >> + def test_stack_info_in_minidump(self): >> + """Test that we can see a trivial stack in a breakpad-generated >> Minidump.""" >> + # target create linux-x86_64 -c linux-x86_64.dmp >> + self.dbg.CreateTarget("linux-x86_64") >> + self.target = self.dbg.GetSelectedTarget() >> + self.process = self.target.LoadCore("linux-x86_64.dmp") >> + self.assertEqual(self.process.GetNumThreads(), 1) >> + thread = self.process.GetThreadAtIndex(0) >> + # frame #0: linux-x86_64`crash() >> + # frame #1: linux-x86_64`_start >> + self.assertEqual(thread.GetNumFrames(), 2) >> + frame = thread.GetFrameAtIndex(0) >> + self.assertTrue(frame.IsValid()) >> + pc = frame.GetPC() >> + eip = frame.FindRegister("pc") >> + self.assertTrue(eip.IsValid()) >> + self.assertEqual(pc, eip.GetValueAsUnsigned()) >> + >> + def test_snapshot_minidump(self): >> + """Test that if we load a snapshot minidump file (meaning the >> process did not crash) there is no stop reason.""" >> + # target create -c linux-x86_64_not_crashed.dmp >> + self.dbg.CreateTarget(None) >> + self.target = self.dbg.GetSelectedTarget() >> + self.process = self.target.LoadCore("linux-x86_64_not_crashed.dmp") >> + self.assertEqual(self.process.GetNumThreads(), 1) >> + thread = self.process.GetThreadAtIndex(0) >> + self.assertEqual(thread.GetStopReason(), lldb.eStopReasonNone) >> + stop_description = thread.GetStopDescription(256) >> + self.assertEqual(stop_description, None) >> + >> + def test_deeper_stack_in_minidump(self): >> + """Test that we can examine a more interesting stack in a >> Minidump.""" >> + # Launch with the Minidump, and inspect the stack. >> + # target create linux-x86_64_not_crashed -c >> linux-x86_64_not_crashed.dmp >> + target = self.dbg.CreateTarget("linux-x86_64_not_crashed") >> + process = target.LoadCore("linux-x86_64_not_crashed.dmp") >> + thread = process.GetThreadAtIndex(0) >> + >> + expected_stack = {1: 'bar', 2: 'foo', 3: '_start'} >> + self.assertGreaterEqual(thread.GetNumFrames(), len(expected_stack)) >> + for index, name in iteritems(expected_stack): >> + frame = thread.GetFrameAtIndex(index) >> + self.assertTrue(frame.IsValid()) >> + function_name = frame.GetFunctionName() >> + self.assertTrue(name in function_name) >> + >> + def test_local_variables_in_minidump(self): >> + """Test that we can examine local variables in a Minidump.""" >> + # Launch with the Minidump, and inspect a local variable. >> + # target create linux-x86_64_not_crashed -c >> linux-x86_64_not_crashed.dmp >> + target = self.dbg.CreateTarget("linux-x86_64_not_crashed") >> + process = target.LoadCore("linux-x86_64_not_crashed.dmp") >> + thread = process.GetThreadAtIndex(0) >> + frame = thread.GetFrameAtIndex(1) >> + value = frame.EvaluateExpression('x') >> + self.assertEqual(value.GetValueAsSigned(), 3) >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/install_breakpad.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/install_breakpad.cpp?rev=285587&view=auto >> ============================================================================== >> --- >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/install_breakpad.cpp >> (added) >> +++ >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/install_breakpad.cpp >> Mon Oct 31 10:35:18 2016 >> @@ -0,0 +1,16 @@ >> +#include "client/linux/handler/exception_handler.h" >> + >> +static bool dumpCallback(const google_breakpad::MinidumpDescriptor >> &descriptor, >> + void *context, bool succeeded) { >> + return succeeded; >> +} >> + >> +google_breakpad::ExceptionHandler *eh; >> + >> +void InstallBreakpad() { >> + google_breakpad::MinidumpDescriptor descriptor("/tmp"); >> + eh = new google_breakpad::ExceptionHandler(descriptor, NULL, dumpCallback, >> + NULL, true, -1); >> +} >> + >> +void WriteMinidump() { eh->WriteMinidump(); } >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64 >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64?rev=285587&view=auto >> ============================================================================== >> Binary files >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64 >> (added) and >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64 >> Mon Oct 31 10:35:18 2016 differ >> >> Propchange: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64 >> ------------------------------------------------------------------------------ >> svn:executable = * >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.cpp?rev=285587&view=auto >> ============================================================================== >> --- >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.cpp >> (added) >> +++ >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.cpp >> Mon Oct 31 10:35:18 2016 >> @@ -0,0 +1,12 @@ >> +void crash() { >> + volatile int *a = (int *)(nullptr); >> + *a = 1; >> +} >> + >> +extern "C" void _start(); >> +void InstallBreakpad(); >> + >> +void _start() { >> + InstallBreakpad(); >> + crash(); >> +} >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.dmp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.dmp?rev=285587&view=auto >> ============================================================================== >> Binary files >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.dmp >> (added) and >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64.dmp >> Mon Oct 31 10:35:18 2016 differ >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed?rev=285587&view=auto >> ============================================================================== >> Binary files >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed >> (added) and >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed >> Mon Oct 31 10:35:18 2016 differ >> >> Propchange: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed >> ------------------------------------------------------------------------------ >> svn:executable = * >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.cpp?rev=285587&view=auto >> ============================================================================== >> --- >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.cpp >> (added) >> +++ >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.cpp >> Mon Oct 31 10:35:18 2016 >> @@ -0,0 +1,22 @@ >> +void InstallBreakpad(); >> +void WriteMinidump(); >> + >> +int global = 42; >> + >> +int bar(int x) { >> + WriteMinidump(); >> + int y = 4 * x + global; >> + return y; >> +} >> + >> +int foo(int x) { >> + int y = 2 * bar(3 * x); >> + return y; >> +} >> + >> +extern "C" void _start(); >> + >> +void _start() { >> + InstallBreakpad(); >> + foo(1); >> +} >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.dmp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.dmp?rev=285587&view=auto >> ============================================================================== >> Binary files >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.dmp >> (added) and >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/linux-x86_64_not_crashed.dmp >> Mon Oct 31 10:35:18 2016 differ >> >> Added: >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/makefile.txt >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/makefile.txt?rev=285587&view=auto >> ============================================================================== >> --- >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/makefile.txt >> (added) >> +++ >> lldb/trunk/packages/Python/lldbsuite/test/functionalities/postmortem/minidump-new/makefile.txt >> Mon Oct 31 10:35:18 2016 >> @@ -0,0 +1,29 @@ >> +# This makefile aims to make the binaries as small as possible, for us not >> to >> +# upload huge binary blobs in the repo. >> +# The binary should have debug symbols because stack unwinding doesn't work >> +# correctly using the information in the Minidump only. Also we want to >> evaluate >> +# local variables, etc. >> +# Breakpad compiles as a static library, so statically linking againts it >> +# makes the binary huge. >> +# Dynamically linking to it does improve things, but we are still >> #include-ing >> +# breakpad headers (which is a lot of source code for which we generate >> debug >> +# symbols) >> +# So, install_breakpad.cpp does the #include-ing and defines a global >> function >> +# "InstallBreakpad" that does all the exception handler registration. >> +# We compile install_breakpad to object file and then link it, alongside the >> +# static libbreakpad, into a shared library. >> +# Then the binaries dynamically link to that lib. >> +# The other optimisation is not using the standard library (hence the _start >> +# instead of main). We only link dynamically to some standard libraries. >> +# This way we have a tiny binary (~8K) that has debug symbols and uses >> breakpad >> +# to generate a Minidump when the binary crashes/requests such. >> +# >> +CC=g++ >> +FLAGS=-g --std=c++11 >> +INCLUDE=-I$HOME/breakpad/src/src/ >> +LINK=-L. -lbreakpad -lpthread -nostdlib -lc -lstdc++ -lgcc_s -fno-exceptions >> +all: >> + $(CC) $(FLAGS) -fPIC -c install_breakpad.cpp $(INCLUDE) -o >> install_breakpad.o >> + ld -shared install_breakpad.o libbreakpad_client.a -o libbreakpad.so >> + $(CC) $(FLAGS) -o linux-x86_64 linux-x86_64.cpp $(LINK) >> + $(CC) $(FLAGS) -o linux-x86_64_not_crashed >> linux-x86_64_not_crashed.cpp $(LINK) >> >> Modified: lldb/trunk/source/API/SystemInitializerFull.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SystemInitializerFull.cpp?rev=285587&r1=285586&r2=285587&view=diff >> ============================================================================== >> --- lldb/trunk/source/API/SystemInitializerFull.cpp (original) >> +++ lldb/trunk/source/API/SystemInitializerFull.cpp Mon Oct 31 10:35:18 2016 >> @@ -78,6 +78,7 @@ >> #include "Plugins/Platform/gdb-server/PlatformRemoteGDBServer.h" >> #include "Plugins/Process/elf-core/ProcessElfCore.h" >> #include "Plugins/Process/gdb-remote/ProcessGDBRemote.h" >> +#include "Plugins/Process/minidump/ProcessMinidump.h" >> #include "Plugins/ScriptInterpreter/None/ScriptInterpreterNone.h" >> #include "Plugins/SymbolFile/DWARF/SymbolFileDWARF.h" >> #include "Plugins/SymbolFile/DWARF/SymbolFileDWARFDebugMap.h" >> @@ -304,6 +305,7 @@ void SystemInitializerFull::Initialize() >> >> JITLoaderGDB::Initialize(); >> ProcessElfCore::Initialize(); >> + minidump::ProcessMinidump::Initialize(); >> #if defined(_MSC_VER) >> ProcessWinMiniDump::Initialize(); >> #endif >> @@ -429,6 +431,7 @@ void SystemInitializerFull::Terminate() >> >> JITLoaderGDB::Terminate(); >> ProcessElfCore::Terminate(); >> + minidump::ProcessMinidump::Terminate(); >> #if defined(_MSC_VER) >> ProcessWinMiniDump::Terminate(); >> #endif >> >> Modified: lldb/trunk/source/Plugins/Process/minidump/CMakeLists.txt >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/CMakeLists.txt?rev=285587&r1=285586&r2=285587&view=diff >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/CMakeLists.txt (original) >> +++ lldb/trunk/source/Plugins/Process/minidump/CMakeLists.txt Mon Oct 31 >> 10:35:18 2016 >> @@ -5,4 +5,6 @@ add_lldb_library(lldbPluginProcessMinidu >> MinidumpParser.cpp >> RegisterContextMinidump_x86_32.cpp >> RegisterContextMinidump_x86_64.cpp >> + ProcessMinidump.cpp >> + ThreadMinidump.cpp >> ) >> >> Modified: lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.cpp?rev=285587&r1=285586&r2=285587&view=diff >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.cpp (original) >> +++ lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.cpp Mon Oct 31 >> 10:35:18 2016 >> @@ -9,6 +9,8 @@ >> >> // Project includes >> #include "MinidumpParser.h" >> +#include "NtStructures.h" >> +#include "RegisterContextMinidump_x86_32.h" >> >> // Other libraries and framework includes >> #include "lldb/Target/MemoryRegionInfo.h" >> @@ -106,11 +108,42 @@ llvm::ArrayRef<MinidumpThread> MinidumpP >> llvm::ArrayRef<uint8_t> >> MinidumpParser::GetThreadContext(const MinidumpThread &td) { >> if (td.thread_context.rva + td.thread_context.data_size > GetData().size()) >> - return llvm::None; >> + return {}; >> >> return GetData().slice(td.thread_context.rva, td.thread_context.data_size); >> } >> >> +llvm::ArrayRef<uint8_t> >> +MinidumpParser::GetThreadContextWow64(const MinidumpThread &td) { >> + // On Windows, a 32-bit process can run on a 64-bit machine under >> + // WOW64. If the minidump was captured with a 64-bit debugger, then >> + // the CONTEXT we just grabbed from the mini_dump_thread is the one >> + // for the 64-bit "native" process rather than the 32-bit "guest" >> + // process we care about. In this case, we can get the 32-bit CONTEXT >> + // from the TEB (Thread Environment Block) of the 64-bit process. >> + auto teb_mem = GetMemory(td.teb, sizeof(TEB64)); >> + if (teb_mem.empty()) >> + return {}; >> + >> + const TEB64 *wow64teb; >> + Error error = consumeObject(teb_mem, wow64teb); >> + if (error.Fail()) >> + return {}; >> + >> + // Slot 1 of the thread-local storage in the 64-bit TEB points to a >> + // structure that includes the 32-bit CONTEXT (after a ULONG). >> + // See: https://msdn.microsoft.com/en-us/library/ms681670.aspx >> + auto context = >> + GetMemory(wow64teb->tls_slots[1] + 4, sizeof(MinidumpContext_x86_32)); >> + if (context.size() < sizeof(MinidumpContext_x86_32)) >> + return {}; >> + >> + return context; >> + // NOTE: We don't currently use the TEB for anything else. If we >> + // need it in the future, the 32-bit TEB is located according to the >> address >> + // stored in the first slot of the 64-bit TEB (wow64teb.Reserved1[0]). >> +} >> + >> const MinidumpSystemInfo *MinidumpParser::GetSystemInfo() { >> llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::SystemInfo); >> >> @@ -229,8 +262,7 @@ llvm::ArrayRef<MinidumpModule> MinidumpP >> >> std::vector<const MinidumpModule *> MinidumpParser::GetFilteredModuleList() { >> llvm::ArrayRef<MinidumpModule> modules = GetModuleList(); >> - // mapping module_name to pair(load_address, pointer to module struct in >> - // memory) >> + // map module_name -> pair(load_address, pointer to module struct in >> memory) >> llvm::StringMap<std::pair<uint64_t, const MinidumpModule *>> lowest_addr; >> >> std::vector<const MinidumpModule *> filtered_modules; >> >> Modified: lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.h >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.h?rev=285587&r1=285586&r2=285587&view=diff >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.h (original) >> +++ lldb/trunk/source/Plugins/Process/minidump/MinidumpParser.h Mon Oct 31 >> 10:35:18 2016 >> @@ -59,6 +59,8 @@ public: >> >> llvm::ArrayRef<uint8_t> GetThreadContext(const MinidumpThread &td); >> >> + llvm::ArrayRef<uint8_t> GetThreadContextWow64(const MinidumpThread &td); >> + >> const MinidumpSystemInfo *GetSystemInfo(); >> >> ArchSpec GetArchitecture(); >> >> Modified: lldb/trunk/source/Plugins/Process/minidump/MinidumpTypes.h >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/MinidumpTypes.h?rev=285587&r1=285586&r2=285587&view=diff >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/MinidumpTypes.h (original) >> +++ lldb/trunk/source/Plugins/Process/minidump/MinidumpTypes.h Mon Oct 31 >> 10:35:18 2016 >> @@ -439,7 +439,8 @@ static_assert(sizeof(MinidumpModule) == >> // Exception stuff >> struct MinidumpException { >> enum { >> - MaxParams = 15, >> + ExceptonInfoMaxParams = 15, >> + DumpRequested = 0xFFFFFFFF, >> }; >> >> llvm::support::ulittle32_t exception_code; >> @@ -448,7 +449,7 @@ struct MinidumpException { >> llvm::support::ulittle64_t exception_address; >> llvm::support::ulittle32_t number_parameters; >> llvm::support::ulittle32_t unused_alignment; >> - llvm::support::ulittle64_t exception_information[MaxParams]; >> + llvm::support::ulittle64_t exception_information[ExceptonInfoMaxParams]; >> }; >> static_assert(sizeof(MinidumpException) == 152, >> "sizeof MinidumpException is not correct!"); >> >> Added: lldb/trunk/source/Plugins/Process/minidump/NtStructures.h >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/NtStructures.h?rev=285587&view=auto >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/NtStructures.h (added) >> +++ lldb/trunk/source/Plugins/Process/minidump/NtStructures.h Mon Oct 31 >> 10:35:18 2016 >> @@ -0,0 +1,37 @@ >> +//===-- NtStructures.h ------------------------------------------*- C++ >> -*-===// >> +// >> +// The LLVM Compiler Infrastructure >> +// >> +// This file is distributed under the University of Illinois Open Source >> +// License. See LICENSE.TXT for details. >> +// >> +//===----------------------------------------------------------------------===// >> + >> +#ifndef liblldb_Plugins_Process_Minidump_NtStructures_h_ >> +#define liblldb_Plugins_Process_Minidump_NtStructures_h_ >> + >> +#include "llvm/Support/Endian.h" >> + >> +namespace lldb_private { >> + >> +namespace minidump { >> + >> +// This describes the layout of a TEB (Thread Environment Block) for a >> 64-bit >> +// process. It's adapted from the 32-bit TEB in winternl.h. Currently, we >> care >> +// only about the position of the tls_slots. >> +struct TEB64 { >> + llvm::support::ulittle64_t reserved1[12]; >> + llvm::support::ulittle64_t process_environment_block; >> + llvm::support::ulittle64_t reserved2[399]; >> + uint8_t reserved3[1952]; >> + llvm::support::ulittle64_t tls_slots[64]; >> + uint8_t reserved4[8]; >> + llvm::support::ulittle64_t reserved5[26]; >> + llvm::support::ulittle64_t reserved_for_ole; // Windows 2000 only >> + llvm::support::ulittle64_t reserved6[4]; >> + llvm::support::ulittle64_t tls_expansion_slots; >> +}; >> + >> +#endif // liblldb_Plugins_Process_Minidump_NtStructures_h_ >> +} // namespace minidump >> +} // namespace lldb_private >> >> Added: lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.cpp?rev=285587&view=auto >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.cpp (added) >> +++ lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.cpp Mon Oct >> 31 10:35:18 2016 >> @@ -0,0 +1,293 @@ >> +//===-- ProcessMinidump.cpp -------------------------------------*- C++ >> -*-===// >> +// >> +// The LLVM Compiler Infrastructure >> +// >> +// This file is distributed under the University of Illinois Open Source >> +// License. See LICENSE.TXT for details. >> +// >> +//===----------------------------------------------------------------------===// >> + >> +// Project includes >> +#include "ProcessMinidump.h" >> +#include "ThreadMinidump.h" >> + >> +// Other libraries and framework includes >> +#include "lldb/Core/DataBufferHeap.h" >> +#include "lldb/Core/Log.h" >> +#include "lldb/Core/Module.h" >> +#include "lldb/Core/ModuleSpec.h" >> +#include "lldb/Core/PluginManager.h" >> +#include "lldb/Core/Section.h" >> +#include "lldb/Core/State.h" >> +#include "lldb/Target/DynamicLoader.h" >> +#include "lldb/Target/MemoryRegionInfo.h" >> +#include "lldb/Target/Target.h" >> +#include "lldb/Target/UnixSignals.h" >> +#include "lldb/Utility/LLDBAssert.h" >> + >> +// C includes >> +// C++ includes >> + >> +using namespace lldb_private; >> +using namespace minidump; >> + >> +ConstString ProcessMinidump::GetPluginNameStatic() { >> + static ConstString g_name("minidump"); >> + return g_name; >> +} >> + >> +const char *ProcessMinidump::GetPluginDescriptionStatic() { >> + return "Minidump plug-in."; >> +} >> + >> +lldb::ProcessSP ProcessMinidump::CreateInstance(lldb::TargetSP target_sp, >> + lldb::ListenerSP >> listener_sp, >> + const FileSpec *crash_file) >> { >> + if (!crash_file) >> + return nullptr; >> + >> + lldb::ProcessSP process_sp; >> + // Read enough data for the Minidump header >> + const size_t header_size = sizeof(MinidumpHeader); >> + lldb::DataBufferSP data_sp(crash_file->MemoryMapFileContents(0, >> header_size)); >> + if (!data_sp) >> + return nullptr; >> + >> + // first, only try to parse the header, beacuse we need to be fast >> + llvm::ArrayRef<uint8_t> header_data(data_sp->GetBytes(), header_size); >> + const MinidumpHeader *header = MinidumpHeader::Parse(header_data); >> + >> + if (data_sp->GetByteSize() != header_size || header == nullptr) >> + return nullptr; >> + >> + lldb::DataBufferSP all_data_sp(crash_file->MemoryMapFileContents()); >> + auto minidump_parser = MinidumpParser::Create(all_data_sp); >> + // check if the parser object is valid >> + // skip if the Minidump file is Windows generated, because we are still >> + // work-in-progress >> + if (!minidump_parser || >> + minidump_parser->GetArchitecture().GetTriple().getOS() == >> + llvm::Triple::OSType::Win32) >> + return nullptr; >> + >> + return std::make_shared<ProcessMinidump>(target_sp, listener_sp, >> *crash_file, >> + minidump_parser.getValue()); >> +} >> + >> +bool ProcessMinidump::CanDebug(lldb::TargetSP target_sp, >> + bool plugin_specified_by_name) { >> + return true; >> +} >> + >> +ProcessMinidump::ProcessMinidump(lldb::TargetSP target_sp, >> + lldb::ListenerSP listener_sp, >> + const FileSpec &core_file, >> + MinidumpParser minidump_parser) >> + : Process(target_sp, listener_sp), m_minidump_parser(minidump_parser), >> + m_core_file(core_file), m_is_wow64(false) {} >> + >> +ProcessMinidump::~ProcessMinidump() { >> + Clear(); >> + // We need to call finalize on the process before destroying ourselves >> + // to make sure all of the broadcaster cleanup goes as planned. If we >> + // destruct this class, then Process::~Process() might have problems >> + // trying to fully destroy the broadcaster. >> + Finalize(); >> +} >> + >> +void ProcessMinidump::Initialize() { >> + static std::once_flag g_once_flag; >> + >> + std::call_once(g_once_flag, []() { >> + PluginManager::RegisterPlugin(GetPluginNameStatic(), >> + GetPluginDescriptionStatic(), >> + ProcessMinidump::CreateInstance); >> + }); >> +} >> + >> +void ProcessMinidump::Terminate() { >> + PluginManager::UnregisterPlugin(ProcessMinidump::CreateInstance); >> +} >> + >> +Error ProcessMinidump::DoLoadCore() { >> + Error error; >> + >> + m_thread_list = m_minidump_parser.GetThreads(); >> + m_active_exception = m_minidump_parser.GetExceptionStream(); >> + ReadModuleList(); >> + GetTarget().SetArchitecture(GetArchitecture()); >> + >> + llvm::Optional<lldb::pid_t> pid = m_minidump_parser.GetPid(); >> + if (!pid) { >> + error.SetErrorString("failed to parse PID"); >> + return error; >> + } >> + SetID(pid.getValue()); >> + >> + return error; >> +} >> + >> +DynamicLoader *ProcessMinidump::GetDynamicLoader() { >> + if (m_dyld_ap.get() == nullptr) >> + m_dyld_ap.reset(DynamicLoader::FindPlugin(this, nullptr)); >> + return m_dyld_ap.get(); >> +} >> + >> +ConstString ProcessMinidump::GetPluginName() { return >> GetPluginNameStatic(); } >> + >> +uint32_t ProcessMinidump::GetPluginVersion() { return 1; } >> + >> +Error ProcessMinidump::DoDestroy() { return Error(); } >> + >> +void ProcessMinidump::RefreshStateAfterStop() { >> + if (!m_active_exception) >> + return; >> + >> + if (m_active_exception->exception_record.exception_code == >> + MinidumpException::DumpRequested) { >> + return; >> + } >> + >> + lldb::StopInfoSP stop_info; >> + lldb::ThreadSP stop_thread; >> + >> + >> Process::m_thread_list.SetSelectedThreadByID(m_active_exception->thread_id); >> + stop_thread = Process::m_thread_list.GetSelectedThread(); >> + ArchSpec arch = GetArchitecture(); >> + >> + if (arch.GetTriple().getOS() == llvm::Triple::Linux) { >> + stop_info = StopInfo::CreateStopReasonWithSignal( >> + *stop_thread, m_active_exception->exception_record.exception_code); >> + } else { >> + std::string desc; >> + llvm::raw_string_ostream desc_stream(desc); >> + desc_stream << "Exception " >> + << llvm::format_hex( >> + m_active_exception->exception_record.exception_code, >> 8) >> + << " encountered at address " >> + << llvm::format_hex( >> + >> m_active_exception->exception_record.exception_address, >> + 8); >> + stop_info = StopInfo::CreateStopReasonWithException( >> + *stop_thread, desc_stream.str().c_str()); >> + } >> + >> + stop_thread->SetStopInfo(stop_info); >> +} >> + >> +bool ProcessMinidump::IsAlive() { return true; } >> + >> +bool ProcessMinidump::WarnBeforeDetach() const { return false; } >> + >> +size_t ProcessMinidump::ReadMemory(lldb::addr_t addr, void *buf, size_t >> size, >> + lldb_private::Error &error) { >> + // Don't allow the caching that lldb_private::Process::ReadMemory does >> + // since we have it all cached in our dump file anyway. >> + return DoReadMemory(addr, buf, size, error); >> +} >> + >> +size_t ProcessMinidump::DoReadMemory(lldb::addr_t addr, void *buf, size_t >> size, >> + lldb_private::Error &error) { >> + >> + llvm::ArrayRef<uint8_t> mem = m_minidump_parser.GetMemory(addr, size); >> + if (mem.empty()) { >> + error.SetErrorString("could not parse memory info"); >> + return 0; >> + } >> + >> + std::memcpy(buf, mem.data(), mem.size()); >> + return mem.size(); >> +} >> + >> +ArchSpec ProcessMinidump::GetArchitecture() { >> + if (!m_is_wow64) { >> + return m_minidump_parser.GetArchitecture(); >> + } >> + >> + llvm::Triple triple; >> + triple.setVendor(llvm::Triple::VendorType::UnknownVendor); >> + triple.setArch(llvm::Triple::ArchType::x86); >> + triple.setOS(llvm::Triple::OSType::Win32); >> + return ArchSpec(triple); >> +} >> + >> +Error ProcessMinidump::GetMemoryRegionInfo( >> + lldb::addr_t load_addr, lldb_private::MemoryRegionInfo &range_info) { >> + Error error; >> + auto info = m_minidump_parser.GetMemoryRegionInfo(load_addr); >> + if (!info) { >> + error.SetErrorString("No valid MemoryRegionInfo found!"); >> + return error; >> + } >> + range_info = info.getValue(); >> + return error; >> +} >> + >> +void ProcessMinidump::Clear() { Process::m_thread_list.Clear(); } >> + >> +bool ProcessMinidump::UpdateThreadList( >> + lldb_private::ThreadList &old_thread_list, >> + lldb_private::ThreadList &new_thread_list) { >> + uint32_t num_threads = 0; >> + if (m_thread_list.size() > 0) >> + num_threads = m_thread_list.size(); >> + >> + for (lldb::tid_t tid = 0; tid < num_threads; ++tid) { >> + llvm::ArrayRef<uint8_t> context; >> + if (!m_is_wow64) >> + context = m_minidump_parser.GetThreadContext(m_thread_list[tid]); >> + else >> + context = m_minidump_parser.GetThreadContextWow64(m_thread_list[tid]); >> + >> + lldb::ThreadSP thread_sp( >> + new ThreadMinidump(*this, m_thread_list[tid], context)); >> + new_thread_list.AddThread(thread_sp); >> + } >> + return new_thread_list.GetSize(false) > 0; >> +} >> + >> +void ProcessMinidump::ReadModuleList() { >> + std::vector<const MinidumpModule *> filtered_modules = >> + m_minidump_parser.GetFilteredModuleList(); >> + >> + for (auto module : filtered_modules) { >> + llvm::Optional<std::string> name = >> + m_minidump_parser.GetMinidumpString(module->module_name_rva); >> + >> + if (!name) >> + continue; >> + >> + Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_MODULES)); >> + if (log) { >> + log->Printf( >> + "ProcessMinidump::%s found module: name: %s %#010lx-%#010lx size: >> %u", >> + __FUNCTION__, name.getValue().c_str(), >> + uint64_t(module->base_of_image), >> + module->base_of_image + module->size_of_image, >> + uint32_t(module->size_of_image)); >> + } >> + >> + // check if the process is wow64 - a 32 bit windows process running on a >> + // 64 bit windows >> + if (llvm::StringRef(name.getValue()).endswith_lower("wow64.dll")) { >> + m_is_wow64 = true; >> + } >> + >> + const auto file_spec = FileSpec(name.getValue(), true); >> + ModuleSpec module_spec = file_spec; >> + Error error; >> + lldb::ModuleSP module_sp = GetTarget().GetSharedModule(module_spec, >> &error); >> + if (!module_sp || error.Fail()) { >> + continue; >> + } >> + >> + if (log) { >> + log->Printf("ProcessMinidump::%s load module: name: %s", __FUNCTION__, >> + name.getValue().c_str()); >> + } >> + >> + bool load_addr_changed = false; >> + module_sp->SetLoadAddress(GetTarget(), module->base_of_image, false, >> + load_addr_changed); >> + } >> +} >> >> Added: lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.h >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.h?rev=285587&view=auto >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.h (added) >> +++ lldb/trunk/source/Plugins/Process/minidump/ProcessMinidump.h Mon Oct 31 >> 10:35:18 2016 >> @@ -0,0 +1,104 @@ >> +//===-- ProcessMinidump.h ---------------------------------------*- C++ >> -*-===// >> +// >> +// The LLVM Compiler Infrastructure >> +// >> +// This file is distributed under the University of Illinois Open Source >> +// License. See LICENSE.TXT for details. >> +// >> +//===----------------------------------------------------------------------===// >> + >> +#ifndef liblldb_ProcessMinidump_h_ >> +#define liblldb_ProcessMinidump_h_ >> + >> +// Project includes >> +#include "MinidumpParser.h" >> +#include "MinidumpTypes.h" >> + >> +// Other libraries and framework includes >> +#include "lldb/Core/ConstString.h" >> +#include "lldb/Core/Error.h" >> +#include "lldb/Target/Process.h" >> +#include "lldb/Target/StopInfo.h" >> +#include "lldb/Target/Target.h" >> + >> +#include "llvm/Support/Format.h" >> +#include "llvm/Support/raw_ostream.h" >> + >> +// C Includes >> +// C++ Includes >> + >> +namespace lldb_private { >> + >> +namespace minidump { >> + >> +class ProcessMinidump : public Process { >> +public: >> + static lldb::ProcessSP CreateInstance(lldb::TargetSP target_sp, >> + lldb::ListenerSP listener_sp, >> + const FileSpec *crash_file_path); >> + >> + static void Initialize(); >> + >> + static void Terminate(); >> + >> + static ConstString GetPluginNameStatic(); >> + >> + static const char *GetPluginDescriptionStatic(); >> + >> + ProcessMinidump(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, >> + const lldb_private::FileSpec &core_file, >> + MinidumpParser minidump_parser); >> + >> + ~ProcessMinidump() override; >> + >> + bool CanDebug(lldb::TargetSP target_sp, >> + bool plugin_specified_by_name) override; >> + >> + Error DoLoadCore() override; >> + >> + DynamicLoader *GetDynamicLoader() override; >> + >> + ConstString GetPluginName() override; >> + >> + uint32_t GetPluginVersion() override; >> + >> + Error DoDestroy() override; >> + >> + void RefreshStateAfterStop() override; >> + >> + bool IsAlive() override; >> + >> + bool WarnBeforeDetach() const override; >> + >> + size_t ReadMemory(lldb::addr_t addr, void *buf, size_t size, >> + Error &error) override; >> + >> + size_t DoReadMemory(lldb::addr_t addr, void *buf, size_t size, >> + Error &error) override; >> + >> + ArchSpec GetArchitecture(); >> + >> + Error GetMemoryRegionInfo(lldb::addr_t load_addr, >> + MemoryRegionInfo &range_info) override; >> + >> + MinidumpParser m_minidump_parser; >> + >> +protected: >> + void Clear(); >> + >> + bool UpdateThreadList(ThreadList &old_thread_list, >> + ThreadList &new_thread_list) override; >> + >> + void ReadModuleList(); >> + >> +private: >> + FileSpec m_core_file; >> + llvm::ArrayRef<MinidumpThread> m_thread_list; >> + const MinidumpExceptionStream *m_active_exception; >> + bool m_is_wow64; >> +}; >> + >> +} // namespace minidump >> +} // namespace lldb_private >> + >> +#endif // liblldb_ProcessMinidump_h_ >> >> Added: lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.cpp?rev=285587&view=auto >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.cpp (added) >> +++ lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.cpp Mon Oct 31 >> 10:35:18 2016 >> @@ -0,0 +1,114 @@ >> +//===-- ThreadMinidump.cpp --------------------------------------*- C++ >> -*-===// >> +// >> +// The LLVM Compiler Infrastructure >> +// >> +// This file is distributed under the University of Illinois Open Source >> +// License. See LICENSE.TXT for details. >> +// >> +//===----------------------------------------------------------------------===// >> + >> +// Project includes >> +#include "ThreadMinidump.h" >> +#include "ProcessMinidump.h" >> + >> +#include "RegisterContextMinidump_x86_32.h" >> +#include "RegisterContextMinidump_x86_64.h" >> + >> +// Other libraries and framework includes >> +#include "Plugins/Process/Utility/RegisterContextLinux_i386.h" >> +#include "Plugins/Process/Utility/RegisterContextLinux_x86_64.h" >> + >> +#include "Plugins/Process/elf-core/RegisterContextPOSIXCore_x86_64.h" >> + >> +#include "lldb/Core/DataExtractor.h" >> +#include "lldb/Core/Log.h" >> +#include "lldb/Target/RegisterContext.h" >> +#include "lldb/Target/StopInfo.h" >> +#include "lldb/Target/Target.h" >> +#include "lldb/Target/Unwind.h" >> + >> +// C Includes >> +// C++ Includes >> + >> +using namespace lldb; >> +using namespace lldb_private; >> +using namespace minidump; >> + >> +ThreadMinidump::ThreadMinidump(Process &process, const MinidumpThread &td, >> + llvm::ArrayRef<uint8_t> gpregset_data) >> + : Thread(process, td.thread_id), m_thread_reg_ctx_sp(), >> + m_gpregset_data(gpregset_data) {} >> + >> +ThreadMinidump::~ThreadMinidump() {} >> + >> +void ThreadMinidump::RefreshStateAfterStop() {} >> + >> +void ThreadMinidump::ClearStackFrames() {} >> + >> +RegisterContextSP ThreadMinidump::GetRegisterContext() { >> + if (!m_reg_context_sp) { >> + m_reg_context_sp = CreateRegisterContextForFrame(nullptr); >> + } >> + return m_reg_context_sp; >> +} >> + >> +RegisterContextSP >> +ThreadMinidump::CreateRegisterContextForFrame(StackFrame *frame) { >> + RegisterContextSP reg_ctx_sp; >> + uint32_t concrete_frame_idx = 0; >> + Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_THREAD)); >> + >> + if (frame) >> + concrete_frame_idx = frame->GetConcreteFrameIndex(); >> + >> + if (concrete_frame_idx == 0) { >> + if (m_thread_reg_ctx_sp) >> + return m_thread_reg_ctx_sp; >> + >> + ProcessMinidump *process = >> + static_cast<ProcessMinidump *>(GetProcess().get()); >> + ArchSpec arch = process->GetArchitecture(); >> + RegisterInfoInterface *reg_interface = nullptr; >> + >> + // TODO write other register contexts and add them here >> + switch (arch.GetMachine()) { >> + case llvm::Triple::x86: { >> + reg_interface = new RegisterContextLinux_i386(arch); >> + lldb::DataBufferSP buf = >> + ConvertMinidumpContext_x86_32(m_gpregset_data, reg_interface); >> + DataExtractor gpregs(buf, lldb::eByteOrderLittle, 4); >> + DataExtractor fpregs; >> + m_thread_reg_ctx_sp.reset(new RegisterContextCorePOSIX_x86_64( >> + *this, reg_interface, gpregs, fpregs)); >> + break; >> + } >> + case llvm::Triple::x86_64: { >> + reg_interface = new RegisterContextLinux_x86_64(arch); >> + lldb::DataBufferSP buf = >> + ConvertMinidumpContext_x86_64(m_gpregset_data, reg_interface); >> + DataExtractor gpregs(buf, lldb::eByteOrderLittle, 8); >> + DataExtractor fpregs; >> + m_thread_reg_ctx_sp.reset(new RegisterContextCorePOSIX_x86_64( >> + *this, reg_interface, gpregs, fpregs)); >> + break; >> + } >> + default: >> + break; >> + } >> + >> + if (!reg_interface) { >> + if (log) >> + log->Printf("elf-core::%s:: Architecture(%d) not supported", >> + __FUNCTION__, arch.GetMachine()); >> + assert(false && "Architecture not supported"); >> + } >> + >> + reg_ctx_sp = m_thread_reg_ctx_sp; >> + } else if (m_unwinder_ap) { >> + reg_ctx_sp = m_unwinder_ap->CreateRegisterContextForFrame(frame); >> + } >> + >> + return reg_ctx_sp; >> +} >> + >> +bool ThreadMinidump::CalculateStopInfo() { return false; } >> >> Added: lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.h >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.h?rev=285587&view=auto >> ============================================================================== >> --- lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.h (added) >> +++ lldb/trunk/source/Plugins/Process/minidump/ThreadMinidump.h Mon Oct 31 >> 10:35:18 2016 >> @@ -0,0 +1,52 @@ >> +//===-- ThreadMinidump.h ---------------------------------------*- C++ >> -*-===// >> +// >> +// The LLVM Compiler Infrastructure >> +// >> +// This file is distributed under the University of Illinois Open Source >> +// License. See LICENSE.TXT for details. >> +// >> +//===----------------------------------------------------------------------===// >> + >> +#ifndef liblldb_ThreadMinidump_h_ >> +#define liblldb_ThreadMinidump_h_ >> + >> +// Project includes >> +#include "MinidumpTypes.h" >> + >> +// Other libraries and framework includes >> +#include "lldb/Target/Thread.h" >> + >> +// C Includes >> +// C++ Includes >> + >> +namespace lldb_private { >> + >> +namespace minidump { >> + >> +class ThreadMinidump : public Thread { >> +public: >> + ThreadMinidump(Process &process, const MinidumpThread &td, >> + llvm::ArrayRef<uint8_t> gpregset_data); >> + >> + ~ThreadMinidump() override; >> + >> + void RefreshStateAfterStop() override; >> + >> + lldb::RegisterContextSP GetRegisterContext() override; >> + >> + lldb::RegisterContextSP >> + CreateRegisterContextForFrame(StackFrame *frame) override; >> + >> + void ClearStackFrames() override; >> + >> +protected: >> + lldb::RegisterContextSP m_thread_reg_ctx_sp; >> + llvm::ArrayRef<uint8_t> m_gpregset_data; >> + >> + bool CalculateStopInfo() override; >> +}; >> + >> +} // namespace minidump >> +} // namespace lldb_private >> + >> +#endif // liblldb_ThreadMinidump_h_ >> >> Modified: lldb/trunk/unittests/Process/minidump/MinidumpParserTest.cpp >> URL: >> http://llvm.org/viewvc/llvm-project/lldb/trunk/unittests/Process/minidump/MinidumpParserTest.cpp?rev=285587&r1=285586&r2=285587&view=diff >> ============================================================================== >> --- lldb/trunk/unittests/Process/minidump/MinidumpParserTest.cpp (original) >> +++ lldb/trunk/unittests/Process/minidump/MinidumpParserTest.cpp Mon Oct 31 >> 10:35:18 2016 >> @@ -134,7 +134,7 @@ TEST_F(MinidumpParserTest, GetModuleList >> llvm::Optional<std::string> name = >> parser->GetMinidumpString(modules[i].module_name_rva); >> ASSERT_TRUE(name.hasValue()); >> - ASSERT_EQ(module_names[i], name.getValue()); >> + EXPECT_EQ(module_names[i], name.getValue()); >> } >> } >> >> @@ -275,8 +275,46 @@ TEST_F(MinidumpParserTest, GetPidWindows >> ASSERT_EQ(4440UL, pid.getValue()); >> } >> >> -// Register stuff >> -// TODO probably split register stuff tests into different file? >> +// wow64 >> +TEST_F(MinidumpParserTest, GetPidWow64) { >> + SetUpData("fizzbuzz_wow64.dmp"); >> + llvm::Optional<lldb::pid_t> pid = parser->GetPid(); >> + ASSERT_TRUE(pid.hasValue()); >> + ASSERT_EQ(7836UL, pid.getValue()); >> +} >> + >> +TEST_F(MinidumpParserTest, GetModuleListWow64) { >> + SetUpData("fizzbuzz_wow64.dmp"); >> + llvm::ArrayRef<MinidumpModule> modules = parser->GetModuleList(); >> + ASSERT_EQ(16UL, modules.size()); >> + std::string module_names[16] = { >> + >> R"(D:\src\llvm\llvm\tools\lldb\packages\Python\lldbsuite\test\functionalities\postmortem\wow64_minidump\fizzbuzz.exe)", >> + R"(C:\Windows\System32\ntdll.dll)", >> + R"(C:\Windows\System32\wow64.dll)", >> + R"(C:\Windows\System32\wow64win.dll)", >> + R"(C:\Windows\System32\wow64cpu.dll)", >> + >> R"(D:\src\llvm\llvm\tools\lldb\packages\Python\lldbsuite\test\functionalities\postmortem\wow64_minidump\fizzbuzz.exe)", >> + R"(C:\Windows\SysWOW64\ntdll.dll)", >> + R"(C:\Windows\SysWOW64\kernel32.dll)", >> + R"(C:\Windows\SysWOW64\KERNELBASE.dll)", >> + R"(C:\Windows\SysWOW64\advapi32.dll)", >> + R"(C:\Windows\SysWOW64\msvcrt.dll)", >> + R"(C:\Windows\SysWOW64\sechost.dll)", >> + R"(C:\Windows\SysWOW64\rpcrt4.dll)", >> + R"(C:\Windows\SysWOW64\sspicli.dll)", >> + R"(C:\Windows\SysWOW64\CRYPTBASE.dll)", >> + R"(C:\Windows\System32\api-ms-win-core-synch-l1-2-0.DLL)", >> + }; >> + >> + for (int i = 0; i < 16; ++i) { >> + llvm::Optional<std::string> name = >> + parser->GetMinidumpString(modules[i].module_name_rva); >> + ASSERT_TRUE(name.hasValue()); >> + EXPECT_EQ(module_names[i], name.getValue()); >> + } >> +} >> + >> +// Register tests >> #define REG_VAL32(x) *(reinterpret_cast<uint32_t *>(x)) >> #define REG_VAL64(x) *(reinterpret_cast<uint64_t *>(x)) >> >> @@ -371,3 +409,45 @@ TEST_F(MinidumpParserTest, ConvertMinidu >> } >> } >> } >> + >> +TEST_F(MinidumpParserTest, ConvertMinidumpContext_x86_32_wow64) { >> + SetUpData("fizzbuzz_wow64.dmp"); >> + llvm::ArrayRef<MinidumpThread> thread_list = parser->GetThreads(); >> + const MinidumpThread thread = thread_list[0]; >> + llvm::ArrayRef<uint8_t> registers(parser->GetThreadContextWow64(thread)); >> + >> + ArchSpec arch = parser->GetArchitecture(); >> + RegisterInfoInterface *reg_interface = new >> RegisterContextLinux_i386(arch); >> + lldb::DataBufferSP buf = >> + ConvertMinidumpContext_x86_32(registers, reg_interface); >> + ASSERT_EQ(reg_interface->GetGPRSize(), buf->GetByteSize()); >> + >> + const RegisterInfo *reg_info = reg_interface->GetRegisterInfo(); >> + >> + std::map<uint64_t, uint32_t> reg_values; >> + >> + reg_values[lldb_eax_i386] = 0x00000000; >> + reg_values[lldb_ebx_i386] = 0x0037f608; >> + reg_values[lldb_ecx_i386] = 0x00e61578; >> + reg_values[lldb_edx_i386] = 0x00000008; >> + reg_values[lldb_edi_i386] = 0x00000000; >> + reg_values[lldb_esi_i386] = 0x00000002; >> + reg_values[lldb_ebp_i386] = 0x0037f654; >> + reg_values[lldb_esp_i386] = 0x0037f5b8; >> + reg_values[lldb_eip_i386] = 0x77ce01fd; >> + reg_values[lldb_eflags_i386] = 0x00000246; >> + reg_values[lldb_cs_i386] = 0x00000023; >> + reg_values[lldb_fs_i386] = 0x00000053; >> + reg_values[lldb_gs_i386] = 0x0000002b; >> + reg_values[lldb_ss_i386] = 0x0000002b; >> + reg_values[lldb_ds_i386] = 0x0000002b; >> + reg_values[lldb_es_i386] = 0x0000002b; >> + >> + for (uint32_t reg_index = 0; reg_index < >> reg_interface->GetRegisterCount(); >> + ++reg_index) { >> + if (reg_values.find(reg_index) != reg_values.end()) { >> + EXPECT_EQ(reg_values[reg_index], >> + REG_VAL32(buf->GetBytes() + >> reg_info[reg_index].byte_offset)); >> + } >> + } >> +} >> \ No newline at end of file >> >> >> _______________________________________________ >> lldb-commits mailing list >> lldb-commits@lists.llvm.org >> http://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits > > _______________________________________________ > lldb-commits mailing list > lldb-commits@lists.llvm.org > http://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits _______________________________________________ lldb-commits mailing list lldb-commits@lists.llvm.org http://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits