cling/lib/Interpreter/ValuePrinter.cpp
2023-02-03 07:29:07 +01:00

1006 lines
35 KiB
C++

//------------------------------------------------------------------------------
// CLING - the C++ LLVM-based InterpreterG :)
// author: Vassil Vassilev <vasil.georgiev.vasilev@cern.ch>
//
// This file is dual-licensed: you can choose to license it under the University
// of Illinois Open Source License or the GNU Lesser General Public License. See
// LICENSE.TXT for details.
//------------------------------------------------------------------------------
#include "cling/Interpreter/Value.h"
#include "EnterUserCodeRAII.h"
#include "cling/Interpreter/CValuePrinter.h"
#include "cling/Interpreter/Interpreter.h"
#include "cling/Interpreter/LookupHelper.h"
#include "cling/Interpreter/Transaction.h"
#include "cling/Interpreter/Value.h"
#include "cling/Interpreter/Visibility.h"
#include "cling/Utils/AST.h"
#include "cling/Utils/Casting.h"
#include "cling/Utils/Output.h"
#include "cling/Utils/Validation.h"
#include "clang/AST/ASTContext.h"
#include "clang/AST/Decl.h"
#include "clang/AST/DeclCXX.h"
#include "clang/AST/Expr.h"
#include "clang/AST/ExprCXX.h"
#include "clang/AST/GlobalDecl.h"
#include "clang/AST/NestedNameSpecifier.h"
#include "clang/AST/Stmt.h"
#include "clang/AST/Type.h"
#include "clang/Frontend/CompilerInstance.h"
#include "clang/Lex/Preprocessor.h"
#include "clang/Parse/Parser.h"
#include "clang/Sema/Lookup.h"
#include "clang/Sema/Sema.h"
#include "llvm/Support/Format.h"
#include <locale>
#include <string>
// GCC 4.x doesn't have the proper UTF-8 conversion routines. So use the
// LLVM conversion routines (which require a buffer 4x string length).
#if !defined(__GLIBCXX__) || (__GNUC__ >= 5)
#include <codecvt>
#else
#define LLVM_UTF8
#include "llvm/Support/ConvertUTF.h"
#endif
using namespace cling;
// Implements the CValuePrinter interface.
extern "C" CLING_LIB_EXPORT
void cling_PrintValue(void * /*cling::Value* V*/) {
//Value* value = (Value*)V;
//std::string typeStr = printTypeInternal(*value);
//std::string valueStr = printValueInternal(*value);
}
// Exported for RuntimePrintValue.h
namespace cling {
namespace valuePrinterInternal {
CLING_LIB_EXPORT
extern const char* const kEmptyCollection = "{}";
struct OpaqueString{};
/// Assign a string to a string*.
void AssignToStringFromStringPtr(OpaqueString* to, const OpaqueString& from) {
std::string* sTo = reinterpret_cast<std::string*>(to);
const std::string& sFrom = reinterpret_cast<const std::string&>(from);
*sTo = sFrom;
}
}
}
namespace {
const static char
* const kNullPtrStr = "nullptr",
* const kNullPtrTStr = "nullptr_t",
* const kTrueStr = "true",
* const kFalseStr = "false",
* const kInvalidAddr = " <invalid memory address>";
static std::string enclose(std::string Mid, const char* Begin,
const char* End, size_t Hint = 0) {
Mid.reserve(Mid.size() + Hint ? Hint : (::strlen(Begin) + ::strlen(End)));
Mid.insert(0, Begin);
Mid.append(End);
return Mid;
}
static std::string enclose(const clang::QualType& Ty, clang::ASTContext& C,
const char* Begin = "(", const char* End = "*)",
size_t Hint = 3) {
return enclose(cling::utils::TypeName::GetFullyQualifiedName(Ty, C),
Begin, End, Hint);
}
static std::string printDeclType(const clang::QualType& QT,
const clang::NamedDecl* D) {
if (!QT.hasQualifiers())
return D->getQualifiedNameAsString();
return QT.getQualifiers().getAsString() + " " + D->getQualifiedNameAsString();
}
static std::string printQualType(clang::ASTContext& Ctx, clang::QualType QT) {
using namespace clang;
const QualType QTNonRef = QT.getNonReferenceType();
PrintingPolicy Policy(Ctx.getPrintingPolicy());
// Print the Allocator in STL containers, for instance.
Policy.SuppressDefaultTemplateArgs = false;
// DefinitionShadower: do not prepend `__cling_N5xxx::` to qualified names
Policy.SuppressUnwrittenScope = true;
// Print 'a<b<c> >' rather than 'a<b<c>>'.
Policy.SplitTemplateClosers = true;
class LocalPrintingPolicyRAII {
public:
LocalPrintingPolicyRAII(ASTContext& Ctx, PrintingPolicy& PPol)
: Context(Ctx), Policy(Ctx.getPrintingPolicy()) { Context.setPrintingPolicy(PPol); }
~LocalPrintingPolicyRAII() { Context.setPrintingPolicy(Policy); }
private:
ASTContext& Context;
PrintingPolicy Policy;
} RAII(Ctx, Policy);
std::string ValueTyStr("(");
if (const TagType *TTy = dyn_cast<TagType>(QTNonRef))
ValueTyStr += printDeclType(QTNonRef, TTy->getDecl());
else if (const RecordType *TRy = dyn_cast<RecordType>(QTNonRef))
ValueTyStr += printDeclType(QTNonRef, TRy->getDecl());
else {
const QualType QTCanon = QTNonRef.getCanonicalType();
if (QTCanon->isBuiltinType() && !QTNonRef->isFunctionPointerType()
&& !QTNonRef->isMemberPointerType()) {
ValueTyStr += QTCanon.getAsString(Ctx.getPrintingPolicy());
}
else if (const TypedefType* TDTy = dyn_cast<TypedefType>(QTNonRef)) {
// FIXME: TemplateSpecializationType & SubstTemplateTypeParmType checks are
// predominately to get STL containers to print nicer and might be better
// handled in GetFullyQualifiedName.
//
// std::vector<Type>::iterator is a TemplateSpecializationType
// std::vector<Type>::value_type is a SubstTemplateTypeParmType
//
QualType SSDesugar = TDTy->getLocallyUnqualifiedSingleStepDesugaredType();
if (dyn_cast<SubstTemplateTypeParmType>(SSDesugar))
ValueTyStr += utils::TypeName::GetFullyQualifiedName(QTCanon, Ctx);
else if (dyn_cast<TemplateSpecializationType>(SSDesugar))
ValueTyStr += utils::TypeName::GetFullyQualifiedName(QTNonRef, Ctx);
else
ValueTyStr += printDeclType(QTNonRef, TDTy->getDecl());
}
else
ValueTyStr += utils::TypeName::GetFullyQualifiedName(QTNonRef, Ctx);
}
if (QT->isReferenceType())
ValueTyStr += " &";
return ValueTyStr + ")";
}
static std::string printAddress(const void* Ptr, const char Prfx = 0) {
if (!Ptr)
return kNullPtrStr;
cling::smallstream Strm;
if (Prfx)
Strm << Prfx;
Strm << Ptr;
if (!utils::isAddressValid(Ptr))
Strm << kInvalidAddr;
return Strm.str().str();
}
} // anonymous namespace
namespace cling {
// General fallback - prints the address
CLING_LIB_EXPORT
std::string printValue(const void *ptr) {
return printAddress(ptr, '@');
}
// void pointer
CLING_LIB_EXPORT
std::string printValue(const void **ptr) {
return printAddress(*ptr);
}
// Bool
CLING_LIB_EXPORT
std::string printValue(const bool *val) {
return *val ? kTrueStr : kFalseStr;
}
// Chars
static std::string printOneChar(char Val,
const std::locale& Locale = std::locale()) {
llvm::SmallString<128> Buf;
llvm::raw_svector_ostream Strm(Buf);
Strm << "'";
if (!std::isprint(Val, Locale)) {
switch (std::isspace(Val, Locale) ? Val : 0) {
case '\t': Strm << "\\t"; break;
case '\n': Strm << "\\n"; break;
case '\r': Strm << "\\r"; break;
case '\f': Strm << "\\f"; break;
case '\v': Strm << "\\v"; break;
default:
Strm << llvm::format_hex(uint64_t(Val)&0xff, 4);
}
}
else
Strm << Val;
Strm << "'";
return Strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const char *val) {
return printOneChar(*val);
}
CLING_LIB_EXPORT
std::string printValue(const signed char *val) {
return printOneChar(*val);
}
CLING_LIB_EXPORT
std::string printValue(const unsigned char *val) {
return printOneChar(*val);
}
// Ints
CLING_LIB_EXPORT
std::string printValue(const short *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const unsigned short *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const int *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const unsigned int *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const long *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const unsigned long *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const long long *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const unsigned long long *val) {
cling::smallstream strm;
strm << *val;
return strm.str().str();
}
// Reals
CLING_LIB_EXPORT
std::string printValue(const float *val) {
cling::smallstream strm;
strm << llvm::format("%#.6g", *val) << 'f';
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const double *val) {
cling::smallstream strm;
strm << llvm::format("%#.8g", *val);
return strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const long double *val) {
cling::smallstream strm;
strm << llvm::format("%#.8Lg", *val) << 'L';
//strm << llvm::format("%Le", *val) << 'L';
return strm.str().str();
}
// Char pointers
static std::string printString(const char *const *Ptr, size_t N = 10000) {
// Assumption is this is a string.
// N is limit to prevent endless loop if Ptr is not really a string.
const char* Start = *Ptr;
if (!Start)
return kNullPtrStr;
const char* End = Start + N;
bool IsValid = utils::isAddressValid(Start);
if (IsValid) {
// If we're gonnd do this, better make sure the end is valid too
// FIXME: getpagesize() & GetSystemInfo().dwPageSize might be better
#if !defined(PAGE_SIZE)
enum { PAGE_SIZE = 1024 };
#endif
while (!(IsValid = utils::isAddressValid(End)) && N > 1024) {
N -= PAGE_SIZE;
End = Start + N;
}
}
if (!IsValid) {
cling::smallstream Strm;
Strm << static_cast<const void*>(Start) << kInvalidAddr;
return Strm.str().str();
}
if (*Start == 0)
return "\"\"";
// Copy the bytes until we get a null-terminator
llvm::SmallString<1024> Buf;
llvm::raw_svector_ostream Strm(Buf);
Strm << "\"";
while (Start < End && *Start)
Strm << *Start++;
Strm << "\"";
return Strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const char *const *val) {
return printString(val);
}
CLING_LIB_EXPORT
std::string printValue(const char **val) {
return printString(val);
}
// std::string
CLING_LIB_EXPORT
std::string printValue(const std::string *val) {
return "\"" + *val + "\"";
}
static std::string quoteString(std::string Str, const char Prefix) {
// No wrap
if (!Prefix)
return Str;
// Quoted wrap
if (Prefix==1)
return enclose(std::move(Str), "\"", "\"", 2);
// Prefix quoted wrap
char Begin[3] = { Prefix, '"', 0 };
return enclose(std::move(Str), Begin, &Begin[1], 3);
}
static std::string quoteString(const char* Str, size_t N, const char Prefix) {
return quoteString(std::string(Str, Str[N-1] == 0 ? (N-1) : N), Prefix);
}
#ifdef LLVM_UTF8
using llvm::ConversionResult;
using llvm::ConversionFlags;
using llvm::lenientConversion;
using llvm::UTF8;
using llvm::UTF16;
using llvm::UTF32;
template <class T> struct CharTraits;
template <> struct CharTraits<char16_t> {
static ConversionResult convert(const char16_t** begin, const char16_t* end,
char** d, char* dEnd, ConversionFlags F ) {
return ConvertUTF16toUTF8(reinterpret_cast<const UTF16**>(begin),
reinterpret_cast<const UTF16*>(end),
reinterpret_cast<UTF8**>(d),
reinterpret_cast<UTF8*>(dEnd), F);
}
};
template <> struct CharTraits<char32_t> {
static ConversionResult convert(const char32_t** begin, const char32_t* end,
char** d, char* dEnd, ConversionFlags F ) {
return ConvertUTF32toUTF8(reinterpret_cast<const UTF32**>(begin),
reinterpret_cast<const UTF32*>(end),
reinterpret_cast<UTF8**>(d),
reinterpret_cast<UTF8*>(dEnd), F);
}
};
template <> struct CharTraits<wchar_t> {
static ConversionResult convert(const wchar_t** src, const wchar_t* srcEnd,
char** dst, char* dEnd, ConversionFlags F) {
switch (sizeof(wchar_t)) {
case sizeof(char16_t):
return CharTraits<char16_t>::convert(
reinterpret_cast<const char16_t**>(src),
reinterpret_cast<const char16_t*>(srcEnd),
dst, dEnd, F);
case sizeof(char32_t):
return CharTraits<char32_t>::convert(
reinterpret_cast<const char32_t**>(src),
reinterpret_cast<const char32_t*>(srcEnd),
dst, dEnd, F);
default: break;
}
llvm_unreachable("wchar_t conversion failure");
}
};
template <typename T>
static std::string encodeUTF8(const T* const Str, size_t N, const char Prfx) {
const T *Bgn = Str,
*End = Str + N;
std::string Result;
Result.resize(UNI_MAX_UTF8_BYTES_PER_CODE_POINT * N);
char *ResultPtr = &Result[0],
*ResultEnd = ResultPtr + Result.size();
CharTraits<T>::convert(&Bgn, End, &ResultPtr, ResultEnd, lenientConversion);
Result.resize(ResultPtr - &Result[0]);
return quoteString(std::move(Result), Prfx);
}
#else // !LLVM_UTF8
template <class T> struct CharTraits { typedef T value_type; };
#if defined(_WIN32) // Likely only to be needed when _MSC_VER < 19??
template <> struct CharTraits<char16_t> { typedef unsigned short value_type; };
template <> struct CharTraits<char32_t> { typedef unsigned int value_type; };
#endif
template <typename T>
static std::string encodeUTF8(const T* const Str, size_t N, const char Prfx) {
typedef typename CharTraits<T>::value_type value_type;
std::wstring_convert<std::codecvt_utf8_utf16<value_type>, value_type> Convert;
const value_type* Src = reinterpret_cast<const value_type*>(Str);
return quoteString(Convert.to_bytes(Src, Src + N), Prfx);
}
#endif // LLVM_UTF8
template <typename T>
std::string utf8Value(const T* const Str, size_t N, const char Prefix,
std::string (*Func)(const T* const Str, size_t N,
const char Prfx) ) {
if (!Str)
return kNullPtrStr;
if (N==0)
return printAddress(Str, '@');
// Drop the null terminator or else it will be encoded into the std::string.
return Func(Str, Str[N-1] == 0 ? (N-1) : N, Prefix);
}
// declaration: cling/Utils/UTF8.h & cling/Interpreter/RuntimePrintValue.h
template <class T>
CLING_LIB_EXPORT
std::string toUTF8(const T* const Str, size_t N, const char Prefix);
template <>
CLING_LIB_EXPORT
std::string toUTF8<char16_t>(const char16_t* const Str, size_t N,
const char Prefix) {
return utf8Value(Str, N, Prefix, encodeUTF8);
}
template <>
CLING_LIB_EXPORT
std::string toUTF8<char32_t>(const char32_t* const Str, size_t N,
const char Prefix) {
return utf8Value(Str, N, Prefix, encodeUTF8);
}
template <>
CLING_LIB_EXPORT
std::string toUTF8<wchar_t>(const wchar_t* const Str, size_t N,
const char Prefix) {
static_assert(sizeof(wchar_t) == sizeof(char16_t) ||
sizeof(wchar_t) == sizeof(char32_t), "Bad wchar_t size");
if (sizeof(wchar_t) == sizeof(char32_t))
return toUTF8(reinterpret_cast<const char32_t *>(Str), N, Prefix);
return toUTF8(reinterpret_cast<const char16_t *>(Str), N, Prefix);
}
template <>
CLING_LIB_EXPORT
std::string toUTF8<char>(const char* const Str, size_t N, const char Prefix) {
return utf8Value(Str, N, Prefix, quoteString);
}
template <typename T>
static std::string toUTF8(
const std::basic_string<T, std::char_traits<T>, std::allocator<T>>* Src,
const char Prefix) {
if (!Src)
return kNullPtrStr;
return encodeUTF8(Src->data(), Src->size(), Prefix);
}
CLING_LIB_EXPORT
std::string printValue(const std::u16string* Val) {
return toUTF8(Val, 'u');
}
CLING_LIB_EXPORT
std::string printValue(const std::u32string* Val) {
return toUTF8(Val, 'U');
}
CLING_LIB_EXPORT
std::string printValue(const std::wstring* Val) {
return toUTF8(Val, 'L');
}
// Unicode chars
template <typename T>
static std::string toUnicode(const T* Src, const char Prefix, char Esc = 0) {
if (!Src)
return kNullPtrStr;
if (!Esc)
Esc = Prefix;
llvm::SmallString<128> Buf;
llvm::raw_svector_ostream Strm(Buf);
Strm << Prefix << "'\\" << Esc
<< llvm::format_hex_no_prefix(unsigned(*Src), sizeof(T)*2) << "'";
return Strm.str().str();
}
CLING_LIB_EXPORT
std::string printValue(const char16_t *Val) {
return toUnicode(Val, 'u');
}
CLING_LIB_EXPORT
std::string printValue(const char32_t *Val) {
return toUnicode(Val, 'U');
}
CLING_LIB_EXPORT
std::string printValue(const wchar_t *Val) {
return toUnicode(Val, 'L', 'x');
}
} // end namespace cling
namespace {
static const char* BuildAndEmitVPWrapperBody(cling::Interpreter &Interp,
clang::Sema &S,
clang::ASTContext &Ctx,
clang::FunctionDecl *WrapperFD,
clang::QualType QT,
const void* ValPtr)
{
const clang::SourceLocation noSrcLoc;
clang::Sema::SynthesizedFunctionScope SemaFScope(S, WrapperFD);
clang::Parser::ParseScope parseScope(&Interp.getParser(),
clang::Scope::FnScope
| clang::Scope::BlockScope);
//Build the following AST (where `S` is `std::string`):
/*
`-FunctionDecl 0x7fc7d4812978 <col:22, col:46> col:24 XYZ_callPrintValue 'struct S (void)'
`-CompoundStmt 0x7fc7d4812ff8 <col:30, col:46>
`-ReturnStmt 0x7fc7d4812fe0 <col:32, col:43>
`-ExprWithCleanups 0x7fc7d4812fc8 <col:39, col:43> 'struct S'
`-CXXConstructExpr 0x7fc7d4812f90 <col:39, col:43> 'struct S' 'void (const struct S &) throw()' elidable
`-MaterializeTemporaryExpr 0x7fc7d4812f20 <col:39, col:43> 'const struct S' lvalue
`-ImplicitCastExpr 0x7fc7d4812f08 <col:39, col:43> 'const struct S' <NoOp>
`-CallExpr 0x7fc7d4812ad0 <col:39, col:43> 'struct S'
`-ImplicitCastExpr 0x7fc7d4812ab8 <col:39> 'struct S (*)(void)' <FunctionToPointerDecay>
`-DeclRefExpr 0x7fc7d4812a68 <col:39> 'struct S (void)' lvalue Function 0x7fc7d4812880 'bar' 'struct S (void)'
*/
clang::NamespaceDecl *clingNS = utils::Lookup::Namespace(&S, "cling");
assert(clingNS && "Cannot find namespace cling!");
clang::NestedNameSpecifierLocBuilder NNSLBld;
NNSLBld.MakeGlobal(Ctx, noSrcLoc);
NNSLBld.Extend(Ctx, clingNS, noSrcLoc, noSrcLoc);
clang::DeclarationName PVDN = S.PP.getIdentifierInfo("printValue");
clang::LookupResult R(S, PVDN, noSrcLoc, clang::Sema::LookupOrdinaryName);
// The subsequent lookup might deserialize or instantiate.
Interpreter::PushTransactionRAII ScopedT(&Interp);
S.LookupQualifiedName(R, clingNS);
clang::Expr *OverldExpr
= clang::UnresolvedLookupExpr::Create(Ctx, nullptr /*namingClass*/,
NNSLBld.getTemporary(),
clang::DeclarationNameInfo(PVDN, noSrcLoc),
/*RequiresADL*/false,
R.isOverloadedResult(),
R.begin(),
R.end());
// For `auto foo = bar;` decls, we are interested in the deduced type, i.e.
// AutoType 0x55e5ac848030 'int *' sugar
// `-PointerType 0x55e5ac847f70 'int *' << this type
// `-BuiltinType 0x55e5ab517420 'int'
if (auto AT = llvm::dyn_cast<clang::AutoType>(QT.getTypePtr())) {
if (AT->isDeduced())
QT = AT->getDeducedType();
}
if (auto PT = llvm::dyn_cast<clang::PointerType>(QT.getTypePtr())) {
// Normalize `X*` to `const void*`, invoke `printValue(const void**)`,
// unless it's a character string.
clang::QualType QTPointeeUnqual = PT->getPointeeType().getUnqualifiedType();
if (!Ctx.hasSameType(QTPointeeUnqual, Ctx.CharTy)
&& !Ctx.hasSameType(QTPointeeUnqual, Ctx.WCharTy)
&& !Ctx.hasSameType(QTPointeeUnqual, Ctx.Char16Ty)
&& !Ctx.hasSameType(QTPointeeUnqual, Ctx.Char32Ty)) {
QT = Ctx.getPointerType(Ctx.VoidTy.withConst());
}
} else if (auto RTy
= llvm::dyn_cast<clang::ReferenceType>(QT.getTypePtr())) {
// X& will be printed as X* (the pointer will be added below).
QT = RTy->getPointeeType();
// Val will be a X**, but we cast this to X*, so dereference here:
ValPtr = *(const void* const*)ValPtr;
}
// `cling::printValue()` takes the *address* of the value to be printed:
clang::QualType QTPtr = Ctx.getPointerType(QT);
clang::Expr *EVPArg
= utils::Synthesize::CStyleCastPtrExpr(&S, QTPtr, (uintptr_t)ValPtr);
llvm::SmallVector<clang::Expr*, 1> CallArgs;
CallArgs.push_back(EVPArg);
clang::ExprResult ExprVP
= S.ActOnCallExpr(S.getCurScope(), OverldExpr, noSrcLoc, CallArgs,
noSrcLoc);
if (ExprVP.isInvalid())
return "ERROR in cling's callPrintValue(): "
"cannot build printValue() expression";
clang::StmtResult RetStmt
= S.ActOnReturnStmt(noSrcLoc, ExprVP.get(), S.getCurScope());
if (RetStmt.isInvalid())
return "ERROR in cling's callPrintValue(): cannot build return expression";
auto *Body
= clang::CompoundStmt::Create(Ctx, {RetStmt.get()}, noSrcLoc, noSrcLoc);
WrapperFD->setBody(Body);
auto &Consumer = Interp.getCI()->getASTConsumer();
Consumer.HandleTopLevelDecl(clang::DeclGroupRef(WrapperFD));
return nullptr; // no error message.
}
static std::string callPrintValue(const Value& V, const void* Val) {
Interpreter *Interp = V.getInterpreter();
assert(Interp && "No cling::Interpreter!");
Value printValueV;
{
clang::ASTContext &Ctx = V.getASTContext();
const clang::SourceLocation noSrcLoc;
clang::Sema &S = Interp->getSema();
const clang::Decl *StdStringDecl
= utils::Lookup::Named(&S, "string",
utils::Lookup::Namespace(&S, "std"));
const auto* StdStringTD = clang::dyn_cast<clang::TypeDecl>(StdStringDecl);
assert(StdStringTD && "Cannot find type of std::string.");
std::string name;
Interp->createUniqueName(name);
name += "_callPrintValue";
clang::DeclarationName DeclName = &Ctx.Idents.get(name);
clang::QualType FnTy
= Ctx.getFunctionType(clang::QualType(StdStringTD->getTypeForDecl(), 0), {},
clang::FunctionProtoType::ExtProtoInfo());
clang::FunctionDecl *WrapperFD
= clang::FunctionDecl::Create(Ctx,
Ctx.getTranslationUnitDecl(),
noSrcLoc,
noSrcLoc,
DeclName,
FnTy,
Ctx.getTrivialTypeSourceInfo(FnTy),
/*StorageClass*/ clang::SC_None
//bool isInlineSpecified = false,
//bool hasWrittenPrototype = true,
//bool isConstexprSpecified = false
);
WrapperFD->setIsUsed();
const void* ValPtr = V.needsManagedAllocation() ? Val : &Val;
if (auto errmsg = BuildAndEmitVPWrapperBody(*Interp, S, Ctx, WrapperFD,
V.getType(), ValPtr))
return errmsg;
clang::GlobalDecl WrapperGD(WrapperFD);
if (void* addr = Interp->getAddressOfGlobal(WrapperGD)) {
auto funptr = cling::utils::VoidToFunctionPtr<std::string(*)()>(addr);
LockCompilationDuringUserCodeExecutionRAII LCDUCER(*Interp);
return funptr();
}
}
return "ERROR in cling's callPrintValue(): missing value string.";
}
template <typename T>
class HasExplicitPrintValue {
template <typename C,
typename = decltype(cling::printValue((C*)nullptr))>
static std::true_type test(int);
static std::false_type test(...);
public:
static constexpr bool value = decltype(test<T>(0))::value;
};
template <typename T> static
typename std::enable_if<!HasExplicitPrintValue<const T>::value, std::string>::type
executePrintValue(const Value& V, const T& val) {
return callPrintValue(V, &val);
}
template <typename T> static
typename std::enable_if<HasExplicitPrintValue<const T>::value, std::string>::type
executePrintValue(const Value& /*V*/, const T& val) {
return printValue(&val);
}
static std::string printEnumValue(const Value &V) {
cling::ostrstream enumString;
clang::ASTContext &C = V.getASTContext();
clang::QualType Ty = V.getType().getDesugaredType(C);
const clang::EnumType *EnumTy = Ty.getNonReferenceType()->getAs<clang::EnumType>();
assert(EnumTy && "ValuePrinter.cpp: ERROR, printEnumValue invoked for a non enum type.");
clang::EnumDecl *ED = EnumTy->getDecl();
uint64_t value = V.getULongLong();
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) {
enumString << " ? ";
}
enumString << "(" << I->getQualifiedNameAsString() << ")";
IsFirst = false;
}
}
enumString << " : " << printQualType(C, ED->getIntegerType()) << " "
<< toString(ValAsAPSInt, /*Radix = */10);
return enumString.str().str();
}
static std::string printFunctionValue(const Value &V, const void *ptr,
clang::QualType Ty) {
cling::largestream o;
o << "Function @" << ptr;
Interpreter &Interp = *const_cast<Interpreter *>(V.getInterpreter());
const Transaction *T = Interp.getLastWrapperTransaction();
if (!T)
return o.str().str();
if (clang::FunctionDecl *WrapperFD = T->getWrapperFD()) {
clang::ASTContext &C = V.getASTContext();
const clang::FunctionDecl *FD = nullptr;
// CE should be the setValueNoAlloc call expr.
if (const clang::CallExpr *CallE
= llvm::dyn_cast_or_null<clang::CallExpr>(
utils::Analyze::GetOrCreateLastExpr(WrapperFD,
/*foundAtPos*/0,
/*omitDS*/false,
&Interp.getSema()))) {
if (const clang::FunctionDecl *FDsetValue
= llvm::dyn_cast_or_null<clang::FunctionDecl>(CallE->getCalleeDecl())) {
if (FDsetValue->getNameAsString() == "setValueNoAlloc" &&
CallE->getNumArgs() == 5) {
const clang::Expr *Arg4 = CallE->getArg(4);
while (const clang::CastExpr *CastE
= clang::dyn_cast<clang::CastExpr>(Arg4))
Arg4 = CastE->getSubExpr();
if (const clang::DeclRefExpr *DeclRefExp
= llvm::dyn_cast<clang::DeclRefExpr>(Arg4))
FD = llvm::dyn_cast<clang::FunctionDecl>(DeclRefExp->getDecl());
}
}
}
if (FD) {
o << '\n';
clang::SourceRange SRange = FD->getSourceRange();
const char *cBegin = 0;
const char *cEnd = 0;
bool Invalid;
if (SRange.isValid()) {
clang::SourceManager &SM = C.getSourceManager();
clang::SourceLocation LocBegin = SRange.getBegin();
LocBegin = SM.getExpansionRange(LocBegin).getBegin();
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).getEnd();
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();
}
// type-based print() never and decl-based print() sometimes does not
// include a final newline:
o << '\n';
}
}
return o.str().str();
}
static std::string printStringType(const Value &V, const clang::Type* Type) {
switch (V.getInterpreter()->getLookupHelper().getStringType(Type)) {
case LookupHelper::kStdString:
return executePrintValue<std::string>(V, *(std::string*)V.getPtr());
case LookupHelper::kWCharString:
return executePrintValue<std::wstring>(V, *(std::wstring*)V.getPtr());
case LookupHelper::kUTF16Str:
return executePrintValue<std::u16string>(V, *(std::u16string*)V.getPtr());
case LookupHelper::kUTF32Str:
return executePrintValue<std::u32string>(V, *(std::u32string*)V.getPtr());
default:
break;
}
return "";
}
static std::string printUnpackedClingValue(const Value &V) {
// Find the Type for `std::string`. We are guaranteed to have that declared
// when this function is called; RuntimePrintValue.h #includes it.
const clang::ASTContext &C = V.getASTContext();
const clang::QualType Td = V.getType().getDesugaredType(C);
const clang::QualType Ty = Td.getNonReferenceType();
if (Ty->isNullPtrType()) {
// special case nullptr_t
return kNullPtrTStr;
} else if (Ty->isEnumeralType()) {
// special case enum printing, using compiled information
return printEnumValue(V);
} else if (Ty->isFunctionType()) {
// special case function printing, using compiled information
return printFunctionValue(V, &V, Ty);
} else if ((Ty->isPointerType() || Ty->isMemberPointerType()) && Ty->getPointeeType()->isFunctionProtoType()) {
// special case function printing, using compiled information
return printFunctionValue(V, V.getPtr(), Ty->getPointeeType());
} else if (clang::CXXRecordDecl *CXXRD = Ty->getAsCXXRecordDecl()) {
if (CXXRD->isLambda())
return printAddress(V.getPtr(), '@');
std::string Str = printStringType(V, CXXRD->getTypeForDecl());
if (!Str.empty())
return Str;
} else if (const clang::BuiltinType *BT
= llvm::dyn_cast<clang::BuiltinType>(Td.getCanonicalType().getTypePtr())) {
switch (BT->getKind()) {
case clang::BuiltinType::Bool:
return executePrintValue<bool>(V, V.castAs<bool>());
case clang::BuiltinType::Char_S:
return executePrintValue<signed char>(V, V.castAs<signed char>());
case clang::BuiltinType::SChar:
return executePrintValue<signed char>(V, V.castAs<signed char>());
case clang::BuiltinType::Short:
return executePrintValue<short>(V, V.castAs<short>());
case clang::BuiltinType::Int:
return executePrintValue<int>(V, V.castAs<int>());
case clang::BuiltinType::Long:
return executePrintValue<long>(V, V.castAs<long>());
case clang::BuiltinType::LongLong:
return executePrintValue<long long>(V, V.castAs<long long>());
case clang::BuiltinType::Char_U:
return executePrintValue<unsigned char>(V, V.castAs<unsigned char>());
case clang::BuiltinType::UChar:
return executePrintValue<unsigned char>(V, V.castAs<unsigned char>());
case clang::BuiltinType::UShort:
return executePrintValue<unsigned short>(V, V.castAs<unsigned short>());
case clang::BuiltinType::UInt:
return executePrintValue<unsigned int>(V, V.castAs<unsigned int>());
case clang::BuiltinType::ULong:
return executePrintValue<unsigned long>(V, V.castAs<unsigned long>());
case clang::BuiltinType::ULongLong:
return executePrintValue<unsigned long long>(V, V.castAs<unsigned long long>());
case clang::BuiltinType::Float:
return executePrintValue<float>(V, V.getFloat());
case clang::BuiltinType::Double:
return executePrintValue<double>(V, V.getDouble());
case clang::BuiltinType::LongDouble:
return executePrintValue<long double>(V, V.getLongDouble());
default:
break;
}
} else
assert(!Ty->isIntegralOrEnumerationType() && "Bad Type.");
if (!V.getPtr())
return kNullPtrStr;
// Print all the other cases by calling into runtime 'cling::printValue()'.
// Ty->isPointerType() || Ty->isReferenceType() || Ty->isArrayType()
// Ty->isObjCObjectPointerType()
return callPrintValue(V, V.getPtr());
}
} // anonymous namespace
namespace cling {
// cling::Value
CLING_LIB_EXPORT
std::string printValue(const Value *value) {
cling::smallstream strm;
if (value->isValid()) {
clang::ASTContext &C = value->getASTContext();
clang::QualType QT = value->getType();
strm << "boxes [";
strm << enclose(QT, C, "(", ") ", 3);
if (!QT->isVoidType()) {
strm << printUnpackedClingValue(*value);
}
strm << "]";
} else
strm << "<<<invalid>>> " << printAddress(value, '@');
return strm.str().str();
}
namespace valuePrinterInternal {
std::string printTypeInternal(const Value &V) {
return printQualType(V.getASTContext(), V.getType());
}
void declarePrintValue(Interpreter &Interp) {
static bool includedRuntimePrintValue = false; // initialized only once as a static function variable
// Include "RuntimePrintValue.h" only on the first printing.
// This keeps the interpreter lightweight and reduces the startup time.
if (!includedRuntimePrintValue) {
Interp.declare("#include \"cling/Interpreter/RuntimePrintValue.h\"");
includedRuntimePrintValue = true;
}
}
std::string printValueInternal(const Value &V) {
Interpreter* Interp = V.getInterpreter();
LockCompilationDuringUserCodeExecutionRAII LCDUCER(*Interp);
declarePrintValue(*Interp);
return printUnpackedClingValue(V);
}
} // end namespace valuePrinterInternal
} // end namespace cling