//===-- llvm-bcanalyzer.cpp - Byte Code Analyzer --------------------------===// // // The LLVM Compiler Infrastructure // // This file was developed by Reid Spencer and is distributed under the // University of Illinois Open Source License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This tool may be invoked in the following manner: // llvm-bcanalyzer [options] - Read LLVM bytecode from stdin // llvm-bcanalyzer [options] x.bc - Read LLVM bytecode from the x.bc file // // Options: // --help - Output information about command line switches // --nodetails - Don't print out detailed informaton about individual // blocks and functions // --dump - Dump low-level bytecode structure in readable format // // This tool provides analytical information about a bytecode file. It is // intended as an aid to developers of bytecode reading and writing software. It // produces on std::out a summary of the bytecode file that shows various // statistics about the contents of the file. By default this information is // detailed and contains information about individual bytecode blocks and the // functions in the module. To avoid this more detailed output, use the // -nodetails option to limit the output to just module level information. // The tool is also able to print a bytecode file in a straight forward text // format that shows the containment and relationships of the information in // the bytecode file (-dump option). // //===----------------------------------------------------------------------===// #include "llvm/Analysis/Verifier.h" #include "llvm/Bitcode/BitstreamReader.h" #include "llvm/Bytecode/Analyzer.h" #include "llvm/Support/CommandLine.h" #include "llvm/Support/Compressor.h" #include "llvm/Support/ManagedStatic.h" #include "llvm/Support/MemoryBuffer.h" #include "llvm/System/Signals.h" #include #include using namespace llvm; static cl::opt InputFilename(cl::Positional, cl::desc(""), cl::init("-")); static cl::opt OutputFilename("-o", cl::init("-"), cl::desc("")); 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")); static cl::opt Bitcode("bitcode", cl::desc("Read a bitcode file")); /// CurStreamType - If we can sniff the flavor of this stream, we can produce /// better dump info. static enum { UnknownBitstream, LLVMIRBitstream } CurStreamType; /// AnalyzeBitcode - Analyze the bitcode file specified by InputFilename. static int AnalyzeBitcode() { // Read the input file. MemoryBuffer *Buffer; if (InputFilename == "-") Buffer = MemoryBuffer::getSTDIN(); else Buffer = MemoryBuffer::getFile(&InputFilename[0], InputFilename.size()); if (Buffer == 0) { std::cerr << "Error reading '" << InputFilename << "'.\n"; return 1; } if (Buffer->getBufferSize() & 3) { std::cerr << "Bitcode stream should be a multiple of 4 bytes in length\n"; return 1; } unsigned char *BufPtr = (unsigned char *)Buffer->getBufferStart(); BitstreamReader Stream(BufPtr, BufPtr+Buffer->getBufferSize()); // Read the stream signature. char Signature[6]; Signature[0] = Stream.Read(8); Signature[1] = Stream.Read(8); Signature[2] = Stream.Read(4); Signature[3] = Stream.Read(4); Signature[4] = Stream.Read(4); Signature[5] = Stream.Read(4); CurStreamType = UnknownBitstream; if (Signature[0] == 'B' && Signature[1] == 'C' && Signature[2] == 0x0 && Signature[3] == 0xC && Signature[4] == 0xE && Signature[5] == 0xD) CurStreamType = LLVMIRBitstream; std::cerr << "Summary of " << InputFilename << ":\n"; std::cerr << " Stream type: "; switch (CurStreamType) { default: assert(0 && "Unknown bitstream type"); case UnknownBitstream: std::cerr << "unknown\n"; break; case LLVMIRBitstream: std::cerr << "LLVM IR\n"; break; } return 0; } int main(int argc, char **argv) { llvm_shutdown_obj X; // Call llvm_shutdown() on exit. cl::ParseCommandLineOptions(argc, argv, " llvm-bcanalyzer file analyzer\n"); sys::PrintStackTraceOnErrorSignal(); if (Bitcode) return AnalyzeBitcode(); try { std::ostream *Out = &std::cout; // Default to printing to stdout... std::string ErrorMessage; BytecodeAnalysis bca; /// Determine what to generate bca.detailedResults = !NoDetails; bca.progressiveVerify = Verify; /// Analyze the bytecode file Module* M = AnalyzeBytecodeFile(InputFilename, bca, Compressor::decompressToNewBuffer, &ErrorMessage, (Dump?Out:0)); // All that bcanalyzer does is write the gathered statistics to the output PrintBytecodeAnalysis(bca,*Out); if (M && Verify) { std::string verificationMsg; if (verifyModule(*M, ReturnStatusAction, &verificationMsg)) std::cerr << "Final Verification Message: " << verificationMsg << "\n"; } if (Out != &std::cout) { ((std::ofstream*)Out)->close(); delete Out; } return 0; } catch (const std::string& msg) { std::cerr << argv[0] << ": " << msg << "\n"; } catch (...) { std::cerr << argv[0] << ": Unexpected unknown exception occurred.\n"; } return 1; }