blob: 9a979335e99e17eb6a1a76515704be61cf31a072 [file] [log] [blame]
Dimitar Vlahovski36e21a32016-10-05 18:11:45 +00001//===-- MinidumpParser.cpp ---------------------------------------*- C++ -*-===//
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
Dimitar Vlahovski36e21a32016-10-05 18:11:45 +00008//===----------------------------------------------------------------------===//
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +00009
10// Project includes
11#include "MinidumpParser.h"
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +000012#include "NtStructures.h"
13#include "RegisterContextMinidump_x86_32.h"
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000014
15// Other libraries and framework includes
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +000016#include "lldb/Target/MemoryRegionInfo.h"
Leonard Mosescu2ae3ec32018-07-12 17:27:18 +000017#include "lldb/Utility/LLDBAssert.h"
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +000018
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000019// C includes
20// C++ includes
Leonard Mosescu2ae3ec32018-07-12 17:27:18 +000021#include <algorithm>
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +000022#include <map>
Leonard Mosescu2ae3ec32018-07-12 17:27:18 +000023#include <vector>
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000024
25using namespace lldb_private;
26using namespace minidump;
27
28llvm::Optional<MinidumpParser>
Kate Stoneb9c1b512016-09-06 20:57:50 +000029MinidumpParser::Create(const lldb::DataBufferSP &data_buf_sp) {
30 if (data_buf_sp->GetByteSize() < sizeof(MinidumpHeader)) {
31 return llvm::None;
32 }
Leonard Mosescu2ae3ec32018-07-12 17:27:18 +000033 return MinidumpParser(data_buf_sp);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000034}
35
Leonard Mosescu2ae3ec32018-07-12 17:27:18 +000036MinidumpParser::MinidumpParser(const lldb::DataBufferSP &data_buf_sp)
37 : m_data_sp(data_buf_sp) {}
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000038
Dimitar Vlahovski4c319072016-09-27 19:05:55 +000039llvm::ArrayRef<uint8_t> MinidumpParser::GetData() {
Dimitar Vlahovski36e21a32016-10-05 18:11:45 +000040 return llvm::ArrayRef<uint8_t>(m_data_sp->GetBytes(),
41 m_data_sp->GetByteSize());
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000042}
43
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000044llvm::ArrayRef<uint8_t>
Kate Stoneb9c1b512016-09-06 20:57:50 +000045MinidumpParser::GetStream(MinidumpStreamType stream_type) {
46 auto iter = m_directory_map.find(static_cast<uint32_t>(stream_type));
47 if (iter == m_directory_map.end())
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000048 return {};
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000049
Kate Stoneb9c1b512016-09-06 20:57:50 +000050 // check if there is enough data
Dimitar Vlahovski36e21a32016-10-05 18:11:45 +000051 if (iter->second.rva + iter->second.data_size > m_data_sp->GetByteSize())
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000052 return {};
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000053
Dimitar Vlahovski36e21a32016-10-05 18:11:45 +000054 return llvm::ArrayRef<uint8_t>(m_data_sp->GetBytes() + iter->second.rva,
55 iter->second.data_size);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000056}
57
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000058llvm::Optional<std::string> MinidumpParser::GetMinidumpString(uint32_t rva) {
Dimitar Vlahovski36e21a32016-10-05 18:11:45 +000059 auto arr_ref = m_data_sp->GetData();
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000060 if (rva > arr_ref.size())
61 return llvm::None;
62 arr_ref = arr_ref.drop_front(rva);
63 return parseMinidumpString(arr_ref);
64}
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000065
Leonard Mosescu9fecd372018-05-02 20:06:17 +000066UUID MinidumpParser::GetModuleUUID(const MinidumpModule *module) {
67 auto cv_record =
68 GetData().slice(module->CV_record.rva, module->CV_record.data_size);
69
70 // Read the CV record signature
71 const llvm::support::ulittle32_t *signature = nullptr;
72 Status error = consumeObject(cv_record, signature);
73 if (error.Fail())
74 return UUID();
75
76 const CvSignature cv_signature =
77 static_cast<CvSignature>(static_cast<const uint32_t>(*signature));
78
79 if (cv_signature == CvSignature::Pdb70) {
80 // PDB70 record
81 const CvRecordPdb70 *pdb70_uuid = nullptr;
82 Status error = consumeObject(cv_record, pdb70_uuid);
83 if (!error.Fail())
Pavel Labath2f93fd12018-06-26 15:12:20 +000084 return UUID::fromData(pdb70_uuid, sizeof(*pdb70_uuid));
85 } else if (cv_signature == CvSignature::ElfBuildId)
86 return UUID::fromData(cv_record);
Leonard Mosescu9fecd372018-05-02 20:06:17 +000087
88 return UUID();
89}
90
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000091llvm::ArrayRef<MinidumpThread> MinidumpParser::GetThreads() {
92 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::ThreadList);
93
94 if (data.size() == 0)
Kate Stoneb9c1b512016-09-06 20:57:50 +000095 return llvm::None;
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000096
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +000097 return MinidumpThread::ParseThreadList(data);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +000098}
99
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000100llvm::ArrayRef<uint8_t>
101MinidumpParser::GetThreadContext(const MinidumpThread &td) {
102 if (td.thread_context.rva + td.thread_context.data_size > GetData().size())
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +0000103 return {};
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000104
105 return GetData().slice(td.thread_context.rva, td.thread_context.data_size);
106}
107
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +0000108llvm::ArrayRef<uint8_t>
109MinidumpParser::GetThreadContextWow64(const MinidumpThread &td) {
Adrian Prantl05097242018-04-30 16:49:04 +0000110 // On Windows, a 32-bit process can run on a 64-bit machine under WOW64. If
111 // the minidump was captured with a 64-bit debugger, then the CONTEXT we just
112 // grabbed from the mini_dump_thread is the one for the 64-bit "native"
113 // process rather than the 32-bit "guest" process we care about. In this
114 // case, we can get the 32-bit CONTEXT from the TEB (Thread Environment
115 // Block) of the 64-bit process.
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +0000116 auto teb_mem = GetMemory(td.teb, sizeof(TEB64));
117 if (teb_mem.empty())
118 return {};
119
120 const TEB64 *wow64teb;
Zachary Turner97206d52017-05-12 04:51:55 +0000121 Status error = consumeObject(teb_mem, wow64teb);
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +0000122 if (error.Fail())
123 return {};
124
Adrian Prantl05097242018-04-30 16:49:04 +0000125 // Slot 1 of the thread-local storage in the 64-bit TEB points to a structure
126 // that includes the 32-bit CONTEXT (after a ULONG). See:
127 // https://msdn.microsoft.com/en-us/library/ms681670.aspx
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +0000128 auto context =
129 GetMemory(wow64teb->tls_slots[1] + 4, sizeof(MinidumpContext_x86_32));
130 if (context.size() < sizeof(MinidumpContext_x86_32))
131 return {};
132
133 return context;
134 // NOTE: We don't currently use the TEB for anything else. If we
135 // need it in the future, the 32-bit TEB is located according to the address
136 // stored in the first slot of the 64-bit TEB (wow64teb.Reserved1[0]).
137}
138
Kate Stoneb9c1b512016-09-06 20:57:50 +0000139const MinidumpSystemInfo *MinidumpParser::GetSystemInfo() {
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000140 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::SystemInfo);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000141
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000142 if (data.size() == 0)
Kate Stoneb9c1b512016-09-06 20:57:50 +0000143 return nullptr;
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000144
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000145 return MinidumpSystemInfo::Parse(data);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000146}
147
Pavel Labath659cee52018-08-03 08:47:22 +0000148ArchSpec MinidumpParser::GetArchitecture() {
149 ArchSpec arch_spec;
150 const MinidumpSystemInfo *system_info = GetSystemInfo();
151
152 if (!system_info)
153 return arch_spec;
154
155 // TODO what to do about big endiand flavors of arm ?
156 // TODO set the arm subarch stuff if the minidump has info about it
157
158 llvm::Triple triple;
159 triple.setVendor(llvm::Triple::VendorType::UnknownVendor);
160
161 const MinidumpCPUArchitecture arch =
162 static_cast<const MinidumpCPUArchitecture>(
163 static_cast<const uint32_t>(system_info->processor_arch));
164
165 switch (arch) {
166 case MinidumpCPUArchitecture::X86:
167 triple.setArch(llvm::Triple::ArchType::x86);
168 break;
169 case MinidumpCPUArchitecture::AMD64:
170 triple.setArch(llvm::Triple::ArchType::x86_64);
171 break;
172 case MinidumpCPUArchitecture::ARM:
173 triple.setArch(llvm::Triple::ArchType::arm);
174 break;
175 case MinidumpCPUArchitecture::ARM64:
176 triple.setArch(llvm::Triple::ArchType::aarch64);
177 break;
178 default:
179 triple.setArch(llvm::Triple::ArchType::UnknownArch);
180 break;
181 }
182
183 const MinidumpOSPlatform os = static_cast<const MinidumpOSPlatform>(
184 static_cast<const uint32_t>(system_info->platform_id));
185
186 // TODO add all of the OSes that Minidump/breakpad distinguishes?
187 switch (os) {
188 case MinidumpOSPlatform::Win32S:
189 case MinidumpOSPlatform::Win32Windows:
190 case MinidumpOSPlatform::Win32NT:
191 case MinidumpOSPlatform::Win32CE:
192 triple.setOS(llvm::Triple::OSType::Win32);
193 break;
194 case MinidumpOSPlatform::Linux:
195 triple.setOS(llvm::Triple::OSType::Linux);
196 break;
197 case MinidumpOSPlatform::MacOSX:
198 triple.setOS(llvm::Triple::OSType::MacOSX);
199 break;
200 case MinidumpOSPlatform::Android:
201 triple.setOS(llvm::Triple::OSType::Linux);
202 triple.setEnvironment(llvm::Triple::EnvironmentType::Android);
203 break;
204 default:
205 triple.setOS(llvm::Triple::OSType::UnknownOS);
206 break;
207 }
208
209 arch_spec.SetTriple(triple);
210
211 return arch_spec;
212}
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000213
Kate Stoneb9c1b512016-09-06 20:57:50 +0000214const MinidumpMiscInfo *MinidumpParser::GetMiscInfo() {
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000215 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::MiscInfo);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000216
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000217 if (data.size() == 0)
Kate Stoneb9c1b512016-09-06 20:57:50 +0000218 return nullptr;
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000219
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000220 return MinidumpMiscInfo::Parse(data);
221}
222
223llvm::Optional<LinuxProcStatus> MinidumpParser::GetLinuxProcStatus() {
224 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::LinuxProcStatus);
225
226 if (data.size() == 0)
227 return llvm::None;
228
229 return LinuxProcStatus::Parse(data);
230}
231
232llvm::Optional<lldb::pid_t> MinidumpParser::GetPid() {
233 const MinidumpMiscInfo *misc_info = GetMiscInfo();
234 if (misc_info != nullptr) {
235 return misc_info->GetPid();
236 }
237
238 llvm::Optional<LinuxProcStatus> proc_status = GetLinuxProcStatus();
239 if (proc_status.hasValue()) {
240 return proc_status->GetPid();
241 }
242
243 return llvm::None;
244}
245
246llvm::ArrayRef<MinidumpModule> MinidumpParser::GetModuleList() {
247 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::ModuleList);
248
249 if (data.size() == 0)
250 return {};
251
252 return MinidumpModule::ParseModuleList(data);
253}
254
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000255std::vector<const MinidumpModule *> MinidumpParser::GetFilteredModuleList() {
256 llvm::ArrayRef<MinidumpModule> modules = GetModuleList();
Dimitar Vlahovski7b18dd42016-10-31 15:35:18 +0000257 // map module_name -> pair(load_address, pointer to module struct in memory)
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000258 llvm::StringMap<std::pair<uint64_t, const MinidumpModule *>> lowest_addr;
259
260 std::vector<const MinidumpModule *> filtered_modules;
261
262 llvm::Optional<std::string> name;
263 std::string module_name;
264
265 for (const auto &module : modules) {
266 name = GetMinidumpString(module.module_name_rva);
267
268 if (!name)
269 continue;
270
271 module_name = name.getValue();
272
273 auto iter = lowest_addr.end();
274 bool exists;
275 std::tie(iter, exists) = lowest_addr.try_emplace(
276 module_name, std::make_pair(module.base_of_image, &module));
277
278 if (exists && module.base_of_image < iter->second.first)
279 iter->second = std::make_pair(module.base_of_image, &module);
280 }
281
282 filtered_modules.reserve(lowest_addr.size());
283 for (const auto &module : lowest_addr) {
284 filtered_modules.push_back(module.second.second);
285 }
286
287 return filtered_modules;
288}
289
Dimitar Vlahovski1d2859e2016-09-13 15:54:38 +0000290const MinidumpExceptionStream *MinidumpParser::GetExceptionStream() {
291 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::Exception);
292
293 if (data.size() == 0)
294 return nullptr;
295
296 return MinidumpExceptionStream::Parse(data);
Dimitar Vlahovski2e50d8e2016-09-01 11:29:53 +0000297}
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000298
299llvm::Optional<minidump::Range>
300MinidumpParser::FindMemoryRange(lldb::addr_t addr) {
301 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::MemoryList);
302 llvm::ArrayRef<uint8_t> data64 = GetStream(MinidumpStreamType::Memory64List);
303
304 if (data.empty() && data64.empty())
305 return llvm::None;
306
307 if (!data.empty()) {
308 llvm::ArrayRef<MinidumpMemoryDescriptor> memory_list =
309 MinidumpMemoryDescriptor::ParseMemoryList(data);
310
311 if (memory_list.empty())
312 return llvm::None;
313
314 for (const auto &memory_desc : memory_list) {
315 const MinidumpLocationDescriptor &loc_desc = memory_desc.memory;
316 const lldb::addr_t range_start = memory_desc.start_of_memory_range;
317 const size_t range_size = loc_desc.data_size;
318
319 if (loc_desc.rva + loc_desc.data_size > GetData().size())
320 return llvm::None;
321
322 if (range_start <= addr && addr < range_start + range_size) {
323 return minidump::Range(range_start,
324 GetData().slice(loc_desc.rva, range_size));
325 }
326 }
327 }
328
Adrian Prantl05097242018-04-30 16:49:04 +0000329 // Some Minidumps have a Memory64ListStream that captures all the heap memory
330 // (full-memory Minidumps). We can't exactly use the same loop as above,
331 // because the Minidump uses slightly different data structures to describe
332 // those
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000333
334 if (!data64.empty()) {
335 llvm::ArrayRef<MinidumpMemoryDescriptor64> memory64_list;
336 uint64_t base_rva;
337 std::tie(memory64_list, base_rva) =
338 MinidumpMemoryDescriptor64::ParseMemory64List(data64);
339
340 if (memory64_list.empty())
341 return llvm::None;
342
343 for (const auto &memory_desc64 : memory64_list) {
344 const lldb::addr_t range_start = memory_desc64.start_of_memory_range;
345 const size_t range_size = memory_desc64.data_size;
346
347 if (base_rva + range_size > GetData().size())
348 return llvm::None;
349
350 if (range_start <= addr && addr < range_start + range_size) {
351 return minidump::Range(range_start,
352 GetData().slice(base_rva, range_size));
353 }
354 base_rva += range_size;
355 }
356 }
357
358 return llvm::None;
359}
360
361llvm::ArrayRef<uint8_t> MinidumpParser::GetMemory(lldb::addr_t addr,
362 size_t size) {
363 // I don't have a sense of how frequently this is called or how many memory
364 // ranges a Minidump typically has, so I'm not sure if searching for the
365 // appropriate range linearly each time is stupid. Perhaps we should build
366 // an index for faster lookups.
367 llvm::Optional<minidump::Range> range = FindMemoryRange(addr);
368 if (!range)
369 return {};
370
371 // There's at least some overlap between the beginning of the desired range
Adrian Prantl05097242018-04-30 16:49:04 +0000372 // (addr) and the current range. Figure out where the overlap begins and how
373 // much overlap there is.
Dimitar Vlahovskib52206d2016-10-19 14:14:18 +0000374
375 const size_t offset = addr - range->start;
376
377 if (addr < range->start || offset >= range->range_ref.size())
378 return {};
379
380 const size_t overlap = std::min(size, range->range_ref.size() - offset);
381 return range->range_ref.slice(offset, overlap);
382}
383
384llvm::Optional<MemoryRegionInfo>
385MinidumpParser::GetMemoryRegionInfo(lldb::addr_t load_addr) {
386 MemoryRegionInfo info;
387 llvm::ArrayRef<uint8_t> data = GetStream(MinidumpStreamType::MemoryInfoList);
388 if (data.empty())
389 return llvm::None;
390
391 std::vector<const MinidumpMemoryInfo *> mem_info_list =
392 MinidumpMemoryInfo::ParseMemoryInfoList(data);
393 if (mem_info_list.empty())
394 return llvm::None;
395
396 const auto yes = MemoryRegionInfo::eYes;
397 const auto no = MemoryRegionInfo::eNo;
398
399 const MinidumpMemoryInfo *next_entry = nullptr;
400 for (const auto &entry : mem_info_list) {
401 const auto head = entry->base_address;
402 const auto tail = head + entry->region_size;
403
404 if (head <= load_addr && load_addr < tail) {
405 info.GetRange().SetRangeBase(
406 (entry->state != uint32_t(MinidumpMemoryInfoState::MemFree))
407 ? head
408 : load_addr);
409 info.GetRange().SetRangeEnd(tail);
410
411 const uint32_t PageNoAccess =
412 static_cast<uint32_t>(MinidumpMemoryProtectionContants::PageNoAccess);
413 info.SetReadable((entry->protect & PageNoAccess) == 0 ? yes : no);
414
415 const uint32_t PageWritable =
416 static_cast<uint32_t>(MinidumpMemoryProtectionContants::PageWritable);
417 info.SetWritable((entry->protect & PageWritable) != 0 ? yes : no);
418
419 const uint32_t PageExecutable = static_cast<uint32_t>(
420 MinidumpMemoryProtectionContants::PageExecutable);
421 info.SetExecutable((entry->protect & PageExecutable) != 0 ? yes : no);
422
423 const uint32_t MemFree =
424 static_cast<uint32_t>(MinidumpMemoryInfoState::MemFree);
425 info.SetMapped((entry->state != MemFree) ? yes : no);
426
427 return info;
428 } else if (head > load_addr &&
429 (next_entry == nullptr || head < next_entry->base_address)) {
430 // In case there is no region containing load_addr keep track of the
431 // nearest region after load_addr so we can return the distance to it.
432 next_entry = entry;
433 }
434 }
435
436 // No containing region found. Create an unmapped region that extends to the
437 // next region or LLDB_INVALID_ADDRESS
438 info.GetRange().SetRangeBase(load_addr);
439 info.GetRange().SetRangeEnd((next_entry != nullptr) ? next_entry->base_address
440 : LLDB_INVALID_ADDRESS);
441 info.SetReadable(no);
442 info.SetWritable(no);
443 info.SetExecutable(no);
444 info.SetMapped(no);
445
446 // Note that the memory info list doesn't seem to contain ranges in kernel
447 // space, so if you're walking a stack that has kernel frames, the stack may
448 // appear truncated.
449 return info;
450}
Leonard Mosescu2ae3ec32018-07-12 17:27:18 +0000451
452Status MinidumpParser::Initialize() {
453 Status error;
454
455 lldbassert(m_directory_map.empty());
456
457 llvm::ArrayRef<uint8_t> header_data(m_data_sp->GetBytes(),
458 sizeof(MinidumpHeader));
459 const MinidumpHeader *header = MinidumpHeader::Parse(header_data);
460 if (header == nullptr) {
461 error.SetErrorString("invalid minidump: can't parse the header");
462 return error;
463 }
464
465 // A minidump without at least one stream is clearly ill-formed
466 if (header->streams_count == 0) {
467 error.SetErrorString("invalid minidump: no streams present");
468 return error;
469 }
470
471 struct FileRange {
472 uint32_t offset = 0;
473 uint32_t size = 0;
474
475 FileRange(uint32_t offset, uint32_t size) : offset(offset), size(size) {}
476 uint32_t end() const { return offset + size; }
477 };
478
479 const uint32_t file_size = m_data_sp->GetByteSize();
480
481 // Build a global minidump file map, checking for:
482 // - overlapping streams/data structures
483 // - truncation (streams pointing past the end of file)
484 std::vector<FileRange> minidump_map;
485
486 // Add the minidump header to the file map
487 if (sizeof(MinidumpHeader) > file_size) {
488 error.SetErrorString("invalid minidump: truncated header");
489 return error;
490 }
491 minidump_map.emplace_back( 0, sizeof(MinidumpHeader) );
492
493 // Add the directory entries to the file map
494 FileRange directory_range(header->stream_directory_rva,
495 header->streams_count *
496 sizeof(MinidumpDirectory));
497 if (directory_range.end() > file_size) {
498 error.SetErrorString("invalid minidump: truncated streams directory");
499 return error;
500 }
501 minidump_map.push_back(directory_range);
502
503 // Parse stream directory entries
504 llvm::ArrayRef<uint8_t> directory_data(
505 m_data_sp->GetBytes() + directory_range.offset, directory_range.size);
506 for (uint32_t i = 0; i < header->streams_count; ++i) {
507 const MinidumpDirectory *directory_entry = nullptr;
508 error = consumeObject(directory_data, directory_entry);
509 if (error.Fail())
510 return error;
511 if (directory_entry->stream_type == 0) {
512 // Ignore dummy streams (technically ill-formed, but a number of
513 // existing minidumps seem to contain such streams)
514 if (directory_entry->location.data_size == 0)
515 continue;
516 error.SetErrorString("invalid minidump: bad stream type");
517 return error;
518 }
519 // Update the streams map, checking for duplicate stream types
520 if (!m_directory_map
521 .insert({directory_entry->stream_type, directory_entry->location})
522 .second) {
523 error.SetErrorString("invalid minidump: duplicate stream type");
524 return error;
525 }
526 // Ignore the zero-length streams for layout checks
527 if (directory_entry->location.data_size != 0) {
528 minidump_map.emplace_back(directory_entry->location.rva,
529 directory_entry->location.data_size);
530 }
531 }
532
533 // Sort the file map ranges by start offset
534 std::sort(minidump_map.begin(), minidump_map.end(),
535 [](const FileRange &a, const FileRange &b) {
536 return a.offset < b.offset;
537 });
538
539 // Check for overlapping streams/data structures
540 for (size_t i = 1; i < minidump_map.size(); ++i) {
541 const auto &prev_range = minidump_map[i - 1];
542 if (prev_range.end() > minidump_map[i].offset) {
543 error.SetErrorString("invalid minidump: overlapping streams");
544 return error;
545 }
546 }
547
548 // Check for streams past the end of file
549 const auto &last_range = minidump_map.back();
550 if (last_range.end() > file_size) {
551 error.SetErrorString("invalid minidump: truncated stream");
552 return error;
553 }
554
555 return error;
556}