From 9ac7d10a3796f16d606d6bb528a2bc96fc667c29 Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Fri, 14 Aug 2015 22:49:01 +0200 Subject: [PATCH 01/10] Minor fixes around printValue, based on comments after the integration. --- core/base/inc/TString.h | 1 + core/base/src/TString.cxx | 9 ++ interpreter/cling/Module.mk | 2 +- .../cling/Interpreter/RuntimePrintValue.h | 124 +++++++++--------- .../cling/lib/Interpreter/ValuePrinter.cpp | 10 +- 5 files changed, 84 insertions(+), 62 deletions(-) diff --git a/core/base/inc/TString.h b/core/base/inc/TString.h index 1aa947f9f8166..99cb36a6bee5e 100644 --- a/core/base/inc/TString.h +++ b/core/base/inc/TString.h @@ -784,6 +784,7 @@ namespace llvm { namespace cling { std::string printValue(const TString &val); std::string printValue(const TSubString &val); + std::string printValue(const std::string_view &val); } #endif diff --git a/core/base/src/TString.cxx b/core/base/src/TString.cxx index c40218c456e3e..5d3ff4e0c32ce 100644 --- a/core/base/src/TString.cxx +++ b/core/base/src/TString.cxx @@ -2633,3 +2633,12 @@ std::string cling::printValue(const TSubString &val) { TString s = TString::Format("\"%.*s\"[%d]", (int)val.Length(), val.Data(), (int)val.Length()); return s.Data(); } + +//////////////////////////////////////////////////////////////////////////////// +/// Print a TString in the cling interpreter: + +std::string cling::printValue(const std::string_view &val) { + std::string str(val); + TString s = TString::Format("\"%s\"[%d]", str.c_str(), (int)val.length()); + return s.Data(); +} diff --git a/interpreter/cling/Module.mk b/interpreter/cling/Module.mk index ac3bee9742572..7499ec44787b8 100644 --- a/interpreter/cling/Module.mk +++ b/interpreter/cling/Module.mk @@ -26,7 +26,7 @@ CLINGETC_CLING := DynamicExprInfo.h DynamicLookupRuntimeUniverse.h \ CLINGETC_LLVM := llvm/ADT/IntrusiveRefCntPtr.h \ llvm/ADT/StringRef.h \ llvm/ADT/SmallVector.h \ - llvm/ADT/iterator_range.h \ + llvm/ADT/iterator_range.h \ llvm/Config/llvm-config.h \ llvm/Support/AlignOf.h \ llvm/Support/Allocator.h \ diff --git a/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h b/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h index adff7e0ace59b..9c4e60fa10f1d 100644 --- a/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h +++ b/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h @@ -12,10 +12,6 @@ #include #include "Value.h" -#ifndef _Bool -#define _Bool bool -#endif - namespace cling { // void pointer @@ -66,25 +62,27 @@ namespace cling { // cling::Value std::string printValue(const Value &value); - // Maps declaration - template - auto printValue_impl(const CollectionType &obj, short) + namespace internal { + // Maps declaration + template + auto printValue_impl(const CollectionType &obj, short) -> decltype( ++(obj.begin()), obj.end(), - obj.begin()->first, obj.begin()->second, - std::string()); + obj.begin()->first, obj.begin()->second, + std::string()); - // Collections like vector, set, deque etc. declaration - template - auto printValue_impl(const CollectionType &obj, int) + // Collections like vector, set, deque etc. declaration + template + auto printValue_impl(const CollectionType &obj, int) -> decltype( ++(obj.begin()), obj.end(), - *(obj.begin()), - std::string()); + *(obj.begin()), + std::string()); - // General fallback - print object address declaration - template - std::string printValue_impl(const T &obj, long); + // General fallback - print object address declaration + template + std::string printValue_impl(const T &obj, long); + } // Collections and general fallback entry function template @@ -99,7 +97,7 @@ namespace cling { std::string printValue(const T (&obj)[N]) { std::string str = "{ "; - for(int i = 0; i < N; ++i) { + for (int i = 0; i < N; ++i) { str = str + printValue(obj[i]); if (i < N-1) str = str + ", "; } @@ -107,54 +105,60 @@ namespace cling { return str + " }"; } - // Maps - template - auto printValue_impl(const CollectionType &obj, short) + namespace internal { + // Maps + template + auto printValue_impl(const CollectionType &obj, short) -> decltype( - ++(obj.begin()), obj.end(), - obj.begin()->first, obj.begin()->second, - std::string()) - { - std::string str = "{ "; - - auto iter = obj.begin(); - auto iterEnd = obj.end(); - while (iter != iterEnd) { - str = str + printValue(iter->first); - str = str + " => "; - str = str + printValue(iter->second); - ++iter; - if (iter != iterEnd) str = str + ", "; + ++(obj.begin()), obj.end(), + obj.begin()->first, obj.begin()->second, + std::string()) + { + std::string str = "{ "; + + auto iter = obj.begin(); + auto iterEnd = obj.end(); + while (iter != iterEnd) { + str = str + printValue(iter->first); + str = str + " => "; + str = str + printValue(iter->second); + ++iter; + if (iter != iterEnd) { + str = str + ", "; + } + } + + return str + " }"; } - return str + " }"; - } - - // Collections like vector, set, deque etc. - template - auto printValue_impl(const CollectionType &obj, int) + // Collections like vector, set, deque etc. + template + auto printValue_impl(const CollectionType &obj, int) -> decltype( - ++(obj.begin()), obj.end(), - *(obj.begin()), - std::string()) - { - std::string str = "{ "; - - auto iter = obj.begin(); - auto iterEnd = obj.end(); - while (iter != iterEnd) { - str = str + printValue(*iter); - ++iter; - if (iter != iterEnd) str = str + ", "; + ++(obj.begin()), obj.end(), + *(obj.begin()), + std::string()) + { + std::string str = "{ "; + + auto iter = obj.begin(); + auto iterEnd = obj.end(); + while (iter != iterEnd) { + str = str + printValue(*iter); + ++iter; + if (iter != iterEnd) { + str = str + ", "; + } + } + + return str + " }"; } - return str + " }"; - } - - // General fallback - print object address - template - std::string printValue_impl(const T &obj, long) { - return "@" + printValue((void *) &obj); + // General fallback - print object address + template + std::string printValue_impl(const T &obj, long) { + return "@" + printValue((void *) &obj); + } } } diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index fbcc9c730ba9a..8cadaa71b9200 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -433,8 +433,10 @@ static std::string getCastValueString(const Value &V) { typeWithOptDeref << "(" << type << ")"; } else if (Ty->isPointerType()) { if (Ty->getPointeeType()->isCharType()) { + // Print char pointers as strings. typeWithOptDeref << "(" << type << ")"; } else { + // Fallback to void pointer for other pointers and print the address. typeWithOptDeref << "(void*)"; } } else if (Ty->isArrayType()) { @@ -445,11 +447,15 @@ static std::string getCastValueString(const Value &V) { const llvm::APInt& APSize = CArrTy->getSize(); size_t size = (size_t)APSize.getZExtValue(); + // typeWithOptDeref example for int[40] array: "((int(&)[40])*(void*)0x5c8f260)" typeWithOptDeref << "(" << ElementTy.getAsString() << "(&)[" << size << "])*(void*)"; } else { typeWithOptDeref << "(void*)"; } } else { + // In other cases, dereference the address of the object. + // If no overload or specific template matches, + // the general template will be used which only prints the address. typeWithOptDeref << "*(" << type << "*)"; } @@ -457,7 +463,6 @@ static std::string getCastValueString(const Value &V) { return strm.str(); } - static std::string printEnumValue(const Value &V){ std::stringstream enumString; clang::ASTContext& C = V.getASTContext(); @@ -766,6 +771,8 @@ namespace valuePrinterInternal { std::string printValue_New(const Value& V) { 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) { V.getInterpreter()->declare("#include \"cling/Interpreter/RuntimePrintValue.h\""); includedRuntimePrintValue = true; @@ -774,6 +781,7 @@ namespace valuePrinterInternal { clang::QualType Ty = V.getType().getDesugaredType(C); if(Ty-> isNullPtrType()) { + // special case nullptr_t return "@0x0"; } else if (Ty->isEnumeralType()) { // special case enum printing, using compiled information From 72874f5e55fde98531307680ba624c1f9c8ae94f Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 12:56:26 +0200 Subject: [PATCH 02/10] Removed old printValue_Default and printType_Default methods. Using fully qualified name in new printType_New method. --- .../cling/lib/Interpreter/ValuePrinter.cpp | 89 +------------------ 1 file changed, 3 insertions(+), 86 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index 8cadaa71b9200..992c09b0b31f4 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -749,7 +749,8 @@ namespace valuePrinterInternal { std::string printType_New(const Value& V) { using namespace clang; std::ostringstream strm; - QualType QT = V.getType().getNonReferenceType(); + clang::ASTContext& C = V.getASTContext(); + QualType QT = V.getType().getDesugaredType(C).getNonReferenceType(); std::string ValueTyStr; if (const TypedefType* TDTy = dyn_cast(QT)) ValueTyStr = TDTy->getDecl()->getQualifiedNameAsString(); @@ -757,7 +758,7 @@ namespace valuePrinterInternal { ValueTyStr = TTy->getDecl()->getQualifiedNameAsString(); if (ValueTyStr.empty()) - ValueTyStr = QT.getAsString(); + ValueTyStr = cling::utils::TypeName::GetFullyQualifiedName(QT, C); else if (QT.hasQualifiers()) ValueTyStr = QT.getQualifiers().getAsString() + " " + ValueTyStr; @@ -797,89 +798,5 @@ namespace valuePrinterInternal { return invokePrintValueOverload(V); } } - - void printValue_Default(llvm::raw_ostream& o, const Value& V) { - clang::ASTContext& C = V.getASTContext(); - clang::QualType Ty = V.getType().getDesugaredType(C); - Interpreter& Interp = *const_cast(V.getInterpreter()); - if (const clang::BuiltinType *BT - = llvm::dyn_cast(Ty.getCanonicalType())) { - switch (BT->getKind()) { - case clang::BuiltinType::Bool: // intentional fall through - case clang::BuiltinType::Char_U: // intentional fall through - case clang::BuiltinType::Char_S: // intentional fall through - case clang::BuiltinType::SChar: // intentional fall through - case clang::BuiltinType::Short: // intentional fall through - case clang::BuiltinType::Int: // intentional fall through - case clang::BuiltinType::Long: // intentional fall through - case clang::BuiltinType::LongLong: { - long long res = V.getLL(); - StreamValue(o, (const void*)&res, Ty, Interp); - } - break; - case clang::BuiltinType::UChar: // intentional fall through - case clang::BuiltinType::UShort: // intentional fall through - case clang::BuiltinType::UInt: // intentional fall through - case clang::BuiltinType::ULong: // intentional fall through - case clang::BuiltinType::ULongLong: { - unsigned long long res = V.getULL(); - StreamValue(o, (const void*)&res, Ty, Interp); - } - break; - case clang::BuiltinType::Float: { - float res = V.getFloat(); - StreamValue(o, (const void*)&res, Ty, Interp); - } - break; - case clang::BuiltinType::Double: { - double res = V.getDouble(); - StreamValue(o, (const void*)&res, Ty, Interp); - } - break; - case clang::BuiltinType::LongDouble: { - long double res = V.getLongDouble(); - StreamValue(o, (const void*)&res, Ty, Interp); - } - break; - default: - StreamValue(o, V.getPtr(), Ty, Interp); - break; - } - } - else if (Ty->isIntegralOrEnumerationType()) { - long long res = V.getLL(); - StreamValue(o, &res, Ty, Interp); - } - else if (Ty->isFunctionType()) - StreamValue(o, &V, Ty, Interp); - else if (Ty->isPointerType() || Ty->isReferenceType() - || Ty->isArrayType()) - StreamValue(o, V.getPtr(), Ty, Interp); - else { - // struct case. - StreamValue(o, V.getPtr(), Ty, Interp); - } - } - - void printType_Default(llvm::raw_ostream& o, const Value& V) { - using namespace clang; - QualType QT = V.getType().getNonReferenceType(); - std::string ValueTyStr; - if (const TypedefType* TDTy = dyn_cast(QT)) - ValueTyStr = TDTy->getDecl()->getQualifiedNameAsString(); - else if (const TagType* TTy = dyn_cast(QT)) - ValueTyStr = TTy->getDecl()->getQualifiedNameAsString(); - - if (ValueTyStr.empty()) - ValueTyStr = QT.getAsString(); - else if (QT.hasQualifiers()) - ValueTyStr = QT.getQualifiers().getAsString() + " " + ValueTyStr; - - o << "("; - o << ValueTyStr; - if (V.getType()->isReferenceType()) - o << " &"; - o << ") "; - } } // end namespace valuePrinterInternal } // end namespace cling From 07294f8744a370981b8264d1d90fe97d69e3068d Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:04:00 +0200 Subject: [PATCH 03/10] Removed unused function hasViableCandidateToCall (overload resolution and general fallback deals with this now). Moved isAddressValid function from Value.cpp to ValuePrinter.cpp and implemented the check for void*. --- interpreter/cling/lib/Interpreter/Value.cpp | 82 ------------------- .../cling/lib/Interpreter/ValuePrinter.cpp | 41 ++++++++++ 2 files changed, 41 insertions(+), 82 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/Value.cpp b/interpreter/cling/lib/Interpreter/Value.cpp index 8ac0fcfa716fc..f7ae76e2c41bd 100644 --- a/interpreter/cling/lib/Interpreter/Value.cpp +++ b/interpreter/cling/lib/Interpreter/Value.cpp @@ -30,41 +30,7 @@ #include #include -// For address validation -#ifdef LLVM_ON_WIN32 -#include -#else -#include -#endif - namespace { - static bool isAddressValid(void* P) { - if (!P || P == (void*)-1) - return false; - -#ifdef LLVM_ON_WIN32 - MEMORY_BASIC_INFORMATION MBI; - if (!VirtualQuery(P, &MBI, sizeof(MBI))) - return false; - if (MBI.State != MEM_COMMIT) - return false; - return true; -#else - // There is a POSIX way of finding whether an address can be accessed for - // reading: write() will return EFAULT if not. - int FD[2]; - if (pipe(FD)) - return false; // error in pipe()? Be conservative... - int NBytes = write(FD[1], P, 1/*byte*/); - close(FD[0]); - close(FD[1]); - if (NBytes != 1) { - assert(errno == EFAULT && "unexpected pipe write error"); - return false; - } - return true; -#endif - } ///\brief The allocation starts with this layout; it is followed by the /// value's object at m_Payload. This class does not inherit from @@ -305,54 +271,6 @@ namespace cling { false /*withAccessControl*/); } - static bool hasViableCandidateToCall(clang::LookupResult& R, - const cling::Value& V) { - if (R.empty()) - return false; - using namespace clang; - ASTContext& C = V.getASTContext(); - Sema& SemaR = R.getSema(); - OverloadCandidateSet overloads(SourceLocation(), - OverloadCandidateSet::CSK_Normal); - QualType Ty = V.getType().getNonReferenceType(); - if (!Ty->isPointerType()) - Ty = C.getPointerType(Ty); - - NamespaceDecl* ClingNSD = utils::Lookup::Namespace(&SemaR, "cling"); - RecordDecl* ClingValueDecl - = dyn_cast(utils::Lookup::Named(&SemaR, "Value", - ClingNSD)); - assert(ClingValueDecl && "Declaration must be found!"); - QualType ClingValueTy = C.getTypeDeclType(ClingValueDecl); - - // The OverloadCandidateSet requires a QualType to be passed in through an - // Expr* as part of Args. We know that we won't be using any node generated. - // We need only an answer whether there is an overload taking these argument - // types. We cannot afford to create useless Expr* on the AST for this - // utility function which may be called thousands of times. Instead, we - // create them on the stack and pretend they are on the heap. We get our - // answer and forget about doing anything wrong. - llvm::SmallVector exprsOnStack; - SourceLocation noLoc; - exprsOnStack.push_back(CXXNullPtrLiteralExpr(Ty, noLoc)); - exprsOnStack.push_back(CXXNullPtrLiteralExpr(Ty, noLoc)); - exprsOnStack.push_back(CXXNullPtrLiteralExpr(ClingValueTy, noLoc)); - llvm::SmallVector exprsFakedOnHeap; - exprsFakedOnHeap.push_back(&exprsOnStack[0]); - exprsFakedOnHeap.push_back(&exprsOnStack[1]); - exprsFakedOnHeap.push_back(&exprsOnStack[2]); - llvm::ArrayRef Args = llvm::makeArrayRef(exprsFakedOnHeap.data(), - exprsFakedOnHeap.size()); - // Could trigger deserialization of decls. - cling::Interpreter::PushTransactionRAII RAII(V.getInterpreter()); - SemaR.AddFunctionCandidates(R.asUnresolvedSet(), Args, overloads); - - OverloadCandidateSet::iterator Best; - OverloadingResult OR = overloads.BestViableFunction(SemaR, - SourceLocation(), Best); - return OR == OR_Success; - } - namespace valuePrinterInternal { std::string printType_New(const Value& V); std::string printValue_New(const Value& V); diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index 992c09b0b31f4..1b62b78c39fea 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -46,6 +46,13 @@ //#endif #endif +// For address validation +#ifdef LLVM_ON_WIN32 +#include +#else +#include +#endif + using namespace cling; // Implements the CValuePrinter interface. @@ -61,6 +68,36 @@ extern "C" void cling_PrintValue(void* /*cling::Value**/ V) { //std::string valueStr = printValue(value->getPtr(), value->getPtr(), *value); } +// Checking whether the pointer points to a valid memory location +// Used for checking of void* output +// Should be moved to earlier stages (ex. IR) in the future +static bool isAddressValid(void* P) { + if (!P || P == (void*)-1) + return false; + +#ifdef LLVM_ON_WIN32 + MEMORY_BASIC_INFORMATION MBI; + if (!VirtualQuery(P, &MBI, sizeof(MBI))) + return false; + if (MBI.State != MEM_COMMIT) + return false; + return true; +#else + // There is a POSIX way of finding whether an address can be accessed for + // reading: write() will return EFAULT if not. + int FD[2]; + if (pipe(FD)) + return false; // error in pipe()? Be conservative... + int NBytes = write(FD[1], P, 1/*byte*/); + close(FD[0]); + close(FD[1]); + if (NBytes != 1) { + assert(errno == EFAULT && "unexpected pipe write error"); + return false; + } + return true; +#endif +} static void StreamValue(llvm::raw_ostream& o, const void* V, clang::QualType QT, cling::Interpreter& interp); @@ -583,6 +620,10 @@ namespace cling { strm << "<<>>"; } else { strm << ptr; + if (isAddressValid(ptr)) + strm << " VALID"; + else + strm << " INVALID"; } return strm.str(); } From d43a8bd9670ed81cba6444e648819afd3c6b390a Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:17:11 +0200 Subject: [PATCH 04/10] Extracted printing of unpacked Values. Minor comment fixes. --- .../cling/lib/Interpreter/ValuePrinter.cpp | 67 +++++++++---------- 1 file changed, 31 insertions(+), 36 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index 1b62b78c39fea..c5e90033f259b 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -64,8 +64,8 @@ extern "C" void cling_PrintValue(void* /*cling::Value**/ V) { // the results in pipes (Savannah #99234). //llvm::raw_fd_ostream outs (STDOUT_FILENO, /*ShouldClose*/false); - //std::string typeStr = printType(value->getPtr(), value->getPtr(), *value); - //std::string valueStr = printValue(value->getPtr(), value->getPtr(), *value); + //std::string typeStr = printType_New(*value); + //std::string valueStr = printValue_New(*value); } // Checking whether the pointer points to a valid memory location @@ -612,6 +612,33 @@ static std::string invokePrintValueOverload(const Value& V) { return *(std::string*)printValueV.getPtr(); } +static std::string printUnpackedClingValue(const Value& V) { + std::stringstream strm; + + clang::ASTContext& C = V.getASTContext(); + clang::QualType QT = V.getType(); + clang::QualType Ty = QT.getDesugaredType(C).getNonReferenceType(); + + if(Ty-> isNullPtrType()) { + // special case nullptr_t + strm << "<<>>"; + } else if (Ty->isEnumeralType()) { + // special case enum printing, using compiled information + strm << printEnumValue(V); + } else if (Ty->isFunctionType()) { + // special case function printing, using compiled information + strm << printFunctionValue(V, &V, Ty); + } else if ((Ty->isPointerType() || Ty->isMemberPointerType()) && Ty->getPointeeType()->isFunctionProtoType()) { + // special case function printing, using compiled information + strm << printFunctionValue(V, V.getPtr(), Ty->getPointeeType()); + } else { + // normal case, modular printing using cling::printValue + strm << invokePrintValueOverload(V); + } + + return strm.str(); +} + namespace cling { std::string printValue(void* ptr) { @@ -764,21 +791,7 @@ namespace cling { strm << "(" << QT.getAsString(C.getPrintingPolicy()) << ") "; - clang::QualType valType = QT.getDesugaredType(C).getNonReferenceType(); - if (C.hasSameType(valType, C.LongDoubleTy)) - strm << value.simplisticCastAs() << "L"; - else if (valType->isFloatingType()) - strm << value.simplisticCastAs(); - else if (valType->isIntegerType()) { - if (valType->hasSignedIntegerRepresentation()) - strm << value.simplisticCastAs(); - else - strm << value.simplisticCastAs(); - } else if (valType->isBooleanType()) - strm << (value.simplisticCastAs() ? "true" : "false"); - else if (!valType->isVoidType()) { - strm << invokePrintValueOverload(value); - } + strm << printUnpackedClingValue(value); strm << "]"; } @@ -819,25 +832,7 @@ namespace valuePrinterInternal { V.getInterpreter()->declare("#include \"cling/Interpreter/RuntimePrintValue.h\""); includedRuntimePrintValue = true; } - clang::ASTContext& C = V.getASTContext(); - clang::QualType Ty = V.getType().getDesugaredType(C); - - if(Ty-> isNullPtrType()) { - // special case nullptr_t - return "@0x0"; - } 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 { - // normal case, modular printing using cling::printValue - return invokePrintValueOverload(V); - } + return printUnpackedClingValue(V); } } // end namespace valuePrinterInternal } // end namespace cling From 61b5564b61302d637cd0da8dfa80800ebe960f43 Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:20:20 +0200 Subject: [PATCH 05/10] Changed using of QualifiedType.getAsString() to cling::utils::TypeName::GetFullyQualifiedName(QualifiedType, ASTCtx). --- interpreter/cling/lib/Interpreter/ValuePrinter.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index c5e90033f259b..2401fcfdb6f91 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -259,7 +259,7 @@ static void StreamClingValue(llvm::raw_ostream& o, const Value* value) { clang::QualType QT = value->getType(); o << "boxes ["; o << "(" - << QT.getAsString(C.getPrintingPolicy()) + << cling::utils::TypeName::GetFullyQualifiedName(QT, C) << ") "; clang::QualType valType = QT.getDesugaredType(C).getNonReferenceType(); if (C.hasSameType(valType, C.LongDoubleTy)) @@ -339,7 +339,7 @@ static void StreamValue(llvm::raw_ostream& o, const void* V, StreamObj(o, V, Ty); } } - else if (Ty.getAsString().compare("std::string") == 0) { + else if (cling::utils::TypeName::GetFullyQualifiedName(Ty, C).compare("std::string") == 0) { StreamObj(o, V, Ty); o << " "; // force a space o <<"c_str: "; @@ -485,7 +485,7 @@ static std::string getCastValueString(const Value &V) { size_t size = (size_t)APSize.getZExtValue(); // typeWithOptDeref example for int[40] array: "((int(&)[40])*(void*)0x5c8f260)" - typeWithOptDeref << "(" << ElementTy.getAsString() << "(&)[" << size << "])*(void*)"; + typeWithOptDeref << "(" << cling::utils::TypeName::GetFullyQualifiedName(ElementTy, C) << "(&)[" << size << "])*(void*)"; } else { typeWithOptDeref << "(void*)"; } @@ -789,7 +789,7 @@ namespace cling { clang::QualType QT = value.getType(); strm << "boxes ["; strm << "(" - << QT.getAsString(C.getPrintingPolicy()) + << cling::utils::TypeName::GetFullyQualifiedName(QT, C) << ") "; strm << printUnpackedClingValue(value); strm << "]"; From 9d22da822a2d5cef92efde36e54a6f3b98a369a2 Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:22:03 +0200 Subject: [PATCH 06/10] Simplified printValue(const char *const val). --- interpreter/cling/lib/Interpreter/ValuePrinter.cpp | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index 2401fcfdb6f91..68bd82678f1b1 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -753,13 +753,9 @@ namespace cling { if (!val) { strm << "<<>>"; } else { - const char *cobj = val; - int i = 0; strm << "\""; - while (*cobj != 0) { + for (const char *cobj = val; *cobj != 0; ++cobj) { strm << *cobj; - cobj++; - i++; } strm << "\""; } From b1320ff2ca4d5eed1ce6f1b0abe14d90f799e6d0 Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:33:35 +0200 Subject: [PATCH 07/10] Removed old Stream functions from ValuePrinter.cpp. --- .../cling/lib/Interpreter/ValuePrinter.cpp | 287 +----------------- 1 file changed, 4 insertions(+), 283 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index 68bd82678f1b1..d961dea28d421 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -99,288 +99,6 @@ static bool isAddressValid(void* P) { #endif } -static void StreamValue(llvm::raw_ostream& o, const void* V, clang::QualType QT, - cling::Interpreter& interp); - -static void StreamChar(llvm::raw_ostream& o, const char v) { - if (isprint(v)) - o << '\'' << v << '\''; - else { - o << "\\0x"; - o.write_hex(v); - } -} - -static void StreamCharPtr(llvm::raw_ostream& o, const char* const v) { - if (!v) { - o << "<<>>"; - return; - } - o << '"'; - const char* p = v; - for (;*p && p - v < 128; ++p) { - o << *p; - } - if (*p) o << "\"..."; - else o << "\""; -} - -static void StreamRef(llvm::raw_ostream& o, const void** V, clang::QualType Ty, - cling::Interpreter& interp){ - const clang::ReferenceType* RTy = llvm::dyn_cast(Ty); - StreamValue(o, *V, RTy->getPointeeTypeAsWritten(), interp); -} - -static void StreamPtr(llvm::raw_ostream& o, const void* v) { - o << v; -} - -static void StreamArr(llvm::raw_ostream& o, const void* V, clang::QualType Ty, - cling::Interpreter& interp) { - clang::ASTContext& C = interp.getCI()->getASTContext(); - const clang::ArrayType* ArrTy = Ty->getAsArrayTypeUnsafe(); - clang::QualType ElementTy = ArrTy->getElementType(); - if (ElementTy->isCharType()) - StreamCharPtr(o, (const char*)V); - 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 << "{ "; - for (size_t i = 0; i < Size; ++i) { - // Handle the case of constant size array of pointers. Eg. const char*[] - if (ElementTy->isPointerType()) - StreamValue(o, *(const char* const *)V + i * ElBytes, ElementTy, interp); - else - StreamValue(o, (const char*)V + i * ElBytes, ElementTy, interp); - - if (i + 1 < Size) { - if (i == 4) { - o << "..."; - break; - } - else o << ", "; - } - } - o << " }"; - } else - StreamPtr(o, V); -} - -static void StreamFunction(llvm::raw_ostream& o, const void* V, - clang::QualType QT, cling::Interpreter& Interp) { - o << "Function @" << V << '\n'; - - clang::ASTContext& C = Interp.getCI()->getASTContext(); - const Transaction* T = Interp.getLastTransaction(); - assert(T->getWrapperFD() && "Must have a wrapper."); - clang::FunctionDecl* WrapperFD = T->getWrapperFD(); - - const clang::FunctionDecl* FD = 0; - // CE should be the setValueNoAlloc call expr. - if (const clang::CallExpr* CallE - = llvm::dyn_cast_or_null( - utils::Analyze::GetOrCreateLastExpr(WrapperFD, - /*foundAtPos*/0, - /*omitDS*/false, - &Interp.getSema()))) { - if (const clang::FunctionDecl* FDsetValue - = llvm::dyn_cast_or_null(CallE->getCalleeDecl())){ - if (FDsetValue->getNameAsString() == "setValueNoAlloc" && - CallE->getNumArgs() == 5) { - const clang::Expr* Arg4 = CallE->getArg(4); - while (const clang::CastExpr* CastE - = clang::dyn_cast(Arg4)) - Arg4 = CastE->getSubExpr(); - if (const clang::DeclRefExpr* DeclRefExp - = llvm::dyn_cast(Arg4)) - FD = llvm::dyn_cast(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 = C.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(Ty)->getDecl(); - } - } - // type-based print() never and decl-based print() sometimes does not include - // a final newline: - o << '\n'; -} - -static void StreamLongDouble(llvm::raw_ostream& o, const Value* value, - clang::ASTContext& C) { - std::stringstream sstr; - sstr << value->simplisticCastAs(); - o << sstr.str() << 'L'; -} - -static void StreamClingValue(llvm::raw_ostream& o, const Value* value) { - if (!value || !value->isValid()) { - o << "<<>> @" << value; - } else { - clang::ASTContext& C = value->getASTContext(); - clang::QualType QT = value->getType(); - o << "boxes ["; - o << "(" - << cling::utils::TypeName::GetFullyQualifiedName(QT, C) - << ") "; - clang::QualType valType = QT.getDesugaredType(C).getNonReferenceType(); - if (C.hasSameType(valType, C.LongDoubleTy)) - StreamLongDouble(o, value, C); - else if (valType->isFloatingType()) - o << value->simplisticCastAs(); - else if (valType->isIntegerType()) { - if (valType->hasSignedIntegerRepresentation()) - o << value->simplisticCastAs(); - else - o << value->simplisticCastAs(); - } else if (valType->isBooleanType()) - o << (value->simplisticCastAs() ? "true" : "false"); - else if (!valType->isVoidType()) - StreamValue(o, value->getPtr(), valType, - *const_cast(value->getInterpreter())); - o << "]"; - } -} - -static void StreamObj(llvm::raw_ostream& o, const void* V, clang::QualType Ty) { - if (clang::CXXRecordDecl* CXXRD = Ty->getAsCXXRecordDecl()) { - std::string QualName = CXXRD->getQualifiedNameAsString(); - if (QualName == "cling::Value"){ - StreamClingValue(o, (const cling::Value*)V); - return; - } - } // if CXXRecordDecl - - // TODO: Print the object members. - o << "@" << V; -} - -static void StreamValue(llvm::raw_ostream& o, const void* V, - clang::QualType Ty, cling::Interpreter& Interp) { - clang::ASTContext& C = Interp.getCI()->getASTContext(); - if (const clang::BuiltinType *BT - = llvm::dyn_cast(Ty.getCanonicalType())) { - switch (BT->getKind()) { - case clang::BuiltinType::Bool: - if (*(const bool*)V) - o << "true"; - else - o << "false"; - break; - case clang::BuiltinType::Char_U: // intentional fall through - case clang::BuiltinType::UChar: // intentional fall through - case clang::BuiltinType::Char_S: // intentional fall through - case clang::BuiltinType::SChar: - StreamChar(o, *(const char*)V); break; - case clang::BuiltinType::Short: - o << *(const short*)V; break; - case clang::BuiltinType::UShort: - o << *(const unsigned short*)V; break; - case clang::BuiltinType::Int: - o << *(const int*)V; break; - case clang::BuiltinType::UInt: - o << *(const unsigned int*)V; break; - case clang::BuiltinType::Long: - o << *(const long*)V; break; - case clang::BuiltinType::ULong: - o << *(const unsigned long*)V; break; - case clang::BuiltinType::LongLong: - o << *(const long long*)V; break; - case clang::BuiltinType::ULongLong: - o << *(const unsigned long long*)V; break; - case clang::BuiltinType::Float: - o << *(const float*)V; break; - case clang::BuiltinType::Double: - o << *(const double*)V; break; - case clang::BuiltinType::LongDouble: { - std::stringstream ssLD; - ssLD << *(const long double*)V; - o << ssLD.str() << 'L'; break; - } - default: - StreamObj(o, V, Ty); - } - } - else if (cling::utils::TypeName::GetFullyQualifiedName(Ty, C).compare("std::string") == 0) { - StreamObj(o, V, Ty); - o << " "; // force a space - o <<"c_str: "; - StreamCharPtr(o, ((const char*) (*(const std::string*)V).c_str())); - } - else if (Ty->isEnumeralType()) { - clang::EnumDecl* ED = Ty->getAs()->getDecl(); - uint64_t value = *(const uint64_t*)V; - 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); - } - else if (Ty->isReferenceType()) - StreamRef(o, (const void**)&V, Ty, Interp); - else if (Ty->isPointerType()) { - clang::QualType PointeeTy = Ty->getPointeeType(); - if (PointeeTy->isCharType()) - StreamCharPtr(o, (const char*)V); - else if (PointeeTy->isFunctionProtoType()) - StreamFunction(o, V, PointeeTy, Interp); - else - StreamPtr(o, V); - } - else if (Ty->isArrayType()) - StreamArr(o, V, Ty, Interp); - else if (Ty->isFunctionType()) - StreamFunction(o, V, Ty, Interp); - else - StreamObj(o, V, Ty); -} - static std::string getValueString(const Value &V) { clang::ASTContext &C = V.getASTContext(); clang::QualType Ty = V.getType().getDesugaredType(C); @@ -444,7 +162,8 @@ static std::string getValueString(const Value &V) { std::ostringstream strm; strm << (const void *) &V; return strm.str(); - } else if (Ty->isPointerType() || Ty->isReferenceType() + } else if (Ty->isPointerType() + || Ty->isReferenceType() || Ty->isArrayType()) { std::ostringstream strm; strm << V.getPtr(); @@ -479,6 +198,8 @@ static std::string getCastValueString(const Value &V) { } else if (Ty->isArrayType()) { const clang::ArrayType* ArrTy = Ty->getAsArrayTypeUnsafe(); clang::QualType ElementTy = ArrTy->getElementType(); + // Not implementing printing as string in case of char ElementTy + // Can be done like this: typeWithOptDeref << "(const char void*)"; if (Ty->isConstantArrayType()) { const clang::ConstantArrayType* CArrTy = C.getAsConstantArrayType(Ty); const llvm::APInt& APSize = CArrTy->getSize(); From bb097651f85182db34728e519f46c707113c02fb Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:54:14 +0200 Subject: [PATCH 08/10] Fixed the bug (missing internal::) from commit 279fc73896036900366fa1596ec4d624b9db1c05. --- .../cling/include/cling/Interpreter/RuntimePrintValue.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h b/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h index 9c4e60fa10f1d..a55a54196be0d 100644 --- a/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h +++ b/interpreter/cling/include/cling/Interpreter/RuntimePrintValue.h @@ -87,9 +87,9 @@ namespace cling { // Collections and general fallback entry function template auto printValue(const CollectionType &obj) - -> decltype(printValue_impl(obj, 0), std::string()) + -> decltype(internal::printValue_impl(obj, 0), std::string()) { - return printValue_impl(obj, (short)0); // short -> int -> long = priority order + return internal::printValue_impl(obj, (short)0); // short -> int -> long = priority order } // Arrays From c9fe04f691c813c2cc3393727969479aaa927d79 Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Mon, 17 Aug 2015 13:55:35 +0200 Subject: [PATCH 09/10] Fixed potential bug found by Coverity (https://coverity.cern.ch/reports.htm#v13142/p10001). --- .../cling/lib/Interpreter/ValuePrinter.cpp | 30 +++++++++++-------- 1 file changed, 18 insertions(+), 12 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index d961dea28d421..627389d61b891 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -225,21 +225,27 @@ static std::string printEnumValue(const Value &V){ std::stringstream enumString; clang::ASTContext& C = V.getASTContext(); clang::QualType Ty = V.getType().getDesugaredType(C); - clang::EnumDecl* ED = Ty.getNonReferenceType()->getAs()->getDecl(); - uint64_t value = *(const uint64_t*)&V; - 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 << " ? "; + const clang::EnumType *EnumTy = Ty.getNonReferenceType()->getAs(); + if(EnumTy != 0) { + clang::EnumDecl *ED = EnumTy->getDecl(); + uint64_t value = *(const uint64_t *) &V; + 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 << "(" << I->getQualifiedNameAsString() << ")"; - IsFirst = false; } + enumString << " : (int) " << ValAsAPSInt.toString(/*Radix = */10); + } else { + // Should not happen, we check for isEnumeralType before invoking this func + enumString << "ERROR: Value is not of enum type!"; } - enumString << " : (int) " << ValAsAPSInt.toString(/*Radix = */10); return enumString.str(); } From a86db5f8442a0b2f339df5212d9a49247f8d5e68 Mon Sep 17 00:00:00 2001 From: Boris Perovic Date: Tue, 18 Aug 2015 11:02:16 +0200 Subject: [PATCH 10/10] Fixes based on cling test suite ("make test"). --- interpreter/cling/lib/Interpreter/ValuePrinter.cpp | 6 ++++-- interpreter/cling/test/Interfaces/evaluate.C | 8 ++++---- interpreter/cling/test/Lookup/scope.C | 2 +- interpreter/cling/test/Prompt/ValuePrinter/Regression.C | 4 ++-- 4 files changed, 11 insertions(+), 9 deletions(-) diff --git a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp index 627389d61b891..4b86c1cb44e4e 100644 --- a/interpreter/cling/lib/Interpreter/ValuePrinter.cpp +++ b/interpreter/cling/lib/Interpreter/ValuePrinter.cpp @@ -514,7 +514,9 @@ namespace cling { strm << "(" << cling::utils::TypeName::GetFullyQualifiedName(QT, C) << ") "; - strm << printUnpackedClingValue(value); + if (!QT->isVoidType()) { + strm << printUnpackedClingValue(value); + } strm << "]"; } @@ -527,7 +529,7 @@ namespace valuePrinterInternal { using namespace clang; std::ostringstream strm; clang::ASTContext& C = V.getASTContext(); - QualType QT = V.getType().getDesugaredType(C).getNonReferenceType(); + QualType QT = V.getType().getNonReferenceType(); std::string ValueTyStr; if (const TypedefType* TDTy = dyn_cast(QT)) ValueTyStr = TDTy->getDecl()->getQualifiedNameAsString(); diff --git a/interpreter/cling/test/Interfaces/evaluate.C b/interpreter/cling/test/Interfaces/evaluate.C index 90012ecb4a0e4..dc412458a32b0 100644 --- a/interpreter/cling/test/Interfaces/evaluate.C +++ b/interpreter/cling/test/Interfaces/evaluate.C @@ -33,7 +33,7 @@ gCling->evaluate("cond = false; if (cond) return \"true\"; else return 0;", V); V // CHECK-NEXT: (cling::Value &) boxes [(int) 0] gCling->evaluate("bool a = [](){return true;};", V); -V // CHECK-NEXT: (cling::Value &) boxes [(bool) 1] +V // CHECK-NEXT: (cling::Value &) boxes [(bool) true] gCling->evaluate("auto a = 12.3; a;", V); V // CHECK: (cling::Value &) boxes [(double) 12.3] @@ -44,13 +44,13 @@ V // CHECK: (cling::Value &) boxes [(long) 17] int* IntP = (int*)0x12; gCling->evaluate("IntP;", V); -V // CHECK: (cling::Value &) boxes [(int *) 0x12] +V // CHECK: (cling::Value &) boxes [(int *) 0x12 INVALID] cling::Value Result; gCling->evaluate("V", Result); // Here we check what happens for record type like cling::Value; they are returned by reference. -Result // CHECK: (cling::Value &) boxes [(cling::Value &) boxes [(int *) 0x12]] -V // CHECK: (cling::Value &) boxes [(int *) 0x12] +Result // CHECK: (cling::Value &) boxes [(cling::Value &) boxes [(int *) 0x12 INVALID]] +V // CHECK: (cling::Value &) boxes [(int *) 0x12 INVALID] // Savannah #96277 gCling->evaluate("gCling->declare(\"double sin(double);\"); double one = sin(3.141/2);", V); diff --git a/interpreter/cling/test/Lookup/scope.C b/interpreter/cling/test/Lookup/scope.C index f90b898d55348..922d47694d310 100644 --- a/interpreter/cling/test/Lookup/scope.C +++ b/interpreter/cling/test/Lookup/scope.C @@ -59,7 +59,7 @@ class A {}; const clang::Decl* cl_A_in_NMP = lookup.findScope("N::M::P::A", diags); cl_A_in_NMP -//CHECK: (const clang::Decl *) 0x{{[1-9a-f][0-9a-f]*$}} +//CHECK: (const clang::Decl *) 0x{{[1-9a-f][0-9a-f]*}} VALID cast(cl_A_in_NMP)->getQualifiedNameAsString().c_str() //CHECK-NEXT: ({{[^)]+}}) "N::M::P::A" diff --git a/interpreter/cling/test/Prompt/ValuePrinter/Regression.C b/interpreter/cling/test/Prompt/ValuePrinter/Regression.C index 0af66b352cce9..0a07f9b14904e 100644 --- a/interpreter/cling/test/Prompt/ValuePrinter/Regression.C +++ b/interpreter/cling/test/Prompt/ValuePrinter/Regression.C @@ -45,9 +45,9 @@ q // CHECK: (const int *) 0x123 0.00001234L // CHECK: (long double) 1.234e-05L // PR ROOT-5467 -&A::someFunc // CHECK: (int (class A::*)(float)) Function @0x{{[0-9a-f]+}} +&A::someFunc // CHECK: (int (A::*)(float)) Function @0x{{[0-9a-f]+}} -nullptr // CHECK: (nullptr_t) @0x0 +nullptr // CHECK: (nullptr_t) <<>> #include std::unordered_multiset {1} // ROOT-7310