//------------------------------------------------------------------------------ // CLING - the C++ LLVM-based InterpreterG :) // version: $Id$ // author: Vassil Vassilev //------------------------------------------------------------------------------ #include "cling/Interpreter/ValuePrinter.h" #include "cling/Interpreter/CValuePrinter.h" #include "cling/Interpreter/ValuePrinterInfo.h" #include "cling/Interpreter/StoredValueRef.h" #include "cling/Interpreter/Value.h" #include "clang/AST/ASTContext.h" #include "clang/AST/Decl.h" #include "clang/AST/DeclCXX.h" #include "clang/AST/Expr.h" #include "clang/AST/Type.h" #include "llvm/Support/raw_ostream.h" #include // Implements the CValuePrinter interface. extern "C" void cling_PrintValue(void* /*clang::Expr**/ E, void* /*clang::ASTContext**/ C, const void* value) { clang::Expr* Exp = (clang::Expr*)E; clang::ASTContext* Context = (clang::ASTContext*)C; cling::ValuePrinterInfo VPI(Exp, Context); cling::printValuePublic(llvm::outs(), value, value, VPI); cling::flushOStream(llvm::outs()); } static void StreamChar(llvm::raw_ostream& o, const char v) { o << '"' << v << "\"\n"; } static void StreamCharPtr(llvm::raw_ostream& o, const char* const v) { o << '"'; const char* p = v; for (;*p && p - v < 128; ++p) { o << *p; } if (*p) o << "\"...\n"; else o << "\"\n"; } static void StreamRef(llvm::raw_ostream& o, const void* v) { o <<"&" << v << "\n"; } static void StreamPtr(llvm::raw_ostream& o, const void* v) { o << v << "\n"; } static void StreamObj(llvm::raw_ostream& o, const void* v, const cling::ValuePrinterInfo& VPI) { const clang::Type* Ty = VPI.getExpr()->getType().getTypePtr(); if (clang::CXXRecordDecl* CXXRD = Ty->getAsCXXRecordDecl()) { const cling::Value* value = 0; if (CXXRD->getQualifiedNameAsString().compare("cling::StoredValueRef") == 0) { const cling::StoredValueRef* VR = (const cling::StoredValueRef*)v; if (VR->isValid()) { value = &VR->get(); } else { o << "<<>> "; } } else if (CXXRD->getQualifiedNameAsString().compare("cling::Value") == 0) { value = (const cling::Value*)v; } if (value) { if (value->isValid()) { o << "boxes ["; const clang::ASTContext& C = *VPI.getASTContext(); o << "(" << value->type.getAsString(C.getPrintingPolicy()) << ") "; clang::QualType valType = value->type.getDesugaredType(C); if (valType->isPointerType()) o << value->value.PointerVal; else if (valType->isFloatingType()) o << value->value.DoubleVal; else if (valType->isIntegerType()) o << value->value.IntVal.getSExtValue(); else if (valType->isBooleanType()) o << value->value.IntVal.getBoolValue(); o << "]\n"; return; } else o << "<<>> "; } } // if CXXRecordDecl // TODO: Print the object members. o << "@" << v << "\n"; } static void StreamValue(llvm::raw_ostream& o, const void* const p, const cling::ValuePrinterInfo& VPI) { clang::QualType Ty = VPI.getExpr()->getType(); const clang::ASTContext& C = *VPI.getASTContext(); Ty = Ty.getDesugaredType(C); if (const clang::BuiltinType *BT = llvm::dyn_cast(Ty.getCanonicalType())) { switch (BT->getKind()) { case clang::BuiltinType::Bool: if (*(const bool*)p) o << "true\n"; else o << "false\n"; break; case clang::BuiltinType::Char_U: case clang::BuiltinType::UChar: case clang::BuiltinType::Char_S: case clang::BuiltinType::SChar: StreamChar(o, *(const char*)p); break; case clang::BuiltinType::Short: o << *(const short*)p << "\n"; break; case clang::BuiltinType::UShort: o << *(const unsigned short*)p << "\n"; break; case clang::BuiltinType::Int: o << *(const int*)p << "\n"; break; case clang::BuiltinType::UInt: o << *(const unsigned int*)p << "\n"; break; case clang::BuiltinType::Long: o << *(const long*)p << "\n"; break; case clang::BuiltinType::ULong: o << *(const unsigned long*)p << "\n"; break; case clang::BuiltinType::LongLong: o << *(const long long*)p << "\n"; break; case clang::BuiltinType::ULongLong: o << *(const unsigned long long*)p << "\n"; break; case clang::BuiltinType::Float: o << *(const float*)p << "\n"; break; case clang::BuiltinType::Double: o << *(const double*)p << "\n"; break; default: StreamObj(o, p, VPI); } } else if (Ty.getAsString().compare("class std::basic_string") == 0) { StreamObj(o, p, VPI); o <<"c_str: "; StreamCharPtr(o, ((const char*) (*(const std::string*)p).c_str())); } else if (Ty->isEnumeralType()) { StreamObj(o, p, VPI); clang::EnumDecl* ED = Ty->getAs()->getDecl(); uint64_t value = *(const uint64_t*)p; bool IsFirst = true; llvm::APSInt ValAsAPSInt = C.MakeIntValue(value, Ty); for (clang::EnumDecl::enumerator_iterator I = ED->enumerator_begin(), E = ED->enumerator_end(); I != E; ++I) { if (I->getInitVal() == ValAsAPSInt) { if (!IsFirst) { o << " ? "; } o << "(" << I->getQualifiedNameAsString() << ")"; IsFirst = false; } } o << " : (int) " << ValAsAPSInt.toString(/*Radix = */10) << "\n"; } else if (Ty->isReferenceType()) StreamRef(o, p); else if (Ty->isPointerType()) { clang::QualType PointeeTy = Ty->getPointeeType(); if (PointeeTy->isCharType()) StreamCharPtr(o, (const char*)p); else StreamPtr(o, p); } else StreamObj(o, p, VPI); } namespace cling { void printValuePublicDefault(llvm::raw_ostream& o, const void* const p, const ValuePrinterInfo& VPI) { const clang::Expr* E = VPI.getExpr(); o << "("; o << E->getType().getAsString(); if (E->isRValue()) // show the user that the var cannot be changed o << " const"; o << ") "; StreamValue(o, p, VPI); } void flushOStream(llvm::raw_ostream& o) { o.flush(); } } // end namespace cling