99dfb99f35
For example, when we have enum e {e1=1}; and we do e1 on the prompt we got the address of e1 and we printed it out. However, this doesn't make sence because we cannot use it later, because it is destroyed. git-svn-id: http://root.cern.ch/svn/root/trunk@48850 27541ba8-7e3a-0410-8455-c3a389f83636
334 lines
11 KiB
C++
334 lines
11 KiB
C++
//------------------------------------------------------------------------------
|
|
// CLING - the C++ LLVM-based InterpreterG :)
|
|
// version: $Id$
|
|
// author: Vassil Vassilev <vasil.georgiev.vasilev@cern.ch>
|
|
//------------------------------------------------------------------------------
|
|
|
|
#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 "llvm/ExecutionEngine/GenericValue.h"
|
|
|
|
#include <string>
|
|
#include <cstdio>
|
|
|
|
// Fragment copied from LLVM's raw_ostream.cpp
|
|
#if defined(_MSC_VER)
|
|
#ifndef STDIN_FILENO
|
|
# define STDIN_FILENO 0
|
|
#endif
|
|
#ifndef STDOUT_FILENO
|
|
# define STDOUT_FILENO 1
|
|
#endif
|
|
#ifndef STDERR_FILENO
|
|
# define STDERR_FILENO 2
|
|
#endif
|
|
#else
|
|
//#if defined(HAVE_UNISTD_H)
|
|
# include <unistd.h>
|
|
//#endif
|
|
#endif
|
|
|
|
using namespace cling;
|
|
|
|
// 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;
|
|
ValuePrinterInfo VPI(Exp->getType(), Context);
|
|
|
|
// We need stream that doesn't close its file descriptor, thus we are not
|
|
// using llvm::outs. Keeping file descriptor open we will be able to use
|
|
// the results in pipes (Savannah #99234).
|
|
llvm::raw_fd_ostream outs (STDOUT_FILENO, /*ShouldClose*/false);
|
|
|
|
printValuePublic(outs, value, value, VPI);
|
|
flushOStream(outs);
|
|
}
|
|
|
|
|
|
static void StreamValue(llvm::raw_ostream& o, const void* const p,
|
|
const ValuePrinterInfo& VPI,
|
|
const char* Sep = "\n");
|
|
|
|
static void StreamChar(llvm::raw_ostream& o, const char v,
|
|
const char* Sep = "\n") {
|
|
o << '"' << v << "\"" << Sep;
|
|
}
|
|
|
|
static void StreamCharPtr(llvm::raw_ostream& o, const char* const v,
|
|
const char* Sep = "\n") {
|
|
if (!v) {
|
|
o << "<<<NULL>>>" << Sep;
|
|
return;
|
|
}
|
|
o << '"';
|
|
const char* p = v;
|
|
for (;*p && p - v < 128; ++p) {
|
|
o << *p;
|
|
}
|
|
if (*p) o << "\"..." << Sep;
|
|
else o << "\"" << Sep;
|
|
}
|
|
|
|
static void StreamRef(llvm::raw_ostream& o, const void* v,
|
|
const char* Sep = "\n") {
|
|
o <<"&" << v << Sep;
|
|
}
|
|
|
|
static void StreamPtr(llvm::raw_ostream& o, const void* v,
|
|
const char* Sep = "\n") {
|
|
o << v << Sep;
|
|
}
|
|
|
|
static void StreamArr(llvm::raw_ostream& o, const void* p,
|
|
const ValuePrinterInfo& VPI,
|
|
const char* Sep = "\n") {
|
|
const clang::QualType& Ty = VPI.getType();
|
|
clang::ASTContext& C = *VPI.getASTContext();
|
|
const clang::ArrayType* ArrTy = Ty->getAsArrayTypeUnsafe();
|
|
clang::QualType ElementTy = ArrTy->getElementType();
|
|
if (ElementTy->isCharType())
|
|
StreamCharPtr(o, (const char*)p);
|
|
else if (Ty->isConstantArrayType()) {
|
|
// Stream a constant array by streaming up to 5 elements.
|
|
const clang::ConstantArrayType* CArrTy
|
|
= C.getAsConstantArrayType(Ty);
|
|
const llvm::APInt& APSize = CArrTy->getSize();
|
|
size_t ElBytes = C.getTypeSize(ElementTy) / C.getCharWidth();
|
|
size_t Size = (size_t)APSize.getZExtValue();
|
|
o << "{ ";
|
|
ValuePrinterInfo ElVPI(ElementTy, &C);
|
|
for (size_t i = 0; i < Size; ++i) {
|
|
StreamValue(o, ((const char*)p) + i * ElBytes, ElVPI, "");
|
|
if (i + 1 < Size) {
|
|
if (i == 4) {
|
|
o << "...";
|
|
break;
|
|
}
|
|
else o << ", ";
|
|
}
|
|
}
|
|
o << " }" << Sep;
|
|
} else
|
|
StreamPtr(o, p, Sep);
|
|
}
|
|
|
|
static void StreamFunction(llvm::raw_ostream& o, const void* addr,
|
|
ValuePrinterInfo VPI, const char* Sep = "\n") {
|
|
o << "Function @" << addr << '\n';
|
|
|
|
const clang::DeclRefExpr* DeclRefExp
|
|
= llvm::dyn_cast_or_null<clang::DeclRefExpr>(VPI.getExpr());
|
|
const clang::FunctionDecl* FD = 0;
|
|
if (DeclRefExp)
|
|
FD = llvm::dyn_cast_or_null<clang::FunctionDecl>(DeclRefExp->getDecl());
|
|
if (FD) {
|
|
clang::SourceRange SRange = FD->getSourceRange();
|
|
const char* cBegin = 0;
|
|
const char* cEnd = 0;
|
|
bool Invalid;
|
|
if (SRange.isValid()) {
|
|
clang::SourceManager& SM = VPI.getASTContext()->getSourceManager();
|
|
clang::SourceLocation LocBegin = SRange.getBegin();
|
|
LocBegin = SM.getExpansionRange(LocBegin).first;
|
|
o << " at " << SM.getFilename(LocBegin);
|
|
unsigned LineNo = SM.getSpellingLineNumber(LocBegin, &Invalid);
|
|
if (!Invalid)
|
|
o << ':' << LineNo;
|
|
o << ":\n";
|
|
bool Invalid = false;
|
|
cBegin = SM.getCharacterData(LocBegin, &Invalid);
|
|
if (!Invalid) {
|
|
clang::SourceLocation LocEnd = SRange.getEnd();
|
|
LocEnd = SM.getExpansionRange(LocEnd).second;
|
|
cEnd = SM.getCharacterData(LocEnd, &Invalid);
|
|
if (Invalid)
|
|
cBegin = 0;
|
|
} else {
|
|
cBegin = 0;
|
|
}
|
|
}
|
|
if (cBegin && cEnd && cEnd > cBegin && cEnd - cBegin < 16 * 1024) {
|
|
o << llvm::StringRef(cBegin, cEnd - cBegin + 1);
|
|
} else {
|
|
const clang::FunctionDecl* FDef;
|
|
if (FD->hasBody(FDef))
|
|
FD = FDef;
|
|
FD->print(o);
|
|
//const clang::FunctionDecl* FD
|
|
// = llvm::cast<const clang::FunctionType>(Ty)->getDecl();
|
|
}
|
|
} else {
|
|
o << ":\n";
|
|
// type-based printing:
|
|
VPI.getType().print(o, VPI.getASTContext()->getPrintingPolicy());
|
|
}
|
|
// type-based print() never and decl-based print() sometimes does not include
|
|
// a final newline:
|
|
o << '\n';
|
|
}
|
|
|
|
static void StreamClingValue(llvm::raw_ostream& o, const Value* value,
|
|
clang::ASTContext& C, const char* Sep = "\n") {
|
|
if (!value || !value->isValid()) {
|
|
o << "<<<invalid>>> @" << value << Sep;
|
|
} else {
|
|
o << "boxes [";
|
|
o << "("
|
|
<< value->getClangType().getAsString(C.getPrintingPolicy())
|
|
<< ") ";
|
|
clang::QualType valType = value->getClangType().getDesugaredType(C);
|
|
if (valType->isFloatingType())
|
|
o << value->getGV().DoubleVal;
|
|
else if (valType->isIntegerType())
|
|
o << value->getGV().IntVal.getSExtValue();
|
|
else if (valType->isBooleanType())
|
|
o << value->getGV().IntVal.getBoolValue();
|
|
else
|
|
StreamValue(o, value->getGV().PointerVal,
|
|
ValuePrinterInfo(valType, &C), "");
|
|
o << "]" << Sep;
|
|
}
|
|
}
|
|
|
|
void cling::StreamStoredValueRef(llvm::raw_ostream& o, const StoredValueRef* VR,
|
|
clang::ASTContext& C,
|
|
const char* Sep /*= "\n"*/) {
|
|
|
|
if (VR->isValid()) {
|
|
StreamClingValue(o, &VR->get(), C, Sep);
|
|
} else {
|
|
o << "<<<invalid>>> @" << VR << Sep;
|
|
}
|
|
}
|
|
|
|
static void StreamObj(llvm::raw_ostream& o, const void* v,
|
|
const ValuePrinterInfo& VPI,
|
|
const char* Sep = "\n") {
|
|
const clang::Type* Ty = VPI.getType().getTypePtr();
|
|
if (clang::CXXRecordDecl* CXXRD = Ty->getAsCXXRecordDecl()) {
|
|
std::string QualName = CXXRD->getQualifiedNameAsString();
|
|
if (QualName == "cling::StoredValueRef"){
|
|
StreamStoredValueRef(o, (const StoredValueRef*)v,
|
|
*VPI.getASTContext(), Sep);
|
|
return;
|
|
} else if (QualName == "cling::Value") {
|
|
StreamClingValue(o, (const Value*)v, *VPI.getASTContext(), Sep);
|
|
return;
|
|
}
|
|
} // if CXXRecordDecl
|
|
|
|
// TODO: Print the object members.
|
|
o << "@" << v << Sep;
|
|
}
|
|
|
|
static void StreamValue(llvm::raw_ostream& o, const void* const p,
|
|
const ValuePrinterInfo& VPI,
|
|
const char* Sep /*= "\n"*/) {
|
|
clang::ASTContext& C = *VPI.getASTContext();
|
|
clang::QualType Ty = VPI.getType().getDesugaredType(C);
|
|
if (const clang::BuiltinType *BT
|
|
= llvm::dyn_cast<clang::BuiltinType>(Ty.getCanonicalType())) {
|
|
switch (BT->getKind()) {
|
|
case clang::BuiltinType::Bool:
|
|
if (*(const bool*)p) o << "true" << Sep;
|
|
else o << "false" << Sep; 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 << Sep; break;
|
|
case clang::BuiltinType::UShort:
|
|
o << *(const unsigned short*)p << Sep;
|
|
break;
|
|
case clang::BuiltinType::Int: o << *(const int*)p << Sep; break;
|
|
case clang::BuiltinType::UInt:
|
|
o << *(const unsigned int*)p << Sep;
|
|
break;
|
|
case clang::BuiltinType::Long: o << *(const long*)p << Sep; break;
|
|
case clang::BuiltinType::ULong:
|
|
o << *(const unsigned long*)p << Sep;
|
|
break;
|
|
case clang::BuiltinType::LongLong:
|
|
o << *(const long long*)p << Sep;
|
|
break;
|
|
case clang::BuiltinType::ULongLong:
|
|
o << *(const unsigned long long*)p << Sep;
|
|
break;
|
|
case clang::BuiltinType::Float: o << *(const float*)p << Sep; break;
|
|
case clang::BuiltinType::Double: o << *(const double*)p << Sep; break;
|
|
default:
|
|
StreamObj(o, p, ValuePrinterInfo(Ty, &C), Sep);
|
|
}
|
|
}
|
|
else if (Ty.getAsString().compare("class std::basic_string<char>") == 0) {
|
|
StreamObj(o, p, ValuePrinterInfo(Ty, &C), Sep);
|
|
if (!Sep[0]) o << " "; // force a space
|
|
o <<"c_str: ";
|
|
StreamCharPtr(o, ((const char*) (*(const std::string*)p).c_str()), Sep);
|
|
}
|
|
else if (Ty->isEnumeralType()) {
|
|
clang::EnumDecl* ED = Ty->getAs<clang::EnumType>()->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) << Sep;
|
|
}
|
|
else if (Ty->isReferenceType())
|
|
StreamRef(o, p, Sep);
|
|
else if (Ty->isPointerType()) {
|
|
clang::QualType PointeeTy = Ty->getPointeeType();
|
|
if (PointeeTy->isCharType())
|
|
StreamCharPtr(o, (const char*)p, Sep);
|
|
else
|
|
StreamPtr(o, p, Sep);
|
|
}
|
|
else if (Ty->isArrayType())
|
|
StreamArr(o, p, ValuePrinterInfo(Ty, &C), Sep);
|
|
else if (Ty->isFunctionType())
|
|
StreamFunction(o, p, VPI, Sep);
|
|
else
|
|
StreamObj(o, p, ValuePrinterInfo(Ty, &C), Sep);
|
|
}
|
|
|
|
namespace cling {
|
|
void printValuePublicDefault(llvm::raw_ostream& o, const void* const p,
|
|
const ValuePrinterInfo& VPI) {
|
|
o << "(";
|
|
o << VPI.getType().getAsString();
|
|
o << ") ";
|
|
StreamValue(o, p, VPI);
|
|
}
|
|
|
|
void flushOStream(llvm::raw_ostream& o) {
|
|
// We want to keep stdout and o in sync if o is different from stdout.
|
|
fflush(stdout);
|
|
o.flush();
|
|
}
|
|
|
|
} // end namespace cling
|