2004-06-08 01:53:43 +08:00
|
|
|
//===-- BytecodeHandler.cpp - Parsing Handler -------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// 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 header file defines the BytecodeHandler class that gets called by the
|
|
|
|
// AbstractBytecodeParser when parsing events occur.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "AnalyzerInternals.h"
|
2004-06-09 14:16:43 +08:00
|
|
|
#include <iostream>
|
2004-06-08 01:53:43 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class AnalyzerHandler : public BytecodeHandler {
|
2004-06-09 14:16:43 +08:00
|
|
|
BytecodeAnalysis& bca;
|
2004-06-11 06:00:54 +08:00
|
|
|
BytecodeAnalysis::BytecodeFunctionInfo* currFunc;
|
2004-06-08 01:53:43 +08:00
|
|
|
public:
|
2004-06-09 14:16:43 +08:00
|
|
|
AnalyzerHandler(BytecodeAnalysis& TheBca)
|
2004-06-11 06:00:54 +08:00
|
|
|
: bca(TheBca)
|
|
|
|
, currFunc(0)
|
|
|
|
{ }
|
2004-06-09 14:16:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual bool handleError(const std::string& str ) {
|
2004-06-08 01:53:43 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleStart() {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.ModuleId.clear();
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numBlocks = 0;
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numTypes = 0;
|
|
|
|
bca.numValues = 0;
|
|
|
|
bca.numFunctions = 0;
|
|
|
|
bca.numConstants = 0;
|
|
|
|
bca.numGlobalVars = 0;
|
|
|
|
bca.numInstructions = 0;
|
|
|
|
bca.numBasicBlocks = 0;
|
|
|
|
bca.numOperands = 0;
|
|
|
|
bca.numCmpctnTables = 0;
|
|
|
|
bca.numSymTab = 0;
|
|
|
|
bca.maxTypeSlot = 0;
|
|
|
|
bca.maxValueSlot = 0;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numAlignment = 0;
|
|
|
|
bca.fileDensity = 0.0;
|
|
|
|
bca.globalsDensity = 0.0;
|
|
|
|
bca.functionDensity = 0.0;
|
|
|
|
bca.vbrCount32 = 0;
|
|
|
|
bca.vbrCount64 = 0;
|
|
|
|
bca.vbrCompBytes = 0;
|
|
|
|
bca.vbrExpdBytes = 0;
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.FunctionInfo.clear();
|
|
|
|
bca.BytecodeDump.clear();
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.BlockSizes[BytecodeFormat::Module] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::Function] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::ConstantPool] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::SymbolTable] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::ModuleGlobalInfo] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::GlobalTypePlane] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::BasicBlock] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::InstructionList] = 0;
|
|
|
|
bca.BlockSizes[BytecodeFormat::CompactionTable] = 0;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleFinish() {
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.fileDensity = double(bca.byteSize) / double( bca.numTypes + bca.numValues );
|
|
|
|
double globalSize = 0.0;
|
|
|
|
globalSize += double(bca.BlockSizes[BytecodeFormat::ConstantPool]);
|
|
|
|
globalSize += double(bca.BlockSizes[BytecodeFormat::ModuleGlobalInfo]);
|
|
|
|
globalSize += double(bca.BlockSizes[BytecodeFormat::GlobalTypePlane]);
|
|
|
|
bca.globalsDensity = globalSize / double( bca.numTypes + bca.numConstants +
|
|
|
|
bca.numGlobalVars );
|
|
|
|
bca.functionDensity = double(bca.BlockSizes[BytecodeFormat::Function]) /
|
|
|
|
double(bca.numFunctions);
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleModuleBegin(const std::string& id) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.ModuleId = id;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleModuleEnd(const std::string& id) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleVersionInfo(
|
2004-06-08 01:53:43 +08:00
|
|
|
unsigned char RevisionNum, ///< Byte code revision number
|
|
|
|
Module::Endianness Endianness, ///< Endianness indicator
|
|
|
|
Module::PointerSize PointerSize ///< PointerSize indicator
|
2004-06-11 06:00:54 +08:00
|
|
|
) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleModuleGlobalsBegin(unsigned size) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleGlobalVariable(
|
2004-06-08 01:53:43 +08:00
|
|
|
const Type* ElemType, ///< The type of the global variable
|
|
|
|
bool isConstant, ///< Whether the GV is constant or not
|
|
|
|
GlobalValue::LinkageTypes ///< The linkage type of the GV
|
2004-06-11 06:00:54 +08:00
|
|
|
) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numGlobalVars++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleInitializedGV(
|
2004-06-08 01:53:43 +08:00
|
|
|
const Type* ElemType, ///< The type of the global variable
|
|
|
|
bool isConstant, ///< Whether the GV is constant or not
|
|
|
|
GlobalValue::LinkageTypes,///< The linkage type of the GV
|
|
|
|
unsigned initSlot ///< Slot number of GV's initializer
|
2004-06-11 06:00:54 +08:00
|
|
|
) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numGlobalVars++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleType( const Type* Ty ) { bca.numTypes++; }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleFunctionDeclaration(
|
|
|
|
Function* Func, ///< The function
|
|
|
|
const FunctionType* FuncType ///< The type of the function
|
|
|
|
) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numFunctions++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleModuleGlobalsEnd() { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleCompactionTableBegin() { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleCompactionTablePlane( unsigned Ty, unsigned NumEntries) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numCmpctnTables++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleCompactionTableType( unsigned i, unsigned TypSlot,
|
|
|
|
const Type* ) {}
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleCompactionTableValue(
|
2004-06-08 01:53:43 +08:00
|
|
|
unsigned i,
|
|
|
|
unsigned ValSlot,
|
2004-06-11 06:00:54 +08:00
|
|
|
const Type* ) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleCompactionTableEnd() { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleSymbolTableBegin() { bca.numSymTab++; }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleSymbolTablePlane( unsigned Ty, unsigned NumEntries,
|
|
|
|
const Type* Typ) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleSymbolTableType( unsigned i, unsigned slot,
|
|
|
|
const std::string& name ) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleSymbolTableValue( unsigned i, unsigned slot,
|
|
|
|
const std::string& name ) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleSymbolTableEnd() { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleFunctionBegin( Function* Func, unsigned Size) {
|
|
|
|
const FunctionType* FType =
|
|
|
|
cast<FunctionType>(Func->getType()->getElementType());
|
|
|
|
currFunc = &bca.FunctionInfo[Func];
|
|
|
|
currFunc->description = FType->getDescription();
|
|
|
|
currFunc->name = Func->getName();
|
|
|
|
currFunc->byteSize = Size;
|
|
|
|
currFunc->numInstructions = 0;
|
|
|
|
currFunc->numBasicBlocks = 0;
|
|
|
|
currFunc->numPhis = 0;
|
|
|
|
currFunc->numOperands = 0;
|
|
|
|
currFunc->density = 0.0;
|
|
|
|
currFunc->vbrCount32 = 0;
|
|
|
|
currFunc->vbrCount64 = 0;
|
|
|
|
currFunc->vbrCompBytes = 0;
|
|
|
|
currFunc->vbrExpdBytes = 0;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleFunctionEnd( Function* Func) {
|
|
|
|
currFunc->density = double(currFunc->byteSize) /
|
|
|
|
double(currFunc->numInstructions+currFunc->numBasicBlocks);
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleBasicBlockBegin( unsigned blocknum) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numBasicBlocks++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-11 06:00:54 +08:00
|
|
|
if ( currFunc ) currFunc->numBasicBlocks++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual bool handleInstruction( unsigned Opcode, const Type* iType,
|
|
|
|
std::vector<unsigned>& Operands, unsigned Size) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numInstructions++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
|
|
|
bca.numOperands += Operands.size();
|
2004-06-11 06:00:54 +08:00
|
|
|
if ( currFunc ) {
|
|
|
|
currFunc->numInstructions++;
|
2004-06-11 11:06:43 +08:00
|
|
|
if ( Opcode == Instruction::PHI ) currFunc->numPhis++;
|
2004-06-11 06:00:54 +08:00
|
|
|
}
|
2004-06-09 14:16:43 +08:00
|
|
|
return Instruction::isTerminator(Opcode);
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleBasicBlockEnd(unsigned blocknum) { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleGlobalConstantsBegin() { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleConstantExpression( unsigned Opcode, const Type* Typ,
|
|
|
|
std::vector<std::pair<const Type*,unsigned> > ArgVec ) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numConstants++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleConstantValue( Constant * c ) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numConstants++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleConstantArray( const ArrayType* AT,
|
|
|
|
std::vector<unsigned>& Elements ) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numConstants++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleConstantStruct(
|
2004-06-10 16:09:13 +08:00
|
|
|
const StructType* ST,
|
|
|
|
std::vector<unsigned>& ElementSlots)
|
2004-06-08 01:53:43 +08:00
|
|
|
{
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numConstants++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleConstantPointer( const PointerType* PT, unsigned Slot) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numConstants++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleConstantString( const ConstantArray* CA ) {
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.numConstants++;
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numValues++;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleGlobalConstantsEnd() { }
|
2004-06-08 01:53:43 +08:00
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleAlignment(unsigned numBytes) {
|
2004-06-10 16:09:13 +08:00
|
|
|
bca.numAlignment += numBytes;
|
|
|
|
}
|
|
|
|
|
2004-06-11 06:00:54 +08:00
|
|
|
virtual void handleBlock(
|
2004-06-10 16:09:13 +08:00
|
|
|
unsigned BType, const unsigned char* StartPtr, unsigned Size) {
|
|
|
|
bca.numBlocks++;
|
|
|
|
bca.BlockSizes[llvm::BytecodeFormat::FileBlockIDs(BType)] += Size;
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
2004-06-10 16:09:13 +08:00
|
|
|
virtual void handleVBR32(unsigned Size ) {
|
|
|
|
bca.vbrCount32++;
|
|
|
|
bca.vbrCompBytes += Size;
|
|
|
|
bca.vbrExpdBytes += sizeof(uint32_t);
|
2004-06-11 06:00:54 +08:00
|
|
|
if (currFunc) {
|
|
|
|
currFunc->vbrCount32++;
|
|
|
|
currFunc->vbrCompBytes += Size;
|
|
|
|
currFunc->vbrExpdBytes += sizeof(uint32_t);
|
|
|
|
}
|
2004-06-10 16:09:13 +08:00
|
|
|
}
|
2004-06-11 06:00:54 +08:00
|
|
|
|
2004-06-10 16:09:13 +08:00
|
|
|
virtual void handleVBR64(unsigned Size ) {
|
|
|
|
bca.vbrCount64++;
|
|
|
|
bca.vbrCompBytes += Size;
|
|
|
|
bca.vbrExpdBytes += sizeof(uint64_t);
|
2004-06-11 06:00:54 +08:00
|
|
|
if ( currFunc ) {
|
|
|
|
currFunc->vbrCount64++;
|
|
|
|
currFunc->vbrCompBytes += Size;
|
|
|
|
currFunc->vbrExpdBytes += sizeof(uint64_t);
|
|
|
|
}
|
2004-06-10 16:09:13 +08:00
|
|
|
}
|
2004-06-08 01:53:43 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void llvm::BytecodeAnalyzer::AnalyzeBytecode(
|
|
|
|
const unsigned char *Buf,
|
|
|
|
unsigned Length,
|
|
|
|
BytecodeAnalysis& bca,
|
|
|
|
const std::string &ModuleID
|
|
|
|
)
|
|
|
|
{
|
2004-06-09 14:16:43 +08:00
|
|
|
bca.byteSize = Length;
|
|
|
|
AnalyzerHandler TheHandler(bca);
|
2004-06-10 16:09:13 +08:00
|
|
|
AbstractBytecodeParser TheParser(&TheHandler, true, true, true);
|
2004-06-08 01:53:43 +08:00
|
|
|
TheParser.ParseBytecode( Buf, Length, ModuleID );
|
2004-06-10 16:09:13 +08:00
|
|
|
TheParser.ParseAllFunctionBodies();
|
2004-06-08 01:53:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// vim: sw=2
|