From 32de6330f1e86e9b95a6ad3c58a754fd94e6f727 Mon Sep 17 00:00:00 2001 From: Chris Lattner Date: Sun, 29 Apr 2007 08:31:14 +0000 Subject: [PATCH] Implement support to read an arbitrary bitcode file. Next up, dumping the file symbolically and actually computing statistics. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@36557 91177308-0d34-0410-b5e6-96231b3b80d8 --- tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp | 87 ++++++++++++++++++++--- 1 file changed, 78 insertions(+), 9 deletions(-) diff --git a/tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp b/tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp index ad130633d2b..04d8dc41104 100644 --- a/tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp +++ b/tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp @@ -51,6 +51,11 @@ static cl::opt static cl::opt NoDetails("nodetails", cl::desc("Skip detailed output")); static cl::opt Dump("dump", cl::desc("Dump low level bytecode trace")); static cl::opt Verify("verify", cl::desc("Progressively verify module")); + +//===----------------------------------------------------------------------===// +// Bitcode specific analysis. +//===----------------------------------------------------------------------===// + static cl::opt Bitcode("bitcode", cl::desc("Read a bitcode file")); /// CurStreamType - If we can sniff the flavor of this stream, we can produce @@ -60,6 +65,54 @@ static enum { LLVMIRBitstream } CurStreamType; +/// Error - All bitcode analysis errors go through this function, making this a +/// good place to breakpoint if debugging. +static bool Error(const std::string &Err) { + std::cerr << Err << "\n"; + return true; +} + +/// ParseBlock - Read a block, updating statistics, etc. +static bool ParseBlock(BitstreamReader &Stream) { + unsigned BlockID = Stream.ReadSubBlockID(); + + // TODO: Compute per-block-id stats. + BlockID = BlockID; + + if (Stream.EnterSubBlock()) + return Error("Malformed block record"); + + SmallVector Record; + + // Read all the records for this block. + while (1) { + if (Stream.AtEndOfStream()) + return Error("Premature end of bitstream"); + + // Read the code for this record. + unsigned AbbrevID = Stream.ReadCode(); + switch (AbbrevID) { + case bitc::END_BLOCK: + if (Stream.ReadBlockEnd()) + return Error("Error at end of block"); + return false; + case bitc::ENTER_SUBBLOCK: + if (ParseBlock(Stream)) + return true; + break; + case bitc::DEFINE_ABBREV: + Stream.ReadAbbrevRecord(); + break; + default: + Record.clear(); + unsigned Code = Stream.ReadRecord(AbbrevID, Record); + // TODO: Compute per-blockid/code stats. + Code = Code; + break; + } + } +} + /// AnalyzeBitcode - Analyze the bitcode file specified by InputFilename. static int AnalyzeBitcode() { // Read the input file. @@ -69,15 +122,11 @@ static int AnalyzeBitcode() { else Buffer = MemoryBuffer::getFile(&InputFilename[0], InputFilename.size()); - if (Buffer == 0) { - std::cerr << "Error reading '" << InputFilename << "'.\n"; - return 1; - } + if (Buffer == 0) + return Error("Error reading '" + InputFilename + "'."); - if (Buffer->getBufferSize() & 3) { - std::cerr << "Bitcode stream should be a multiple of 4 bytes in length\n"; - return 1; - } + if (Buffer->getBufferSize() & 3) + return Error("Bitcode stream should be a multiple of 4 bytes in length"); unsigned char *BufPtr = (unsigned char *)Buffer->getBufferStart(); BitstreamReader Stream(BufPtr, BufPtr+Buffer->getBufferSize()); @@ -92,12 +141,25 @@ static int AnalyzeBitcode() { Signature[4] = Stream.Read(4); Signature[5] = Stream.Read(4); + // Autodetect the file contents, if it is one we know. CurStreamType = UnknownBitstream; if (Signature[0] == 'B' && Signature[1] == 'C' && Signature[2] == 0x0 && Signature[3] == 0xC && Signature[4] == 0xE && Signature[5] == 0xD) CurStreamType = LLVMIRBitstream; + // Parse the top-level structure. We only allow blocks at the top-level. + while (!Stream.AtEndOfStream()) { + unsigned Code = Stream.ReadCode(); + if (Code != bitc::ENTER_SUBBLOCK) + return Error("Invalid record at top-level"); + + if (ParseBlock(Stream)) + return true; + } + + // Print a summary of the read file. + std::cerr << "Summary of " << InputFilename << ":\n"; std::cerr << " Stream type: "; switch (CurStreamType) { @@ -105,10 +167,17 @@ static int AnalyzeBitcode() { case UnknownBitstream: std::cerr << "unknown\n"; break; case LLVMIRBitstream: std::cerr << "LLVM IR\n"; break; } - + + // TODO: Stats! + return 0; } + +//===----------------------------------------------------------------------===// +// Bytecode specific analysis. +//===----------------------------------------------------------------------===// + int main(int argc, char **argv) { llvm_shutdown_obj X; // Call llvm_shutdown() on exit. cl::ParseCommandLineOptions(argc, argv, " llvm-bcanalyzer file analyzer\n");