blob: 6ecbb5c8cfadaf68a002f0dea6219f0e870ef16e [file] [log] [blame]
Bob Haarman653baa22016-10-21 19:43:19 +00001//===- GSI.cpp - Common Functions for GlobalsStream and PublicsStream ----===//
2//
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//
8//===----------------------------------------------------------------------===//
9
10#include "GSI.h"
11
12#include "llvm/DebugInfo/MSF/StreamArray.h"
13#include "llvm/DebugInfo/MSF/StreamReader.h"
14#include "llvm/DebugInfo/PDB/Raw/RawError.h"
15#include "llvm/DebugInfo/PDB/Raw/RawTypes.h"
16
17#include "llvm/Support/Error.h"
18
19namespace llvm {
20namespace pdb {
21
22static Error checkHashHdrVersion(const GSIHashHeader *HashHdr) {
23 if (HashHdr->VerHdr != GSIHashHeader::HdrVersion)
24 return make_error<RawError>(
25 raw_error_code::feature_unsupported,
26 "Encountered unsupported globals stream version.");
27
28 return Error::success();
29}
30
31Error readGSIHashBuckets(
32 msf::FixedStreamArray<support::ulittle32_t> &HashBuckets,
33 const GSIHashHeader *HashHdr, msf::StreamReader &Reader) {
34 if (auto EC = checkHashHdrVersion(HashHdr))
35 return EC;
36
37 // Before the actual hash buckets, there is a bitmap of length determined by
38 // IPHR_HASH.
39 ArrayRef<uint8_t> Bitmap;
40 size_t BitmapSizeInBits = alignTo(IPHR_HASH + 1, 32);
41 uint32_t NumBitmapEntries = BitmapSizeInBits / 8;
42 if (auto EC = Reader.readBytes(Bitmap, NumBitmapEntries))
43 return joinErrors(std::move(EC),
44 make_error<RawError>(raw_error_code::corrupt_file,
45 "Could not read a bitmap."));
46 uint32_t NumBuckets = 0;
47 for (uint8_t B : Bitmap)
48 NumBuckets += countPopulation(B);
49
50 // Hash buckets follow.
51 if (auto EC = Reader.readArray(HashBuckets, NumBuckets))
52 return joinErrors(std::move(EC),
53 make_error<RawError>(raw_error_code::corrupt_file,
54 "Hash buckets corrupted."));
55
56 return Error::success();
57}
58
59Error readGSIHashHeader(const GSIHashHeader *&HashHdr,
60 msf::StreamReader &Reader) {
61 if (Reader.readObject(HashHdr))
62 return make_error<RawError>(raw_error_code::corrupt_file,
63 "Stream does not contain a GSIHashHeader.");
64
65 if (HashHdr->VerSignature != GSIHashHeader::HdrSignature)
66 return make_error<RawError>(
67 raw_error_code::feature_unsupported,
68 "GSIHashHeader signature (0xffffffff) not found.");
69
70 return Error::success();
71}
72
73Error readGSIHashRecords(msf::FixedStreamArray<PSHashRecord> &HashRecords,
74 const GSIHashHeader *HashHdr,
75 msf::StreamReader &Reader) {
76 if (auto EC = checkHashHdrVersion(HashHdr))
77 return EC;
78
79 // HashHdr->HrSize specifies the number of bytes of PSHashRecords we have.
80 // Verify that we can read them all.
81 if (HashHdr->HrSize % sizeof(PSHashRecord))
82 return make_error<RawError>(raw_error_code::corrupt_file,
83 "Invalid HR array size.");
84 uint32_t NumHashRecords = HashHdr->HrSize / sizeof(PSHashRecord);
85 if (auto EC = Reader.readArray(HashRecords, NumHashRecords))
86 return joinErrors(std::move(EC),
87 make_error<RawError>(raw_error_code::corrupt_file,
88 "Error reading hash records."));
89
90 return Error::success();
91}
92}
93}