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