30#include "llvm/ADT/StringRef.h"
31#include "llvm/BinaryFormat/Minidump.h"
32#include "llvm/Support/ConvertUTF.h"
33#include "llvm/Support/Endian.h"
34#include "llvm/Support/Error.h"
35#include "llvm/TargetParser/Triple.h"
50using namespace llvm::minidump;
63 llvm::Triple::OSType::Linux)
67 std::vector<lldb::ThreadSP> threads =
69 for (
const ThreadSP &thread_sp : threads) {
70 StopInfoSP stop_info_sp = thread_sp->GetStopInfo();
72 const StopReason &stop_reason = stop_info_sp->GetStopReason();
90 "Failed to fill in header and directory "
91 "sections. Written / Expected (%" PRIx64
" / %" PRIx64
")",
101 uint64_t stream_size) {
106 "Unable to add directory for stream type "
107 "%x, offset is greater then 32 bit limit.",
114 "Unable to add directory for stream type %x, exceeded expected number "
115 "of directories %zu.",
120 LocationDescriptor loc;
121 loc.DataSize =
static_cast<llvm::support::ulittle32_t
>(stream_size);
126 dir.Type =
static_cast<llvm::support::little_t<StreamType>
>(type);
135 StreamType type = StreamType::LLDBGenerated;
141 const llvm::Triple &target_triple =
142 m_process_sp->GetTarget().GetArchitecture().GetTriple();
144 AddDirectory(StreamType::SystemInfo,
sizeof(llvm::minidump::SystemInfo));
148 llvm::minidump::ProcessorArchitecture arch;
149 switch (target_triple.getArch()) {
150 case llvm::Triple::ArchType::x86_64:
151 arch = ProcessorArchitecture::AMD64;
153 case llvm::Triple::ArchType::x86:
154 arch = ProcessorArchitecture::X86;
156 case llvm::Triple::ArchType::arm:
157 arch = ProcessorArchitecture::ARM;
159 case llvm::Triple::ArchType::aarch64:
160 arch = ProcessorArchitecture::ARM64;
162 case llvm::Triple::ArchType::mips64:
163 case llvm::Triple::ArchType::mips64el:
164 case llvm::Triple::ArchType::mips:
165 case llvm::Triple::ArchType::mipsel:
166 arch = ProcessorArchitecture::MIPS;
168 case llvm::Triple::ArchType::ppc64:
169 case llvm::Triple::ArchType::ppc:
170 case llvm::Triple::ArchType::ppc64le:
171 arch = ProcessorArchitecture::PPC;
175 "Architecture %s not supported.",
176 target_triple.getArchName().str().c_str());
180 llvm::support::little_t<OSPlatform> platform_id;
181 switch (target_triple.getOS()) {
182 case llvm::Triple::OSType::Linux:
183 if (target_triple.getEnvironment() ==
184 llvm::Triple::EnvironmentType::Android)
185 platform_id = OSPlatform::Android;
187 platform_id = OSPlatform::Linux;
189 case llvm::Triple::OSType::Win32:
190 platform_id = OSPlatform::Win32NT;
192 case llvm::Triple::OSType::MacOSX:
193 platform_id = OSPlatform::MacOSX;
195 case llvm::Triple::OSType::IOS:
196 platform_id = OSPlatform::IOS;
200 "OS %s not supported.", target_triple.getOSName().str().c_str());
204 llvm::minidump::SystemInfo sys_info{};
205 sys_info.ProcessorArch =
206 static_cast<llvm::support::little_t<ProcessorArchitecture>
>(arch);
208 sys_info.CSDVersionRVA =
static_cast<llvm::support::ulittle32_t
>(
210 sys_info.PlatformId = platform_id;
211 m_data.AppendData(&sys_info,
sizeof(llvm::minidump::SystemInfo));
213 std::string csd_string;
229 llvm::StringRef to_write_ref(to_write.c_str(), to_write.size() + 1);
230 llvm::SmallVector<llvm::UTF16, 128> to_write_utf16;
232 bool converted = convertUTF8ToUTF16String(to_write_ref, to_write_utf16);
235 "Unable to convert the string to UTF16. Failed to convert %s",
242 llvm::support::ulittle32_t to_write_size(to_write_utf16.size_in_bytes() - 2);
244 buffer->
AppendData(&to_write_size,
sizeof(llvm::support::ulittle32_t));
245 buffer->
AppendData(to_write_utf16.data(), to_write_utf16.size_in_bytes());
253 uint64_t SizeOfImage = 0;
255 for (
const auto §ion : *mod->GetObjectFile()->GetSectionList()) {
256 SizeOfImage += section->GetByteSize();
260 SectionSP sect_sp = mod->GetObjectFile()->GetBaseAddress().GetSection();
263 return llvm::createStringError(std::errc::operation_not_supported,
264 "Couldn't obtain the section information.");
266 lldb::addr_t sect_addr = sect_sp->GetLoadBaseAddress(&target);
273 mod->GetObjectFile()->GetBaseAddress().GetLoadAddress(&target) -
275 SizeOfImage = sect_size - base_sect_offset;
280 while (next_sect_sp &&
281 next_sect_sp->GetLoadBaseAddress(&target) == next_sect_addr) {
282 sect_size = sect_sp->GetByteSize();
283 SizeOfImage += sect_size;
284 next_sect_addr += sect_size;
298 constexpr size_t minidump_module_size =
sizeof(llvm::minidump::Module);
303 llvm::support::ulittle32_t modules_count =
304 static_cast<llvm::support::ulittle32_t
>(modules.
GetSize());
317 std::vector<std::pair<ModuleSP, uint64_t>> valid_modules;
319 for (
size_t i = 0; i < modules_count; ++i) {
321 std::string module_name = mod->GetSpecificationDescription();
323 if (!maybe_mod_size) {
324 llvm::Error mod_size_err = maybe_mod_size.takeError();
326 llvm::handleAllErrors(
327 std::move(mod_size_err), [&](
const llvm::ErrorInfoBase &E) {
328 LLDB_LOGF(log,
"Unable to get the size of module %s: %s",
329 module_name.c_str(), E.message().c_str());
333 valid_modules.emplace_back(mod, *maybe_mod_size);
336 size_t module_stream_size =
sizeof(llvm::support::ulittle32_t) +
337 valid_modules.size() * minidump_module_size;
344 llvm::support::ulittle32_t count =
345 static_cast<llvm::support::ulittle32_t
>(valid_modules.size());
346 m_data.AppendData(&count,
sizeof(llvm::support::ulittle32_t));
348 for (
const auto &valid_module : valid_modules) {
350 uint64_t module_size = valid_module.second;
351 std::string module_name = mod->GetSpecificationDescription();
353 llvm::support::ulittle32_t signature =
354 static_cast<llvm::support::ulittle32_t
>(
356 auto uuid = mod->GetUUID().GetBytes();
358 VSFixedFileInfo info;
359 info.Signature =
static_cast<llvm::support::ulittle32_t
>(0u);
360 info.StructVersion =
static_cast<llvm::support::ulittle32_t
>(0u);
361 info.FileVersionHigh =
static_cast<llvm::support::ulittle32_t
>(0u);
362 info.FileVersionLow =
static_cast<llvm::support::ulittle32_t
>(0u);
363 info.ProductVersionHigh =
static_cast<llvm::support::ulittle32_t
>(0u);
364 info.ProductVersionLow =
static_cast<llvm::support::ulittle32_t
>(0u);
365 info.FileFlagsMask =
static_cast<llvm::support::ulittle32_t
>(0u);
366 info.FileFlags =
static_cast<llvm::support::ulittle32_t
>(0u);
367 info.FileOS =
static_cast<llvm::support::ulittle32_t
>(0u);
368 info.FileType =
static_cast<llvm::support::ulittle32_t
>(0u);
369 info.FileSubtype =
static_cast<llvm::support::ulittle32_t
>(0u);
370 info.FileDateHigh =
static_cast<llvm::support::ulittle32_t
>(0u);
371 info.FileDateLow =
static_cast<llvm::support::ulittle32_t
>(0u);
373 LocationDescriptor ld;
374 ld.DataSize =
static_cast<llvm::support::ulittle32_t
>(0u);
375 ld.RVA =
static_cast<llvm::support::ulittle32_t
>(0u);
379 LocationDescriptor ld_cv;
380 ld_cv.DataSize =
static_cast<llvm::support::ulittle32_t
>(
381 sizeof(llvm::support::ulittle32_t) + uuid.size());
382 ld_cv.RVA =
static_cast<llvm::support::ulittle32_t
>(
383 size_before + module_stream_size + helper_data.
GetByteSize());
385 helper_data.
AppendData(&signature,
sizeof(llvm::support::ulittle32_t));
386 helper_data.
AppendData(uuid.begin(), uuid.size());
388 llvm::minidump::Module m{};
389 m.BaseOfImage =
static_cast<llvm::support::ulittle64_t
>(
390 mod->GetObjectFile()->GetBaseAddress().GetLoadAddress(&target));
391 m.SizeOfImage =
static_cast<llvm::support::ulittle32_t
>(module_size);
392 m.Checksum =
static_cast<llvm::support::ulittle32_t
>(0);
394 static_cast<llvm::support::ulittle32_t
>(std::time(
nullptr));
395 m.ModuleNameRVA =
static_cast<llvm::support::ulittle32_t
>(
396 size_before + module_stream_size + helper_data.
GetByteSize());
397 m.VersionInfo = info;
406 m_data.AppendData(&m,
sizeof(llvm::minidump::Module));
414 llvm::StringRef reg_name) {
419 bool success = reg_ctx->
ReadRegister(reg_info, reg_value);
426 llvm::StringRef reg_name) {
431 bool success = reg_ctx->
ReadRegister(reg_info, reg_value);
438 llvm::StringRef reg_name) {
443 bool success = reg_ctx->
ReadRegister(reg_info, reg_value);
450 llvm::StringRef reg_name) {
451 return static_cast<llvm::support::ulittle16_t
>(
456 llvm::StringRef reg_name) {
457 return static_cast<llvm::support::ulittle32_t
>(
462 llvm::StringRef reg_name) {
463 return static_cast<llvm::support::ulittle64_t
>(
476 if (bytes_copied == 16)
521 return thread_context;
532 for (uint32_t i = 0; i < 31; ++i) {
533 snprintf(reg_name,
sizeof(reg_name),
"x%u", i);
543 for (uint32_t i = 0; i < 32; ++i) {
544 snprintf(reg_name,
sizeof(reg_name),
"v%u", i);
547 return thread_context;
562 case llvm::Triple::ArchType::x86_64:
565 case llvm::Triple::ArchType::aarch64:
578 case llvm::Triple::ArchType::x86_64:
580 case llvm::Triple::ArchType::aarch64:
581 return sizeof(
arm64);
597 const llvm::minidump::Thread &thread = pair.second;
598 size_t bytes_to_write =
sizeof(llvm::minidump::Thread);
599 size_t bytes_written = bytes_to_write;
601 if (
error.Fail() || bytes_to_write != bytes_written) {
603 "Wrote incorrect number of bytes to minidump file. (written %zd/%zd)",
604 bytes_written, bytes_to_write);
613 constexpr size_t minidump_thread_size =
sizeof(llvm::minidump::Thread);
614 std::vector<ThreadSP> thread_list =
619 size_t thread_stream_size =
sizeof(llvm::support::ulittle32_t) +
620 thread_list.size() * minidump_thread_size;
628 llvm::support::ulittle32_t thread_count =
629 static_cast<llvm::support::ulittle32_t
>(thread_list.size());
630 m_data.AppendData(&thread_count,
sizeof(llvm::support::ulittle32_t));
637 for (
const ThreadSP &thread_sp : thread_list) {
650 "architecture %s not supported.",
651 arch.
GetTriple().getArchName().str().c_str());
655 uint64_t
sp = reg_ctx->
GetSP();
660 MemoryDescriptor stack;
661 LocationDescriptor empty_label;
662 empty_label.DataSize = 0;
664 stack.Memory = empty_label;
665 stack.StartOfMemoryRange = 0;
666 LocationDescriptor thread_context_memory_locator;
667 thread_context_memory_locator.DataSize =
668 static_cast<llvm::support::ulittle32_t
>(thread_context.
size());
669 thread_context_memory_locator.RVA =
static_cast<llvm::support::ulittle32_t
>(
670 size_before + thread_stream_size + helper_data.
GetByteSize());
674 LLDB_LOGF(log,
"AddThreadList for thread %d: thread_context %zu bytes",
675 thread_sp->GetIndexID(), thread_context.
size());
678 llvm::minidump::Thread t;
679 t.ThreadId =
static_cast<llvm::support::ulittle32_t
>(thread_sp->GetID());
680 t.SuspendCount =
static_cast<llvm::support::ulittle32_t
>(
682 t.PriorityClass =
static_cast<llvm::support::ulittle32_t
>(0);
683 t.Priority =
static_cast<llvm::support::ulittle32_t
>(0);
684 t.EnvironmentBlock =
static_cast<llvm::support::ulittle64_t
>(0);
685 t.Stack = stack, t.Context = thread_context_memory_locator;
689 m_data.AppendData(&t,
sizeof(llvm::minidump::Thread));
692 LLDB_LOGF(log,
"AddThreadList(): total helper_data %" PRIx64
" bytes",
699 std::vector<ThreadSP> thread_list =
702 for (
const ThreadSP &thread_sp : thread_list) {
703 StopInfoSP stop_info_sp = thread_sp->GetStopInfo();
709 constexpr size_t minidump_exception_size =
710 sizeof(llvm::minidump::ExceptionStream);
717 exp_record.ExceptionCode =
718 static_cast<llvm::support::ulittle32_t
>(stop_info_sp->GetValue());
719 exp_record.ExceptionFlags =
720 static_cast<llvm::support::ulittle32_t
>(Exception::LLDB_FLAG);
721 exp_record.ExceptionRecord =
static_cast<llvm::support::ulittle64_t
>(0);
722 exp_record.ExceptionAddress = reg_ctx_sp->GetPC();
723 exp_record.NumberParameters =
static_cast<llvm::support::ulittle32_t
>(1);
724 std::string description = stop_info_sp->GetDescription();
727 memcpy(&exp_record.ExceptionInformation, description.c_str(),
728 std::min(description.size(), Exception::MaxParameterBytes));
729 exp_record.UnusedAlignment =
static_cast<llvm::support::ulittle32_t
>(0);
730 ExceptionStream exp_stream;
731 exp_stream.ThreadId =
732 static_cast<llvm::support::ulittle32_t
>(thread_sp->GetID());
733 exp_stream.UnusedAlignment =
static_cast<llvm::support::ulittle32_t
>(0);
734 exp_stream.ExceptionRecord = exp_record;
737 exp_stream.ThreadContext = Iter->second;
739 exp_stream.ThreadContext.DataSize = 0;
740 exp_stream.ThreadContext.RVA = 0;
742 m_data.AppendData(&exp_stream, minidump_exception_size);
756 misc_info.
size =
static_cast<llvm::support::ulittle32_t
>(
760 misc_info.
flags1 =
static_cast<llvm::support::ulittle32_t
>(0);
767 static_cast<llvm::support::ulittle32_t
>(
static_cast<uint32_t
>(
770 static_cast<llvm::support::ulittle32_t
>(process_info.
GetProcessID());
773 m_data.AppendData(&misc_info,
778std::unique_ptr<llvm::MemoryBuffer>
780 auto maybe_stream = llvm::MemoryBuffer::getFileAsStream(path);
783 return std::move(maybe_stream.get());
789 if (
m_process_sp->GetTarget().GetArchitecture().GetTriple().getOS() !=
793 std::vector<std::pair<StreamType, std::string>> files_with_stream_types = {
794 {StreamType::LinuxCPUInfo,
"/proc/cpuinfo"},
795 {StreamType::LinuxLSBRelease,
"/etc/lsb-release"},
802 std::string pid_str = std::to_string(pid);
803 files_with_stream_types.push_back(
804 {StreamType::LinuxProcStatus,
"/proc/" + pid_str +
"/status"});
805 files_with_stream_types.push_back(
806 {StreamType::LinuxCMDLine,
"/proc/" + pid_str +
"/cmdline"});
807 files_with_stream_types.push_back(
808 {StreamType::LinuxEnviron,
"/proc/" + pid_str +
"/environ"});
809 files_with_stream_types.push_back(
810 {StreamType::LinuxAuxv,
"/proc/" + pid_str +
"/auxv"});
811 files_with_stream_types.push_back(
812 {StreamType::LinuxMaps,
"/proc/" + pid_str +
"/maps"});
813 files_with_stream_types.push_back(
814 {StreamType::LinuxProcStat,
"/proc/" + pid_str +
"/stat"});
815 files_with_stream_types.push_back(
816 {StreamType::LinuxProcFD,
"/proc/" + pid_str +
"/fd"});
819 for (
const auto &entry : files_with_stream_types) {
820 StreamType stream = entry.first;
821 std::string path = entry.second;
825 size_t size = memory_buffer->getBufferSize();
831 m_data.AppendData(memory_buffer->getBufferStart(), size);
845 std::vector<CoreFileMemoryRange> ranges_32;
846 llvm::Expected<CoreFileMemoryRanges> all_core_memory_ranges_maybe =
848 if (!all_core_memory_ranges_maybe)
852 *all_core_memory_ranges_maybe;
855 all_core_memory_ranges.
GetSize());
856 std::vector<CoreFileMemoryRange> all_core_memory_vec;
859 for (
const auto &core_range : all_core_memory_ranges)
860 all_core_memory_vec.push_back(core_range.data);
865 auto iterator = all_core_memory_vec.begin();
866 while (iterator != all_core_memory_vec.end()) {
869 ranges_32.push_back(*iterator);
871 iterator->range.size() +
sizeof(llvm::minidump::MemoryDescriptor);
872 iterator = all_core_memory_vec.erase(iterator);
880 total_size +=
sizeof(llvm::minidump::MemoryListHeader);
884 "Unable to write minidump. Stack memory "
885 "exceeds 32b limit. (Num Stacks %zu)",
897 if (!all_core_memory_ranges.
IsEmpty()) {
908 llvm::minidump::Header header;
909 header.Signature =
static_cast<llvm::support::ulittle32_t
>(
910 llvm::minidump::Header::MagicSignature);
911 header.Version =
static_cast<llvm::support::ulittle32_t
>(
912 llvm::minidump::Header::MagicVersion);
913 header.NumberOfStreams =
914 static_cast<llvm::support::ulittle32_t
>(
m_directories.size());
916 header.StreamDirectoryRVA =
917 static_cast<llvm::support::ulittle32_t
>(
HEADER_SIZE);
918 header.Checksum =
static_cast<llvm::support::ulittle32_t
>(
920 header.TimeDateStamp =
921 static_cast<llvm::support::ulittle32_t
>(std::time(
nullptr));
923 static_cast<llvm::support::ulittle64_t
>(0u);
926 size_t bytes_written;
934 "Unable to write the minidump header (written %zd/%zd)",
947 size_t bytes_written;
955 "unable to write the directory (written %zd/%zd)", bytes_written,
971 uint64_t total_bytes_read = 0;
976 if (
error.Fail() || bytes_read == 0) {
978 "Failed to read memory region at: 0x%" PRIx64
979 ". Bytes read: 0x%" PRIx64
", error: %s",
980 current_addr, bytes_read,
error.AsCString());
992 if (current_addr != addr + total_bytes_read) {
994 "Current addr is at unexpected address, 0x%" PRIx64
995 ", expected at 0x%" PRIx64,
996 current_addr, addr + total_bytes_read);
1001 "Unexpected address encounterd when reading memory in chunks "
1002 "0x%" PRIx64
" expected 0x%" PRIx64,
1003 current_addr, addr + total_bytes_read);
1012 addDataError =
AddData(buf, bytes_read);
1013 if (addDataError.
Fail())
1016 total_bytes_read += bytes_read;
1019 if (bytes_read != data_buffer.
GetByteSize() && total_bytes_read != size) {
1021 "Memory region at: 0x%" PRIx64
" partial read 0x%" PRIx64
1022 " bytes out of 0x%" PRIx64
" bytes.",
1023 current_addr, bytes_read,
1037 return addDataError;
1042 uint64_t max_size = 0;
1043 for (
const auto &core_range : ranges)
1044 max_size = std::max(max_size, core_range.range.size());
1051 std::vector<MemoryDescriptor> descriptors;
1053 if (ranges.size() == 0)
1057 size_t region_index = 0;
1060 for (
const auto &core_range : ranges) {
1063 const addr_t addr = core_range.range.start();
1064 const addr_t size = core_range.range.size();
1065 const addr_t end = core_range.range.end();
1068 "AddMemoryList %zu/%zu reading memory for region "
1069 "(0x%" PRIx64
" bytes) [0x%" PRIx64
", 0x%" PRIx64
")",
1070 region_index, ranges.size(), size, addr, addr + size);
1073 progress.
Increment(1,
"Adding Memory Range " + core_range.Dump());
1074 uint64_t bytes_read = 0;
1081 if (bytes_read == 0)
1084 MemoryDescriptor descriptor;
1085 descriptor.StartOfMemoryRange =
1086 static_cast<llvm::support::ulittle64_t
>(addr);
1087 descriptor.Memory.DataSize =
1088 static_cast<llvm::support::ulittle32_t
>(bytes_read);
1089 descriptor.Memory.RVA =
1090 static_cast<llvm::support::ulittle32_t
>(offset_for_data);
1091 descriptors.push_back(descriptor);
1100 sizeof(llvm::minidump::MemoryListHeader) +
1101 descriptors.size() *
1102 sizeof(llvm::minidump::MemoryDescriptor));
1106 llvm::minidump::MemoryListHeader list_header;
1107 llvm::support::ulittle32_t memory_ranges_num =
1108 static_cast<llvm::support::ulittle32_t
>(descriptors.size());
1109 list_header.NumberOfMemoryRanges = memory_ranges_num;
1110 m_data.AppendData(&list_header,
sizeof(llvm::minidump::MemoryListHeader));
1113 m_data.AppendData(descriptors.data(),
1114 descriptors.size() *
sizeof(MemoryDescriptor));
1127 (
sizeof(llvm::minidump::Memory64ListHeader)) +
1129 sizeof(llvm::minidump::MemoryDescriptor_64));
1133 llvm::minidump::Memory64ListHeader list_header;
1134 llvm::support::ulittle64_t memory_ranges_num =
1135 static_cast<llvm::support::ulittle64_t
>(ranges.size());
1136 list_header.NumberOfMemoryRanges = memory_ranges_num;
1145 (ranges.size() *
sizeof(llvm::minidump::MemoryDescriptor_64));
1146 llvm::support::ulittle64_t memory_ranges_base_rva =
1147 static_cast<llvm::support::ulittle64_t
>(base_rva);
1148 list_header.BaseRVA = memory_ranges_base_rva;
1149 m_data.AppendData(&list_header,
sizeof(llvm::minidump::Memory64ListHeader));
1153 bool cleanup_required =
false;
1154 std::vector<MemoryDescriptor_64> descriptors;
1157 for (
const auto core_range : ranges) {
1159 MemoryDescriptor_64 memory_desc;
1160 memory_desc.StartOfMemoryRange =
1161 static_cast<llvm::support::ulittle64_t
>(core_range.range.start());
1162 memory_desc.DataSize =
1163 static_cast<llvm::support::ulittle64_t
>(core_range.range.size());
1164 descriptors.push_back(memory_desc);
1166 m_data.AppendData(&memory_desc,
sizeof(MemoryDescriptor_64));
1170 size_t region_index = 0;
1171 for (
const auto &core_range : ranges) {
1172 const addr_t addr = core_range.range.start();
1173 const addr_t size = core_range.range.size();
1176 "AddMemoryList_64 %zu/%zu reading memory for region "
1177 "(%" PRIx64
"bytes) "
1178 "[%" PRIx64
", %" PRIx64
")",
1179 region_index, ranges.size(), size, addr, addr + size);
1181 progress.
Increment(1,
"Adding Memory Range " + core_range.Dump());
1182 uint64_t bytes_read = 0;
1187 if (bytes_read == 0) {
1188 cleanup_required =
true;
1189 descriptors[region_index].DataSize = 0;
1191 if (bytes_read != size) {
1192 cleanup_required =
true;
1193 descriptors[region_index].DataSize = bytes_read;
1200 if (!cleanup_required) {
1207 size_t bytes_written =
sizeof(MemoryDescriptor_64) * descriptors.size();
1210 bytes_written !=
sizeof(MemoryDescriptor_64) * descriptors.size()) {
1212 "unable to write the memory descriptors (written %zd/%zd)",
1213 bytes_written,
sizeof(MemoryDescriptor_64) * descriptors.size());
1222 m_data.AppendData(data, size);
1234 addr_t remaining_bytes = starting_size;
1237 while (remaining_bytes > 0) {
1238 size_t bytes_written = remaining_bytes;
1244 "Wrote incorrect number of bytes to minidump file. (written %" PRIx64
1246 starting_size - remaining_bytes, starting_size);
1250 offset += bytes_written;
1251 remaining_bytes -= bytes_written;
1285 LLDB_LOGF(log,
"Failed to close minidump file: %s",
error.AsCString());
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_LOGF(log,...)
llvm::support::ulittle16_t read_register_u16(RegisterContext *reg_ctx, llvm::StringRef reg_name)
static uint64_t GetLargestRangeSize(const std::vector< CoreFileMemoryRange > &ranges)
uint64_t read_register_u64_raw(RegisterContext *reg_ctx, llvm::StringRef reg_name)
Status WriteString(const std::string &to_write, lldb_private::DataBufferHeap *buffer)
lldb_private::minidump::MinidumpContext_x86_64 GetThreadContext_x86_64(RegisterContext *reg_ctx)
llvm::Expected< uint64_t > getModuleFileSize(Target &target, const ModuleSP &mod)
uint16_t read_register_u16_raw(RegisterContext *reg_ctx, llvm::StringRef reg_name)
llvm::support::ulittle32_t read_register_u32(RegisterContext *reg_ctx, llvm::StringRef reg_name)
void read_register_u128(RegisterContext *reg_ctx, llvm::StringRef reg_name, uint8_t *dst)
std::unique_ptr< llvm::MemoryBuffer > getFileStreamHelper(const std::string &path)
llvm::support::ulittle64_t read_register_u64(RegisterContext *reg_ctx, llvm::StringRef reg_name)
minidump::RegisterContextMinidump_ARM64::Context GetThreadContext_ARM64(RegisterContext *reg_ctx)
uint32_t read_register_u32_raw(RegisterContext *reg_ctx, llvm::StringRef reg_name)
Structure holding data neccessary for minidump file creation.
lldb_private::Status WriteString(const std::string &to_write, lldb_private::DataBufferHeap *buffer)
const void * data() const
llvm::Triple::ArchType m_arch
lldb_private::minidump::MinidumpContext_x86_64 x86_64
lldb_private::minidump::RegisterContextMinidump_ARM64::Context arm64
bool prepareRegisterContext(RegisterContext *reg_ctx)
ArchThreadContexts(llvm::Triple::ArchType arch)
lldb_private::Status AddExceptions()
std::unordered_map< lldb::tid_t, llvm::minidump::LocationDescriptor > m_tid_to_reg_ctx
lldb_private::Status AddLLDBGeneratedStream()
lldb_private::Status AddThreadList()
lldb_private::Status AddHeaderAndCalculateDirectories()
uint64_t m_saved_data_size
static constexpr uint64_t MAX_WRITE_CHUNK_SIZE
static constexpr size_t DIRECTORY_SIZE
lldb_private::DataBufferHeap m_data
lldb_private::Status AddSystemInfo()
lldb_private::Status ReadWriteMemoryInChunks(lldb_private::DataBufferHeap &data_buffer, const lldb_private::CoreFileMemoryRange &range, uint64_t &bytes_read)
lldb_private::Status AddMemoryList()
std::unordered_map< lldb::addr_t, llvm::minidump::Thread > m_thread_by_range_end
lldb_private::Status AddData(const void *data, uint64_t size)
lldb_private::Status DumpFile()
lldb_private::SaveCoreOptions m_save_core_options
lldb_private::Status FlushBufferToDisk()
size_t m_expected_directories
lldb::offset_t m_thread_list_start
lldb_private::Status AddDirectory(llvm::minidump::StreamType type, uint64_t stream_size)
lldb_private::Status FixThreadStacks()
void DeleteFile() noexcept
lldb::ProcessSP m_process_sp
static constexpr size_t HEADER_SIZE
lldb::offset_t GetCurrentDataEndOffset() const
lldb_private::Status AddMemoryList_32(std::vector< lldb_private::CoreFileMemoryRange > &ranges, lldb_private::Progress &progress)
std::vector< llvm::minidump::Directory > m_directories
lldb_private::Status DumpDirectories() const
lldb_private::Status AddLinuxFileStreams()
lldb_private::Status AddModuleList()
lldb_private::Status AddMiscInfo()
lldb_private::Status DumpHeader() const
lldb_private::Status AddMemoryList_64(std::vector< lldb_private::CoreFileMemoryRange > &ranges, lldb_private::Progress &progress)
A section + offset based address class.
lldb::SectionSP GetSection() const
Get const accessor for the section.
An architecture specification class.
llvm::Triple & GetTriple()
Architecture triple accessor.
llvm::Triple::ArchType GetMachine() const
Returns a machine family for the current architecture.
A subclass of DataBuffer that stores a data buffer on the heap.
lldb::offset_t GetByteSize() const override
Get the number of bytes in the data buffer.
void AppendData(const void *src, uint64_t src_len)
A collection class for Module objects.
lldb::ModuleSP GetModuleAtIndex(size_t idx) const
Get the module shared pointer for the module at index idx.
size_t GetSize() const
Gets the size of the module list.
@ eTypeJIT
JIT code that has symbols, sections and possibly debug info.
bool ProcessIDIsValid() const
lldb::pid_t GetProcessID() const
std::function< IterationAction(lldb_private::Status &error, lldb::addr_t bytes_addr, const void *bytes, lldb::offset_t bytes_size)> ReadMemoryChunkCallback
A Progress indicator helper class.
void Increment(uint64_t amount=1, std::optional< std::string > updated_detail={})
Increment the progress and send a notification to the installed callback.
uint64_t GetSP(uint64_t fail_value=LLDB_INVALID_ADDRESS)
const RegisterInfo * GetRegisterInfoByName(llvm::StringRef reg_name, uint32_t start_idx=0)
virtual bool ReadRegister(const RegisterInfo *reg_info, RegisterValue ®_value)=0
uint16_t GetAsUInt16(uint16_t fail_value=UINT16_MAX, bool *success_ptr=nullptr) const
uint32_t GetAsMemoryData(const RegisterInfo ®_info, void *dst, uint32_t dst_len, lldb::ByteOrder dst_byte_order, Status &error) const
uint64_t GetAsUInt64(uint64_t fail_value=UINT64_MAX, bool *success_ptr=nullptr) const
uint32_t GetAsUInt32(uint32_t fail_value=UINT32_MAX, bool *success_ptr=nullptr) const
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
static Status FromErrorString(const char *str)
bool Fail() const
Test for error condition.
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
bool ResolveLoadAddress(lldb::addr_t load_addr, Address &so_addr, uint32_t stop_id=SectionLoadHistory::eStopIDNow, bool allow_section_end=false)
const ModuleList & GetImages() const
Get accessor for the images for this process.
const ArchSpec & GetArchitecture() const
uint8_t * GetBytes()
Get a pointer to the data.
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.
IterationAction
Useful for callbacks whose return type indicates whether to continue iteration or short-circuit.
std::shared_ptr< lldb_private::Thread > ThreadSP
@ eStateSuspended
Process or thread is in a suspended state as far as the debugger is concerned while other processes o...
std::shared_ptr< lldb_private::StopInfo > StopInfoSP
std::shared_ptr< lldb_private::Section > SectionSP
StopReason
Thread stop reasons.
std::shared_ptr< lldb_private::RegisterContext > RegisterContextSP
std::shared_ptr< lldb_private::Module > ModuleSP
BaseType GetRangeEnd() const
Every register is described in detail including its name, alternate name (optional),...
llvm::support::ulittle16_t fs
llvm::support::ulittle64_t rdx
llvm::support::ulittle16_t cs
llvm::support::ulittle64_t rsp
llvm::support::ulittle16_t ss
llvm::support::ulittle64_t rbx
llvm::support::ulittle16_t ds
llvm::support::ulittle64_t gs_base
llvm::support::ulittle64_t r8
llvm::support::ulittle16_t gs
llvm::support::ulittle64_t rcx
llvm::support::ulittle64_t r10
llvm::support::ulittle64_t r15
llvm::support::ulittle64_t rbp
llvm::support::ulittle64_t r13
llvm::support::ulittle64_t r12
llvm::support::ulittle64_t rsi
llvm::support::ulittle64_t p1_home
llvm::support::ulittle64_t rax
llvm::support::ulittle32_t context_flags
llvm::support::ulittle64_t rdi
llvm::support::ulittle64_t r14
llvm::support::ulittle32_t eflags
llvm::support::ulittle64_t r9
llvm::support::ulittle64_t fs_base
llvm::support::ulittle64_t rip
llvm::support::ulittle64_t r11
llvm::support::ulittle32_t process_id
llvm::support::ulittle32_t flags1
llvm::support::ulittle32_t size