LLDB mainline
ProcessMinidump.cpp
Go to the documentation of this file.
1//===-- ProcessMinidump.cpp -----------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#include "ProcessMinidump.h"
10
11#include "ThreadMinidump.h"
12
14#include "lldb/Core/Module.h"
17#include "lldb/Core/Section.h"
28#include "lldb/Target/Target.h"
33#include "lldb/Utility/Log.h"
34#include "lldb/Utility/State.h"
35#include "llvm/BinaryFormat/Magic.h"
36#include "llvm/Support/MemoryBuffer.h"
37
41
42#include <memory>
43#include <optional>
44
45using namespace lldb;
46using namespace lldb_private;
47using namespace minidump;
48
50
51namespace {
52
53/// Duplicate the HashElfTextSection() from the breakpad sources.
54///
55/// Breakpad, a Google crash log reporting tool suite, creates minidump files
56/// for many different architectures. When using Breakpad to create ELF
57/// minidumps, it will check for a GNU build ID when creating a minidump file
58/// and if one doesn't exist in the file, it will say the UUID of the file is a
59/// checksum of up to the first 4096 bytes of the .text section. Facebook also
60/// uses breakpad and modified this hash to avoid collisions so we can
61/// calculate and check for this as well.
62///
63/// The breakpad code might end up hashing up to 15 bytes that immediately
64/// follow the .text section in the file, so this code must do exactly what it
65/// does so we can get an exact match for the UUID.
66///
67/// \param[in] module_sp The module to grab the .text section from.
68///
69/// \param[in,out] breakpad_uuid A vector that will receive the calculated
70/// breakpad .text hash.
71///
72/// \param[in,out] facebook_uuid A vector that will receive the calculated
73/// facebook .text hash.
74///
75void HashElfTextSection(ModuleSP module_sp, std::vector<uint8_t> &breakpad_uuid,
76 std::vector<uint8_t> &facebook_uuid) {
77 SectionList *sect_list = module_sp->GetSectionList();
78 if (sect_list == nullptr)
79 return;
80 SectionSP sect_sp = sect_list->FindSectionByName(ConstString(".text"));
81 if (!sect_sp)
82 return;
83 constexpr size_t kMDGUIDSize = 16;
84 constexpr size_t kBreakpadPageSize = 4096;
85 // The breakpad code has a bug where it might access beyond the end of a
86 // .text section by up to 15 bytes, so we must ensure we round up to the
87 // next kMDGUIDSize byte boundary.
88 DataExtractorSP extractor_sp;
89 const size_t text_size = sect_sp->GetFileSize();
90 const size_t read_size = std::min<size_t>(
91 llvm::alignTo(text_size, kMDGUIDSize), kBreakpadPageSize);
92 sect_sp->GetObjectFile()->GetData(sect_sp->GetFileOffset(), read_size,
93 extractor_sp);
94
95 breakpad_uuid.assign(kMDGUIDSize, 0);
96 facebook_uuid.assign(kMDGUIDSize, 0);
97
98 // The only difference between the breakpad hash and the facebook hash is the
99 // hashing of the text section size into the hash prior to hashing the .text
100 // contents.
101 for (size_t i = 0; i < kMDGUIDSize; i++)
102 facebook_uuid[i] ^= text_size % 255;
103
104 // This code carefully duplicates how the hash was created in Breakpad
105 // sources, including the error where it might has an extra 15 bytes past the
106 // end of the .text section if the .text section is less than a page size in
107 // length.
108 const uint8_t *ptr = extractor_sp->GetDataStart();
109 const uint8_t *ptr_end = extractor_sp->GetDataEnd();
110 while (ptr < ptr_end) {
111 for (unsigned i = 0; i < kMDGUIDSize; i++) {
112 breakpad_uuid[i] ^= ptr[i];
113 facebook_uuid[i] ^= ptr[i];
114 }
115 ptr += kMDGUIDSize;
116 }
117}
118
119} // namespace
120
122 return "Minidump plug-in.";
123}
124
126 lldb::ListenerSP listener_sp,
127 const FileSpec *crash_file,
128 bool can_connect) {
129 if (!crash_file || can_connect)
130 return nullptr;
131
132 lldb::ProcessSP process_sp;
133 // Read enough data for the Minidump header
134 constexpr size_t header_size = sizeof(Header);
135 auto DataPtr = FileSystem::Instance().CreateDataBuffer(crash_file->GetPath(),
136 header_size, 0);
137 if (!DataPtr)
138 return nullptr;
139
140 lldbassert(DataPtr->GetByteSize() == header_size);
141 if (identify_magic(toStringRef(DataPtr->GetData())) != llvm::file_magic::minidump)
142 return nullptr;
143
144 auto AllData =
145 FileSystem::Instance().CreateDataBuffer(crash_file->GetPath(), -1, 0);
146 if (!AllData)
147 return nullptr;
148
149 return std::make_shared<ProcessMinidump>(target_sp, listener_sp, *crash_file,
150 std::move(AllData));
151}
152
154 bool plugin_specified_by_name) {
155 return true;
156}
157
159 lldb::ListenerSP listener_sp,
160 const FileSpec &core_file,
161 DataBufferSP core_data)
162 : PostMortemProcess(target_sp, listener_sp, core_file),
163 m_core_data(std::move(core_data)), m_is_wow64(false) {}
164
166 Clear();
167 // We need to call finalize on the process before destroying ourselves to
168 // make sure all of the broadcaster cleanup goes as planned. If we destruct
169 // this class, then Process::~Process() might have problems trying to fully
170 // destroy the broadcaster.
171 Finalize(true /* destructing */);
172}
173
179
183
185 auto expected_parser = MinidumpParser::Create(m_core_data);
186 if (!expected_parser)
187 return Status::FromError(expected_parser.takeError());
188 m_minidump_parser = std::move(*expected_parser);
189
191
192 // Do we support the minidump's architecture?
193 ArchSpec arch = GetArchitecture();
194 switch (arch.GetMachine()) {
195 case llvm::Triple::x86:
196 case llvm::Triple::x86_64:
197 case llvm::Triple::arm:
198 case llvm::Triple::aarch64:
199 // Any supported architectures must be listed here and also supported in
200 // ThreadMinidump::CreateRegisterContextForFrame().
201 break;
202 default:
204 "unsupported minidump architecture: %s", arch.GetArchitectureName());
205 return error;
206 }
207 GetTarget().SetArchitecture(arch, true /*set_platform*/);
208
209 m_thread_list = m_minidump_parser->GetThreads();
210 auto exception_stream_it = m_minidump_parser->GetExceptionStreams();
211 for (auto exception_stream : exception_stream_it) {
212 // If we can't read an exception stream skip it
213 // We should probably serve a warning
214 if (!exception_stream)
215 continue;
216
218 .try_emplace(exception_stream->ThreadId, exception_stream.get())
219 .second) {
221 "Duplicate exception stream for tid {0}", exception_stream->ThreadId);
222 }
223 }
224
226
228 if (ModuleSP module = GetTarget().GetExecutableModule())
229 GetTarget().MergeArchitecture(module->GetArchitecture());
230 std::optional<lldb::pid_t> pid = m_minidump_parser->GetPid();
231 if (!pid) {
232 Debugger::ReportWarning("unable to retrieve process ID from minidump file, "
233 "setting process ID to 1",
234 GetTarget().GetDebugger().GetID());
235 pid = 1;
236 }
237 SetID(*pid);
238
239 return error;
240}
241
243
245
246 for (const auto &[_, exception_stream] : m_exceptions_by_tid) {
247 constexpr uint32_t BreakpadDumpRequested = 0xFFFFFFFF;
248 if (exception_stream.ExceptionRecord.ExceptionCode ==
249 BreakpadDumpRequested) {
250 // This "ExceptionCode" value is a sentinel that is sometimes used
251 // when generating a dump for a process that hasn't crashed.
252
253 // TODO: The definition and use of this "dump requested" constant
254 // in Breakpad are actually Linux-specific, and for similar use
255 // cases on Mac/Windows it defines different constants, referring
256 // to them as "simulated" exceptions; consider moving this check
257 // down to the OS-specific paths and checking each OS for its own
258 // constant.
259 return;
260 }
261
262 lldb::StopInfoSP stop_info;
263 lldb::ThreadSP stop_thread;
264
265 Process::m_thread_list.SetSelectedThreadByID(exception_stream.ThreadId);
266 stop_thread = Process::m_thread_list.GetSelectedThread();
267 ArchSpec arch = GetArchitecture();
268
269 if (arch.GetTriple().getOS() == llvm::Triple::Linux) {
270 uint32_t signo = exception_stream.ExceptionRecord.ExceptionCode;
271 if (signo == 0) {
272 // No stop.
273 return;
274 }
275 const char *description = nullptr;
276 if (exception_stream.ExceptionRecord.ExceptionFlags ==
277 llvm::minidump::Exception::LLDB_FLAG)
278 description = reinterpret_cast<const char *>(
279 exception_stream.ExceptionRecord.ExceptionInformation);
280
281 llvm::StringRef description_str(description,
282 Exception::MaxParameterBytes);
284 *stop_thread, signo, description_str.str().c_str());
285 } else if (arch.GetTriple().getVendor() == llvm::Triple::Apple) {
287 *stop_thread, exception_stream.ExceptionRecord.ExceptionCode, 2,
288 exception_stream.ExceptionRecord.ExceptionFlags,
289 exception_stream.ExceptionRecord.ExceptionAddress, 0);
290 } else {
291 std::string desc;
292 llvm::raw_string_ostream desc_stream(desc);
293 desc_stream << "Exception "
294 << llvm::format_hex(
295 exception_stream.ExceptionRecord.ExceptionCode, 8)
296 << " encountered at address "
297 << llvm::format_hex(
298 exception_stream.ExceptionRecord.ExceptionAddress, 8);
299 stop_info =
300 StopInfo::CreateStopReasonWithException(*stop_thread, desc.c_str());
301 }
302
303 stop_thread->SetStopInfo(stop_info);
304 }
305}
306
307bool ProcessMinidump::IsAlive() { return true; }
308
309bool ProcessMinidump::WarnBeforeDetach() const { return false; }
310
311size_t ProcessMinidump::ReadMemory(lldb::addr_t addr, void *buf, size_t size,
312 Status &error) {
313 // Don't allow the caching that lldb_private::Process::ReadMemory does since
314 // we have it all cached in our dump file anyway.
315 return DoReadMemory(addr, buf, size, error);
316}
317
318size_t ProcessMinidump::DoReadMemory(lldb::addr_t addr, void *buf, size_t size,
319 Status &error) {
320
321 llvm::Expected<llvm::ArrayRef<uint8_t>> mem_maybe =
322 m_minidump_parser->GetMemory(addr, size);
323 if (!mem_maybe) {
324 error = Status::FromError(mem_maybe.takeError());
325 return 0;
326 }
327
328 llvm::ArrayRef<uint8_t> mem = *mem_maybe;
329
330 std::memcpy(buf, mem.data(), mem.size());
331 return mem.size();
332}
333
335 if (!m_is_wow64) {
336 return m_minidump_parser->GetArchitecture();
337 }
338
339 llvm::Triple triple;
340 triple.setVendor(llvm::Triple::VendorType::UnknownVendor);
341 triple.setArch(llvm::Triple::ArchType::x86);
342 triple.setOS(llvm::Triple::OSType::Win32);
343 return ArchSpec(triple);
344}
345
347 std::optional<llvm::ArrayRef<uint8_t>> auxv =
348 m_minidump_parser->GetStream(StreamType::LinuxAuxv);
349 if (!auxv)
350 return DataExtractor();
351
352 return DataExtractor(auxv->data(), auxv->size(), GetByteOrder(),
354}
355
357 std::optional<llvm::ArrayRef<uint8_t>> lldb_generated_section =
358 m_minidump_parser->GetRawStream(StreamType::LLDBGenerated);
359 return lldb_generated_section.has_value();
360}
361
363 // This is a workaround for the dynamic loader not playing nice in issue
364 // #119598. The specific reason we use the dynamic loader is to get the TLS
365 // info sections, which we can assume are not being written to the minidump
366 // unless it's an LLDB generate minidump.
367 if (IsLLDBMinidump())
369 return nullptr;
370}
371
374 return;
375 m_memory_regions.emplace();
376 bool is_complete;
377 std::tie(*m_memory_regions, is_complete) =
378 m_minidump_parser->BuildMemoryRegions();
379
380 if (is_complete)
381 return;
382
383 MemoryRegionInfos to_add;
384 ModuleList &modules = GetTarget().GetImages();
385 Target &target = GetTarget();
386 modules.ForEach([&](const ModuleSP &module_sp) {
387 SectionList *sections = module_sp->GetSectionList();
388 for (size_t i = 0; i < sections->GetSize(); ++i) {
389 SectionSP section_sp = sections->GetSectionAtIndex(i);
390 addr_t load_addr = target.GetSectionLoadAddress(section_sp);
391 if (load_addr == LLDB_INVALID_ADDRESS)
392 continue;
393 MemoryRegionInfo::RangeType section_range(load_addr,
394 section_sp->GetByteSize());
395 MemoryRegionInfo region =
397 if (region.GetMapped() != MemoryRegionInfo::eYes &&
398 region.GetRange().GetRangeBase() <= section_range.GetRangeBase() &&
399 section_range.GetRangeEnd() <= region.GetRange().GetRangeEnd()) {
400 to_add.emplace_back();
401 to_add.back().GetRange() = section_range;
402 to_add.back().SetLLDBPermissions(section_sp->GetPermissions());
403 to_add.back().SetMapped(MemoryRegionInfo::eYes);
404 to_add.back().SetName(module_sp->GetFileSpec().GetPath().c_str());
405 }
406 }
408 });
409 m_memory_regions->insert(m_memory_regions->end(), to_add.begin(),
410 to_add.end());
411 llvm::sort(*m_memory_regions);
412}
413
420
423 region_list = *m_memory_regions;
424 return Status();
425}
426
428
430 ThreadList &new_thread_list) {
431 for (const minidump::Thread &thread : m_thread_list) {
432 LocationDescriptor context_location = thread.Context;
433
434 // If the minidump contains an exception context, use it
435 if (auto it = m_exceptions_by_tid.find(thread.ThreadId);
436 it != m_exceptions_by_tid.end())
437 context_location = it->second.ThreadContext;
438
439 llvm::ArrayRef<uint8_t> context;
440 if (!m_is_wow64)
441 context = m_minidump_parser->GetThreadContext(context_location);
442 else
443 context = m_minidump_parser->GetThreadContextWow64(thread);
444
445 lldb::ThreadSP thread_sp(new ThreadMinidump(*this, thread, context));
446 new_thread_list.AddThread(thread_sp);
447 }
448 return new_thread_list.GetSize(false) > 0;
449}
450
452 llvm::StringRef name,
453 ModuleSpec module_spec) {
456
457 ModuleSP module_sp =
458 GetTarget().GetOrCreateModule(module_spec, true /* notify */, &error);
459 if (!module_sp)
460 return module_sp;
461 // We consider the module to be a match if the minidump UUID is a
462 // prefix of the actual UUID, or if either of the UUIDs are empty.
463 const auto dmp_bytes = minidump_uuid.GetBytes();
464 const auto mod_bytes = module_sp->GetUUID().GetBytes();
465 const bool match = dmp_bytes.empty() || mod_bytes.empty() ||
466 mod_bytes.take_front(dmp_bytes.size()) == dmp_bytes;
467 if (match) {
468 LLDB_LOG(log, "Partial uuid match for {0}.", name);
469 return module_sp;
470 }
471
472 // Breakpad generates minindump files, and if there is no GNU build
473 // ID in the binary, it will calculate a UUID by hashing first 4096
474 // bytes of the .text section and using that as the UUID for a module
475 // in the minidump. Facebook uses a modified breakpad client that
476 // uses a slightly modified this hash to avoid collisions. Check for
477 // UUIDs from the minindump that match these cases and accept the
478 // module we find if they do match.
479 std::vector<uint8_t> breakpad_uuid;
480 std::vector<uint8_t> facebook_uuid;
481 HashElfTextSection(module_sp, breakpad_uuid, facebook_uuid);
482 if (dmp_bytes == llvm::ArrayRef<uint8_t>(breakpad_uuid)) {
483 LLDB_LOG(log, "Breakpad .text hash match for {0}.", name);
484 return module_sp;
485 }
486 if (dmp_bytes == llvm::ArrayRef<uint8_t>(facebook_uuid)) {
487 LLDB_LOG(log, "Facebook .text hash match for {0}.", name);
488 return module_sp;
489 }
490 // The UUID wasn't a partial match and didn't match the .text hash
491 // so remove the module from the target, we will need to create a
492 // placeholder object file.
493 GetTarget().GetImages().Remove(module_sp);
494 module_sp.reset();
495 return module_sp;
496}
497
499 std::vector<const minidump::Module *> filtered_modules =
500 m_minidump_parser->GetFilteredModuleList();
501
503
504 for (auto module : filtered_modules) {
505 std::string name = cantFail(m_minidump_parser->GetMinidumpFile().getString(
506 module->ModuleNameRVA));
507 const uint64_t load_addr = module->BaseOfImage;
508 const uint64_t load_size = module->SizeOfImage;
509 LLDB_LOG(log, "found module: name: {0} {1:x10}-{2:x10} size: {3}", name,
510 load_addr, load_addr + load_size, load_size);
511
512 // check if the process is wow64 - a 32 bit windows process running on a
513 // 64 bit windows
514 if (llvm::StringRef(name).ends_with_insensitive("wow64.dll")) {
515 m_is_wow64 = true;
516 }
517
518 const auto uuid = m_minidump_parser->GetModuleUUID(module);
519 auto file_spec = FileSpec(name, GetArchitecture().GetTriple());
520 ModuleSpec module_spec(file_spec, uuid);
521 module_spec.GetArchitecture() = GetArchitecture();
523 // Try and find a module with a full UUID that matches. This function will
524 // add the module to the target if it finds one.
525 lldb::ModuleSP module_sp = GetTarget().GetOrCreateModule(module_spec,
526 true /* notify */, &error);
527 if (module_sp) {
528 LLDB_LOG(log, "Full uuid match for {0}.", name);
529 } else {
530 // We couldn't find a module with an exactly-matching UUID. Sometimes
531 // a minidump UUID is only a partial match or is a hash. So try again
532 // without specifying the UUID, then again without specifying the
533 // directory if that fails. This will allow us to find modules with
534 // partial matches or hash UUIDs in user-provided sysroots or search
535 // directories (target.exec-search-paths).
536 ModuleSpec partial_module_spec = module_spec;
537 partial_module_spec.GetUUID().Clear();
538 module_sp = GetOrCreateModule(uuid, name, partial_module_spec);
539 if (!module_sp) {
540 partial_module_spec.GetFileSpec().ClearDirectory();
541 module_sp = GetOrCreateModule(uuid, name, partial_module_spec);
542 }
543 }
544 if (module_sp) {
545 // Watch out for place holder modules that have different paths, but the
546 // same UUID. If the base address is different, create a new module. If
547 // we don't then we will end up setting the load address of a different
548 // ObjectFilePlaceholder and an assertion will fire.
549 auto *objfile = module_sp->GetObjectFile();
550 if (objfile &&
551 objfile->GetPluginName() ==
553 if (((ObjectFilePlaceholder *)objfile)->GetBaseImageAddress() !=
554 load_addr)
555 module_sp.reset();
556 }
557 }
558 if (!module_sp) {
559 // We failed to locate a matching local object file. Fortunately, the
560 // minidump format encodes enough information about each module's memory
561 // range to allow us to create placeholder modules.
562 //
563 // This enables most LLDB functionality involving address-to-module
564 // translations (ex. identifing the module for a stack frame PC) and
565 // modules/sections commands (ex. target modules list, ...)
566 LLDB_LOG(log,
567 "Unable to locate the matching object file, creating a "
568 "placeholder module for: {0}",
569 name);
570
572 module_spec, load_addr, load_size);
573 // If we haven't loaded a main executable yet, set the first module to be
574 // main executable
575 if (!GetTarget().GetExecutableModule())
576 GetTarget().SetExecutableModule(module_sp);
577 else
578 GetTarget().GetImages().Append(module_sp, true /* notify */);
579 }
580
581 bool load_addr_changed = false;
582 module_sp->SetLoadAddress(GetTarget(), load_addr, false,
583 load_addr_changed);
584 }
585}
586
588 info.Clear();
589 info.SetProcessID(GetID());
592 if (module_sp) {
593 const bool add_exe_file_as_first_arg = false;
594 info.SetExecutableFile(GetTarget().GetExecutableModule()->GetFileSpec(),
595 add_exe_file_as_first_arg);
596 }
597 return true;
598}
599
600// For minidumps there's no runtime generated code so we don't need JITLoader(s)
601// Avoiding them will also speed up minidump loading since JITLoaders normally
602// try to set up symbolic breakpoints, which in turn may force loading more
603// debug information than needed.
605 if (!m_jit_loaders_up) {
606 m_jit_loaders_up = std::make_unique<JITLoaderList>();
607 }
608 return *m_jit_loaders_up;
609}
610
611#define INIT_BOOL(VAR, LONG, SHORT, DESC) \
612 VAR(LLDB_OPT_SET_1, false, LONG, SHORT, DESC, false, true)
613#define APPEND_OPT(VAR) \
614 m_option_group.Append(&VAR, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1)
615
617private:
644
646 if (m_dump_all.GetOptionValue().GetCurrentValue() ||
647 m_dump_linux_all.GetOptionValue().GetCurrentValue() ||
648 m_fb_all.GetOptionValue().GetCurrentValue() ||
649 m_dump_directory.GetOptionValue().GetCurrentValue() ||
650 m_dump_linux_cpuinfo.GetOptionValue().GetCurrentValue() ||
651 m_dump_linux_proc_status.GetOptionValue().GetCurrentValue() ||
652 m_dump_linux_lsb_release.GetOptionValue().GetCurrentValue() ||
653 m_dump_linux_cmdline.GetOptionValue().GetCurrentValue() ||
654 m_dump_linux_environ.GetOptionValue().GetCurrentValue() ||
655 m_dump_linux_auxv.GetOptionValue().GetCurrentValue() ||
656 m_dump_linux_maps.GetOptionValue().GetCurrentValue() ||
657 m_dump_linux_proc_stat.GetOptionValue().GetCurrentValue() ||
658 m_dump_linux_proc_uptime.GetOptionValue().GetCurrentValue() ||
659 m_dump_linux_proc_fd.GetOptionValue().GetCurrentValue() ||
660 m_fb_app_data.GetOptionValue().GetCurrentValue() ||
661 m_fb_build_id.GetOptionValue().GetCurrentValue() ||
662 m_fb_version.GetOptionValue().GetCurrentValue() ||
663 m_fb_java_stack.GetOptionValue().GetCurrentValue() ||
664 m_fb_dalvik.GetOptionValue().GetCurrentValue() ||
665 m_fb_unwind.GetOptionValue().GetCurrentValue() ||
666 m_fb_error_log.GetOptionValue().GetCurrentValue() ||
667 m_fb_app_state.GetOptionValue().GetCurrentValue() ||
668 m_fb_abort.GetOptionValue().GetCurrentValue() ||
669 m_fb_thread.GetOptionValue().GetCurrentValue() ||
670 m_fb_logcat.GetOptionValue().GetCurrentValue())
671 return;
672 // If no options were set, then dump everything
673 m_dump_all.GetOptionValue().SetCurrentValue(true);
674 }
675 bool DumpAll() const {
676 return m_dump_all.GetOptionValue().GetCurrentValue();
677 }
678 bool DumpDirectory() const {
679 return DumpAll() ||
680 m_dump_directory.GetOptionValue().GetCurrentValue();
681 }
682 bool DumpLinux() const {
683 return DumpAll() || m_dump_linux_all.GetOptionValue().GetCurrentValue();
684 }
685 bool DumpLinuxCPUInfo() const {
686 return DumpLinux() ||
687 m_dump_linux_cpuinfo.GetOptionValue().GetCurrentValue();
688 }
689 bool DumpLinuxProcStatus() const {
690 return DumpLinux() ||
691 m_dump_linux_proc_status.GetOptionValue().GetCurrentValue();
692 }
693 bool DumpLinuxProcStat() const {
694 return DumpLinux() ||
695 m_dump_linux_proc_stat.GetOptionValue().GetCurrentValue();
696 }
697 bool DumpLinuxLSBRelease() const {
698 return DumpLinux() ||
699 m_dump_linux_lsb_release.GetOptionValue().GetCurrentValue();
700 }
701 bool DumpLinuxCMDLine() const {
702 return DumpLinux() ||
703 m_dump_linux_cmdline.GetOptionValue().GetCurrentValue();
704 }
705 bool DumpLinuxEnviron() const {
706 return DumpLinux() ||
707 m_dump_linux_environ.GetOptionValue().GetCurrentValue();
708 }
709 bool DumpLinuxAuxv() const {
710 return DumpLinux() ||
711 m_dump_linux_auxv.GetOptionValue().GetCurrentValue();
712 }
713 bool DumpLinuxMaps() const {
714 return DumpLinux() ||
715 m_dump_linux_maps.GetOptionValue().GetCurrentValue();
716 }
717 bool DumpLinuxProcUptime() const {
718 return DumpLinux() ||
719 m_dump_linux_proc_uptime.GetOptionValue().GetCurrentValue();
720 }
721 bool DumpLinuxProcFD() const {
722 return DumpLinux() ||
723 m_dump_linux_proc_fd.GetOptionValue().GetCurrentValue();
724 }
725 bool DumpFacebook() const {
726 return DumpAll() || m_fb_all.GetOptionValue().GetCurrentValue();
727 }
728 bool DumpFacebookAppData() const {
729 return DumpFacebook() || m_fb_app_data.GetOptionValue().GetCurrentValue();
730 }
731 bool DumpFacebookBuildID() const {
732 return DumpFacebook() || m_fb_build_id.GetOptionValue().GetCurrentValue();
733 }
735 return DumpFacebook() || m_fb_version.GetOptionValue().GetCurrentValue();
736 }
738 return DumpFacebook() || m_fb_java_stack.GetOptionValue().GetCurrentValue();
739 }
741 return DumpFacebook() || m_fb_dalvik.GetOptionValue().GetCurrentValue();
742 }
744 return DumpFacebook() || m_fb_unwind.GetOptionValue().GetCurrentValue();
745 }
746 bool DumpFacebookErrorLog() const {
747 return DumpFacebook() || m_fb_error_log.GetOptionValue().GetCurrentValue();
748 }
750 return DumpFacebook() || m_fb_app_state.GetOptionValue().GetCurrentValue();
751 }
753 return DumpFacebook() || m_fb_abort.GetOptionValue().GetCurrentValue();
754 }
756 return DumpFacebook() || m_fb_thread.GetOptionValue().GetCurrentValue();
757 }
758 bool DumpFacebookLogcat() const {
759 return DumpFacebook() || m_fb_logcat.GetOptionValue().GetCurrentValue();
760 }
761public:
763 : CommandObjectParsed(interpreter, "process plugin dump",
764 "Dump information from the minidump file.", nullptr),
766 INIT_BOOL(m_dump_all, "all", 'a',
767 "Dump the everything in the minidump."),
768 INIT_BOOL(m_dump_directory, "directory", 'd',
769 "Dump the minidump directory map."),
770 INIT_BOOL(m_dump_linux_cpuinfo, "cpuinfo", 'C',
771 "Dump linux /proc/cpuinfo."),
772 INIT_BOOL(m_dump_linux_proc_status, "status", 's',
773 "Dump linux /proc/<pid>/status."),
774 INIT_BOOL(m_dump_linux_lsb_release, "lsb-release", 'r',
775 "Dump linux /etc/lsb-release."),
776 INIT_BOOL(m_dump_linux_cmdline, "cmdline", 'c',
777 "Dump linux /proc/<pid>/cmdline."),
778 INIT_BOOL(m_dump_linux_environ, "environ", 'e',
779 "Dump linux /proc/<pid>/environ."),
780 INIT_BOOL(m_dump_linux_auxv, "auxv", 'x',
781 "Dump linux /proc/<pid>/auxv."),
782 INIT_BOOL(m_dump_linux_maps, "maps", 'm',
783 "Dump linux /proc/<pid>/maps."),
785 "Dump linux /proc/<pid>/stat."),
786 INIT_BOOL(m_dump_linux_proc_uptime, "uptime", 'u',
787 "Dump linux process uptime."),
789 "Dump linux /proc/<pid>/fd."),
790 INIT_BOOL(m_dump_linux_all, "linux", 'l',
791 "Dump all linux streams."),
792 INIT_BOOL(m_fb_app_data, "fb-app-data", 1,
793 "Dump Facebook application custom data."),
794 INIT_BOOL(m_fb_build_id, "fb-build-id", 2,
795 "Dump the Facebook build ID."),
796 INIT_BOOL(m_fb_version, "fb-version", 3,
797 "Dump Facebook application version string."),
798 INIT_BOOL(m_fb_java_stack, "fb-java-stack", 4,
799 "Dump Facebook java stack."),
800 INIT_BOOL(m_fb_dalvik, "fb-dalvik-info", 5,
801 "Dump Facebook Dalvik info."),
802 INIT_BOOL(m_fb_unwind, "fb-unwind-symbols", 6,
803 "Dump Facebook unwind symbols."),
804 INIT_BOOL(m_fb_error_log, "fb-error-log", 7,
805 "Dump Facebook error log."),
806 INIT_BOOL(m_fb_app_state, "fb-app-state-log", 8,
807 "Dump Facebook java stack."),
808 INIT_BOOL(m_fb_abort, "fb-abort-reason", 9,
809 "Dump Facebook abort reason."),
810 INIT_BOOL(m_fb_thread, "fb-thread-name", 10,
811 "Dump Facebook thread name."),
812 INIT_BOOL(m_fb_logcat, "fb-logcat", 11,
813 "Dump Facebook logcat."),
814 INIT_BOOL(m_fb_all, "facebook", 12, "Dump all Facebook streams.") {
840 m_option_group.Finalize();
841 }
842
844
845 Options *GetOptions() override { return &m_option_group; }
846
847 void DoExecute(Args &command, CommandReturnObject &result) override {
848 const size_t argc = command.GetArgumentCount();
849 if (argc > 0) {
850 result.AppendErrorWithFormat("'%s' take no arguments, only options",
851 m_cmd_name.c_str());
852 return;
853 }
855
856 ProcessMinidump *process = static_cast<ProcessMinidump *>(
857 m_interpreter.GetExecutionContext().GetProcessPtr());
859 Stream &s = result.GetOutputStream();
861 if (DumpDirectory()) {
862 s.Printf("RVA SIZE TYPE StreamType\n");
863 s.Printf("---------- ---------- ---------- --------------------------\n");
864 for (const auto &stream_desc : minidump.GetMinidumpFile().streams())
865 s.Printf(
866 "0x%8.8x 0x%8.8x 0x%8.8x %s\n", (uint32_t)stream_desc.Location.RVA,
867 (uint32_t)stream_desc.Location.DataSize,
868 (unsigned)(StreamType)stream_desc.Type,
869 MinidumpParser::GetStreamTypeAsString(stream_desc.Type).data());
870 s.Printf("\n");
871 }
872 auto DumpTextStream = [&](StreamType stream_type,
873 llvm::StringRef label) -> void {
874 auto bytes = minidump.GetStream(stream_type);
875 if (!bytes.empty()) {
876 if (label.empty())
877 label = MinidumpParser::GetStreamTypeAsString(stream_type);
878 s.Printf("%s:\n%s\n\n", label.data(), bytes.data());
879 }
880 };
881 auto DumpBinaryStream = [&](StreamType stream_type,
882 llvm::StringRef label) -> void {
883 auto bytes = minidump.GetStream(stream_type);
884 if (!bytes.empty()) {
885 if (label.empty())
886 label = MinidumpParser::GetStreamTypeAsString(stream_type);
887 s.Printf("%s:\n", label.data());
888 DataExtractor data(bytes.data(), bytes.size(), eByteOrderLittle,
889 process->GetAddressByteSize());
891 bytes.size(), 16, 0, 0, 0);
892 s.Printf("\n\n");
893 }
894 };
895
896 if (DumpLinuxCPUInfo())
897 DumpTextStream(StreamType::LinuxCPUInfo, "/proc/cpuinfo");
899 DumpTextStream(StreamType::LinuxProcStatus, "/proc/PID/status");
901 DumpTextStream(StreamType::LinuxLSBRelease, "/etc/lsb-release");
902 if (DumpLinuxCMDLine())
903 DumpTextStream(StreamType::LinuxCMDLine, "/proc/PID/cmdline");
904 if (DumpLinuxEnviron())
905 DumpTextStream(StreamType::LinuxEnviron, "/proc/PID/environ");
906 if (DumpLinuxAuxv())
907 DumpBinaryStream(StreamType::LinuxAuxv, "/proc/PID/auxv");
908 if (DumpLinuxMaps())
909 DumpTextStream(StreamType::LinuxMaps, "/proc/PID/maps");
910 if (DumpLinuxProcStat())
911 DumpTextStream(StreamType::LinuxProcStat, "/proc/PID/stat");
913 DumpTextStream(StreamType::LinuxProcUptime, "uptime");
914 if (DumpLinuxProcFD())
915 DumpTextStream(StreamType::LinuxProcFD, "/proc/PID/fd");
917 DumpTextStream(StreamType::FacebookAppCustomData,
918 "Facebook App Data");
919 if (DumpFacebookBuildID()) {
920 auto bytes = minidump.GetStream(StreamType::FacebookBuildID);
921 if (bytes.size() >= 4) {
922 DataExtractor data(bytes.data(), bytes.size(), eByteOrderLittle,
923 process->GetAddressByteSize());
924 lldb::offset_t offset = 0;
925 uint32_t build_id = data.GetU32(&offset);
926 s.Printf("Facebook Build ID:\n");
927 s.Printf("%u\n", build_id);
928 s.Printf("\n");
929 }
930 }
932 DumpTextStream(StreamType::FacebookAppVersionName,
933 "Facebook Version String");
935 DumpTextStream(StreamType::FacebookJavaStack,
936 "Facebook Java Stack");
938 DumpTextStream(StreamType::FacebookDalvikInfo,
939 "Facebook Dalvik Info");
941 DumpBinaryStream(StreamType::FacebookUnwindSymbols,
942 "Facebook Unwind Symbols Bytes");
944 DumpTextStream(StreamType::FacebookDumpErrorLog,
945 "Facebook Error Log");
947 DumpTextStream(StreamType::FacebookAppStateLog,
948 "Faceook Application State Log");
950 DumpTextStream(StreamType::FacebookAbortReason,
951 "Facebook Abort Reason");
953 DumpTextStream(StreamType::FacebookThreadName,
954 "Facebook Thread Name");
955 if (DumpFacebookLogcat())
956 DumpTextStream(StreamType::FacebookLogcat, "Facebook Logcat");
957 }
958};
959
961public:
963 : CommandObjectMultiword(interpreter, "process plugin",
964 "Commands for operating on a ProcessMinidump process.",
965 "process plugin <subcommand> [<subcommand-options>]") {
966 LoadSubCommand("dump",
968 }
969
971};
972
974 if (!m_command_sp)
975 m_command_sp = std::make_shared<CommandObjectMultiwordProcessMinidump>(
976 GetTarget().GetDebugger().GetCommandInterpreter());
977 return m_command_sp.get();
978}
static void DumpDirectory(Stream &strm, const FileSpec *file_spec_ptr, uint32_t width)
static llvm::raw_ostream & error(Stream &strm)
#define lldbassert(x)
Definition LLDBAssert.h:16
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
Definition Log.h:369
#define LLDB_PLUGIN_DEFINE(PluginName)
#define INIT_BOOL(VAR, LONG, SHORT, DESC)
#define APPEND_OPT(VAR)
CommandObjectMultiwordProcessMinidump(CommandInterpreter &interpreter)
~CommandObjectMultiwordProcessMinidump() override=default
OptionGroupBoolean m_dump_linux_lsb_release
OptionGroupBoolean m_dump_linux_proc_uptime
OptionGroupBoolean m_dump_linux_proc_status
CommandObjectProcessMinidumpDump(CommandInterpreter &interpreter)
~CommandObjectProcessMinidumpDump() override=default
void DoExecute(Args &command, CommandReturnObject &result) override
A minimal ObjectFile implementation providing a dummy object file for the cases when the real module ...
static llvm::StringRef GetPluginNameStatic()
An architecture specification class.
Definition ArchSpec.h:32
llvm::Triple & GetTriple()
Architecture triple accessor.
Definition ArchSpec.h:457
llvm::Triple::ArchType GetMachine() const
Returns a machine family for the current architecture.
Definition ArchSpec.cpp:673
const char * GetArchitectureName() const
Returns a static string representing the current architecture.
Definition ArchSpec.cpp:548
A command line argument class.
Definition Args.h:33
size_t GetArgumentCount() const
Gets the number of arguments left in this command object.
Definition Args.h:120
bool LoadSubCommand(llvm::StringRef cmd_name, const lldb::CommandObjectSP &command_obj) override
CommandObjectMultiword(CommandInterpreter &interpreter, const char *name, const char *help=nullptr, const char *syntax=nullptr, uint32_t flags=0)
CommandObjectParsed(CommandInterpreter &interpreter, const char *name, const char *help=nullptr, const char *syntax=nullptr, uint32_t flags=0)
CommandInterpreter & m_interpreter
void SetStatus(lldb::ReturnStatus status)
void AppendErrorWithFormat(const char *format,...) __attribute__((format(printf
A uniqued constant string class.
Definition ConstString.h:40
An data extractor class.
uint32_t GetU32(lldb::offset_t *offset_ptr) const
Extract a uint32_t value from *offset_ptr.
static void ReportWarning(std::string message, std::optional< lldb::user_id_t > debugger_id=std::nullopt, std::once_flag *once=nullptr)
Report warning events.
A file utility class.
Definition FileSpec.h:57
void ClearDirectory()
Clear the directory in this object.
Definition FileSpec.cpp:367
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
Definition FileSpec.cpp:374
static FileSystem & Instance()
std::shared_ptr< DataBuffer > CreateDataBuffer(const llvm::Twine &path, uint64_t size=0, uint64_t offset=0)
Create memory buffer from path.
Class used by the Process to hold a list of its JITLoaders.
Range< lldb::addr_t, lldb::addr_t > RangeType
A collection class for Module objects.
Definition ModuleList.h:125
bool Remove(const lldb::ModuleSP &module_sp, bool notify=true)
Remove a module from the module list.
void Append(const lldb::ModuleSP &module_sp, bool notify=true)
Append a module to the module list.
void ForEach(std::function< IterationAction(const lldb::ModuleSP &module_sp)> const &callback) const
Applies 'callback' to each module in this ModuleList.
FileSpec & GetFileSpec()
Definition ModuleSpec.h:57
ArchSpec & GetArchitecture()
Definition ModuleSpec.h:93
static lldb::ModuleSP CreateModuleFromObjectFile(Args &&...args)
Definition Module.h:135
A command line option parsing protocol class.
Definition Options.h:58
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
PostMortemProcess(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, const FileSpec &core_file)
void SetExecutableFile(const FileSpec &exe_file, bool add_exe_file_as_first_arg)
void SetArchitecture(const ArchSpec &arch)
Definition ProcessInfo.h:66
void SetProcessID(lldb::pid_t pid)
Definition ProcessInfo.h:70
lldb::pid_t GetID() const
Returns the pid of the process or LLDB_INVALID_PROCESS_ID if there is no known pid.
Definition Process.h:537
lldb::JITLoaderListUP m_jit_loaders_up
Definition Process.h:3408
void SetUnixSignals(lldb::UnixSignalsSP &&signals_sp)
Definition Process.cpp:3711
lldb::ByteOrder GetByteOrder() const
Definition Process.cpp:3721
void SetID(lldb::pid_t new_pid)
Sets the stored pid.
Definition Process.h:542
friend class Target
Definition Process.h:360
uint32_t GetAddressByteSize() const
Definition Process.cpp:3725
virtual DynamicLoader * GetDynamicLoader()
Get the dynamic loader plug-in for this process.
Definition Process.cpp:2913
virtual void Finalize(bool destructing)
This object is about to be destroyed, do any necessary cleanup.
Definition Process.cpp:537
ThreadList m_thread_list
The threads for this process as the user will see them.
Definition Process.h:3380
friend class DynamicLoader
Definition Process.h:357
friend class ThreadList
Definition Process.h:361
Target & GetTarget()
Get the target object pointer for this module.
Definition Process.h:1250
lldb::SectionSP FindSectionByName(ConstString section_dstr) const
Definition Section.cpp:560
size_t GetSize() const
Definition Section.h:77
lldb::SectionSP GetSectionAtIndex(size_t idx) const
Definition Section.cpp:553
An error handling class.
Definition Status.h:118
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
Definition Status.cpp:106
static Status static Status FromErrorStringWithFormatv(const char *format, Args &&...args)
Definition Status.h:151
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
Definition Status.cpp:136
static lldb::StopInfoSP CreateStopReasonWithMachException(Thread &thread, uint32_t exc_type, uint32_t exc_data_count, uint64_t exc_code, uint64_t exc_sub_code, uint64_t exc_sub_sub_code, bool pc_already_adjusted=true, bool adjust_pc_if_needed=false)
static lldb::StopInfoSP CreateStopReasonWithSignal(Thread &thread, int signo, const char *description=nullptr, std::optional< int > code=std::nullopt)
static lldb::StopInfoSP CreateStopReasonWithException(Thread &thread, const char *description)
A stream class that can stream formatted output to a file.
Definition Stream.h:28
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
Definition Stream.cpp:134
lldb::addr_t GetSectionLoadAddress(const lldb::SectionSP &section_sp)
Definition Target.cpp:5347
lldb::ModuleSP GetOrCreateModule(const ModuleSpec &module_spec, bool notify, Status *error_ptr=nullptr)
Find a binary on the system and return its Module, or return an existing Module that is already in th...
Definition Target.cpp:2352
bool SetArchitecture(const ArchSpec &arch_spec, bool set_platform=false, bool merge=true)
Set the architecture for this target.
Definition Target.cpp:1705
lldb::ModuleSP GetExecutableModule()
Gets the module for the main executable.
Definition Target.cpp:1525
const ModuleList & GetImages() const
Get accessor for the images for this process.
Definition Target.h:1141
void SetExecutableModule(lldb::ModuleSP &module_sp, LoadDependentFiles load_dependent_files=eLoadDependentsDefault)
Set the main executable module.
Definition Target.cpp:1576
bool MergeArchitecture(const ArchSpec &arch_spec)
Definition Target.cpp:1796
void AddThread(const lldb::ThreadSP &thread_sp)
uint32_t GetSize(bool can_update=true)
Represents UUID's of various sizes.
Definition UUID.h:27
llvm::ArrayRef< uint8_t > GetBytes() const
Definition UUID.h:66
void Clear()
Definition UUID.h:62
static lldb::UnixSignalsSP Create(const ArchSpec &arch)
static MemoryRegionInfo GetMemoryRegionInfo(const MemoryRegionInfos &regions, lldb::addr_t load_addr)
static llvm::Expected< MinidumpParser > Create(const lldb::DataBufferSP &data_buf_sp)
static llvm::StringRef GetStreamTypeAsString(StreamType stream_type)
bool DoUpdateThreadList(ThreadList &old_thread_list, ThreadList &new_thread_list) override
Update the thread list following process plug-in's specific logic.
bool GetProcessInfo(ProcessInstanceInfo &info) override
static llvm::StringRef GetPluginNameStatic()
ProcessMinidump(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, const FileSpec &core_file, lldb::DataBufferSP code_data)
bool IsAlive() override
Check if a process is still alive.
lldb::ModuleSP GetOrCreateModule(lldb_private::UUID minidump_uuid, llvm::StringRef name, lldb_private::ModuleSpec module_spec)
size_t DoReadMemory(lldb::addr_t addr, void *buf, size_t size, Status &error) override
Actually do the reading of memory from a process.
CommandObject * GetPluginCommandObject() override
Return a multi-word command object that can be used to expose plug-in specific commands.
Status GetMemoryRegions(lldb_private::MemoryRegionInfos &region_list) override
Obtain all the mapped memory regions within this process.
std::optional< MemoryRegionInfos > m_memory_regions
static llvm::StringRef GetPluginDescriptionStatic()
static lldb::ProcessSP CreateInstance(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, const FileSpec *crash_file_path, bool can_connect)
bool WarnBeforeDetach() const override
Before lldb detaches from a process, it warns the user that they are about to lose their debug sessio...
llvm::ArrayRef< minidump::Thread > m_thread_list
lldb_private::DataExtractor GetAuxvData() override
std::unordered_map< uint32_t, const minidump::ExceptionStream > m_exceptions_by_tid
DynamicLoader * GetDynamicLoader() override
Get the dynamic loader plug-in for this process.
bool CanDebug(lldb::TargetSP target_sp, bool plugin_specified_by_name) override
Check if a plug-in instance can debug the file in module.
Status DoGetMemoryRegionInfo(lldb::addr_t load_addr, MemoryRegionInfo &range_info) override
DoGetMemoryRegionInfo is called by GetMemoryRegionInfo after it has removed non address bits from loa...
void RefreshStateAfterStop() override
Currently called as part of ShouldStop.
std::optional< MinidumpParser > m_minidump_parser
size_t ReadMemory(lldb::addr_t addr, void *buf, size_t size, Status &error) override
Read of memory from a process.
#define LLDB_INVALID_ADDRESS
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
Definition Log.h:332
lldb::offset_t DumpDataExtractor(const DataExtractor &DE, Stream *s, lldb::offset_t offset, lldb::Format item_format, size_t item_byte_size, size_t item_count, size_t num_per_line, uint64_t base_addr, uint32_t item_bit_size, uint32_t item_bit_offset, ExecutionContextScope *exe_scope=nullptr, bool show_memory_tags=false)
Dumps item_count objects into the stream s.
std::shared_ptr< lldb_private::Thread > ThreadSP
std::shared_ptr< lldb_private::CommandObject > CommandObjectSP
@ eFormatBytesWithASCII
uint64_t offset_t
Definition lldb-types.h:85
std::shared_ptr< lldb_private::Process > ProcessSP
@ eReturnStatusSuccessFinishResult
std::shared_ptr< lldb_private::Listener > ListenerSP
std::shared_ptr< lldb_private::DataBuffer > DataBufferSP
std::shared_ptr< lldb_private::StopInfo > StopInfoSP
std::shared_ptr< lldb_private::Section > SectionSP
uint64_t addr_t
Definition lldb-types.h:80
std::shared_ptr< lldb_private::Target > TargetSP
std::shared_ptr< lldb_private::DataExtractor > DataExtractorSP
std::shared_ptr< lldb_private::Module > ModuleSP
BaseType GetRangeBase() const
Definition RangeMap.h:45
BaseType GetRangeEnd() const
Definition RangeMap.h:78