1*9880d681SAndroid Build Coastguard Worker //===-- llvm-pdbdump-fuzzer.cpp - Fuzz the llvm-pdbdump tool --------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker ///
10*9880d681SAndroid Build Coastguard Worker /// \file
11*9880d681SAndroid Build Coastguard Worker /// \brief This file implements a function that runs llvm-pdbdump
12*9880d681SAndroid Build Coastguard Worker /// on a single input. This function is then linked into the Fuzzer library.
13*9880d681SAndroid Build Coastguard Worker ///
14*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
15*9880d681SAndroid Build Coastguard Worker #include "llvm/ADT/STLExtras.h"
16*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/CodeView/ByteStream.h"
17*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/CodeView/SymbolDumper.h"
18*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/CodeView/TypeDumper.h"
19*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/PDB/Raw/DbiStream.h"
20*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/PDB/Raw/IPDBStreamData.h"
21*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/PDB/Raw/MappedBlockStream.h"
22*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/PDB/Raw/ModStream.h"
23*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/PDB/Raw/PDBFile.h"
24*9880d681SAndroid Build Coastguard Worker #include "llvm/DebugInfo/PDB/Raw/RawSession.h"
25*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/MemoryBuffer.h"
26*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ScopedPrinter.h"
27*9880d681SAndroid Build Coastguard Worker
28*9880d681SAndroid Build Coastguard Worker using namespace llvm;
29*9880d681SAndroid Build Coastguard Worker
30*9880d681SAndroid Build Coastguard Worker namespace {
31*9880d681SAndroid Build Coastguard Worker // We need a class which behaves like an immutable ByteStream, but whose data
32*9880d681SAndroid Build Coastguard Worker // is backed by an llvm::MemoryBuffer. It also needs to own the underlying
33*9880d681SAndroid Build Coastguard Worker // MemoryBuffer, so this simple adapter is a good way to achieve that.
34*9880d681SAndroid Build Coastguard Worker class InputByteStream : public codeview::ByteStream<false> {
35*9880d681SAndroid Build Coastguard Worker public:
InputByteStream(std::unique_ptr<MemoryBuffer> Buffer)36*9880d681SAndroid Build Coastguard Worker explicit InputByteStream(std::unique_ptr<MemoryBuffer> Buffer)
37*9880d681SAndroid Build Coastguard Worker : ByteStream(ArrayRef<uint8_t>(Buffer->getBuffer().bytes_begin(),
38*9880d681SAndroid Build Coastguard Worker Buffer->getBuffer().bytes_end())),
39*9880d681SAndroid Build Coastguard Worker MemBuffer(std::move(Buffer)) {}
40*9880d681SAndroid Build Coastguard Worker
41*9880d681SAndroid Build Coastguard Worker std::unique_ptr<MemoryBuffer> MemBuffer;
42*9880d681SAndroid Build Coastguard Worker };
43*9880d681SAndroid Build Coastguard Worker }
44*9880d681SAndroid Build Coastguard Worker
LLVMFuzzerTestOneInput(uint8_t * data,size_t size)45*9880d681SAndroid Build Coastguard Worker extern "C" int LLVMFuzzerTestOneInput(uint8_t *data, size_t size) {
46*9880d681SAndroid Build Coastguard Worker std::unique_ptr<MemoryBuffer> Buff = MemoryBuffer::getMemBuffer(
47*9880d681SAndroid Build Coastguard Worker StringRef((const char *)data, size), "", false);
48*9880d681SAndroid Build Coastguard Worker
49*9880d681SAndroid Build Coastguard Worker ScopedPrinter P(nulls());
50*9880d681SAndroid Build Coastguard Worker codeview::CVTypeDumper TD(&P, false);
51*9880d681SAndroid Build Coastguard Worker
52*9880d681SAndroid Build Coastguard Worker auto InputStream = llvm::make_unique<InputByteStream>(std::move(Buff));
53*9880d681SAndroid Build Coastguard Worker std::unique_ptr<pdb::PDBFile> File(new pdb::PDBFile(std::move(InputStream)));
54*9880d681SAndroid Build Coastguard Worker if (auto E = File->parseFileHeaders()) {
55*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
56*9880d681SAndroid Build Coastguard Worker return 0;
57*9880d681SAndroid Build Coastguard Worker }
58*9880d681SAndroid Build Coastguard Worker if (auto E = File->parseStreamData()) {
59*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
60*9880d681SAndroid Build Coastguard Worker return 0;
61*9880d681SAndroid Build Coastguard Worker }
62*9880d681SAndroid Build Coastguard Worker
63*9880d681SAndroid Build Coastguard Worker auto DbiS = File->getPDBDbiStream();
64*9880d681SAndroid Build Coastguard Worker if (auto E = DbiS.takeError()) {
65*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
66*9880d681SAndroid Build Coastguard Worker return 0;
67*9880d681SAndroid Build Coastguard Worker }
68*9880d681SAndroid Build Coastguard Worker auto TpiS = File->getPDBTpiStream();
69*9880d681SAndroid Build Coastguard Worker if (auto E = TpiS.takeError()) {
70*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
71*9880d681SAndroid Build Coastguard Worker return 0;
72*9880d681SAndroid Build Coastguard Worker }
73*9880d681SAndroid Build Coastguard Worker auto IpiS = File->getPDBIpiStream();
74*9880d681SAndroid Build Coastguard Worker if (auto E = IpiS.takeError()) {
75*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
76*9880d681SAndroid Build Coastguard Worker return 0;
77*9880d681SAndroid Build Coastguard Worker }
78*9880d681SAndroid Build Coastguard Worker auto InfoS = File->getPDBInfoStream();
79*9880d681SAndroid Build Coastguard Worker if (auto E = InfoS.takeError()) {
80*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
81*9880d681SAndroid Build Coastguard Worker return 0;
82*9880d681SAndroid Build Coastguard Worker }
83*9880d681SAndroid Build Coastguard Worker pdb::DbiStream &DS = DbiS.get();
84*9880d681SAndroid Build Coastguard Worker
85*9880d681SAndroid Build Coastguard Worker for (auto &Modi : DS.modules()) {
86*9880d681SAndroid Build Coastguard Worker auto ModStreamData = pdb::MappedBlockStream::createIndexedStream(
87*9880d681SAndroid Build Coastguard Worker Modi.Info.getModuleStreamIndex(), *File);
88*9880d681SAndroid Build Coastguard Worker if (!ModStreamData) {
89*9880d681SAndroid Build Coastguard Worker consumeError(ModStreamData.takeError());
90*9880d681SAndroid Build Coastguard Worker return 0;
91*9880d681SAndroid Build Coastguard Worker }
92*9880d681SAndroid Build Coastguard Worker pdb::ModStream ModS(Modi.Info, std::move(*ModStreamData));
93*9880d681SAndroid Build Coastguard Worker if (auto E = ModS.reload()) {
94*9880d681SAndroid Build Coastguard Worker consumeError(std::move(E));
95*9880d681SAndroid Build Coastguard Worker return 0;
96*9880d681SAndroid Build Coastguard Worker }
97*9880d681SAndroid Build Coastguard Worker codeview::CVSymbolDumper SD(P, TD, nullptr, false);
98*9880d681SAndroid Build Coastguard Worker bool HadError = false;
99*9880d681SAndroid Build Coastguard Worker for (auto &S : ModS.symbols(&HadError)) {
100*9880d681SAndroid Build Coastguard Worker SD.dump(S);
101*9880d681SAndroid Build Coastguard Worker }
102*9880d681SAndroid Build Coastguard Worker }
103*9880d681SAndroid Build Coastguard Worker return 0;
104*9880d681SAndroid Build Coastguard Worker }
105