X-Git-Url: http://demsky.eecs.uci.edu/git/?a=blobdiff_plain;f=lib%2FVMCore%2FAsmWriter.cpp;h=8c7b06ba0f17f264409a724303051e88e1dc959f;hb=61b91bc156bb9e3d9017a9e93d567f8dccfa3f68;hp=9523c9428a190f42a1960b8b3ab7073966d081c2;hpb=7061dc50b2513731d7b346ab16183cda4a44619f;p=oota-llvm.git diff --git a/lib/VMCore/AsmWriter.cpp b/lib/VMCore/AsmWriter.cpp index 9523c9428a1..8c7b06ba0f1 100644 --- a/lib/VMCore/AsmWriter.cpp +++ b/lib/VMCore/AsmWriter.cpp @@ -1,99 +1,73 @@ -//===-- Writer.cpp - Library for Printing VM assembly files ------*- C++ -*--=// +//===-- AsmWriter.cpp - Printing LLVM as an assembly file -----------------===// // // This library implements the functionality defined in llvm/Assembly/Writer.h // -// This library uses the Analysis library to figure out offsets for -// variables in the method tables... -// -// TODO: print out the type name instead of the full type if a particular type -// is in the symbol table... +// Note that these routines must be extremely tolerant of various errors in the +// LLVM code, because of of the primary uses of it is for debugging +// transformations. // //===----------------------------------------------------------------------===// #include "llvm/Assembly/CachedWriter.h" -#include "llvm/Analysis/SlotCalculator.h" +#include "llvm/Assembly/Writer.h" +#include "llvm/Assembly/PrintModulePass.h" +#include "llvm/SlotCalculator.h" +#include "llvm/DerivedTypes.h" +#include "llvm/Instruction.h" #include "llvm/Module.h" -#include "llvm/Method.h" -#include "llvm/GlobalVariable.h" -#include "llvm/BasicBlock.h" -#include "llvm/ConstPoolVals.h" +#include "llvm/Constants.h" #include "llvm/iMemory.h" #include "llvm/iTerminators.h" #include "llvm/iPHINode.h" #include "llvm/iOther.h" #include "llvm/SymbolTable.h" +#include "llvm/Support/CFG.h" #include "Support/StringExtras.h" #include "Support/STLExtras.h" #include -#include +using std::string; +using std::map; +using std::vector; +using std::ostream; + +static RegisterPass +X("printm", "Print module to stderr",PassInfo::Analysis|PassInfo::Optimization); +static RegisterPass +Y("print","Print function to stderr",PassInfo::Analysis|PassInfo::Optimization); + +static void WriteAsOperandInternal(ostream &Out, const Value *V, bool PrintName, + map &TypeTable, + SlotCalculator *Table); static const Module *getModuleFromVal(const Value *V) { - if (const MethodArgument *MA =dyn_cast(V)) + if (const Argument *MA = dyn_cast(V)) return MA->getParent() ? MA->getParent()->getParent() : 0; else if (const BasicBlock *BB = dyn_cast(V)) return BB->getParent() ? BB->getParent()->getParent() : 0; else if (const Instruction *I = dyn_cast(V)) { - const Method *M = I->getParent() ? I->getParent()->getParent() : 0; + const Function *M = I->getParent() ? I->getParent()->getParent() : 0; return M ? M->getParent() : 0; - } else if (const GlobalValue *GV =dyn_cast(V)) + } else if (const GlobalValue *GV = dyn_cast(V)) return GV->getParent(); - else if (const Module *Mod = dyn_cast(V)) - return Mod; return 0; } static SlotCalculator *createSlotCalculator(const Value *V) { assert(!isa(V) && "Can't create an SC for a type!"); - if (const MethodArgument *MA =dyn_cast(V)){ - return new SlotCalculator(MA->getParent(), true); + if (const Argument *FA = dyn_cast(V)) { + return new SlotCalculator(FA->getParent(), true); } else if (const Instruction *I = dyn_cast(V)) { return new SlotCalculator(I->getParent()->getParent(), true); } else if (const BasicBlock *BB = dyn_cast(V)) { return new SlotCalculator(BB->getParent(), true); - } else if (const GlobalVariable *GV =dyn_cast(V)){ + } else if (const GlobalVariable *GV = dyn_cast(V)){ return new SlotCalculator(GV->getParent(), true); - } else if (const Method *Meth = dyn_cast(V)) { - return new SlotCalculator(Meth, true); - } else if (const Module *Mod = dyn_cast(V)) { - return new SlotCalculator(Mod, true); + } else if (const Function *Func = dyn_cast(V)) { + return new SlotCalculator(Func, true); } return 0; } -// WriteAsOperand - Write the name of the specified value out to the specified -// ostream. This can be useful when you just want to print int %reg126, not the -// whole instruction that generated it. -// -static void WriteAsOperandInternal(ostream &Out, const Value *V, bool PrintName, - SlotCalculator *Table) { - if (PrintName && V->hasName()) { - Out << " %" << V->getName(); - } else { - if (const ConstPoolVal *CPV = dyn_cast(V)) { - Out << " " << CPV->getStrValue(); - } else { - int Slot; - if (Table) { - Slot = Table->getValSlot(V); - } else { - if (const Type *Ty = dyn_cast(V)) { - Out << " " << Ty->getDescription(); - return; - } - - Table = createSlotCalculator(V); - if (Table == 0) { Out << "BAD VALUE TYPE!"; return; } - - Slot = Table->getValSlot(V); - delete Table; - } - if (Slot >= 0) Out << " %" << Slot; - else if (PrintName) - Out << ""; // Not embeded into a location? - } - } -} - // If the module has a symbol table, take all global types and stuff their // names into the TypeNames map. @@ -111,8 +85,8 @@ static void fillTypeNameTable(const Module *M, // const Type *Ty = cast(I->second); if (!isa(Ty) || - !cast(Ty)->getValueType()->isPrimitiveType()) - TypeNames.insert(make_pair(Ty, "%"+I->first)); + !cast(Ty)->getElementType()->isPrimitiveType()) + TypeNames.insert(std::make_pair(Ty, "%"+I->first)); } } } @@ -143,18 +117,18 @@ static string calcTypeName(const Type *Ty, vector &TypeStack, string Result; switch (Ty->getPrimitiveID()) { - case Type::MethodTyID: { - const MethodType *MTy = cast(Ty); - Result = calcTypeName(MTy->getReturnType(), TypeStack, TypeNames) + " ("; - for (MethodType::ParamTypes::const_iterator - I = MTy->getParamTypes().begin(), - E = MTy->getParamTypes().end(); I != E; ++I) { - if (I != MTy->getParamTypes().begin()) + case Type::FunctionTyID: { + const FunctionType *FTy = cast(Ty); + Result = calcTypeName(FTy->getReturnType(), TypeStack, TypeNames) + " ("; + for (FunctionType::ParamTypes::const_iterator + I = FTy->getParamTypes().begin(), + E = FTy->getParamTypes().end(); I != E; ++I) { + if (I != FTy->getParamTypes().begin()) Result += ", "; Result += calcTypeName(*I, TypeStack, TypeNames); } - if (MTy->isVarArg()) { - if (!MTy->getParamTypes().empty()) Result += ", "; + if (FTy->isVarArg()) { + if (!FTy->getParamTypes().empty()) Result += ", "; Result += "..."; } Result += ")"; @@ -174,20 +148,17 @@ static string calcTypeName(const Type *Ty, vector &TypeStack, break; } case Type::PointerTyID: - Result = calcTypeName(cast(Ty)->getValueType(), - TypeStack, TypeNames) + " *"; + Result = calcTypeName(cast(Ty)->getElementType(), + TypeStack, TypeNames) + "*"; break; case Type::ArrayTyID: { const ArrayType *ATy = cast(Ty); - int NumElements = ATy->getNumElements(); - Result = "["; - if (NumElements != -1) Result += itostr(NumElements) + " x "; + Result = "[" + utostr(ATy->getNumElements()) + " x "; Result += calcTypeName(ATy->getElementType(), TypeStack, TypeNames) + "]"; break; } default: - assert(0 && "Unhandled case in getTypeProps!"); - Result = ""; + Result = ""; } TypeStack.pop_back(); // Remove self from stack... @@ -215,7 +186,7 @@ static ostream &printTypeInt(ostream &Out, const Type *Ty, // vector TypeStack; string TypeName = calcTypeName(Ty, TypeStack, TypeNames); - TypeNames.insert(make_pair(Ty, TypeName)); // Cache type name for later use + TypeNames.insert(std::make_pair(Ty, TypeName));//Cache type name for later use return Out << TypeName; } @@ -239,17 +210,209 @@ ostream &WriteTypeSymbolic(ostream &Out, const Type *Ty, const Module *M) { } } +static void WriteConstantInt(ostream &Out, const Constant *CV, bool PrintName, + map &TypeTable, + SlotCalculator *Table) { + if (const ConstantBool *CB = dyn_cast(CV)) { + Out << (CB == ConstantBool::True ? "true" : "false"); + } else if (const ConstantSInt *CI = dyn_cast(CV)) { + Out << CI->getValue(); + } else if (const ConstantUInt *CI = dyn_cast(CV)) { + Out << CI->getValue(); + } else if (const ConstantFP *CFP = dyn_cast(CV)) { + // We would like to output the FP constant value in exponential notation, + // but we cannot do this if doing so will lose precision. Check here to + // make sure that we only output it in exponential format if we can parse + // the value back and get the same value. + // + std::string StrVal = ftostr(CFP->getValue()); + + // Check to make sure that the stringized number is not some string like + // "Inf" or NaN, that atof will accept, but the lexer will not. Check that + // the string matches the "[-+]?[0-9]" regex. + // + if ((StrVal[0] >= '0' && StrVal[0] <= '9') || + ((StrVal[0] == '-' || StrVal[0] == '+') && + (StrVal[0] >= '0' && StrVal[0] <= '9'))) + // Reparse stringized version! + if (atof(StrVal.c_str()) == CFP->getValue()) { + Out << StrVal; return; + } + + // Otherwise we could not reparse it to exactly the same value, so we must + // output the string in hexadecimal format! + // + // Behave nicely in the face of C TBAA rules... see: + // http://www.nullstone.com/htmls/category/aliastyp.htm + // + double Val = CFP->getValue(); + char *Ptr = (char*)&Val; + assert(sizeof(double) == sizeof(uint64_t) && sizeof(double) == 8 && + "assuming that double is 64 bits!"); + Out << "0x" << utohexstr(*(uint64_t*)Ptr); + + } else if (const ConstantArray *CA = dyn_cast(CV)) { + // As a special case, print the array as a string if it is an array of + // ubytes or an array of sbytes with positive values. + // + const Type *ETy = CA->getType()->getElementType(); + bool isString = (ETy == Type::SByteTy || ETy == Type::UByteTy); + + if (ETy == Type::SByteTy) + for (unsigned i = 0; i < CA->getNumOperands(); ++i) + if (cast(CA->getOperand(i))->getValue() < 0) { + isString = false; + break; + } + + if (isString) { + Out << "c\""; + for (unsigned i = 0; i < CA->getNumOperands(); ++i) { + unsigned char C = (ETy == Type::SByteTy) ? + (unsigned char)cast(CA->getOperand(i))->getValue() : + (unsigned char)cast(CA->getOperand(i))->getValue(); + + if (isprint(C) && C != '"' && C != '\\') { + Out << C; + } else { + Out << '\\' + << (char) ((C/16 < 10) ? ( C/16 +'0') : ( C/16 -10+'A')) + << (char)(((C&15) < 10) ? ((C&15)+'0') : ((C&15)-10+'A')); + } + } + Out << "\""; + + } else { // Cannot output in string format... + Out << "["; + if (CA->getNumOperands()) { + Out << " "; + printTypeInt(Out, ETy, TypeTable); + WriteAsOperandInternal(Out, CA->getOperand(0), + PrintName, TypeTable, Table); + for (unsigned i = 1, e = CA->getNumOperands(); i != e; ++i) { + Out << ", "; + printTypeInt(Out, ETy, TypeTable); + WriteAsOperandInternal(Out, CA->getOperand(i), PrintName, + TypeTable, Table); + } + } + Out << " ]"; + } + } else if (const ConstantStruct *CS = dyn_cast(CV)) { + Out << "{"; + if (CS->getNumOperands()) { + Out << " "; + printTypeInt(Out, CS->getOperand(0)->getType(), TypeTable); + + WriteAsOperandInternal(Out, CS->getOperand(0), + PrintName, TypeTable, Table); + + for (unsigned i = 1; i < CS->getNumOperands(); i++) { + Out << ", "; + printTypeInt(Out, CS->getOperand(i)->getType(), TypeTable); + + WriteAsOperandInternal(Out, CS->getOperand(i), + PrintName, TypeTable, Table); + } + } + + Out << " }"; + } else if (isa(CV)) { + Out << "null"; + + } else if (const ConstantPointerRef *PR = dyn_cast(CV)) { + const GlobalValue *V = PR->getValue(); + if (V->hasName()) { + Out << "%" << V->getName(); + } else if (Table) { + int Slot = Table->getValSlot(V); + if (Slot >= 0) + Out << "%" << Slot; + else + Out << ""; + } else { + Out << ""; + } + + } else if (const ConstantExpr *CE = dyn_cast(CV)) { + Out << CE->getOpcodeName(); + + bool isGEP = CE->getOpcode() == Instruction::GetElementPtr; + Out << " ("; + + for (User::const_op_iterator OI=CE->op_begin(); OI != CE->op_end(); ++OI) { + printTypeInt(Out, (*OI)->getType(), TypeTable); + WriteAsOperandInternal(Out, *OI, PrintName, TypeTable, Table); + if (OI+1 != CE->op_end()) + Out << ", "; + } + + if (CE->getOpcode() == Instruction::Cast) { + Out << " to "; + printTypeInt(Out, CE->getType(), TypeTable); + } + Out << ")"; + + } else { + Out << ""; + } +} + + +// WriteAsOperand - Write the name of the specified value out to the specified +// ostream. This can be useful when you just want to print int %reg126, not the +// whole instruction that generated it. +// +static void WriteAsOperandInternal(ostream &Out, const Value *V, bool PrintName, + map &TypeTable, + SlotCalculator *Table) { + Out << " "; + if (PrintName && V->hasName()) { + Out << "%" << V->getName(); + } else { + if (const Constant *CV = dyn_cast(V)) { + WriteConstantInt(Out, CV, PrintName, TypeTable, Table); + } else { + int Slot; + if (Table) { + Slot = Table->getValSlot(V); + } else { + if (const Type *Ty = dyn_cast(V)) { + Out << Ty->getDescription(); + return; + } + + Table = createSlotCalculator(V); + if (Table == 0) { Out << "BAD VALUE TYPE!"; return; } + + Slot = Table->getValSlot(V); + delete Table; + } + if (Slot >= 0) Out << "%" << Slot; + else if (PrintName) + Out << ""; // Not embeded into a location? + } + } +} + + // WriteAsOperand - Write the name of the specified value out to the specified // ostream. This can be useful when you just want to print int %reg126, not the // whole instruction that generated it. // ostream &WriteAsOperand(ostream &Out, const Value *V, bool PrintType, - bool PrintName, SlotCalculator *Table) { - if (PrintType) - WriteTypeSymbolic(Out, V->getType(), getModuleFromVal(V)); + bool PrintName, const Module *Context) { + map TypeNames; + if (Context == 0) Context = getModuleFromVal(V); + + if (Context && Context->hasSymbolTable()) + fillTypeNameTable(Context, TypeNames); - WriteAsOperandInternal(Out, V, PrintName, Table); + if (PrintType) + printTypeInt(Out, V->getType(), TypeNames); + + WriteAsOperandInternal(Out, V, PrintName, TypeNames, 0); return Out; } @@ -272,35 +435,90 @@ public: inline void write(const Module *M) { printModule(M); } inline void write(const GlobalVariable *G) { printGlobal(G); } - inline void write(const Method *M) { printMethod(M); } + inline void write(const Function *F) { printFunction(F); } inline void write(const BasicBlock *BB) { printBasicBlock(BB); } - inline void write(const Instruction *I) { printInstruction(I); } - inline void write(const ConstPoolVal *CPV) { printConstant(CPV); } + inline void write(const Instruction *I) { printInstruction(*I); } + inline void write(const Constant *CPV) { printConstant(CPV); } inline void write(const Type *Ty) { printType(Ty); } + void writeOperand(const Value *Op, bool PrintType, bool PrintName = true); + private : void printModule(const Module *M); void printSymbolTable(const SymbolTable &ST); - void printConstant(const ConstPoolVal *CPV); + void printConstant(const Constant *CPV); void printGlobal(const GlobalVariable *GV); - void printMethod(const Method *M); - void printMethodArgument(const MethodArgument *MA); + void printFunction(const Function *F); + void printArgument(const Argument *FA); void printBasicBlock(const BasicBlock *BB); - void printInstruction(const Instruction *I); - ostream &printType(const Type *Ty); + void printInstruction(const Instruction &I); - void writeOperand(const Value *Op, bool PrintType, bool PrintName = true); + // printType - Go to extreme measures to attempt to print out a short, + // symbolic version of a type name. + // + ostream &printType(const Type *Ty) { + return printTypeInt(Out, Ty, TypeNames); + } + + // printTypeAtLeastOneLevel - Print out one level of the possibly complex type + // without considering any symbolic types that we may have equal to it. + // + ostream &printTypeAtLeastOneLevel(const Type *Ty); // printInfoComment - Print a little comment after the instruction indicating // which slot it occupies. - void printInfoComment(const Value *V); + void printInfoComment(const Value &V); }; +// printTypeAtLeastOneLevel - Print out one level of the possibly complex type +// without considering any symbolic types that we may have equal to it. +// +ostream &AssemblyWriter::printTypeAtLeastOneLevel(const Type *Ty) { + if (const FunctionType *FTy = dyn_cast(Ty)) { + printType(FTy->getReturnType()) << " ("; + for (FunctionType::ParamTypes::const_iterator + I = FTy->getParamTypes().begin(), + E = FTy->getParamTypes().end(); I != E; ++I) { + if (I != FTy->getParamTypes().begin()) + Out << ", "; + printType(*I); + } + if (FTy->isVarArg()) { + if (!FTy->getParamTypes().empty()) Out << ", "; + Out << "..."; + } + Out << ")"; + } else if (const StructType *STy = dyn_cast(Ty)) { + Out << "{ "; + for (StructType::ElementTypes::const_iterator + I = STy->getElementTypes().begin(), + E = STy->getElementTypes().end(); I != E; ++I) { + if (I != STy->getElementTypes().begin()) + Out << ", "; + printType(*I); + } + Out << " }"; + } else if (const PointerType *PTy = dyn_cast(Ty)) { + printType(PTy->getElementType()) << "*"; + } else if (const ArrayType *ATy = dyn_cast(Ty)) { + Out << "[" << ATy->getNumElements() << " x "; + printType(ATy->getElementType()) << "]"; + } else if (const OpaqueType *OTy = dyn_cast(Ty)) { + Out << OTy->getDescription(); + } else { + if (!Ty->isPrimitiveType()) + Out << ""; + printType(Ty); + } + return Out; +} + + void AssemblyWriter::writeOperand(const Value *Operand, bool PrintType, bool PrintName) { if (PrintType) { Out << " "; printType(Operand->getType()); } - WriteAsOperandInternal(Out, Operand, PrintName, &Table); + WriteAsOperandInternal(Out, Operand, PrintName, TypeNames, &Table); } @@ -309,13 +527,14 @@ void AssemblyWriter::printModule(const Module *M) { if (M->hasSymbolTable()) printSymbolTable(*M->getSymbolTable()); - for_each(M->gbegin(), M->gend(), - bind_obj(this, &AssemblyWriter::printGlobal)); + for (Module::const_giterator I = M->gbegin(), E = M->gend(); I != E; ++I) + printGlobal(I); - Out << "implementation\n"; + Out << "\nimplementation ; Functions:\n"; - // Output all of the methods... - for_each(M->begin(), M->end(), bind_obj(this,&AssemblyWriter::printMethod)); + // Output all of the functions... + for (Module::const_iterator I = M->begin(), E = M->end(); I != E; ++I) + printFunction(I); } void AssemblyWriter::printGlobal(const GlobalVariable *GV) { @@ -325,13 +544,13 @@ void AssemblyWriter::printGlobal(const GlobalVariable *GV) { if (!GV->hasInitializer()) Out << "uninitialized "; Out << (GV->isConstant() ? "constant " : "global "); - printType(GV->getType()->getValueType()); + printType(GV->getType()->getElementType()); if (GV->hasInitializer()) writeOperand(GV->getInitializer(), false, false); - printInfoComment(GV); - Out << endl; + printInfoComment(*GV); + Out << "\n"; } @@ -345,10 +564,15 @@ void AssemblyWriter::printSymbolTable(const SymbolTable &ST) { for (; I != End; ++I) { const Value *V = I->second; - if (const ConstPoolVal *CPV = dyn_cast(V)) { + if (const Constant *CPV = dyn_cast(V)) { printConstant(CPV); } else if (const Type *Ty = dyn_cast(V)) { - Out << "\t%" << I->first << " = type " << Ty->getDescription() << endl; + Out << "\t%" << I->first << " = type "; + + // Make sure we print out at least one level of the type structure, so + // that we do not get %FILE = type %FILE + // + printTypeAtLeastOneLevel(Ty) << "\n"; } } } @@ -357,85 +581,72 @@ void AssemblyWriter::printSymbolTable(const SymbolTable &ST) { // printConstant - Print out a constant pool entry... // -void AssemblyWriter::printConstant(const ConstPoolVal *CPV) { +void AssemblyWriter::printConstant(const Constant *CPV) { // Don't print out unnamed constants, they will be inlined if (!CPV->hasName()) return; // Print out name... - Out << "\t%" << CPV->getName() << " = "; - - // Print out the constant type... - printType(CPV->getType()); + Out << "\t%" << CPV->getName() << " ="; // Write the value out now... - writeOperand(CPV, false, false); + writeOperand(CPV, true, false); - if (!CPV->hasName() && CPV->getType() != Type::VoidTy) { - int Slot = Table.getValSlot(CPV); // Print out the def slot taken... - Out << "\t\t; <"; - printType(CPV->getType()) << ">:"; - if (Slot >= 0) Out << Slot; - else Out << ""; - } - - Out << endl; + printInfoComment(*CPV); + Out << "\n"; } -// printMethod - Print all aspects of a method. +// printFunction - Print all aspects of a function. // -void AssemblyWriter::printMethod(const Method *M) { +void AssemblyWriter::printFunction(const Function *F) { // Print out the return type and name... - Out << "\n" << (M->isExternal() ? "declare " : "") - << (M->hasInternalLinkage() ? "internal " : ""); - printType(M->getReturnType()) << " \"" << M->getName() << "\"("; - Table.incorporateMethod(M); + Out << "\n" << (F->isExternal() ? "declare " : "") + << (F->hasInternalLinkage() ? "internal " : ""); + printType(F->getReturnType()) << " %" << F->getName() << "("; + Table.incorporateFunction(F); // Loop over the arguments, printing them... - const MethodType *MT = cast(M->getMethodType()); + const FunctionType *FT = F->getFunctionType(); - if (!M->isExternal()) { - for_each(M->getArgumentList().begin(), M->getArgumentList().end(), - bind_obj(this, &AssemblyWriter::printMethodArgument)); + if (!F->isExternal()) { + for(Function::const_aiterator I = F->abegin(), E = F->aend(); I != E; ++I) + printArgument(I); } else { // Loop over the arguments, printing them... - const MethodType *MT = cast(M->getMethodType()); - for (MethodType::ParamTypes::const_iterator I = MT->getParamTypes().begin(), - E = MT->getParamTypes().end(); I != E; ++I) { - if (I != MT->getParamTypes().begin()) Out << ", "; + for (FunctionType::ParamTypes::const_iterator I = FT->getParamTypes().begin(), + E = FT->getParamTypes().end(); I != E; ++I) { + if (I != FT->getParamTypes().begin()) Out << ", "; printType(*I); } } // Finish printing arguments... - if (MT->isVarArg()) { - if (MT->getParamTypes().size()) Out << ", "; + if (FT->isVarArg()) { + if (FT->getParamTypes().size()) Out << ", "; Out << "..."; // Output varargs portion of signature! } - Out << ")\n"; - - if (!M->isExternal()) { - // Loop over the symbol table, emitting all named constants... - if (M->hasSymbolTable()) - printSymbolTable(*M->getSymbolTable()); + Out << ")"; - Out << "begin"; + if (F->isExternal()) { + Out << "\n"; + } else { + Out << " {"; - // Output all of its basic blocks... for the method - for_each(M->begin(), M->end(), - bind_obj(this, &AssemblyWriter::printBasicBlock)); + // Output all of its basic blocks... for the function + for (Function::const_iterator I = F->begin(), E = F->end(); I != E; ++I) + printBasicBlock(I); - Out << "end\n"; + Out << "}\n"; } - Table.purgeMethod(); + Table.purgeFunction(); } -// printMethodArgument - This member is called for every argument that -// is passed into the method. Simply print it out +// printArgument - This member is called for every argument that +// is passed into the function. Simply print it out // -void AssemblyWriter::printMethodArgument(const MethodArgument *Arg) { +void AssemblyWriter::printArgument(const Argument *Arg) { // Insert commas as we go... the first arg doesn't get a comma - if (Arg != Arg->getParent()->getArgumentList().front()) Out << ", "; + if (Arg != &Arg->getParent()->afront()) Out << ", "; // Output type... printType(Arg->getType()); @@ -452,7 +663,7 @@ void AssemblyWriter::printMethodArgument(const MethodArgument *Arg) { void AssemblyWriter::printBasicBlock(const BasicBlock *BB) { if (BB->hasName()) { // Print out the label if it exists... Out << "\n" << BB->getName() << ":"; - } else { + } else if (!BB->use_empty()) { // Don't print block # of no uses... int Slot = Table.getValSlot(BB); Out << "\n;