//===-- Writer.cpp - Library for writing C files --------------------------===// // // This library implements the functionality defined in llvm/Assembly/CWriter.h // and CLocalVars.h // // TODO : Recursive types. // //===-----------------------------------------------------------------------==// #include "llvm/Assembly/CWriter.h" #include "CLocalVars.h" #include "llvm/SlotCalculator.h" #include "llvm/Constants.h" #include "llvm/DerivedTypes.h" #include "llvm/Module.h" #include "llvm/GlobalVariable.h" #include "llvm/Function.h" #include "llvm/Argument.h" #include "llvm/BasicBlock.h" #include "llvm/iMemory.h" #include "llvm/iTerminators.h" #include "llvm/iPHINode.h" #include "llvm/iOther.h" #include "llvm/iOperators.h" #include "llvm/SymbolTable.h" #include "llvm/Support/InstVisitor.h" #include "Support/StringExtras.h" #include "Support/STLExtras.h" #include #include using std::string; using std::map; using std::vector; using std::ostream; //===-----------------------------------------------------------------------==// // // Implementation of the CLocalVars methods // Appends a variable to the LocalVars map if it does not already exist // Also check that the type exists on the map. void CLocalVars::addLocalVar(const Type *t, const string & var) { if (!LocalVars.count(t) || find(LocalVars[t].begin(), LocalVars[t].end(), var) == LocalVars[t].end()) { LocalVars[t].push_back(var); } } static std::string getConstStrValue(const Constant* CPV); static std::string getConstArrayStrValue(const Constant* CPV) { std::string Result; // 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 = cast(CPV->getType())->getElementType(); bool isString = (ETy == Type::SByteTy || ETy == Type::UByteTy); if (ETy == Type::SByteTy) { for (unsigned i = 0; i < CPV->getNumOperands(); ++i) if (ETy == Type::SByteTy && cast(CPV->getOperand(i))->getValue() < 0) { isString = false; break; } } if (isString) { // Make sure the last character is a null char, as automatically added by C if (CPV->getNumOperands() == 0 || !cast(*(CPV->op_end()-1))->isNullValue()) isString = false; } if (isString) { Result = "\""; // Do not include the last character, which we know is null for (unsigned i = 0, e = CPV->getNumOperands()-1; i != e; ++i) { unsigned char C = (ETy == Type::SByteTy) ? (unsigned char)cast(CPV->getOperand(i))->getValue() : (unsigned char)cast(CPV->getOperand(i))->getValue(); if (isprint(C)) { Result += C; } else { switch (C) { case '\n': Result += "\\n"; break; case '\t': Result += "\\t"; break; case '\r': Result += "\\r"; break; case '\v': Result += "\\v"; break; case '\a': Result += "\\a"; break; default: Result += "\\x"; Result += ( C/16 < 10) ? ( C/16 +'0') : ( C/16 -10+'A'); Result += ((C&15) < 10) ? ((C&15)+'0') : ((C&15)-10+'A'); break; } } } Result += "\""; } else { Result = "{"; if (CPV->getNumOperands()) { Result += " " + getConstStrValue(cast(CPV->getOperand(0))); for (unsigned i = 1; i < CPV->getNumOperands(); i++) Result += ", " + getConstStrValue(cast(CPV->getOperand(i))); } Result += " }"; } return Result; } static std::string getConstStrValue(const Constant* CPV) { switch (CPV->getType()->getPrimitiveID()) { case Type::BoolTyID: return CPV == ConstantBool::False ? "0" : "1"; case Type::SByteTyID: case Type::ShortTyID: case Type::IntTyID: return itostr(cast(CPV)->getValue()); case Type::LongTyID: return itostr(cast(CPV)->getValue()) + "ll"; case Type::UByteTyID: return utostr(cast(CPV)->getValue()); case Type::UShortTyID: return utostr(cast(CPV)->getValue()); case Type::UIntTyID: return utostr(cast(CPV)->getValue())+"u"; case Type::ULongTyID: return utostr(cast(CPV)->getValue())+"ull"; case Type::FloatTyID: case Type::DoubleTyID: return ftostr(cast(CPV)->getValue()); case Type::ArrayTyID: return getConstArrayStrValue(CPV); case Type::StructTyID: { std::string Result = "{"; if (CPV->getNumOperands()) { Result += " " + getConstStrValue(cast(CPV->getOperand(0))); for (unsigned i = 1; i < CPV->getNumOperands(); i++) Result += ", " + getConstStrValue(cast(CPV->getOperand(i))); } return Result + " }"; } default: cerr << "Unknown constant type: " << CPV << "\n"; abort(); } } // Pass the Type* variable and and the variable name and this prints out the // variable declaration. // static string calcTypeNameVar(const Type *Ty, map &TypeNames, const string &NameSoFar, bool ignoreName = false){ if (Ty->isPrimitiveType()) switch (Ty->getPrimitiveID()) { case Type::VoidTyID: return "void " + NameSoFar; case Type::BoolTyID: return "bool " + NameSoFar; case Type::UByteTyID: return "unsigned char " + NameSoFar; case Type::SByteTyID: return "signed char " + NameSoFar; case Type::UShortTyID: return "unsigned short " + NameSoFar; case Type::ShortTyID: return "short " + NameSoFar; case Type::UIntTyID: return "unsigned " + NameSoFar; case Type::IntTyID: return "int " + NameSoFar; case Type::ULongTyID: return "unsigned long long " + NameSoFar; case Type::LongTyID: return "signed long long " + NameSoFar; case Type::FloatTyID: return "float " + NameSoFar; case Type::DoubleTyID: return "double " + NameSoFar; default : cerr << "Unknown primitive type: " << Ty << "\n"; abort(); } // Check to see if the type is named. if (!ignoreName) { map::iterator I = TypeNames.find(Ty); if (I != TypeNames.end()) return I->second + " " + NameSoFar; } string Result; switch (Ty->getPrimitiveID()) { case Type::FunctionTyID: { const FunctionType *MTy = cast(Ty); Result += calcTypeNameVar(MTy->getReturnType(), TypeNames, ""); Result += " " + NameSoFar; Result += " ("; for (FunctionType::ParamTypes::const_iterator I = MTy->getParamTypes().begin(), E = MTy->getParamTypes().end(); I != E; ++I) { if (I != MTy->getParamTypes().begin()) Result += ", "; Result += calcTypeNameVar(*I, TypeNames, ""); } if (MTy->isVarArg()) { if (!MTy->getParamTypes().empty()) Result += ", "; Result += "..."; } Result += ")"; break; } case Type::StructTyID: { const StructType *STy = cast(Ty); Result = NameSoFar + " {\n"; unsigned indx = 0; for (StructType::ElementTypes::const_iterator I = STy->getElementTypes().begin(), E = STy->getElementTypes().end(); I != E; ++I) { Result += " " +calcTypeNameVar(*I, TypeNames, "field" + utostr(indx++)); Result += ";\n"; } Result += "}"; break; } case Type::PointerTyID: { Result = calcTypeNameVar(cast(Ty)->getElementType(), TypeNames, "*" + NameSoFar); break; } case Type::ArrayTyID: { const ArrayType *ATy = cast(Ty); int NumElements = ATy->getNumElements(); Result = calcTypeNameVar(ATy->getElementType(), TypeNames, NameSoFar + "[" + itostr(NumElements) + "]"); break; } default: assert(0 && "Unhandled case in getTypeProps!"); Result = ""; } return Result; } namespace { class CWriter { ostream& Out; SlotCalculator &Table; const Module *TheModule; map TypeNames; public: inline CWriter(ostream &o, SlotCalculator &Tab, const Module *M) : Out(o), Table(Tab), TheModule(M) { } inline void write(Module *M) { printModule(M); } ostream& printTypeVar(const Type *Ty, const string &VariableName) { return Out << calcTypeNameVar(Ty, TypeNames, VariableName); } ostream& printType(const Type *Ty) { return Out << calcTypeNameVar(Ty, TypeNames, ""); } void writeOperand(const Value *Operand); string getValueName(const Value *V); private : void printModule(Module *M); void printSymbolTable(const SymbolTable &ST); void printGlobal(const GlobalVariable *GV); void printFunctionSignature(const Function *F); void printFunctionDecl(const Function *F); // Print just the forward decl void printFunction(Function *); }; /* END class CWriter */ /* CLASS InstLocalVarsVisitor */ class InstLocalVarsVisitor : public InstVisitor { CWriter& CW; void handleTerminator(TerminatorInst *tI, int indx); public: CLocalVars CLV; InstLocalVarsVisitor(CWriter &cw) : CW(cw) {} void visitInstruction(Instruction *I) { if (I->getType() != Type::VoidTy) CLV.addLocalVar(I->getType(), CW.getValueName(I)); } void visitBranchInst(BranchInst *I) { handleTerminator(I, 0); if (I->isConditional()) handleTerminator(I, 1); } }; } void InstLocalVarsVisitor::handleTerminator(TerminatorInst *tI,int indx) { BasicBlock *bb = tI->getSuccessor(indx); BasicBlock::const_iterator insIt = bb->begin(); while (insIt != bb->end()) { if (const PHINode *pI = dyn_cast(*insIt)) { // Its a phinode! // Calculate the incoming index for this assert(pI->getBasicBlockIndex(tI->getParent()) != -1); CLV.addLocalVar(pI->getType(), CW.getValueName(pI)); } else break; insIt++; } } namespace { /* CLASS CInstPrintVisitor */ class CInstPrintVisitor: public InstVisitor { CWriter& CW; SlotCalculator& Table; ostream &Out; void outputLValue(Instruction *); void printBranchToBlock(BasicBlock *CurBlock, BasicBlock *SuccBlock, unsigned Indent); void printIndexingExpr(MemAccessInst *MAI); public: CInstPrintVisitor (CWriter &cw, SlotCalculator& table, ostream& o) : CW(cw), Table(table), Out(o) {} void visitCastInst(CastInst *I); void visitCallInst(CallInst *I); void visitShiftInst(ShiftInst *I) { visitBinaryOperator(I); } void visitReturnInst(ReturnInst *I); void visitBranchInst(BranchInst *I); void visitSwitchInst(SwitchInst *I); void visitInvokeInst(InvokeInst *I) ; void visitMallocInst(MallocInst *I); void visitAllocaInst(AllocaInst *I); void visitFreeInst(FreeInst *I); void visitLoadInst(LoadInst *I); void visitStoreInst(StoreInst *I); void visitGetElementPtrInst(GetElementPtrInst *I); void visitPHINode(PHINode *I) {} void visitNot(GenericUnaryInst *I); void visitBinaryOperator(Instruction *I); }; } void CInstPrintVisitor::outputLValue(Instruction *I) { Out << " " << CW.getValueName(I) << " = "; } // Implement all "other" instructions, except for PHINode void CInstPrintVisitor::visitCastInst(CastInst *I) { outputLValue(I); Out << "("; CW.printType(I->getType()); Out << ")"; CW.writeOperand(I->getOperand(0)); Out << ";\n"; } void CInstPrintVisitor::visitCallInst(CallInst *I) { if (I->getType() != Type::VoidTy) outputLValue(I); else Out << " "; const PointerType *PTy = cast(I->getCalledValue()->getType()); const FunctionType *FTy = cast(PTy->getElementType()); const Type *RetTy = FTy->getReturnType(); Out << CW.getValueName(I->getOperand(0)) << "("; if (I->getNumOperands() > 1) { CW.writeOperand(I->getOperand(1)); for (unsigned op = 2, Eop = I->getNumOperands(); op != Eop; ++op) { Out << ", "; CW.writeOperand(I->getOperand(op)); } } Out << ");\n"; } // Specific Instruction type classes... note that all of the casts are // neccesary because we use the instruction classes as opaque types... // void CInstPrintVisitor::visitReturnInst(ReturnInst *I) { // Don't output a void return if this is the last basic block in the function if (I->getNumOperands() == 0 && *(I->getParent()->getParent()->end()-1) == I->getParent()) return; Out << " return"; if (I->getNumOperands()) { Out << " "; CW.writeOperand(I->getOperand(0)); } Out << ";\n"; } // Return true if BB1 immediately preceeds BB2. static bool BBFollowsBB(BasicBlock *BB1, BasicBlock *BB2) { Function *F = BB1->getParent(); Function::iterator I = find(F->begin(), F->end(), BB1); assert(I != F->end() && "BB not in function!"); return *(I+1) == BB2; } static bool isGotoCodeNeccessary(BasicBlock *From, BasicBlock *To) { // If PHI nodes need copies, we need the copy code... if (isa(To->front()) || !BBFollowsBB(From, To)) // Not directly successor, need goto return true; // Otherwise we don't need the code. return false; } void CInstPrintVisitor::printBranchToBlock(BasicBlock *CurBB, BasicBlock *Succ, unsigned Indent) { for (BasicBlock::iterator I = Succ->begin(); PHINode *PN = dyn_cast(*I); ++I) { // now we have to do the printing Out << string(Indent, ' '); outputLValue(PN); CW.writeOperand(PN->getIncomingValue(PN->getBasicBlockIndex(CurBB))); Out << "; /* for PHI node */\n"; } if (!BBFollowsBB(CurBB, Succ)) { Out << string(Indent, ' ') << " goto "; CW.writeOperand(Succ); Out << ";\n"; } } // Brach instruction printing - Avoid printing out a brach to a basic block that // immediately succeeds the current one. // void CInstPrintVisitor::visitBranchInst(BranchInst *I) { if (I->isConditional()) { if (isGotoCodeNeccessary(I->getParent(), I->getSuccessor(0))) { Out << " if ("; CW.writeOperand(I->getCondition()); Out << ") {\n"; printBranchToBlock(I->getParent(), I->getSuccessor(0), 2); if (isGotoCodeNeccessary(I->getParent(), I->getSuccessor(1))) { Out << " } else {\n"; printBranchToBlock(I->getParent(), I->getSuccessor(1), 2); } } else { // First goto not neccesary, assume second one is... Out << " if (!"; CW.writeOperand(I->getCondition()); Out << ") {\n"; printBranchToBlock(I->getParent(), I->getSuccessor(1), 2); } Out << " }\n"; } else { printBranchToBlock(I->getParent(), I->getSuccessor(0), 0); } Out << "\n"; } void CInstPrintVisitor::visitSwitchInst(SwitchInst *I) { assert(0 && "Switch not implemented!"); } void CInstPrintVisitor::visitInvokeInst(InvokeInst *I) { assert(0 && "Invoke not implemented!"); } void CInstPrintVisitor::visitMallocInst(MallocInst *I) { outputLValue(I); Out << "("; CW.printType(I->getType()); Out << ")malloc(sizeof("; CW.printType(I->getType()->getElementType()); Out << ")"; if (I->isArrayAllocation()) { Out << " * " ; CW.writeOperand(I->getOperand(0)); } Out << ");\n"; } void CInstPrintVisitor::visitAllocaInst(AllocaInst *I) { outputLValue(I); Out << "("; CW.printType(I->getType()); Out << ") alloca(sizeof("; CW.printType(I->getType()->getElementType()); Out << ")"; if (I->isArrayAllocation()) { Out << " * " ; CW.writeOperand(I->getOperand(0)); } Out << ");\n"; } void CInstPrintVisitor::visitFreeInst(FreeInst *I) { Out << " free("; CW.writeOperand(I->getOperand(0)); Out << ");\n"; } void CInstPrintVisitor::printIndexingExpr(MemAccessInst *MAI) { MemAccessInst::op_iterator I = MAI->idx_begin(), E = MAI->idx_end(); if (I == E) Out << "*"; // Implicit zero first argument: '*x' is equivalent to 'x[0]' CW.writeOperand(MAI->getPointerOperand()); if (I == E) return; // Print out the -> operator if possible... Constant *CI = dyn_cast(*I); if (CI && CI->isNullValue() && I+1 != E && (*(I+1))->getType() == Type::UByteTy) { ++I; Out << "->field" << cast(*I)->getValue(); ++I; } for (; I != E; ++I) if ((*I)->getType() == Type::UIntTy) { Out << "["; CW.writeOperand(*I); Out << "]"; } else { Out << ".field" << cast(*I)->getValue(); } } void CInstPrintVisitor::visitLoadInst(LoadInst *I) { outputLValue(I); printIndexingExpr(I); Out << ";\n"; } void CInstPrintVisitor::visitStoreInst(StoreInst *I) { Out << " "; printIndexingExpr(I); Out << " = "; CW.writeOperand(I->getOperand(0)); Out << ";\n"; } void CInstPrintVisitor::visitGetElementPtrInst(GetElementPtrInst *I) { outputLValue(I); Out << "&"; printIndexingExpr(I); Out << ";\n"; } void CInstPrintVisitor::visitNot(GenericUnaryInst *I) { outputLValue(I); Out << "~"; CW.writeOperand(I->getOperand(0)); Out << ";\n"; } void CInstPrintVisitor::visitBinaryOperator(Instruction *I) { // binary instructions, shift instructions, setCond instructions. outputLValue(I); if (isa(I->getType())) { Out << "("; CW.printType(I->getType()); Out << ")"; } if (isa(I->getType())) Out << "(long long)"; CW.writeOperand(I->getOperand(0)); switch (I->getOpcode()) { case Instruction::Add: Out << " + "; break; case Instruction::Sub: Out << " - "; break; case Instruction::Mul: Out << "*"; break; case Instruction::Div: Out << "/"; break; case Instruction::Rem: Out << "%"; break; case Instruction::And: Out << " & "; break; case Instruction::Or: Out << " | "; break; case Instruction::Xor: Out << " ^ "; break; case Instruction::SetEQ: Out << " == "; break; case Instruction::SetNE: Out << " != "; break; case Instruction::SetLE: Out << " <= "; break; case Instruction::SetGE: Out << " >= "; break; case Instruction::SetLT: Out << " < "; break; case Instruction::SetGT: Out << " > "; break; case Instruction::Shl : Out << " << "; break; case Instruction::Shr : Out << " >> "; break; default: cerr << "Invalid operator type!" << I; abort(); } if (isa(I->getType())) Out << "(long long)"; CW.writeOperand(I->getOperand(1)); Out << ";\n"; } /* END : CInstPrintVisitor implementation */ // We dont want identifier names with ., space, - in them. // So we replace them with _ static string makeNameProper(string x) { string tmp; for (string::iterator sI = x.begin(), sEnd = x.end(); sI != sEnd; sI++) switch (*sI) { case '.': tmp += "d_"; break; case ' ': tmp += "s_"; break; case '-': tmp += "D_"; break; case '_': tmp += "__"; break; default: tmp += *sI; } return tmp; } string CWriter::getValueName(const Value *V) { if (V->hasName()) { // Print out the label if it exists... if (isa(V)) // Do not mangle globals... return makeNameProper(V->getName()); return "l" + utostr(V->getType()->getUniqueID()) + "_" + makeNameProper(V->getName()); } int Slot = Table.getValSlot(V); assert(Slot >= 0 && "Invalid value!"); return "ltmp_" + itostr(Slot) + "_" + utostr(V->getType()->getUniqueID()); } void CWriter::printModule(Module *M) { // printing stdlib inclusion // Out << "#include \n"; // get declaration for alloca Out << "/* Provide Declarations */\n" << "#include \n\n" // Provide a definition for null if one does not already exist. << "#ifndef NULL\n#define NULL 0\n#endif\n\n" << "typedef unsigned char bool;\n" << "\n\n/* Global Symbols */\n"; // Loop over the symbol table, emitting all named constants... if (M->hasSymbolTable()) printSymbolTable(*M->getSymbolTable()); Out << "\n\n/* Global Data */\n"; for (Module::const_giterator I = M->gbegin(), E = M->gend(); I != E; ++I) { GlobalVariable *GV = *I; if (GV->hasInternalLinkage()) Out << "static "; printTypeVar(GV->getType()->getElementType(), getValueName(GV)); if (GV->hasInitializer()) { Out << " = " ; writeOperand(GV->getInitializer()); } Out << ";\n"; } // First output all the declarations of the functions as C requires Functions // be declared before they are used. // Out << "\n\n/* Function Declarations */\n"; for_each(M->begin(), M->end(), bind_obj(this, &CWriter::printFunctionDecl)); // Output all of the functions... Out << "\n\n/* Function Bodies */\n"; for_each(M->begin(), M->end(), bind_obj(this, &CWriter::printFunction)); } // printSymbolTable - Run through symbol table looking for named constants // if a named constant is found, emit it's declaration... // Assuming that symbol table has only types and constants. void CWriter::printSymbolTable(const SymbolTable &ST) { for (SymbolTable::const_iterator TI = ST.begin(); TI != ST.end(); ++TI) { SymbolTable::type_const_iterator I = ST.type_begin(TI->first); SymbolTable::type_const_iterator End = ST.type_end(TI->first); for (; I != End; ++I) if (const Type *Ty = dyn_cast(I->second)) { string Name = "struct l_" + I->first; Out << Name << ";\n"; TypeNames.insert(std::make_pair(Ty, Name)); } } Out << "\n"; for (SymbolTable::const_iterator TI = ST.begin(); TI != ST.end(); ++TI) { SymbolTable::type_const_iterator I = ST.type_begin(TI->first); SymbolTable::type_const_iterator End = ST.type_end(TI->first); for (; I != End; ++I) { const Value *V = I->second; if (const Type *Ty = dyn_cast(V)) { string Name = "l_" + I->first; if (isa(Ty)) Name = "struct " + Name; else Out << "typedef "; Out << calcTypeNameVar(Ty, TypeNames, Name, true) << ";\n"; } } } } // printFunctionDecl - Print function declaration // void CWriter::printFunctionDecl(const Function *F) { printFunctionSignature(F); Out << ";\n"; } void CWriter::printFunctionSignature(const Function *F) { if (F->hasInternalLinkage()) Out << "static "; // Loop over the arguments, printing them... const FunctionType *FT = cast(F->getFunctionType()); // Print out the return type and name... printType(F->getReturnType()); Out << getValueName(F) << "("; if (!F->isExternal()) { if (!F->getArgumentList().empty()) { printTypeVar(F->getArgumentList().front()->getType(), getValueName(F->getArgumentList().front())); for (Function::ArgumentListType::const_iterator I = F->getArgumentList().begin()+1, E = F->getArgumentList().end(); I != E; ++I) { Out << ", "; printTypeVar((*I)->getType(), getValueName(*I)); } } } else { // Loop over the arguments, printing them... 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 (FT->isVarArg()) { if (FT->getParamTypes().size()) Out << ", "; Out << "..."; // Output varargs portion of signature! } Out << ")"; } void CWriter::printFunction(Function *F) { if (F->isExternal()) return; Table.incorporateFunction(F); printFunctionSignature(F); Out << " {\n"; // Process each of the basic blocks, gather information and call the // output methods on the CLocalVars and Function* objects. // gather local variable information for each basic block InstLocalVarsVisitor ILV(*this); ILV.visit(F); // print the local variables // we assume that every local variable is alloca'ed in the C code. std::map &locals = ILV.CLV.LocalVars; map::iterator iter; for (iter = locals.begin(); iter != locals.end(); ++iter) { VarListType::iterator listiter; for (listiter = iter->second.begin(); listiter != iter->second.end(); ++listiter) { Out << " "; printTypeVar(iter->first, *listiter); Out << ";\n"; } } // print the basic blocks for (Function::iterator I = F->begin(), E = F->end(); I != E; ++I) { BasicBlock *BB = *I, *Prev = I != F->begin() ? *(I-1) : 0; // Don't print the label for the basic block if there are no uses, or if the // only terminator use is the precessor basic block's terminator. We have // to scan the use list because PHI nodes use basic blocks too but do not // require a label to be generated. // bool NeedsLabel = false; for (Value::use_iterator UI = BB->use_begin(), UE = BB->use_end(); UI != UE; ++UI) if (TerminatorInst *TI = dyn_cast(*UI)) if (TI != Prev->getTerminator()) { NeedsLabel = true; break; } if (NeedsLabel) Out << getValueName(BB) << ":\n"; // Output all of the instructions in the basic block... // print the basic blocks CInstPrintVisitor CIPV(*this, Table, Out); CIPV.visit(BB); } Out << "}\n\n"; Table.purgeFunction(); } void CWriter::writeOperand(const Value *Operand) { if (isa(Operand)) Out << "(&"; // Global variables are references as their addresses by llvm if (Operand->hasName()) { Out << getValueName(Operand); } else if (const Constant *CPV = dyn_cast(Operand)) { if (isa(CPV)) { Out << "(("; printTypeVar(CPV->getType(), ""); Out << ")NULL)"; } else Out << getConstStrValue(CPV); } else { int Slot = Table.getValSlot(Operand); assert(Slot >= 0 && "Malformed LLVM!"); Out << "ltmp_" << Slot << "_" << Operand->getType()->getUniqueID(); } if (isa(Operand)) Out << ")"; } //===----------------------------------------------------------------------===// // External Interface declaration //===----------------------------------------------------------------------===// void WriteToC(const Module *M, ostream &Out) { assert(M && "You can't write a null module!!"); SlotCalculator SlotTable(M, false); CWriter W(Out, SlotTable, M); W.write((Module*)M); Out.flush(); }