2012-09-05 09:37:39 +00:00
//------------------------------------------------------------------------------
// CLING - the C++ LLVM-based InterpreterG :)
// author: Lukasz Janyst <ljanyst@cern.ch>
2014-01-07 11:08:37 +01:00
//
// 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.
2012-09-05 09:37:39 +00:00
//------------------------------------------------------------------------------
# include "cling/Interpreter/Interpreter.h"
# include "DynamicLookup.h"
# include "ExecutionContext.h"
# include "IncrementalParser.h"
# include "cling/Interpreter/CIFactory.h"
2013-09-17 17:13:38 +02:00
# include "cling/Interpreter/ClangInternalState.h"
2012-10-02 10:30:25 +00:00
# include "cling/Interpreter/CompilationOptions.h"
2013-09-24 11:41:22 +02:00
# include "cling/Interpreter/DynamicLibraryManager.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"
2013-10-24 13:16:56 -05:00
# include "clang/AST/GlobalDecl.h"
2012-09-05 09:37:39 +00:00
# include "clang/Basic/TargetInfo.h"
2013-09-07 00:00:26 +02:00
# include "clang/Basic/SourceManager.h"
2012-09-05 09:37:39 +00:00
# include "clang/CodeGen/ModuleBuilder.h"
# include "clang/Frontend/CompilerInstance.h"
# include "clang/Frontend/Utils.h"
# include "clang/Lex/Preprocessor.h"
2012-12-05 10:04:42 +00:00
# include "clang/Parse/ParseDiagnostic.h" // FIXME: remove this back-dependency!
// when clang is ready.
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"
2013-04-24 16:28:08 +00:00
# include "llvm/IR/LLVMContext.h"
2013-12-05 16:24:43 +01:00
# include "llvm/IR/Module.h"
2012-09-05 09:37:39 +00:00
# include "llvm/Support/Path.h"
2013-08-21 12:25:48 +02:00
# include "llvm/Support/raw_ostream.h"
2012-09-05 09:37:39 +00:00
# include <sstream>
2013-08-21 12:25:48 +02:00
# include <string>
2012-09-05 09:37:39 +00:00
# include <vector>
2013-08-19 16:07:25 +02:00
using namespace clang ;
2013-04-26 11:45:17 +00:00
2012-09-05 09:37:39 +00:00
namespace {
2012-11-20 16:24:02 +00:00
static cling : : Interpreter : : ExecutionResult
ConvertExecutionResult ( cling : : ExecutionContext : : ExecutionResult ExeRes ) {
switch ( ExeRes ) {
case cling : : ExecutionContext : : kExeSuccess :
return cling : : Interpreter : : kExeSuccess ;
case cling : : ExecutionContext : : kExeFunctionNotCompiled :
return cling : : Interpreter : : kExeFunctionNotCompiled ;
case cling : : ExecutionContext : : kExeUnresolvedSymbols :
return cling : : Interpreter : : kExeUnresolvedSymbols ;
default : break ;
}
return cling : : Interpreter : : kExeSuccess ;
}
2012-09-05 09:37:39 +00:00
} // unnamed namespace
2013-09-19 11:24:31 +02:00
namespace cling {
namespace runtime {
namespace internal {
// "Declared" to the JIT in RuntimeUniverse.h
int local_cxa_atexit ( void ( * func ) ( void * ) , void * arg , void * dso ,
void * interp ) {
Interpreter * cling = ( cling : : Interpreter * ) interp ;
IncrementalParser * incrP = cling - > m_IncrParser . get ( ) ;
// FIXME: Bind to the module symbols.
Decl * lastTLD = incrP - > getLastTransaction ( ) - > getLastDecl ( ) . getSingleDecl ( ) ;
int result = cling - > m_ExecutionContext - > CXAAtExit ( func , arg , dso , lastTLD ) ;
return result ;
}
} // end namespace internal
} // end namespace runtime
2012-11-02 11:51:05 +00:00
}
2013-09-11 19:21:26 +02:00
2012-11-02 11:51:05 +00:00
namespace cling {
2013-10-03 09:47:56 +02:00
// FIXME: workaround until JIT supports exceptions
jmp_buf * Interpreter : : m_JumpBuf ;
2012-11-02 11:51:05 +00:00
# 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 ( ) ;
2013-02-21 17:31:30 +00:00
} /*S*/ ;
2012-11-02 11:51:05 +00:00
__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
2013-08-26 17:21:07 +02:00
Interpreter : : PushTransactionRAII : : PushTransactionRAII ( const Interpreter * i )
2013-04-24 12:53:08 +00:00
: m_Interpreter ( i ) {
2013-04-03 16:25:08 +00:00
CompilationOptions CO ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = 0 ;
CO . ResultEvaluation = 0 ;
CO . DynamicScoping = 0 ;
CO . Debug = 0 ;
CO . CodeGeneration = 1 ;
CO . CodeGenerationForModule = 0 ;
2013-04-24 12:53:08 +00:00
m_Transaction = m_Interpreter - > m_IncrParser - > beginTransaction ( CO ) ;
2013-04-03 16:25:08 +00:00
}
Interpreter : : PushTransactionRAII : : ~ PushTransactionRAII ( ) {
pop ( ) ;
}
2013-08-26 17:21:07 +02:00
void Interpreter : : PushTransactionRAII : : pop ( ) const {
2013-07-06 13:04:31 +02:00
if ( Transaction * T
= m_Interpreter - > m_IncrParser - > endTransaction ( m_Transaction ) ) {
assert ( T = = m_Transaction & & " Ended different transaction? " ) ;
m_Interpreter - > m_IncrParser - > commitTransaction ( T ) ;
}
2013-10-16 14:22:22 +02:00
}
2013-03-13 09:26:55 +00:00
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).
2013-09-20 16:01:06 +02:00
std : : string GetExecutablePath ( const char * Argv0 ) {
2012-09-05 09:37:39 +00:00
// 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 ;
2013-09-20 16:01:06 +02:00
return llvm : : sys : : fs : : getMainExecutable ( Argv0 , MainAddr ) ;
2012-09-05 09:37:39 +00:00
}
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
Interpreter : : Interpreter ( int argc , const char * const * argv ,
const char * llvmdir /*= 0*/ ) :
2013-08-22 10:28:43 +02:00
m_UniqueCounter ( 0 ) , m_PrintAST ( false ) , m_PrintIR ( false ) ,
m_DynamicLookupEnabled ( false ) , m_RawInputEnabled ( false ) {
2012-09-05 09:37:39 +00:00
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 ] ] ) ;
}
2013-09-24 11:41:22 +02:00
m_DyLibManager . reset ( new DynamicLibraryManager ( getOptions ( ) ) ) ;
2012-09-05 09:37:39 +00:00
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 ,
2013-04-03 16:25:08 +00:00
/*isTemp*/ true ) , this ) ) ;
2012-09-05 09:37:39 +00:00
2013-06-10 15:44:36 +02:00
if ( m_IncrParser - > hasCodeGenerator ( ) ) {
llvm : : Module * theModule = m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
m_ExecutionContext . reset ( new ExecutionContext ( theModule ) ) ;
}
2012-09-05 09:37:39 +00:00
2013-03-11 13:11:15 +00:00
m_IncrParser - > Initialize ( ) ;
2013-09-30 18:03:35 +02:00
// Add configuration paths to interpreter's include files.
# ifdef CLING_INCLUDE_PATHS
llvm : : StringRef InclPaths ( CLING_INCLUDE_PATHS ) ;
for ( std : : pair < llvm : : StringRef , llvm : : StringRef > Split = InclPaths . split ( ' : ' ) ;
! Split . second . empty ( ) ; Split = InclPaths . split ( ' : ' ) ) {
if ( llvm : : sys : : fs : : is_directory ( Split . first ) )
AddIncludePath ( Split . first ) ;
InclPaths = Split . second ;
}
// Add remaining part
AddIncludePath ( InclPaths ) ;
# else
2013-09-20 16:01:06 +02:00
llvm : : SmallString < 512 > P ( GetExecutablePath ( argv [ 0 ] ) ) ;
if ( ! P . empty ( ) ) {
// Remove /cling from foo/bin/clang
llvm : : StringRef ExeIncl = llvm : : sys : : path : : parent_path ( P ) ;
// Remove /bin from foo/bin
ExeIncl = llvm : : sys : : path : : parent_path ( ExeIncl ) ;
P . resize ( ExeIncl . size ( ) + 1 ) ;
P [ ExeIncl . size ( ) ] = 0 ;
2012-09-05 09:37:39 +00:00
// Get foo/include
2013-09-20 16:01:06 +02:00
llvm : : sys : : path : : append ( P , " include " ) ;
2013-09-20 18:24:19 +02:00
if ( llvm : : sys : : fs : : is_directory ( P . str ( ) ) )
2012-09-05 09:37:39 +00:00
AddIncludePath ( P . str ( ) ) ;
}
2013-09-30 18:03:35 +02:00
# endif
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 ( ) ;
2013-02-27 15:05:21 +00:00
handleFrontendOptions ( ) ;
// Tell the diagnostic client that we are entering file parsing mode.
DiagnosticConsumer & DClient = getCI ( ) - > getDiagnosticClient ( ) ;
DClient . BeginSourceFile ( getCI ( ) - > getLangOpts ( ) ,
& getCI ( ) - > getPreprocessor ( ) ) ;
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
2013-09-19 11:24:31 +02:00
// We have to use the #defined __CLING__ on windows first.
2012-09-05 09:37:39 +00:00
//FIXME: Find proper fix.
declare ( " #ifdef __CLING__ \n #endif " ) ;
# endif
2012-09-26 13:27:48 +00:00
declare ( " #include \" cling/Interpreter/RuntimeUniverse.h \" " ) ;
2013-03-11 13:11:15 +00:00
if ( getCodeGenerator ( ) ) {
// Set up the gCling variable if it can be used
std : : stringstream initializer ;
initializer < < " namespace cling {namespace runtime { "
" cling::Interpreter *gCling=(cling::Interpreter*) "
< < ( uintptr_t ) this < < " ;} } " ;
declare ( initializer . str ( ) ) ;
2013-10-30 21:11:30 -05:00
m_ExecutionContext - > remapCXAAtExit ( ) ;
2013-03-11 13:11:15 +00:00
}
2013-09-19 11:24:31 +02:00
2013-10-30 17:40:00 -05:00
declare ( " #include \" cling/Interpreter/ValuePrinter.h \" " ) ;
2012-09-05 09:37:39 +00:00
}
else {
declare ( " #include \" cling/Interpreter/CValuePrinter.h \" " ) ;
}
}
2013-10-08 11:07:43 +02:00
Interpreter : : ~ Interpreter ( ) {
if ( m_ExecutionContext )
m_ExecutionContext - > shuttingDown ( ) ;
2013-10-14 14:31:05 +02:00
assert ( ! m_StoredStates . size ( ) & & " Unbalanced store/compare state. " ) ;
2013-08-02 17:05:12 +02:00
getCI ( ) - > getDiagnostics ( ) . getClient ( ) - > EndSourceFile ( ) ;
2012-09-05 09:37:39 +00:00
}
const char * Interpreter : : getVersion ( ) const {
return " $Id$ " ;
}
void Interpreter : : handleFrontendOptions ( ) {
if ( m_Opts . ShowVersion ) {
2013-01-17 15:27:14 +00:00
llvm : : errs ( ) < < getVersion ( ) < < ' \n ' ;
2012-09-05 09:37:39 +00:00
}
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 ( ) ;
const bool IsFramework = false ;
const bool IsSysRootRelative = true ;
2013-04-24 16:28:08 +00:00
headerOpts . AddPath ( incpath , frontend : : Angled , IsFramework ,
2012-09-05 09:37:39 +00:00
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 ) {
2013-01-17 15:27:14 +00:00
llvm : : errs ( ) < < IncPaths [ i ] < < " \n " ;
2012-11-15 17:07:44 +00:00
}
}
2013-08-26 17:21:07 +02:00
void Interpreter : : storeInterpreterState ( const std : : string & name ) const {
2013-10-09 15:26:04 +02:00
// This may induce deserialization
PushTransactionRAII RAII ( this ) ;
2013-09-17 16:34:41 +02:00
ClangInternalState * state
2013-10-21 17:31:21 +02:00
= new ClangInternalState ( getCI ( ) - > getASTContext ( ) ,
getCI ( ) - > getPreprocessor ( ) , * getModule ( ) , name ) ;
2013-09-17 16:34:41 +02:00
m_StoredStates . push_back ( state ) ;
2013-09-07 00:00:26 +02:00
}
void Interpreter : : compareInterpreterState ( const std : : string & name ) const {
2013-10-09 15:26:04 +02:00
// This may induce deserialization
PushTransactionRAII RAII ( this ) ;
2013-10-25 15:22:45 +02:00
ClangInternalState state ( getCI ( ) - > getASTContext ( ) ,
2013-10-29 10:43:48 +01:00
getCI ( ) - > getPreprocessor ( ) , * getModule ( ) , name ) ;
2013-09-17 16:34:41 +02:00
for ( unsigned i = 0 , e = m_StoredStates . size ( ) ; i ! = e ; + + i ) {
if ( m_StoredStates [ i ] - > getName ( ) = = name ) {
2013-10-09 15:28:06 +02:00
m_StoredStates [ i ] - > compare ( state ) ;
2013-10-14 14:31:05 +02:00
// Remove from the stack and free the storage.
delete * m_StoredStates . erase ( m_StoredStates . begin ( ) + i ) ;
2013-09-17 16:34:41 +02:00
break ;
2013-09-07 00:00:26 +02:00
}
}
}
2013-09-17 16:34:41 +02:00
void Interpreter : : printIncludedFiles ( llvm : : raw_ostream & Out ) const {
ClangInternalState : : printIncludedFiles ( Out , getCI ( ) - > getSourceManager ( ) ) ;
2013-09-07 00:00:26 +02:00
}
2013-09-06 23:58:31 +02:00
2013-09-07 00:00:26 +02:00
2012-11-15 17:07:44 +00:00
// 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 ] ;
2013-04-24 16:28:08 +00:00
if ( E . IsFramework & & E . Group ! = frontend : : Angled )
2012-09-05 09:37:39 +00:00
llvm : : report_fatal_error ( " Invalid option set! " ) ;
2013-04-24 16:28:08 +00:00
switch ( E . Group ) {
case frontend : : After :
if ( withFlags ) incpaths . push_back ( " -idirafter " ) ;
break ;
case frontend : : Quoted :
if ( withFlags ) incpaths . push_back ( " -iquote " ) ;
break ;
case frontend : : System :
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -isystem " ) ;
break ;
case frontend : : IndexHeaderMap :
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -index-header-map " ) ;
if ( withFlags ) incpaths . push_back ( E . IsFramework ? " -F " : " -I " ) ;
break ;
case frontend : : CSystem :
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -c-isystem " ) ;
break ;
case frontend : : ExternCSystem :
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -extern-c-isystem " ) ;
break ;
case frontend : : CXXSystem :
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -cxx-isystem " ) ;
break ;
case frontend : : ObjCSystem :
if ( ! withSystem ) continue ;
if ( withFlags ) incpaths . push_back ( " -objc-isystem " ) ;
break ;
case frontend : : ObjCXXSystem :
2012-11-15 17:07:44 +00:00
if ( ! withSystem ) continue ;
2013-04-24 16:28:08 +00:00
if ( withFlags ) incpaths . push_back ( " -objcxx-isystem " ) ;
break ;
case frontend : : Angled :
if ( withFlags ) incpaths . push_back ( E . IsFramework ? " -F " : " -I " ) ;
break ;
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 ( ) ;
}
2013-12-02 10:49:37 +01:00
llvm : : ExecutionEngine * Interpreter : : getExecutionEngine ( ) const {
if ( ! m_ExecutionContext ) return 0 ;
2012-09-05 09:37:39 +00:00
return m_ExecutionContext - > getExecutionEngine ( ) ;
}
llvm : : Module * Interpreter : : getModule ( ) const {
2013-06-04 16:15:43 +02:00
return m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
2012-09-05 09:37:39 +00:00
}
///\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 */ ,
2013-07-15 13:53:36 +03:00
Transaction * * T /* = 0 */ ) {
2013-03-28 14:34:42 +00:00
if ( isRawInputEnabled ( ) | | ! ShouldWrapInput ( input ) )
2013-07-15 13:53:36 +03:00
return declare ( input , T ) ;
2012-12-06 10:38:56 +00:00
2012-09-05 09:37:39 +00:00
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 ( ) ;
2013-06-17 20:49:08 +02:00
CO . IRDebug = isPrintingIR ( ) ;
2012-09-05 09:37:39 +00:00
2013-07-15 13:53:36 +03:00
if ( EvaluateInternal ( input , CO , V , T ) = = Interpreter : : kFailure ) {
2012-09-05 09:37:39 +00:00
return Interpreter : : kFailure ;
}
return Interpreter : : kSuccess ;
}
2013-07-15 13:53:36 +03:00
Interpreter : : CompilationResult
Interpreter : : parse ( const std : : string & input , Transaction * * T /*=0*/ ) const {
2012-09-05 09:37:39 +00:00
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 ( ) ;
2013-06-17 20:49:08 +02:00
CO . IRDebug = isPrintingIR ( ) ;
2012-09-05 09:37:39 +00:00
2013-07-15 13:53:36 +03:00
return DeclareInternal ( input , CO , T ) ;
2012-09-05 09:37:39 +00:00
}
2013-01-23 14:41:15 +00:00
Interpreter : : CompilationResult
Interpreter : : loadModuleForHeader ( const std : : string & headerFile ) {
Preprocessor & PP = getCI ( ) - > getPreprocessor ( ) ;
//Copied from clang's PPDirectives.cpp
bool isAngled = false ;
// Clang doc says:
// "LookupFrom is set when this is a \#include_next directive, it specifies
// the file to start searching from."
const DirectoryLookup * LookupFrom = 0 ;
const DirectoryLookup * CurDir = 0 ;
2013-09-23 15:36:32 +02:00
ModuleMap : : KnownHeader suggestedModule ;
// PP::LookupFile uses it to issue 'nice' diagnostic
SourceLocation fileNameLoc ;
PP . LookupFile ( fileNameLoc , headerFile , isAngled , LookupFrom , CurDir ,
/*SearchPath*/ 0 , /*RelativePath*/ 0 , & suggestedModule ,
/*SkipCache*/ false ) ;
if ( ! suggestedModule )
2013-01-23 14:41:15 +00:00
return Interpreter : : kFailure ;
2013-09-23 15:36:32 +02:00
// Copied from PPDirectives.cpp
SmallVector < std : : pair < IdentifierInfo * , SourceLocation > , 2 > path ;
for ( Module * mod = suggestedModule . getModule ( ) ; mod ; mod = mod - > Parent ) {
IdentifierInfo * II
= & getSema ( ) . getPreprocessor ( ) . getIdentifierTable ( ) . get ( mod - > Name ) ;
path . push_back ( std : : make_pair ( II , fileNameLoc ) ) ;
}
std : : reverse ( path . begin ( ) , path . end ( ) ) ;
2013-01-23 14:41:15 +00:00
2013-01-24 10:54:57 +00:00
// Pretend that the module came from an inclusion directive, so that clang
// will create an implicit import declaration to capture it in the AST.
bool isInclude = true ;
2013-01-23 14:41:15 +00:00
SourceLocation includeLoc ;
2013-01-24 10:54:57 +00:00
if ( getCI ( ) - > loadModule ( includeLoc , path , Module : : AllVisible , isInclude ) ) {
// After module load we need to "force" Sema to generate the code for
// things like dynamic classes.
getSema ( ) . ActOnEndOfTranslationUnit ( ) ;
2013-01-23 14:41:15 +00:00
return Interpreter : : kSuccess ;
2013-01-24 10:54:57 +00:00
}
2013-01-23 14:41:15 +00:00
return Interpreter : : kFailure ;
}
2012-12-25 04:57:07 +00:00
Interpreter : : CompilationResult
Interpreter : : parseForModule ( const std : : string & input ) {
CompilationOptions CO ;
CO . CodeGeneration = 1 ;
CO . CodeGenerationForModule = 1 ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = 0 ;
CO . ResultEvaluation = 0 ;
CO . DynamicScoping = isDynamicLookupEnabled ( ) ;
CO . Debug = isPrintingAST ( ) ;
2013-06-17 20:49:08 +02:00
CO . IRDebug = isPrintingIR ( ) ;
2012-12-25 04:57:07 +00:00
2013-02-14 17:10:38 +00:00
// When doing parseForModule avoid warning about the user code
// being loaded ... we probably might as well extend this to
// ALL warnings ... but this will suffice for now (working
// around a real bug in QT :().
DiagnosticsEngine & Diag = getCI ( ) - > getDiagnostics ( ) ;
Diag . setDiagnosticMapping ( clang : : diag : : warn_field_is_uninit ,
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
2012-12-25 04:57:07 +00:00
return DeclareInternal ( input , CO ) ;
}
2012-09-05 09:37:39 +00:00
Interpreter : : CompilationResult
2013-07-15 13:53:36 +03:00
Interpreter : : declare ( const std : : string & input , Transaction * * T /*=0 */ ) {
2012-09-05 09:37:39 +00:00
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 ( ) ;
2013-06-17 20:49:08 +02:00
CO . IRDebug = isPrintingIR ( ) ;
2012-09-05 09:37:39 +00:00
2013-07-15 13:53:36 +03:00
return DeclareInternal ( input , CO , T ) ;
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 ( ) ;
2013-06-17 20:49:08 +02:00
CO . IRDebug = isPrintingIR ( ) ;
2013-10-21 14:31:00 -05:00
return EvaluateInternal ( input , CO ) ;
2012-10-18 11:56:20 +00:00
}
2013-06-25 10:53:29 +02:00
Interpreter : : CompilationResult Interpreter : : emitAllDecls ( Transaction * T ) {
2013-06-11 12:36:27 +02:00
assert ( getCodeGenerator ( ) & & " No CodeGenerator? " ) ;
2013-06-25 10:53:29 +02:00
m_IncrParser - > markWholeTransactionAsUsed ( T ) ;
2013-06-25 11:49:04 +02:00
m_IncrParser - > codeGenTransaction ( T ) ;
2013-06-27 14:30:07 +02:00
// The static initializers might run anything and can thus cause more
// decls that need to end up in a transaction. But this one is done
// with CodeGen...
T - > setState ( Transaction : : kCommitted ) ;
2013-07-06 12:56:25 +02:00
if ( runStaticInitializersOnce ( * T ) )
2012-11-21 02:28:31 +00:00
return Interpreter : : kSuccess ;
2013-06-27 14:30:07 +02:00
2012-11-21 02:28:31 +00:00
return Interpreter : : kFailure ;
}
2012-12-11 14:43:40 +00:00
bool Interpreter : : ShouldWrapInput ( const std : : string & input ) {
2013-10-23 10:48:18 -05:00
// TODO: For future reference.
// Parser* P = const_cast<clang::Parser*>(m_IncrParser->getParser());
// Parser::TentativeParsingAction TA(P);
// TPResult result = P->isCXXDeclarationSpecifier();
// TA.Revert();
// return result == TPResult::True();
2012-12-10 11:17:43 +00:00
llvm : : OwningPtr < llvm : : MemoryBuffer > buf ;
buf . reset ( llvm : : MemoryBuffer : : getMemBuffer ( input , " Cling Preparse Buf " ) ) ;
2012-12-11 14:43:40 +00:00
Lexer WrapLexer ( SourceLocation ( ) , getSema ( ) . getLangOpts ( ) , input . c_str ( ) ,
input . c_str ( ) , input . c_str ( ) + input . size ( ) ) ;
2012-12-10 11:17:43 +00:00
Token Tok ;
2013-09-23 15:36:32 +02:00
WrapLexer . LexFromRawLexer ( Tok ) ;
2012-12-10 11:17:43 +00:00
2012-12-11 14:43:40 +00:00
const tok : : TokenKind kind = Tok . getKind ( ) ;
2012-12-10 11:17:43 +00:00
if ( kind = = tok : : raw_identifier & & ! Tok . needsCleaning ( ) ) {
StringRef keyword ( Tok . getRawIdentifierData ( ) , Tok . getLength ( ) ) ;
if ( keyword . equals ( " using " ) )
return false ;
if ( keyword . equals ( " extern " ) )
return false ;
if ( keyword . equals ( " namespace " ) )
return false ;
2013-02-18 13:22:39 +00:00
if ( keyword . equals ( " template " ) )
return false ;
2012-12-10 11:17:43 +00:00
}
2012-12-11 14:43:40 +00:00
else if ( kind = = tok : : hash ) {
2013-09-23 15:36:32 +02:00
WrapLexer . LexFromRawLexer ( Tok ) ;
2012-12-11 14:43:40 +00:00
if ( Tok . is ( tok : : raw_identifier ) & & ! Tok . needsCleaning ( ) ) {
StringRef keyword ( Tok . getRawIdentifierData ( ) , Tok . getLength ( ) ) ;
if ( keyword . equals ( " include " ) )
return false ;
}
2012-12-10 11:17:43 +00:00
}
2012-12-11 14:43:40 +00:00
return true ;
2012-12-10 11:17:43 +00:00
}
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 ;
2013-10-22 17:08:14 -05:00
utils : : Analyze : : maybeMangleDeclName ( FD , mangledNameIfNeeded ) ;
2012-11-15 20:55:43 +00:00
ExecutionContext : : ExecutionResult ExeRes =
m_ExecutionContext - > executeFunction ( mangledNameIfNeeded . c_str ( ) ,
2013-12-04 11:26:02 +01:00
* this ,
2012-11-15 20:55:43 +00:00
FD - > getResultType ( ) , res ) ;
Add support in TClingCallFunc for calling virtual functions: when we have a
CallFunc representation of a member function, we have its concrete address, and
if there is derived class there is no way how to evaluate the vtable and call the
correct virtual.
In order to evaluate the member's vtable, we either have to calculate it (which
is platform/implementation dependent) or create a trampoline function which calls
the target function. Thus codegen will generate code for vtable evaluation. The
trampolines are created for functions that could be possibly virtual - non-static
member functions, declared as virutal. A trampoline looks like this:
void unique_name(Base* This, MyClass2* a, MyClass3* b, double c, MyClass* ret) {
if (ret)
*ret = This->function(*a, *b, c);
else
This->function(*a, *b, c);
}
where we pass in explicitly the address that CallFunc is given (This), followed
by set of arguments (possibly needing conversion) and if the target is non-void
a last argument that is used to provide a storage for the return result.
The given arguments and the required by *LLVM* IR may differ - in that case
simple argument conversions are done. For now primary pointers and builtins, which
covers pretty much entire LLVM type system.
Note that the first implementation is very rough and suboptimal, however it is
very good starting point (milestone):
* Common code is extracted out in its separate routines (here a lot of refactoring more
could be done).
* cling::Value now stores besides clang::QualType, describing (soft-of inaccurately)
the llvm::GenericValue, LLVM-type which is the actual description of the generic value.
* cling::Value naming conventions improved;
* cling::Interpreter::getLLVMType is added temporarily and soon will be removed.
* TClingCallFunc now preallocates space for the this ptr and the return result.
(instead of multiple array copies)
* TClingCallFunc now stores not llvm::GenericValues but cling::StoredValueRefs
because the type conversions need llvm::GenericValue's description (which
essentially is the type).
* Does better default argument evaluation: it could be even *more betterer* if
we query Sema to fold for us the constant expressions instead of decompiling them
to source and using cling for evaluation.
* Return storage implemented.
And, yes that is the longest commit message in my life...
Implementation of argument conversion is revised.
git-svn-id: http://root.cern.ch/svn/root/trunk@48537 27541ba8-7e3a-0410-8455-c3a389f83636
2013-02-11 17:30:03 +00:00
if ( res & & res - > isValid ( ) )
res - > get ( ) . setLLVMType ( getLLVMType ( res - > get ( ) . getClangType ( ) ) ) ;
2012-11-20 16:24:02 +00:00
return ConvertExecutionResult ( ExeRes ) ;
2012-09-05 09:37:39 +00:00
}
2013-12-05 16:24:43 +01:00
const FunctionDecl * Interpreter : : ParseCFunction ( StringRef name ,
StringRef code ,
bool withAccessControl ) {
/*
In CallFunc we currently always ( intentionally and somewhat necessarily )
always fully specify member function template , however this can lead to
an ambiguity with a class template . For example in
roottest / cling / functionTemplate we get :
input_line_171 : 3 : 15 : warning : lookup of ' set ' in member access expression
is ambiguous ; using member of ' t '
( ( t * ) obj ) - > set < int > ( * ( int * ) args [ 0 ] ) ;
^
roottest / cling / functionTemplate / t . h : 19 : 9 : note : lookup in the object type
' t ' refers here
void set ( T targ ) {
^
/ usr / include / c + + / 4.4 .5 / bits / stl_set . h : 87 : 11 : note : lookup from the
current scope refers here
class set
^
This is an intention warning implemented in clang , see
http : //llvm.org/viewvc/llvm-project?view=revision&revision=105518
which ' should have been ' an error :
C + + [ basic . lookup . classref ] requires this to be an error , but ,
because it ' s hard to work around , Clang downgrades it to a warning as
an extension . < / p >
// C++98 [basic.lookup.classref]p1:
// In a class member access expression (5.2.5), if the . or -> token is
// immediately followed by an identifier followed by a <, the identifier
// must be looked up to determine whether the < is the beginning of a
// template argument list (14.2) or a less-than operator. The identifier
// is first looked up in the class of the object expression. If the
// identifier is not found, it is then looked up in the context of the
// entire postfix-expression and shall name a class or function template. If
// the lookup in the class of the object expression finds a template, the
// name is also looked up in the context of the entire postfix-expression
// and
// -- if the name is not found, the name found in the class of the
// object expression is used, otherwise
// -- if the name is found in the context of the entire postfix-expression
// and does not name a class template, the name found in the class of the
// object expression is used, otherwise
// -- if the name found is a class template, it must refer to the same
// entity as the one found in the class of the object expression,
// otherwise the program is ill-formed.
See - Wambiguous - member - template
An alternative to disabling the diagnostics is to use a pointer to
member function :
# include <set>
using namespace std ;
extern " C " int printf ( const char * , . . . ) ;
struct S {
template < typename T >
void set ( T ) { } ;
virtual void virtua ( ) { printf ( " S \n " ) ; }
} ;
struct T : public S {
void virtua ( ) { printf ( " T \n " ) ; }
} ;
int main ( ) {
S * s = new T ( ) ;
typedef void ( S : : * Func_p ) ( int ) ;
Func_p p = & S : : set < int > ;
( s - > * p ) ( 12 ) ;
typedef void ( S : : * Vunc_p ) ( void ) ;
Vunc_p q = & S : : virtua ;
( s - > * q ) ( ) ; // prints "T"
return 0 ;
}
*/
DiagnosticsEngine & Diag = getCI ( ) - > getDiagnostics ( ) ;
Diag . setDiagnosticMapping (
clang : : diag : : ext_nested_name_member_ref_lookup_ambiguous ,
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
LangOptions & LO = const_cast < LangOptions & > ( getCI ( ) - > getLangOpts ( ) ) ;
bool savedAccessControl = LO . AccessControl ;
LO . AccessControl = withAccessControl ;
cling : : Transaction * T = 0 ;
cling : : Interpreter : : CompilationResult CR = declare ( code , & T ) ;
LO . AccessControl = savedAccessControl ;
if ( CR ! = cling : : Interpreter : : kSuccess )
return 0 ;
for ( cling : : Transaction : : const_iterator I = T - > decls_begin ( ) ,
E = T - > decls_end ( ) ; I ! = E ; + + I ) {
if ( I - > m_Call ! = cling : : Transaction : : kCCIHandleTopLevelDecl )
continue ;
2013-12-06 10:30:31 +01:00
if ( const LinkageSpecDecl * LSD
= dyn_cast < LinkageSpecDecl > ( * I - > m_DGR . begin ( ) ) ) {
DeclContext : : decl_iterator DeclBegin = LSD - > decls_begin ( ) ;
if ( DeclBegin = = LSD - > decls_end ( ) )
continue ;
if ( const FunctionDecl * D = dyn_cast < FunctionDecl > ( * DeclBegin ) ) {
const IdentifierInfo * II = D - > getDeclName ( ) . getAsIdentifierInfo ( ) ;
if ( II & & II - > getName ( ) = = name )
return D ;
}
}
2013-12-05 16:24:43 +01:00
}
return 0 ;
}
void *
Interpreter : : compileFunction ( llvm : : StringRef name , llvm : : StringRef code ,
bool ifUnique , bool withAccessControl ) {
//
// Compile the wrapper code.
//
const llvm : : GlobalValue * GV = 0 ;
if ( ifUnique )
GV = getModule ( ) - > getNamedValue ( name ) ;
if ( ! GV ) {
const FunctionDecl * FD = ParseCFunction ( name , code , withAccessControl ) ;
if ( ! FD ) return 0 ;
//
// Get the wrapper function pointer
// from the ExecutionEngine (the JIT).
//
GV = getModule ( ) - > getNamedValue ( name ) ;
}
if ( ! GV )
return 0 ;
return m_ExecutionContext - > getPointerToGlobalFromJIT ( * GV ) ;
}
2012-09-05 09:37:39 +00:00
void Interpreter : : createUniqueName ( std : : string & out ) {
Add support in TClingCallFunc for calling virtual functions: when we have a
CallFunc representation of a member function, we have its concrete address, and
if there is derived class there is no way how to evaluate the vtable and call the
correct virtual.
In order to evaluate the member's vtable, we either have to calculate it (which
is platform/implementation dependent) or create a trampoline function which calls
the target function. Thus codegen will generate code for vtable evaluation. The
trampolines are created for functions that could be possibly virtual - non-static
member functions, declared as virutal. A trampoline looks like this:
void unique_name(Base* This, MyClass2* a, MyClass3* b, double c, MyClass* ret) {
if (ret)
*ret = This->function(*a, *b, c);
else
This->function(*a, *b, c);
}
where we pass in explicitly the address that CallFunc is given (This), followed
by set of arguments (possibly needing conversion) and if the target is non-void
a last argument that is used to provide a storage for the return result.
The given arguments and the required by *LLVM* IR may differ - in that case
simple argument conversions are done. For now primary pointers and builtins, which
covers pretty much entire LLVM type system.
Note that the first implementation is very rough and suboptimal, however it is
very good starting point (milestone):
* Common code is extracted out in its separate routines (here a lot of refactoring more
could be done).
* cling::Value now stores besides clang::QualType, describing (soft-of inaccurately)
the llvm::GenericValue, LLVM-type which is the actual description of the generic value.
* cling::Value naming conventions improved;
* cling::Interpreter::getLLVMType is added temporarily and soon will be removed.
* TClingCallFunc now preallocates space for the this ptr and the return result.
(instead of multiple array copies)
* TClingCallFunc now stores not llvm::GenericValues but cling::StoredValueRefs
because the type conversions need llvm::GenericValue's description (which
essentially is the type).
* Does better default argument evaluation: it could be even *more betterer* if
we query Sema to fold for us the constant expressions instead of decompiling them
to source and using cling for evaluation.
* Return storage implemented.
And, yes that is the longest commit message in my life...
Implementation of argument conversion is revised.
git-svn-id: http://root.cern.ch/svn/root/trunk@48537 27541ba8-7e3a-0410-8455-c3a389f83636
2013-02-11 17:30:03 +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 ( ) {
2013-12-05 16:24:43 +01:00
const size_t size
2012-10-17 08:52:24 +00:00
= 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 ,
2013-07-15 13:53:36 +03:00
Transaction * * T /* = 0 */ ) const {
2013-05-21 16:15:22 +02:00
// Disable warnings which doesn't make sense when using the prompt
// This gets reset with the clang::Diagnostics().Reset()
ignoreFakeDiagnostics ( ) ;
2012-09-05 09:37:39 +00:00
2013-10-24 13:17:31 -05:00
if ( Transaction * lastT = m_IncrParser - > Compile ( input , CO ) ) {
2013-10-24 11:49:32 -05:00
if ( lastT - > getIssuedDiags ( ) ! = Transaction : : kErrors ) {
if ( T )
* T = lastT ;
return Interpreter : : kSuccess ;
}
2013-10-24 13:17:31 -05:00
return Interpreter : : kFailure ;
}
2012-09-05 09:37:39 +00:00
2013-10-24 13:17:31 -05:00
// Even if the transaction was empty it is still success.
return Interpreter : : kSuccess ;
2012-09-05 09:37:39 +00:00
}
Interpreter : : CompilationResult
2012-09-18 15:14:58 +00:00
Interpreter : : EvaluateInternal ( const std : : string & input ,
const CompilationOptions & CO ,
2013-07-15 13:53:36 +03:00
StoredValueRef * V , /* = 0 */
Transaction * * T /* = 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 ) ;
2013-10-24 11:49:32 -05:00
if ( Transaction * lastT = m_IncrParser - > Compile ( Wrapper , CO ) ) {
2013-10-29 13:09:23 -05:00
//FIXME: uncomment when the macro support comes in the transaction
//assert((!V || lastT->size()) && "No decls created!?");
2013-10-24 11:49:32 -05:00
assert ( ( lastT - > getState ( ) = = Transaction : : kCommitted
| | lastT - > getState ( ) = = Transaction : : kRolledBack )
& & " Not committed? " ) ;
2013-10-29 13:09:23 -05:00
if ( lastT - > getIssuedDiags ( ) ! = Transaction : : kErrors ) {
if ( ! lastT - > getWrapperFD ( ) ) // no wrapper to run
return Interpreter : : kSuccess ;
else if ( RunFunction ( lastT - > getWrapperFD ( ) , V ) < kExeFirstError )
2013-10-24 11:49:32 -05:00
return Interpreter : : kSuccess ;
2013-10-29 13:09:23 -05:00
}
2013-10-24 11:49:32 -05:00
if ( V )
* V = StoredValueRef : : invalidValue ( ) ;
2012-09-05 09:37:39 +00:00
2013-10-24 11:49:32 -05:00
return Interpreter : : kFailure ;
}
return Interpreter : : kSuccess ;
2012-09-05 09:37:39 +00:00
}
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 ;
2013-09-24 11:41:22 +02:00
if ( getDynamicLibraryManager ( ) - > loadLibrary ( filename , false , & tryCode )
= = DynamicLibraryManager : : kLoadLibSuccess )
2012-11-16 13:39:09 +00:00
return kSuccess ;
2013-10-31 11:25:20 +01:00
if ( ! tryCode ) {
2013-10-31 14:37:55 +01:00
llvm : : errs ( ) < < " ERROR in cling::Interpreter::loadFile(): cannot find "
< < filename < < " ! \n " ;
2012-11-16 13:39:09 +00:00
return kFailure ;
2013-10-31 11:25:20 +01:00
}
2012-11-16 13:39:09 +00:00
}
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 ) ;
return res ;
2012-09-05 09:37:39 +00:00
}
2013-10-28 23:18:40 -05:00
void Interpreter : : unload ( unsigned numberOfTransactions ) {
m_IncrParser - > unloadLastNTransactions ( numberOfTransactions ) ;
}
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 ( ) ;
2013-05-19 10:57:58 -06:00
// The evaluation should happen on the global scope, because of the wrapper
// that is created.
//
2013-02-14 17:29:30 +00:00
// We can't PushDeclContext, because we don't have scope.
2013-05-19 10:57:58 -06:00
Sema : : ContextRAII pushDC ( TheSema ,
TheSema . getASTContext ( ) . getTranslationUnitDecl ( ) ) ;
2012-09-05 09:37:39 +00:00
2012-10-05 12:09:51 +00:00
StoredValueRef Result ;
2013-02-14 17:29:30 +00:00
getCallbacks ( ) - > SetIsRuntime ( true ) ;
if ( ValuePrinterReq )
echo ( expr , & Result ) ;
else
evaluate ( expr , Result ) ;
getCallbacks ( ) - > SetIsRuntime ( false ) ;
2012-09-05 09:37:39 +00:00
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 ) ;
2013-08-20 11:35:25 +02:00
// FIXME: We should add a multiplexer in the ASTContext, too.
llvm : : OwningPtr < ExternalASTSource > astContextExternalSource ;
astContextExternalSource . reset ( getSema ( ) . getExternalSource ( ) ) ;
clang : : ASTContext & Ctx = getSema ( ) . getASTContext ( ) ;
// FIXME: This is a gross hack. We must make multiplexer in the astcontext,
// or a derived class that extends what we need.
Ctx . ExternalSource . take ( ) ; // FIXME: make sure we delete it.
Ctx . setExternalSource ( astContextExternalSource ) ;
2012-09-05 09:37:39 +00:00
}
2013-08-22 10:28:43 +02:00
//FIXME: Get rid of that.
2013-04-10 15:15:49 +00:00
clang : : ASTDeserializationListener *
Interpreter : : getASTDeserializationListener ( ) const {
2013-08-22 10:28:43 +02:00
if ( ! m_Callbacks )
return 0 ;
return m_Callbacks - > getInterpreterDeserializationListener ( ) ;
2013-04-10 15:15:49 +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 ( ) ) {
2013-02-22 16:35:15 +00:00
if ( loadModuleForHeader ( " cling/Interpreter/DynamicLookupRuntimeUniverse.h " )
! = kSuccess )
declare ( " #include \" cling/Interpreter/DynamicLookupRuntimeUniverse.h \" " ) ;
2012-09-05 09:37:39 +00:00
}
}
2012-11-20 16:24:02 +00:00
Interpreter : : ExecutionResult
2013-06-27 14:29:44 +02:00
Interpreter : : runStaticInitializersOnce ( const Transaction & T ) const {
assert ( m_IncrParser - > hasCodeGenerator ( ) & & " Running on what? " ) ;
assert ( T . getState ( ) = = Transaction : : kCommitted & & " Must be committed " ) ;
2012-09-05 09:37:39 +00:00
// Forward to ExecutionContext; should not be called by
// anyone except for IncrementalParser.
2013-06-04 16:15:43 +02:00
llvm : : Module * module = m_IncrParser - > getCodeGenerator ( ) - > GetModule ( ) ;
2012-11-20 16:24:02 +00:00
ExecutionContext : : ExecutionResult ExeRes
2013-06-04 16:15:43 +02:00
= m_ExecutionContext - > runStaticInitializersOnce ( module ) ;
2013-03-28 14:12:23 +00:00
// Reset the module builder to clean up global initializers, c'tors, d'tors
2013-04-10 11:52:26 +00:00
getCodeGenerator ( ) - > HandleTranslationUnit ( getCI ( ) - > getASTContext ( ) ) ;
2013-03-28 14:12:23 +00:00
2012-11-20 16:24:02 +00:00
return ConvertExecutionResult ( ExeRes ) ;
2012-09-05 09:37:39 +00:00
}
2013-09-19 11:24:31 +02:00
void Interpreter : : runStaticDestructorsOnce ( ) {
m_ExecutionContext - > runStaticDestructorsOnce ( getModule ( ) ) ;
2012-09-05 09:37:39 +00:00
}
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 ,
2012-12-05 10:04:42 +00:00
clang : : diag : : MAP_IGNORE , SourceLocation ( ) ) ;
// Very very ugly. TODO: Revisit and extract out as interpreter arg
Diag . setDiagnosticMapping ( clang : : diag : : ext_auto_type_specifier ,
2012-11-11 12:53:21 +00:00
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 ;
2013-09-19 11:24:31 +02:00
return m_ExecutionContext - > addSymbol ( symbolName , symbolAddress ) ;
2012-09-05 09:37:39 +00:00
}
2013-10-24 13:16:56 -05:00
void * Interpreter : : getAddressOfGlobal ( const GlobalDecl & GD ,
2012-09-24 09:57:43 +00:00
bool * fromJIT /*=0*/ ) const {
// Return a symbol's address, and whether it was jitted.
std : : string mangledName ;
2013-10-24 13:16:56 -05:00
utils : : Analyze : : maybeMangleDeclName ( GD , 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.
2013-06-04 16:15:43 +02: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
}
Add support in TClingCallFunc for calling virtual functions: when we have a
CallFunc representation of a member function, we have its concrete address, and
if there is derived class there is no way how to evaluate the vtable and call the
correct virtual.
In order to evaluate the member's vtable, we either have to calculate it (which
is platform/implementation dependent) or create a trampoline function which calls
the target function. Thus codegen will generate code for vtable evaluation. The
trampolines are created for functions that could be possibly virtual - non-static
member functions, declared as virutal. A trampoline looks like this:
void unique_name(Base* This, MyClass2* a, MyClass3* b, double c, MyClass* ret) {
if (ret)
*ret = This->function(*a, *b, c);
else
This->function(*a, *b, c);
}
where we pass in explicitly the address that CallFunc is given (This), followed
by set of arguments (possibly needing conversion) and if the target is non-void
a last argument that is used to provide a storage for the return result.
The given arguments and the required by *LLVM* IR may differ - in that case
simple argument conversions are done. For now primary pointers and builtins, which
covers pretty much entire LLVM type system.
Note that the first implementation is very rough and suboptimal, however it is
very good starting point (milestone):
* Common code is extracted out in its separate routines (here a lot of refactoring more
could be done).
* cling::Value now stores besides clang::QualType, describing (soft-of inaccurately)
the llvm::GenericValue, LLVM-type which is the actual description of the generic value.
* cling::Value naming conventions improved;
* cling::Interpreter::getLLVMType is added temporarily and soon will be removed.
* TClingCallFunc now preallocates space for the this ptr and the return result.
(instead of multiple array copies)
* TClingCallFunc now stores not llvm::GenericValues but cling::StoredValueRefs
because the type conversions need llvm::GenericValue's description (which
essentially is the type).
* Does better default argument evaluation: it could be even *more betterer* if
we query Sema to fold for us the constant expressions instead of decompiling them
to source and using cling for evaluation.
* Return storage implemented.
And, yes that is the longest commit message in my life...
Implementation of argument conversion is revised.
git-svn-id: http://root.cern.ch/svn/root/trunk@48537 27541ba8-7e3a-0410-8455-c3a389f83636
2013-02-11 17:30:03 +00:00
const llvm : : Type * Interpreter : : getLLVMType ( QualType QT ) {
if ( ! m_IncrParser - > hasCodeGenerator ( ) )
return 0 ;
// Note: The first thing this routine does is getCanonicalType(), so we
// do not need to do that first.
2013-04-10 11:52:26 +00:00
return getCodeGenerator ( ) - > ConvertType ( QT ) ;
Add support in TClingCallFunc for calling virtual functions: when we have a
CallFunc representation of a member function, we have its concrete address, and
if there is derived class there is no way how to evaluate the vtable and call the
correct virtual.
In order to evaluate the member's vtable, we either have to calculate it (which
is platform/implementation dependent) or create a trampoline function which calls
the target function. Thus codegen will generate code for vtable evaluation. The
trampolines are created for functions that could be possibly virtual - non-static
member functions, declared as virutal. A trampoline looks like this:
void unique_name(Base* This, MyClass2* a, MyClass3* b, double c, MyClass* ret) {
if (ret)
*ret = This->function(*a, *b, c);
else
This->function(*a, *b, c);
}
where we pass in explicitly the address that CallFunc is given (This), followed
by set of arguments (possibly needing conversion) and if the target is non-void
a last argument that is used to provide a storage for the return result.
The given arguments and the required by *LLVM* IR may differ - in that case
simple argument conversions are done. For now primary pointers and builtins, which
covers pretty much entire LLVM type system.
Note that the first implementation is very rough and suboptimal, however it is
very good starting point (milestone):
* Common code is extracted out in its separate routines (here a lot of refactoring more
could be done).
* cling::Value now stores besides clang::QualType, describing (soft-of inaccurately)
the llvm::GenericValue, LLVM-type which is the actual description of the generic value.
* cling::Value naming conventions improved;
* cling::Interpreter::getLLVMType is added temporarily and soon will be removed.
* TClingCallFunc now preallocates space for the this ptr and the return result.
(instead of multiple array copies)
* TClingCallFunc now stores not llvm::GenericValues but cling::StoredValueRefs
because the type conversions need llvm::GenericValue's description (which
essentially is the type).
* Does better default argument evaluation: it could be even *more betterer* if
we query Sema to fold for us the constant expressions instead of decompiling them
to source and using cling for evaluation.
* Return storage implemented.
And, yes that is the longest commit message in my life...
Implementation of argument conversion is revised.
git-svn-id: http://root.cern.ch/svn/root/trunk@48537 27541ba8-7e3a-0410-8455-c3a389f83636
2013-02-11 17:30:03 +00:00
}
2012-09-05 09:37:39 +00:00
} // namespace cling