2012-09-05 09:37:39 +00:00
//------------------------------------------------------------------------------
// CLING - the C++ LLVM-based InterpreterG :)
// version: $Id$
// author: Lukasz Janyst <ljanyst@cern.ch>
//------------------------------------------------------------------------------
# include "cling/Interpreter/Interpreter.h"
# include "DynamicLookup.h"
# include "ExecutionContext.h"
# include "IncrementalParser.h"
# include "cling/Interpreter/CIFactory.h"
2012-10-02 10:30:25 +00:00
# include "cling/Interpreter/CompilationOptions.h"
2012-09-05 09:37:39 +00:00
# include "cling/Interpreter/InterpreterCallbacks.h"
2012-10-02 10:30:25 +00:00
# include "cling/Interpreter/LookupHelper.h"
2012-10-05 12:09:51 +00:00
# include "cling/Interpreter/StoredValueRef.h"
2012-10-02 10:34:22 +00:00
# include "cling/Interpreter/Transaction.h"
2012-10-02 10:30:25 +00:00
# include "cling/Utils/AST.h"
2012-09-05 09:37:39 +00:00
# include "clang/AST/ASTContext.h"
# include "clang/AST/Mangle.h"
# include "clang/Basic/TargetInfo.h"
# include "clang/CodeGen/ModuleBuilder.h"
# include "clang/Frontend/CompilerInstance.h"
# include "clang/Frontend/Utils.h"
# include "clang/Lex/Preprocessor.h"
2012-11-15 00:11:14 +00:00
# include "clang/Parse/Parser.h"
2012-09-05 09:37:39 +00:00
# include "clang/Sema/Sema.h"
# include "clang/Sema/SemaInternal.h"
# include "llvm/Linker.h"
# include "llvm/LLVMContext.h"
# include "llvm/Support/Path.h"
# include <sstream>
# include <vector>
using namespace clang ;
namespace {
static bool canWrapForCall ( const std : : string & input_line ) {
// Whether input_line can be wrapped into a function.
// "1" can, "#include <vector>" can't.
if ( input_line . length ( ) > 1 & & input_line [ 0 ] = = ' # ' ) return false ;
if ( input_line . compare ( 0 , strlen ( " extern " ) , " extern " ) = = 0 ) return false ;
if ( input_line . compare ( 0 , strlen ( " using " ) , " using " ) = = 0 ) return false ;
return true ;
}
} // unnamed namespace
2012-11-02 11:51:05 +00:00
// "Declared" to the JIT in RuntimeUniverse.h
extern " C "
int cling__runtime__internal__local_cxa_atexit ( void ( * func ) ( void * ) , void * arg ,
void * dso ,
void * interp ) {
return ( ( cling : : Interpreter * ) interp ) - > CXAAtExit ( func , arg , dso ) ;
}
namespace cling {
# if (!_WIN32)
2012-09-05 09:37:39 +00:00
// "Declared" to the JIT in RuntimeUniverse.h
namespace runtime {
namespace internal {
struct __trigger__cxa_atexit {
~ __trigger__cxa_atexit ( ) ;
2012-11-02 11:51:05 +00:00
} S ;
__trigger__cxa_atexit : : ~ __trigger__cxa_atexit ( ) {
if ( std : : getenv ( " bar " ) = = ( char * ) - 1 ) {
llvm : : errs ( ) < <
" UNEXPECTED cling::runtime::internal::__trigger__cxa_atexit \n " ;
}
}
2012-09-05 09:37:39 +00:00
}
2012-11-02 12:56:40 +00:00
}
2012-11-02 11:51:05 +00:00
# endif
2012-09-05 09:37:39 +00:00
// This function isn't referenced outside its translation unit, but it
// can't use the "static" keyword because its address is used for
// GetMainExecutable (since some platforms don't support taking the
// address of main, and some platforms can't implement GetMainExecutable
// without being given the address of a function in the main executable).
llvm : : sys : : Path GetExecutablePath ( const char * Argv0 ) {
// This just needs to be some symbol in the binary; C++ doesn't
// allow taking the address of ::main however.
void * MainAddr = ( void * ) ( intptr_t ) GetExecutablePath ;
return llvm : : sys : : Path : : GetMainExecutable ( Argv0 , MainAddr ) ;
}
2012-11-15 23:51:52 +00:00
const Parser & Interpreter : : getParser ( ) const {
return * m_IncrParser - > getParser ( ) ;
}
2012-11-14 21:23:31 +00:00
CodeGenerator * Interpreter : : getCodeGenerator ( ) const {
return m_IncrParser - > getCodeGenerator ( ) ;
}
2012-09-05 09:37:39 +00:00
void Interpreter : : unload ( ) {
m_IncrParser - > unloadTransaction ( 0 ) ;
}
Interpreter : : Interpreter ( int argc , const char * const * argv ,
const char * llvmdir /*= 0*/ ) :
m_UniqueCounter ( 0 ) , m_PrintAST ( false ) , m_DynamicLookupEnabled ( false ) {
m_LLVMContext . reset ( new llvm : : LLVMContext ) ;
std : : vector < unsigned > LeftoverArgsIdx ;
m_Opts = InvocationOptions : : CreateFromArgs ( argc , argv , LeftoverArgsIdx ) ;
std : : vector < const char * > LeftoverArgs ;
for ( size_t I = 0 , N = LeftoverArgsIdx . size ( ) ; I < N ; + + I ) {
LeftoverArgs . push_back ( argv [ LeftoverArgsIdx [ I ] ] ) ;
}
m_IncrParser . reset ( new IncrementalParser ( this , LeftoverArgs . size ( ) ,
& LeftoverArgs [ 0 ] ,
llvmdir ) ) ;
2012-11-15 00:11:14 +00:00
Sema & SemaRef = getSema ( ) ;
m_LookupHelper . reset ( new LookupHelper ( new Parser ( SemaRef . getPreprocessor ( ) ,
SemaRef ,
/*SkipFunctionBodies*/ false ,
/*isTemp*/ true ) ) ) ;
2012-09-05 09:37:39 +00:00
m_ExecutionContext . reset ( new ExecutionContext ( ) ) ;
// Add path to interpreter's include files
// Try to find the headers in the src folder first
# ifdef CLING_SRCDIR_INCL
llvm : : sys : : Path SrcP ( CLING_SRCDIR_INCL ) ;
if ( SrcP . canRead ( ) )
AddIncludePath ( SrcP . str ( ) ) ;
# endif
llvm : : sys : : Path P = GetExecutablePath ( argv [ 0 ] ) ;
if ( ! P . isEmpty ( ) ) {
P . eraseComponent ( ) ; // Remove /cling from foo/bin/clang
P . eraseComponent ( ) ; // Remove /bin from foo/bin
// Get foo/include
P . appendComponent ( " include " ) ;
if ( P . canRead ( ) )
AddIncludePath ( P . str ( ) ) ;
else {
# ifdef CLING_INSTDIR_INCL
llvm : : sys : : Path InstP ( CLING_INSTDIR_INCL ) ;
if ( InstP . canRead ( ) )
AddIncludePath ( InstP . str ( ) ) ;
# endif
}
}
2012-11-02 11:51:05 +00:00
m_ExecutionContext - > addSymbol ( " cling__runtime__internal__local_cxa_atexit " ,
( void * ) ( intptr_t ) & cling__runtime__internal__local_cxa_atexit ) ;
2012-09-05 09:37:39 +00:00
2012-10-04 13:09:52 +00:00
// Enable incremental processing, which prevents the preprocessor destroying
// the lexer on EOF token.
getSema ( ) . getPreprocessor ( ) . enableIncrementalProcessing ( ) ;
2012-09-05 09:37:39 +00:00
if ( getCI ( ) - > getLangOpts ( ) . CPlusPlus ) {
// Set up common declarations which are going to be available
// only at runtime
// Make sure that the universe won't be included to compile time by using
// -D __CLING__ as CompilerInstance's arguments
# ifdef _WIN32
// We have to use the #defined __CLING__ on windows first.
//FIXME: Find proper fix.
declare ( " #ifdef __CLING__ \n #endif " ) ;
# endif
2012-09-26 13:27:48 +00:00
declare ( " #include \" cling/Interpreter/RuntimeUniverse.h \" " ) ;
2012-09-05 09:37:39 +00:00
declare ( " #include \" cling/Interpreter/ValuePrinter.h \" " ) ;
// Set up the gCling variable
std : : stringstream initializer ;
initializer < < " gCling=(cling::Interpreter*) " < < ( uintptr_t ) this < < " ; " ;
2012-10-18 11:56:20 +00:00
execute ( initializer . str ( ) ) ;
2012-09-05 09:37:39 +00:00
}
else {
declare ( " #include \" cling/Interpreter/CValuePrinter.h \" " ) ;
}
handleFrontendOptions ( ) ;
}
Interpreter : : ~ Interpreter ( ) {
for ( size_t I = 0 , N = m_AtExitFuncs . size ( ) ; I < N ; + + I ) {
const CXAAtExitElement & AEE = m_AtExitFuncs [ N - I - 1 ] ;
( * AEE . m_Func ) ( AEE . m_Arg ) ;
}
}
const char * Interpreter : : getVersion ( ) const {
return " $Id$ " ;
}
void Interpreter : : handleFrontendOptions ( ) {
if ( m_Opts . ShowVersion ) {
llvm : : outs ( ) < < getVersion ( ) < < ' \n ' ;
}
if ( m_Opts . Help ) {
m_Opts . PrintHelp ( ) ;
}
}
void Interpreter : : AddIncludePath ( llvm : : StringRef incpath )
{
// Add the given path to the list of directories in which the interpreter
// looks for include files. Only one path item can be specified at a
// time, i.e. "path1:path2" is not supported.
CompilerInstance * CI = getCI ( ) ;
HeaderSearchOptions & headerOpts = CI - > getHeaderSearchOpts ( ) ;
2012-11-15 17:07:44 +00:00
const bool IsUserSupplied = true ;
2012-09-05 09:37:39 +00:00
const bool IsFramework = false ;
const bool IsSysRootRelative = true ;
headerOpts . AddPath ( incpath , frontend : : Angled , IsUserSupplied , IsFramework ,
IsSysRootRelative ) ;
Preprocessor & PP = CI - > getPreprocessor ( ) ;
ApplyHeaderSearchOptions ( PP . getHeaderSearchInfo ( ) , headerOpts ,
PP . getLangOpts ( ) ,
PP . getTargetInfo ( ) . getTriple ( ) ) ;
}
void Interpreter : : DumpIncludePath ( ) {
2012-11-15 17:07:44 +00:00
llvm : : SmallVector < std : : string , 100 > IncPaths ;
GetIncludePaths ( IncPaths , true /*withSystem*/ , true /*withFlags*/ ) ;
// print'em all
for ( unsigned i = 0 ; i < IncPaths . size ( ) ; + + i ) {
llvm : : outs ( ) < < IncPaths [ i ] < < " \n " ;
}
}
// Adapted from clang/lib/Frontend/CompilerInvocation.cpp
void Interpreter : : GetIncludePaths ( llvm : : SmallVectorImpl < std : : string > & incpaths ,
bool withSystem , bool withFlags ) {
2012-09-05 09:37:39 +00:00
const HeaderSearchOptions Opts ( getCI ( ) - > getHeaderSearchOpts ( ) ) ;
2012-11-15 17:07:44 +00:00
if ( withFlags & & Opts . Sysroot ! = " / " ) {
incpaths . push_back ( " -isysroot " ) ;
incpaths . push_back ( Opts . Sysroot ) ;
2012-09-05 09:37:39 +00:00
}
/// User specified include entries.
for ( unsigned i = 0 , e = Opts . UserEntries . size ( ) ; i ! = e ; + + i ) {
const HeaderSearchOptions : : Entry & E = Opts . UserEntries [ i ] ;
if ( E . IsFramework & & ( E . Group ! = frontend : : Angled | | ! E . IsUserSupplied ) )
llvm : : report_fatal_error ( " Invalid option set! " ) ;
if ( E . IsUserSupplied ) {
switch ( E . Group ) {
case frontend : : After :
2012-11-15 17:07:44 +00:00
if ( withFlags ) incpaths . push_back ( " -idirafter " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : Quoted :
2012-11-15 17:07:44 +00:00
if ( withFlags ) incpaths . push_back ( " -iquote " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : System :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -isystem " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : IndexHeaderMap :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -index-header-map " ) ;
if ( withFlags ) incpaths . push_back ( E . IsFramework ? " -F " : " -I " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : CSystem :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -c-isystem " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : CXXSystem :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -cxx-isystem " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : ObjCSystem :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -objc-isystem " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : ObjCXXSystem :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -objcxx-isystem " ) ;
2012-09-05 09:37:39 +00:00
break ;
case frontend : : Angled :
2012-11-15 17:07:44 +00:00
if ( withFlags ) incpaths . push_back ( E . IsFramework ? " -F " : " -I " ) ;
2012-09-05 09:37:39 +00:00
break ;
}
} else {
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
2012-09-05 09:37:39 +00:00
if ( E . Group ! = frontend : : Angled & & E . Group ! = frontend : : System )
llvm : : report_fatal_error ( " Invalid option set! " ) ;
2012-11-15 17:07:44 +00:00
if ( withFlags )
incpaths . push_back ( E . Group = = frontend : : Angled ?
" -iwithprefixbefore " :
" -iwithprefix " ) ;
2012-09-05 09:37:39 +00:00
}
2012-11-15 17:07:44 +00:00
incpaths . push_back ( E . Path ) ;
2012-09-05 09:37:39 +00:00
}
2012-11-15 17:07:44 +00:00
if ( withSystem & & ! Opts . ResourceDir . empty ( ) ) {
if ( withFlags ) incpaths . push_back ( " -resource-dir " ) ;
incpaths . push_back ( Opts . ResourceDir ) ;
2012-09-05 09:37:39 +00:00
}
2012-11-15 17:07:44 +00:00
if ( withSystem & & withFlags & & ! Opts . ModuleCachePath . empty ( ) ) {
incpaths . push_back ( " -fmodule-cache-path " ) ;
incpaths . push_back ( Opts . ModuleCachePath ) ;
2012-09-05 09:37:39 +00:00
}
2012-11-15 17:07:44 +00:00
if ( withSystem & & withFlags & & ! Opts . UseStandardSystemIncludes )
incpaths . push_back ( " -nostdinc " ) ;
if ( withSystem & & withFlags & & ! Opts . UseStandardCXXIncludes )
incpaths . push_back ( " -nostdinc++ " ) ;
if ( withSystem & & withFlags & & Opts . UseLibcxx )
incpaths . push_back ( " -stdlib=libc++ " ) ;
if ( withSystem & & withFlags & & Opts . Verbose )
incpaths . push_back ( " -v " ) ;
2012-09-05 09:37:39 +00:00
}
CompilerInstance * Interpreter : : getCI ( ) const {
return m_IncrParser - > getCI ( ) ;
}
2012-10-03 12:40:41 +00:00
const Sema & Interpreter : : getSema ( ) const {
return getCI ( ) - > getSema ( ) ;
}
2012-10-03 12:57:09 +00:00
Sema & Interpreter : : getSema ( ) {
return getCI ( ) - > getSema ( ) ;
}
2012-09-05 09:37:39 +00:00
llvm : : ExecutionEngine * Interpreter : : getExecutionEngine ( ) const {
return m_ExecutionContext - > getExecutionEngine ( ) ;
}
llvm : : Module * Interpreter : : getModule ( ) const {
return m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
}
///\brief Maybe transform the input line to implement cint command line
/// semantics (declarations are global) and compile to produce a module.
///
Interpreter : : CompilationResult
2012-10-05 12:09:51 +00:00
Interpreter : : process ( const std : : string & input , StoredValueRef * V /* = 0 */ ,
2012-09-05 09:37:39 +00:00
const Decl * * D /* = 0 */ ) {
CompilationOptions CO ;
CO . DeclarationExtraction = 1 ;
CO . ValuePrinting = CompilationOptions : : VPAuto ;
2012-10-15 13:42:09 +00:00
CO . ResultEvaluation = ( bool ) V ;
2012-09-05 09:37:39 +00:00
CO . DynamicScoping = isDynamicLookupEnabled ( ) ;
CO . Debug = isPrintingAST ( ) ;
if ( ! canWrapForCall ( input ) )
return declare ( input , D ) ;
2012-09-18 15:14:58 +00:00
if ( EvaluateInternal ( input , CO , V ) = = Interpreter : : kFailure ) {
2012-09-05 09:37:39 +00:00
if ( D )
* D = 0 ;
return Interpreter : : kFailure ;
}
if ( D )
* D = m_IncrParser - > getLastTransaction ( ) - > getFirstDecl ( ) . getSingleDecl ( ) ;
return Interpreter : : kSuccess ;
}
Interpreter : : CompilationResult
Interpreter : : parse ( const std : : string & input ) {
CompilationOptions CO ;
CO . CodeGeneration = 0 ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = 0 ;
2012-10-15 13:42:09 +00:00
CO . ResultEvaluation = 0 ;
2012-09-05 09:37:39 +00:00
CO . DynamicScoping = isDynamicLookupEnabled ( ) ;
CO . Debug = isPrintingAST ( ) ;
2012-09-18 15:14:58 +00:00
return DeclareInternal ( input , CO ) ;
2012-09-05 09:37:39 +00:00
}
Interpreter : : CompilationResult
Interpreter : : declare ( const std : : string & input , const Decl * * D /* = 0 */ ) {
CompilationOptions CO ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = 0 ;
2012-10-15 13:42:09 +00:00
CO . ResultEvaluation = 0 ;
2012-09-05 09:37:39 +00:00
CO . DynamicScoping = isDynamicLookupEnabled ( ) ;
CO . Debug = isPrintingAST ( ) ;
2012-09-18 15:14:58 +00:00
return DeclareInternal ( input , CO , D ) ;
2012-09-05 09:37:39 +00:00
}
Interpreter : : CompilationResult
2012-10-18 12:24:04 +00:00
Interpreter : : evaluate ( const std : : string & input , StoredValueRef & V ) {
2012-09-05 09:37:39 +00:00
// Here we might want to enforce further restrictions like: Only one
// ExprStmt can be evaluated and etc. Such enforcement cannot happen in the
// worker, because it is used from various places, where there is no such
// rule
CompilationOptions CO ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = 0 ;
2012-10-18 12:24:04 +00:00
CO . ResultEvaluation = 1 ;
2012-09-05 09:37:39 +00:00
2012-10-18 12:24:04 +00:00
return EvaluateInternal ( input , CO , & V ) ;
2012-09-05 09:37:39 +00:00
}
Interpreter : : CompilationResult
2012-10-05 12:09:51 +00:00
Interpreter : : echo ( const std : : string & input , StoredValueRef * V /* = 0 */ ) {
2012-09-05 09:37:39 +00:00
CompilationOptions CO ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = CompilationOptions : : VPEnabled ;
2012-10-15 13:42:09 +00:00
CO . ResultEvaluation = 0 ;
2012-09-05 09:37:39 +00:00
2012-09-18 15:14:58 +00:00
return EvaluateInternal ( input , CO , V ) ;
2012-09-05 09:37:39 +00:00
}
2012-10-18 11:56:20 +00:00
Interpreter : : CompilationResult
Interpreter : : execute ( const std : : string & input ) {
CompilationOptions CO ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = 0 ;
CO . ResultEvaluation = 0 ;
CO . DynamicScoping = 0 ;
CO . Debug = isPrintingAST ( ) ;
// Disable warnings which doesn't make sense when using the prompt
// This gets reset with the clang::Diagnostics().Reset()
2012-10-18 12:05:53 +00:00
// TODO: Here might be useful to issue unused variable diagnostic,
// because we don't do declaration extraction and the decl won't be visible
// anymore.
ignoreFakeDiagnostics ( ) ;
2012-10-18 11:56:20 +00:00
// Wrap the expression
std : : string WrapperName ;
std : : string Wrapper = input ;
WrapInput ( Wrapper , WrapperName ) ;
2012-10-19 11:56:52 +00:00
if ( m_IncrParser - > Compile ( Wrapper , CO ) = = IncrementalParser : : kSuccess ) {
const Transaction * lastT = m_IncrParser - > getLastTransaction ( ) ;
2012-10-19 12:23:34 +00:00
if ( lastT - > getState ( ) = = Transaction : : kCommitted
2012-11-15 20:55:43 +00:00
& & RunFunction ( lastT - > getWrapperFD ( ) ) < kExeFirstError )
2012-10-18 11:56:20 +00:00
return Interpreter : : kSuccess ;
2012-10-19 11:56:52 +00:00
}
2012-10-18 11:56:20 +00:00
return Interpreter : : kFailure ;
}
2012-09-05 09:37:39 +00:00
void Interpreter : : WrapInput ( std : : string & input , std : : string & fname ) {
fname = createUniqueWrapper ( ) ;
input . insert ( 0 , " void " + fname + " () { \n " ) ;
input . append ( " \n ; \n } " ) ;
}
2012-11-15 20:55:43 +00:00
Interpreter : : ExecutionResult
Interpreter : : RunFunction ( const FunctionDecl * FD , StoredValueRef * res /*=0*/ ) {
2012-09-05 09:37:39 +00:00
if ( getCI ( ) - > getDiagnostics ( ) . hasErrorOccurred ( ) )
2012-11-15 20:55:43 +00:00
return kExeCompilationError ;
2012-09-05 09:37:39 +00:00
if ( ! m_IncrParser - > hasCodeGenerator ( ) ) {
2012-11-15 20:55:43 +00:00
return kExeNoCodeGen ;
2012-09-05 09:37:39 +00:00
}
2012-10-19 11:56:52 +00:00
if ( ! FD )
2012-11-15 20:55:43 +00:00
return kExeUnkownFunction ;
2012-09-05 09:37:39 +00:00
2012-10-19 11:56:52 +00:00
std : : string mangledNameIfNeeded ;
mangleName ( FD , mangledNameIfNeeded ) ;
2012-11-15 20:55:43 +00:00
ExecutionContext : : ExecutionResult ExeRes =
m_ExecutionContext - > executeFunction ( mangledNameIfNeeded . c_str ( ) ,
getCI ( ) - > getASTContext ( ) ,
FD - > getResultType ( ) , res ) ;
switch ( ExeRes ) {
case ExecutionContext : : kExeSuccess :
return kExeSuccess ;
case ExecutionContext : : kExeFunctionNotCompiled :
return kExeFunctionNotCompiled ;
case ExecutionContext : : kExeUnresolvedSymbols :
return kExeUnresolvedSymbols ;
default : break ;
}
// Should not end up here...
return kExeSuccess ;
2012-09-05 09:37:39 +00:00
}
void Interpreter : : createUniqueName ( std : : string & out ) {
2012-10-17 08:52:24 +00:00
out = utils : : Synthesize : : UniquePrefix ;
2012-09-05 09:37:39 +00:00
llvm : : raw_string_ostream ( out ) < < m_UniqueCounter + + ;
}
2012-10-09 09:48:54 +00:00
bool Interpreter : : isUniqueName ( llvm : : StringRef name ) {
2012-10-17 08:52:24 +00:00
return name . startswith ( utils : : Synthesize : : UniquePrefix ) ;
2012-10-09 09:48:54 +00:00
}
llvm : : StringRef Interpreter : : createUniqueWrapper ( ) {
2012-10-17 08:52:24 +00:00
const size_t size
= sizeof ( utils : : Synthesize : : UniquePrefix ) + sizeof ( m_UniqueCounter ) ;
llvm : : SmallString < size > out ( utils : : Synthesize : : UniquePrefix ) ;
2012-10-09 09:48:54 +00:00
llvm : : raw_svector_ostream ( out ) < < m_UniqueCounter + + ;
return ( getCI ( ) - > getASTContext ( ) . Idents . getOwn ( out ) ) . getName ( ) ;
}
bool Interpreter : : isUniqueWrapper ( llvm : : StringRef name ) {
2012-10-17 08:52:24 +00:00
return name . startswith ( utils : : Synthesize : : UniquePrefix ) ;
2012-10-09 09:48:54 +00:00
}
2012-09-05 09:37:39 +00:00
Interpreter : : CompilationResult
2012-09-18 15:14:58 +00:00
Interpreter : : DeclareInternal ( const std : : string & input ,
const CompilationOptions & CO ,
const clang : : Decl * * D /* = 0 */ ) {
2012-09-05 09:37:39 +00:00
if ( m_IncrParser - > Compile ( input , CO ) ! = IncrementalParser : : kFailed ) {
if ( D )
* D = m_IncrParser - > getLastTransaction ( ) - > getFirstDecl ( ) . getSingleDecl ( ) ;
return Interpreter : : kSuccess ;
}
return Interpreter : : kFailure ;
}
Interpreter : : CompilationResult
2012-09-18 15:14:58 +00:00
Interpreter : : EvaluateInternal ( const std : : string & input ,
const CompilationOptions & CO ,
2012-10-05 12:09:51 +00:00
StoredValueRef * V /* = 0 */ ) {
2012-09-05 09:37:39 +00:00
// Disable warnings which doesn't make sense when using the prompt
// This gets reset with the clang::Diagnostics().Reset()
2012-10-18 12:05:53 +00:00
ignoreFakeDiagnostics ( ) ;
2012-09-05 09:37:39 +00:00
// Wrap the expression
std : : string WrapperName ;
std : : string Wrapper = input ;
WrapInput ( Wrapper , WrapperName ) ;
if ( V ) {
2012-10-19 11:56:52 +00:00
Transaction * CurT = m_IncrParser - > Parse ( Wrapper , CO ) ;
2012-09-05 09:37:39 +00:00
assert ( CurT - > size ( ) & & " No decls created by Parse! " ) ;
2012-11-14 22:06:58 +00:00
m_IncrParser - > commitTransaction ( CurT ) ;
2012-09-05 09:37:39 +00:00
}
else
m_IncrParser - > Compile ( Wrapper , CO ) ;
// get the result
2012-10-19 11:56:52 +00:00
const Transaction * lastT = m_IncrParser - > getLastTransaction ( ) ;
2012-10-19 12:23:34 +00:00
if ( lastT - > getState ( ) = = Transaction : : kCommitted
2012-11-15 20:55:43 +00:00
& & RunFunction ( lastT - > getWrapperFD ( ) , V ) < kExeFirstError )
2012-09-05 09:37:39 +00:00
return Interpreter : : kSuccess ;
2012-10-18 09:38:39 +00:00
else if ( V )
2012-10-18 09:33:29 +00:00
* V = StoredValueRef : : invalidValue ( ) ;
2012-09-05 09:37:39 +00:00
return Interpreter : : kFailure ;
}
2012-11-15 18:23:11 +00:00
void Interpreter : : addLoadedFile ( const std : : string & name ,
Interpreter : : LoadedFileInfo : : FileType type ,
const llvm : : sys : : DynamicLibrary * dyLib ) {
m_LoadedFiles . push_back ( new LoadedFileInfo ( name , type , dyLib ) ) ;
}
2012-11-15 14:50:59 +00:00
Interpreter : : CompilationResult
Interpreter : : loadFile ( const std : : string & filename ,
bool allowSharedLib /*=true*/ ) {
2012-11-16 13:39:09 +00:00
if ( allowSharedLib ) {
bool tryCode ;
if ( loadLibrary ( filename , false , & tryCode )
= = kLoadLibSuccess )
return kSuccess ;
if ( ! tryCode )
return kFailure ;
}
2012-11-15 21:36:41 +00:00
2012-09-05 09:37:39 +00:00
std : : string code ;
code + = " #include \" " + filename + " \" " ;
2012-11-15 18:23:11 +00:00
CompilationResult res = declare ( code ) ;
if ( res = = kSuccess )
2012-11-16 13:39:09 +00:00
addLoadedFile ( filename , LoadedFileInfo : : kSource ) ;
2012-11-15 18:23:11 +00:00
return res ;
2012-09-05 09:37:39 +00:00
}
2012-11-15 16:41:30 +00:00
2012-11-15 14:50:59 +00:00
Interpreter : : LoadLibResult
2012-11-15 21:36:41 +00:00
Interpreter : : tryLinker ( const std : : string & filename , bool permanent ,
bool & tryCode ) {
2012-11-15 16:41:30 +00:00
using namespace llvm : : sys ;
2012-11-15 21:36:41 +00:00
tryCode = true ;
2012-11-15 14:50:59 +00:00
llvm : : Module * module = m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
2012-11-15 16:41:30 +00:00
assert ( module & & " Module must exist for linking! " ) ;
llvm : : Linker L ( " cling " , module , llvm : : Linker : : QuietWarnings
| llvm : : Linker : : QuietErrors ) ;
2012-11-15 18:23:11 +00:00
struct LinkerModuleReleaseRAII {
LinkerModuleReleaseRAII ( llvm : : Linker & L ) : m_L ( L ) { }
~ LinkerModuleReleaseRAII ( ) { m_L . releaseModule ( ) ; }
llvm : : Linker & m_L ;
} LinkerModuleReleaseRAII_ ( L ) ;
2012-11-15 16:41:30 +00:00
const InvocationOptions & Opts = getOptions ( ) ;
for ( std : : vector < Path > : : const_iterator I
= Opts . LibSearchPath . begin ( ) , E = Opts . LibSearchPath . end ( ) ; I ! = E ;
+ + I ) {
L . addPath ( * I ) ;
}
L . addSystemPaths ( ) ;
bool Native = true ;
if ( L . LinkInLibrary ( filename , Native ) ) {
// that didn't work, try bitcode:
Path FilePath ( filename ) ;
std : : string Magic ;
if ( ! FilePath . getMagicNumber ( Magic , 64 ) ) {
// filename doesn't exist...
2012-11-15 21:36:41 +00:00
// tryCode because it might be found through -I
2012-11-15 18:23:11 +00:00
return kLoadLibError ;
2012-11-15 14:50:59 +00:00
}
2012-11-15 18:23:11 +00:00
if ( IdentifyFileType ( Magic . c_str ( ) , 64 ) ! = Bitcode_FileType ) {
2012-11-15 16:41:30 +00:00
// Nothing the linker can handle
2012-11-15 18:23:11 +00:00
return kLoadLibError ;
2012-11-15 14:50:59 +00:00
}
2012-11-15 18:23:11 +00:00
// We are promised a bitcode file, complain if it fails
L . setFlags ( 0 ) ;
if ( L . LinkInFile ( Path ( filename ) , Native ) ) {
2012-11-15 21:36:41 +00:00
tryCode = false ;
2012-11-15 18:23:11 +00:00
return kLoadLibError ;
}
addLoadedFile ( filename , LoadedFileInfo : : kBitcode ) ;
return kLoadLibSuccess ;
}
if ( Native ) {
2012-11-15 21:36:41 +00:00
tryCode = false ;
2012-11-15 16:41:30 +00:00
// native shared library, load it!
Path SoFile = L . FindLib ( filename ) ;
assert ( ! SoFile . isEmpty ( ) & & " The shared lib exists but can't find it! " ) ;
std : : string errMsg ;
// TODO: !permanent case
DynamicLibrary DyLib
= DynamicLibrary : : getPermanentLibrary ( SoFile . str ( ) . c_str ( ) , & errMsg ) ;
2012-11-15 21:36:41 +00:00
if ( ! DyLib . isValid ( ) ) {
llvm : : errs ( ) < < " cling::Interpreter::tryLinker(): " < < errMsg < < ' \n ' ;
2012-11-15 18:23:11 +00:00
return kLoadLibError ;
2012-11-15 21:36:41 +00:00
}
2012-11-15 18:23:11 +00:00
std : : pair < std : : set < llvm : : sys : : DynamicLibrary > : : iterator , bool > insRes
= m_DyLibs . insert ( DyLib ) ;
if ( ! insRes . second )
return kLoadLibExists ;
2012-11-16 11:30:31 +00:00
addLoadedFile ( SoFile . str ( ) , LoadedFileInfo : : kDynamicLibrary ,
2012-11-15 18:23:11 +00:00
& ( * insRes . first ) ) ;
return kLoadLibSuccess ;
2012-11-15 16:41:30 +00:00
}
2012-11-15 18:23:11 +00:00
return kLoadLibError ;
2012-11-15 16:41:30 +00:00
}
Interpreter : : LoadLibResult
2012-11-15 21:36:41 +00:00
Interpreter : : loadLibrary ( const std : : string & filename , bool permanent ,
bool * tryCode ) {
bool tryCodeDummy ;
LoadLibResult res = tryLinker ( filename , permanent ,
tryCode ? * tryCode : tryCodeDummy ) ;
2012-11-15 18:23:11 +00:00
if ( res ! = kLoadLibError ) {
2012-11-15 16:41:30 +00:00
return res ;
}
if ( filename . compare ( 0 , 3 , " lib " ) = = 0 ) {
// starts with "lib", try without (the llvm::Linker forces
// a "lib" in front, which makes it liblib...
2012-11-15 21:36:41 +00:00
res = tryLinker ( filename . substr ( 3 , std : : string : : npos ) , permanent ,
tryCodeDummy ) ;
2012-11-15 18:23:11 +00:00
if ( res ! = kLoadLibError )
2012-11-15 16:41:30 +00:00
return res ;
2012-11-15 14:50:59 +00:00
}
2012-11-15 18:23:11 +00:00
return kLoadLibError ;
2012-11-15 14:50:59 +00:00
}
2012-09-26 13:27:48 +00:00
void Interpreter : : installLazyFunctionCreator ( void * ( * fp ) ( const std : : string & ) ) {
2012-09-05 09:37:39 +00:00
m_ExecutionContext - > installLazyFunctionCreator ( fp ) ;
}
2012-10-25 14:07:19 +00:00
void Interpreter : : suppressLazyFunctionCreatorDiags ( bool suppressed /*=true*/ ) {
m_ExecutionContext - > suppressLazyFunctionCreatorDiags ( suppressed ) ;
2012-10-25 09:17:45 +00:00
}
2012-10-05 12:09:51 +00:00
StoredValueRef Interpreter : : Evaluate ( const char * expr , DeclContext * DC ,
bool ValuePrinterReq ) {
2012-09-05 09:37:39 +00:00
Sema & TheSema = getCI ( ) - > getSema ( ) ;
if ( ! DC )
DC = TheSema . getASTContext ( ) . getTranslationUnitDecl ( ) ;
// Set up the declaration context
DeclContext * CurContext ;
CurContext = TheSema . CurContext ;
TheSema . CurContext = DC ;
2012-10-05 12:09:51 +00:00
StoredValueRef Result ;
2012-10-09 10:21:17 +00:00
if ( TheSema . getExternalSource ( ) ) {
2012-10-18 12:24:04 +00:00
( ValuePrinterReq ) ? echo ( expr , & Result ) : evaluate ( expr , Result ) ;
2012-09-05 09:37:39 +00:00
}
else
2012-10-18 12:24:04 +00:00
( ValuePrinterReq ) ? echo ( expr , & Result ) : evaluate ( expr , Result ) ;
2012-09-05 09:37:39 +00:00
TheSema . CurContext = CurContext ;
return Result ;
}
void Interpreter : : setCallbacks ( InterpreterCallbacks * C ) {
2012-10-10 13:00:17 +00:00
// We need it to enable LookupObject callback.
2012-10-02 10:30:25 +00:00
m_Callbacks . reset ( C ) ;
2012-09-05 09:37:39 +00:00
}
2012-10-19 13:30:06 +00:00
const Transaction * Interpreter : : getFirstTransaction ( ) const {
return m_IncrParser - > getFirstTransaction ( ) ;
}
2012-09-05 09:37:39 +00:00
void Interpreter : : enableDynamicLookup ( bool value /*=true*/ ) {
m_DynamicLookupEnabled = value ;
if ( isDynamicLookupEnabled ( ) ) {
declare ( " #include \" cling/Interpreter/DynamicLookupRuntimeUniverse.h \" " ) ;
}
2012-10-10 13:00:17 +00:00
else
setCallbacks ( 0 ) ;
2012-09-05 09:37:39 +00:00
}
void Interpreter : : runStaticInitializersOnce ( ) const {
// Forward to ExecutionContext; should not be called by
// anyone except for IncrementalParser.
llvm : : Module * module = m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
m_ExecutionContext - > runStaticInitializersOnce ( module ) ;
}
int Interpreter : : CXAAtExit ( void ( * func ) ( void * ) , void * arg , void * dso ) {
// Register a CXAAtExit function
Decl * LastTLD
= m_IncrParser - > getLastTransaction ( ) - > getLastDecl ( ) . getSingleDecl ( ) ;
m_AtExitFuncs . push_back ( CXAAtExitElement ( func , arg , dso , LastTLD ) ) ;
return 0 ; // happiness
}
2012-09-24 15:20:19 +00:00
void Interpreter : : mangleName ( const clang : : NamedDecl * D ,
std : : string & mangledName ) const {
2012-09-24 09:57:43 +00:00
///Get the mangled name of a NamedDecl.
///
///D - mangle this decl's name
///mangledName - put the mangled name in here
2012-09-24 12:11:46 +00:00
if ( ! m_MangleCtx ) {
m_MangleCtx . reset ( getCI ( ) - > getASTContext ( ) . createMangleContext ( ) ) ;
}
if ( m_MangleCtx - > shouldMangleDeclName ( D ) ) {
2012-09-24 09:57:43 +00:00
llvm : : raw_string_ostream RawStr ( mangledName ) ;
2012-09-24 12:11:46 +00:00
m_MangleCtx - > mangleName ( D , RawStr ) ;
2012-09-24 09:57:43 +00:00
RawStr . flush ( ) ;
} else {
mangledName = D - > getNameAsString ( ) ;
}
}
2012-10-18 12:05:53 +00:00
void Interpreter : : ignoreFakeDiagnostics ( ) const {
DiagnosticsEngine & Diag = getCI ( ) - > getDiagnostics ( ) ;
// Disable warnings which doesn't make sense when using the prompt
// This gets reset with the clang::Diagnostics().Reset()
Diag . setDiagnosticMapping ( clang : : diag : : warn_unused_expr ,
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
Diag . setDiagnosticMapping ( clang : : diag : : warn_unused_call ,
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
Diag . setDiagnosticMapping ( clang : : diag : : warn_unused_comparison ,
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
2012-11-11 12:53:21 +00:00
Diag . setDiagnosticMapping ( clang : : diag : : ext_return_has_expr ,
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
2012-10-18 12:05:53 +00:00
}
2012-09-24 09:57:43 +00:00
bool Interpreter : : addSymbol ( const char * symbolName , void * symbolAddress ) {
2012-09-05 09:37:39 +00:00
// Forward to ExecutionContext;
if ( ! symbolName | | ! symbolAddress )
return false ;
return m_ExecutionContext - > addSymbol ( symbolName , symbolAddress ) ;
}
2012-09-24 09:57:43 +00:00
void * Interpreter : : getAddressOfGlobal ( const clang : : NamedDecl * D ,
bool * fromJIT /*=0*/ ) const {
// Return a symbol's address, and whether it was jitted.
std : : string mangledName ;
2012-09-24 15:20:19 +00:00
mangleName ( D , mangledName ) ;
2012-11-16 15:32:43 +00:00
return getAddressOfGlobal ( mangledName . c_str ( ) , fromJIT ) ;
}
void * Interpreter : : getAddressOfGlobal ( const char * SymName ,
bool * fromJIT /*=0*/ ) const {
// Return a symbol's address, and whether it was jitted.
2012-09-24 12:11:46 +00:00
llvm : : Module * module = m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
2012-11-16 15:32:43 +00:00
return m_ExecutionContext - > getAddressOfGlobal ( module , SymName , fromJIT ) ;
2012-09-24 09:57:43 +00:00
}
2012-09-05 09:37:39 +00:00
} // namespace cling