2012-09-05 09:37:39 +00:00
//------------------------------------------------------------------------------
// CLING - the C++ LLVM-based InterpreterG :)
// author: Axel Naumann <axel@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 "IncrementalParser.h"
2014-03-27 17:30:09 +01:00
2016-09-15 14:55:14 -04:00
# include "ASTTransformer.h"
2013-02-27 15:28:38 +00:00
# include "AutoSynthesizer.h"
2015-02-11 21:58:21 +01:00
# include "BackendPasses.h"
2013-10-24 11:54:06 -05:00
# include "CheckEmptyTransactionTransformer.h"
2015-02-18 10:44:12 +01:00
# include "ClingPragmas.h"
2012-09-05 09:37:39 +00:00
# include "DeclCollector.h"
# include "DeclExtractor.h"
# include "DynamicLookup.h"
2015-01-07 10:22:52 +01:00
# include "IncrementalExecutor.h"
2013-07-14 23:35:00 +03:00
# include "NullDerefProtectionTransformer.h"
2013-08-01 12:27:40 +02:00
# include "TransactionPool.h"
2016-09-15 14:55:14 -04:00
# include "ValueExtractionSynthesizer.h"
2012-09-05 09:37:39 +00:00
# include "ValuePrinterSynthesizer.h"
# include "cling/Interpreter/CIFactory.h"
# include "cling/Interpreter/Interpreter.h"
2012-10-02 10:30:25 +00:00
# include "cling/Interpreter/InterpreterCallbacks.h"
2012-10-02 10:34:22 +00:00
# include "cling/Interpreter/Transaction.h"
2016-09-10 15:04:39 -04:00
# include "cling/Utils/Output.h"
2012-09-05 09:37:39 +00:00
# include "clang/AST/ASTContext.h"
2016-09-15 14:55:14 -04:00
# include "clang/AST/Attr.h"
2012-09-05 09:37:39 +00:00
# include "clang/AST/Decl.h"
# include "clang/AST/DeclGroup.h"
2013-03-01 15:15:31 +00:00
# include "clang/AST/RecursiveASTVisitor.h"
2017-01-09 16:32:49 +01:00
# include "clang/Basic/Diagnostic.h"
2012-09-05 09:37:39 +00:00
# include "clang/Basic/FileManager.h"
# include "clang/CodeGen/ModuleBuilder.h"
# include "clang/Frontend/CompilerInstance.h"
2016-09-15 14:55:14 -04:00
# include "clang/Lex/Preprocessor.h"
# include "clang/Parse/Parser.h"
2014-12-02 17:43:18 +01:00
# include "clang/Sema/Sema.h"
2015-01-23 10:49:18 +01:00
# include "clang/Sema/SemaDiagnostic.h"
2012-09-05 09:37:39 +00:00
# include "clang/Serialization/ASTWriter.h"
2013-04-24 16:28:08 +00:00
# include "llvm/IR/LLVMContext.h"
# include "llvm/IR/Module.h"
2012-09-05 09:37:39 +00:00
# include "llvm/Support/CrashRecoveryContext.h"
# include "llvm/Support/MemoryBuffer.h"
# include <iostream>
# include <sstream>
2016-09-15 14:55:14 -04:00
# include <stdio.h>
2014-10-14 08:58:40 +02:00
2012-09-05 09:37:39 +00:00
using namespace clang ;
2014-10-14 08:58:40 +02:00
namespace {
2016-07-05 03:37:26 -04:00
static const Token * getMacroToken ( const Preprocessor & PP , const char * Macro ) {
if ( const IdentifierInfo * II = PP . getIdentifierInfo ( Macro ) ) {
if ( const DefMacroDirective * MD = llvm : : dyn_cast_or_null
< DefMacroDirective > ( PP . getLocalMacroDirective ( II ) ) ) {
if ( const clang : : MacroInfo * MI = MD - > getMacroInfo ( ) ) {
if ( MI - > getNumTokens ( ) = = 1 )
return MI - > tokens_begin ( ) ;
}
}
}
return nullptr ;
}
2014-10-14 08:58:40 +02:00
///\brief Check the compile-time C++ ABI version vs the run-time ABI version,
/// a mismatch could cause havoc. Reports if ABI versions differ.
2016-07-05 03:37:26 -04:00
static bool CheckABICompatibility ( clang : : CompilerInstance * CI ) {
2016-10-04 17:06:20 -04:00
# if defined(__GLIBCXX__)
# define CLING_CXXABI_VERS std::to_string(__GLIBCXX__)
const char * CLING_CXXABI_NAME = " __GLIBCXX__ " ;
# elif defined(_LIBCPP_VERSION)
# define CLING_CXXABI_VERS std::to_string(_LIBCPP_VERSION)
const char * CLING_CXXABI_NAME = " _LIBCPP_VERSION " ;
# elif defined(_CRT_MSVCP_CURRENT)
# define CLING_CXXABI_VERS _CRT_MSVCP_CURRENT
const char * CLING_CXXABI_NAME = " _CRT_MSVCP_CURRENT " ;
# else
# error "Unknown platform for ABI check";
# endif
2016-07-05 03:37:26 -04:00
llvm : : StringRef CurABI ;
const clang : : Preprocessor & PP = CI - > getPreprocessor ( ) ;
2016-10-04 17:06:20 -04:00
const clang : : Token * Tok = getMacroToken ( PP , CLING_CXXABI_NAME ) ;
2016-07-05 03:37:26 -04:00
if ( Tok & & Tok - > isLiteral ( ) ) {
2016-10-04 17:06:20 -04:00
// Strip any quotation marks.
CurABI = llvm : : StringRef ( Tok - > getLiteralData ( ) ,
Tok - > getLength ( ) ) . trim ( " \" " ) ;
if ( CurABI . equals ( CLING_CXXABI_VERS ) )
2016-07-05 03:37:26 -04:00
return true ;
}
2014-10-14 08:58:40 +02:00
2016-09-10 15:04:39 -04:00
cling : : errs ( ) < <
2016-07-05 03:37:26 -04:00
" Warning in cling::IncrementalParser::CheckABICompatibility(): \n "
2016-10-04 17:06:20 -04:00
" Possible C++ standard library mismatch, compiled with "
< < CLING_CXXABI_NAME < < " ' " < < CLING_CXXABI_VERS < < " ' \n "
2016-07-05 03:37:26 -04:00
" Extraction of runtime standard library version was: ' "
< < CurABI < < " ' \n " ;
2016-10-04 17:06:20 -04:00
2016-07-05 03:37:26 -04:00
return false ;
2014-10-14 08:58:40 +02:00
}
2017-01-09 16:32:49 +01:00
class FilteringDiagConsumer : public ForwardingDiagnosticConsumer {
std : : unique_ptr < DiagnosticConsumer > fOwnedTarget ;
2017-01-12 10:54:30 +01:00
std : : stack < bool > fIgnorePromptDiags ;
2017-01-09 16:32:49 +01:00
2017-01-12 10:54:13 +01:00
void SyncDiagCountWithTarget ( ) {
NumWarnings = fOwnedTarget - > getNumWarnings ( ) ;
NumErrors = fOwnedTarget - > getNumErrors ( ) ;
}
2017-01-09 16:32:49 +01:00
public :
FilteringDiagConsumer ( std : : unique_ptr < DiagnosticConsumer > & & Target ) :
2017-01-12 10:54:30 +01:00
ForwardingDiagnosticConsumer ( * Target ) ,
2017-01-09 16:32:49 +01:00
fOwnedTarget ( std : : move ( Target ) ) { }
void BeginSourceFile ( const LangOptions & LangOpts ,
const Preprocessor * PP = nullptr ) override {
fOwnedTarget - > BeginSourceFile ( LangOpts , PP ) ;
}
void EndSourceFile ( ) override {
fOwnedTarget - > EndSourceFile ( ) ;
2017-01-12 10:54:13 +01:00
SyncDiagCountWithTarget ( ) ;
2017-01-09 16:32:49 +01:00
}
void finish ( ) override {
fOwnedTarget - > finish ( ) ;
2017-01-12 10:54:13 +01:00
SyncDiagCountWithTarget ( ) ;
2017-01-09 16:32:49 +01:00
}
void clear ( ) override {
fOwnedTarget - > clear ( ) ;
2017-01-12 10:54:13 +01:00
SyncDiagCountWithTarget ( ) ;
2017-01-09 16:32:49 +01:00
}
void HandleDiagnostic ( DiagnosticsEngine : : Level DiagLevel ,
const Diagnostic & Info ) override {
if ( Ignoring ( ) ) {
if ( Info . getID ( ) = = diag : : warn_unused_expr
| | Info . getID ( ) = = diag : : warn_unused_call
2017-01-10 13:58:24 +01:00
| | Info . getID ( ) = = diag : : warn_unused_comparison )
2017-01-09 16:32:49 +01:00
return ; // ignore!
if ( Info . getID ( ) = = diag : : warn_falloff_nonvoid_function ) {
DiagLevel = DiagnosticsEngine : : Error ;
2017-01-10 13:58:24 +01:00
}
if ( Info . getID ( ) = = diag : : ext_return_has_expr ) {
// An error that we need to suppress.
2017-01-09 16:32:49 +01:00
auto Diags = const_cast < DiagnosticsEngine * > ( Info . getDiags ( ) ) ;
2017-01-10 13:58:24 +01:00
assert ( Diags - > hasErrorOccurred ( ) & & " Expected ErrorOccurred " ) ;
if ( fOwnedTarget - > getNumErrors ( ) = = 0 ) { // first error
Diags - > Reset ( true /*soft - only counts, not mappings*/ ) ;
} // else we had other errors, too.
return ; // ignore!
2017-01-09 16:32:49 +01:00
}
}
ForwardingDiagnosticConsumer : : HandleDiagnostic ( DiagLevel , Info ) ;
2017-01-12 10:54:13 +01:00
SyncDiagCountWithTarget ( ) ;
2017-01-09 16:32:49 +01:00
}
void push ( bool ignoreDiags ) { fIgnorePromptDiags . push ( ignoreDiags ) ; }
void pop ( ) {
assert ( ! fIgnorePromptDiags . empty ( ) & & " Unignoring non-ignored prompt diags! " ) ;
fIgnorePromptDiags . pop ( ) ;
}
bool Ignoring ( ) const {
return ! fIgnorePromptDiags . empty ( ) & & fIgnorePromptDiags . top ( ) ;
}
} ;
2014-10-14 08:58:40 +02:00
} // unnamed namespace
2012-09-05 09:37:39 +00:00
namespace cling {
2016-08-08 16:29:06 -04:00
IncrementalParser : : IncrementalParser ( Interpreter * interp , const char * llvmdir ) :
m_Interpreter ( interp ) ,
m_CI ( CIFactory : : createCI ( " " , interp - > getOptions ( ) , llvmdir ) ) ,
m_Consumer ( nullptr ) , m_ModuleNo ( 0 ) {
2012-09-05 09:37:39 +00:00
2016-07-02 00:20:48 -04:00
if ( ! m_CI ) {
cling : : errs ( ) < < " Compiler instance could not be created. \n " ;
return ;
}
2012-09-05 09:37:39 +00:00
2016-07-02 00:20:48 -04:00
m_Consumer = dyn_cast < DeclCollector > ( & m_CI - > getSema ( ) . getASTConsumer ( ) ) ;
if ( ! m_Consumer ) {
cling : : errs ( ) < < " No AST consumer available. \n " ;
return ;
}
2017-01-09 16:32:49 +01:00
DiagnosticsEngine & Diag = m_CI - > getDiagnostics ( ) ;
2016-08-08 16:29:06 -04:00
if ( m_CI - > getFrontendOpts ( ) . ProgramAction ! = frontend : : ParseSyntaxOnly ) {
m_CodeGen . reset ( CreateLLVMCodeGen (
2017-01-09 16:32:49 +01:00
Diag , " cling-module-0 " , m_CI - > getHeaderSearchOpts ( ) ,
2016-08-08 16:29:06 -04:00
m_CI - > getPreprocessorOpts ( ) , m_CI - > getCodeGenOpts ( ) ,
* m_Interpreter - > getLLVMContext ( ) ) ) ;
2015-03-16 14:49:28 +01:00
m_Consumer - > setContext ( this , m_CodeGen . get ( ) ) ;
} else {
m_Consumer - > setContext ( this , 0 ) ;
2012-09-05 09:37:39 +00:00
}
2017-01-09 16:32:49 +01:00
DiagnosticConsumer * DC
2017-01-10 16:00:31 +01:00
= new FilteringDiagConsumer ( Diag . takeClient ( ) ) ;
2017-01-09 16:32:49 +01:00
Diag . setClient ( DC , true /*own*/ ) ;
2013-08-22 14:11:12 +02:00
initializeVirtualFile ( ) ;
2013-03-11 13:11:15 +00:00
}
2013-06-26 20:51:05 +02:00
2016-07-02 00:20:48 -04:00
bool
2015-03-18 09:38:35 +01:00
IncrementalParser : : Initialize ( llvm : : SmallVectorImpl < ParseResultTransaction > &
2016-01-07 17:19:20 +01:00
result , bool isChildInterpreter ) {
2016-06-26 19:43:40 -04:00
m_TransactionPool . reset ( new TransactionPool ) ;
2016-11-15 15:12:47 +01:00
if ( hasCodeGenerator ( ) )
2013-03-11 13:11:15 +00:00
getCodeGenerator ( ) - > Initialize ( getCI ( ) - > getASTContext ( ) ) ;
2013-05-23 11:05:02 +02:00
2013-03-26 09:59:19 +00:00
CompilationOptions CO ;
CO . DeclarationExtraction = 0 ;
CO . ValuePrinting = CompilationOptions : : VPDisabled ;
CO . CodeGeneration = hasCodeGenerator ( ) ;
2013-05-23 14:44:52 +02:00
2016-05-18 16:08:14 +02:00
Transaction * CurT = beginTransaction ( CO ) ;
Preprocessor & PP = m_CI - > getPreprocessor ( ) ;
2016-07-02 00:20:48 -04:00
DiagnosticsEngine & Diags = m_CI - > getSema ( ) . getDiagnostics ( ) ;
2016-05-18 16:08:14 +02:00
// Pull in PCH.
2013-05-23 14:44:52 +02:00
const std : : string & PCHFileName
2015-07-03 17:41:33 +02:00
= m_CI - > getInvocation ( ) . getPreprocessorOpts ( ) . ImplicitPCHInclude ;
2013-03-11 13:11:15 +00:00
if ( ! PCHFileName . empty ( ) ) {
2013-04-27 01:52:55 +02:00
Transaction * CurT = beginTransaction ( CO ) ;
2013-03-11 13:11:15 +00:00
m_CI - > createPCHExternalASTSource ( PCHFileName ,
true /*DisablePCHValidation*/ ,
true /*AllowPCHWithCompilerErrors*/ ,
2014-06-06 10:19:14 +02:00
0 /*DeserializationListener*/ ,
true /*OwnsDeserializationListener*/ ) ;
2015-03-18 09:38:35 +01:00
result . push_back ( endTransaction ( CurT ) ) ;
2013-03-11 13:11:15 +00:00
}
2012-09-05 09:37:39 +00:00
2015-02-18 10:44:12 +01:00
addClingPragmas ( * m_Interpreter ) ;
2016-05-18 16:08:14 +02:00
// Must happen after attaching the PCH, else PCH elements will end up
// being lexed.
PP . EnterMainSourceFile ( ) ;
Sema * TheSema = & m_CI - > getSema ( ) ;
2016-07-02 00:20:48 -04:00
m_Parser . reset ( new Parser ( PP , * TheSema , false /*skipFuncBodies*/ ) ) ;
2016-05-18 16:08:14 +02:00
// Initialize the parser after PP has entered the main source file.
2012-09-05 09:37:39 +00:00
m_Parser - > Initialize ( ) ;
2013-03-11 13:11:15 +00:00
ExternalASTSource * External = TheSema - > getASTContext ( ) . getExternalSource ( ) ;
if ( External )
External - > StartTranslationUnit ( m_Consumer ) ;
2013-04-27 01:52:55 +02:00
2016-11-09 17:53:50 +01:00
// Start parsing the "main file" to warm up lexing (enter caching lex mode
// for ParseInternal()'s call EnterSourceFile() to make sense.
2017-01-29 14:39:53 -05:00
while ( ! m_Parser - > ParseTopLevelDecl ( ) ) { }
2016-11-09 17:53:50 +01:00
2015-12-18 16:12:00 +01:00
// If I belong to the parent Interpreter, only then do
2016-01-06 14:48:00 +01:00
// the #include <new>
2016-01-07 17:19:20 +01:00
if ( ! isChildInterpreter & & m_CI - > getLangOpts ( ) . CPlusPlus ) {
2015-02-21 22:37:50 +01:00
// <new> is needed by the ValuePrinter so it's a good thing to include it.
// We need to include it to determine the version number of the standard
// library implementation.
ParseInternal ( " #include <new> " ) ;
// That's really C++ ABI compatibility. C has other problems ;-)
CheckABICompatibility ( m_CI . get ( ) ) ;
}
2014-10-14 08:58:40 +02:00
2014-03-23 21:08:48 +01:00
// DO NOT commit the transactions here: static initialization in these
// transactions requires gCling through local_cxa_atexit(), but that has not
// been defined yet!
2015-03-16 11:25:01 +01:00
ParseResultTransaction PRT = endTransaction ( CurT ) ;
2015-03-18 09:38:35 +01:00
result . push_back ( PRT ) ;
2016-07-02 00:20:48 -04:00
return true ;
}
bool IncrementalParser : : isValid ( bool initialized ) const {
return m_CI & & m_CI - > hasFileManager ( ) & & m_Consumer
& & ! m_VirtualFileID . isInvalid ( )
& & ( ! initialized | | ( m_TransactionPool & & m_Parser ) ) ;
2012-09-05 09:37:39 +00:00
}
2014-05-14 14:29:44 +02:00
const Transaction * IncrementalParser : : getCurrentTransaction ( ) const {
return m_Consumer - > getTransaction ( ) ;
}
2014-04-01 16:44:59 +02:00
SourceLocation IncrementalParser : : getLastMemoryBufferEndLoc ( ) const {
const SourceManager & SM = getCI ( ) - > getSourceManager ( ) ;
SourceLocation Result = SM . getLocForStartOfFile ( m_VirtualFileID ) ;
return Result . getLocWithOffset ( m_MemoryBuffers . size ( ) + 1 ) ;
}
2012-09-05 09:37:39 +00:00
IncrementalParser : : ~ IncrementalParser ( ) {
2016-06-26 19:43:40 -04:00
Transaction * T = const_cast < Transaction * > ( getFirstTransaction ( ) ) ;
2012-11-20 13:21:00 +00:00
while ( T ) {
2013-09-15 13:18:49 -05:00
assert ( ( T - > getState ( ) = = Transaction : : kCommitted
2014-08-03 21:05:42 -05:00
| | T - > getState ( ) = = Transaction : : kRolledBackWithErrors
2013-11-27 14:28:40 +01:00
| | T - > getState ( ) = = Transaction : : kNumStates // reset from the pool
2013-06-17 12:45:18 +02:00
| | T - > getState ( ) = = Transaction : : kRolledBack )
& & " Not committed? " ) ;
2016-06-26 19:43:40 -04:00
const Transaction * nextT = T - > getNext ( ) ;
m_TransactionPool - > releaseTransaction ( T , false ) ;
T = const_cast < Transaction * > ( nextT ) ;
2012-11-20 13:21:00 +00:00
}
2012-09-05 09:37:39 +00:00
}
2015-02-18 14:05:30 +01:00
void IncrementalParser : : addTransaction ( Transaction * T ) {
if ( ! T - > isNestedTransaction ( ) & & T ! = getLastTransaction ( ) ) {
if ( getLastTransaction ( ) )
m_Transactions . back ( ) - > setNext ( T ) ;
m_Transactions . push_back ( T ) ;
}
}
2014-08-03 21:05:42 -05:00
Transaction * IncrementalParser : : beginTransaction ( const CompilationOptions &
2012-11-14 22:06:58 +00:00
Opts ) {
2012-09-05 09:37:39 +00:00
Transaction * OldCurT = m_Consumer - > getTransaction ( ) ;
2016-06-26 19:43:40 -04:00
Transaction * NewCurT = m_TransactionPool - > takeTransaction ( m_CI - > getSema ( ) ) ;
2013-07-06 13:04:31 +02:00
NewCurT - > setCompilationOpts ( Opts ) ;
2014-08-03 21:05:42 -05:00
// If we are in the middle of transaction and we see another begin
2012-09-05 09:37:39 +00:00
// transaction - it must be nested transaction.
2014-08-03 21:05:42 -05:00
if ( OldCurT & & OldCurT ! = NewCurT
2013-08-04 18:17:41 +02:00
& & ( OldCurT - > getState ( ) = = Transaction : : kCollecting
| | OldCurT - > getState ( ) = = Transaction : : kCompleted ) ) {
2013-07-06 13:04:31 +02:00
OldCurT - > addNestedTransaction ( NewCurT ) ; // takes the ownership
2012-09-05 09:37:39 +00:00
}
2013-04-24 06:55:38 +00:00
m_Consumer - > setTransaction ( NewCurT ) ;
2012-11-14 22:06:58 +00:00
return NewCurT ;
2012-09-05 09:37:39 +00:00
}
2015-03-16 11:25:01 +01:00
IncrementalParser : : ParseResultTransaction
IncrementalParser : : endTransaction ( Transaction * T ) {
2013-04-27 01:52:55 +02:00
assert ( T & & " Null transaction!? " ) ;
assert ( T - > getState ( ) = = Transaction : : kCollecting ) ;
2013-06-27 12:34:05 +02:00
# ifndef NDEBUG
if ( T - > hasNestedTransactions ( ) ) {
for ( Transaction : : const_nested_iterator I = T - > nested_begin ( ) ,
E = T - > nested_end ( ) ; I ! = E ; + + I )
assert ( ( * I ) - > isCompleted ( ) & & " Nested transaction not completed!? " ) ;
}
# endif
2013-04-27 01:52:55 +02:00
T - > setState ( Transaction : : kCompleted ) ;
2015-03-16 11:25:01 +01:00
2015-03-16 20:26:27 +01:00
DiagnosticsEngine & Diags = getCI ( ) - > getSema ( ) . getDiagnostics ( ) ;
2015-03-16 11:25:01 +01:00
//TODO: Make the enum orable.
EParseResult ParseResult = kSuccess ;
2015-03-16 20:53:42 +01:00
if ( Diags . hasErrorOccurred ( ) | | Diags . hasFatalErrorOccurred ( )
| | T - > getIssuedDiags ( ) = = Transaction : : kErrors ) {
2015-03-16 11:25:01 +01:00
T - > setIssuedDiags ( Transaction : : kErrors ) ;
ParseResult = kFailed ;
2015-03-16 20:53:42 +01:00
} else if ( Diags . getNumWarnings ( ) > 0 ) {
T - > setIssuedDiags ( Transaction : : kWarnings ) ;
ParseResult = kSuccessWithWarnings ;
2015-03-16 11:25:01 +01:00
}
2015-06-08 14:48:24 +02:00
// Empty transaction, send it back to the pool.
2013-07-06 13:04:31 +02:00
if ( T - > empty ( ) ) {
2014-08-03 21:05:42 -05:00
assert ( ( ! m_Consumer - > getTransaction ( )
2013-08-05 15:15:37 +02:00
| | ( m_Consumer - > getTransaction ( ) = = T ) )
2013-08-04 18:18:27 +02:00
& & " Cannot release different T " ) ;
2014-08-03 21:05:42 -05:00
// If a nested transaction the active one should be its parent
2013-08-04 18:18:27 +02:00
// from now on. FIXME: Merge conditional with commitTransaction
if ( T - > isNestedTransaction ( ) )
m_Consumer - > setTransaction ( T - > getParent ( ) ) ;
else
m_Consumer - > setTransaction ( ( Transaction * ) 0 ) ;
2013-07-06 13:04:31 +02:00
m_TransactionPool - > releaseTransaction ( T ) ;
2015-03-16 11:25:01 +01:00
return ParseResultTransaction ( nullptr , ParseResult ) ;
2013-10-24 11:54:06 -05:00
}
2013-07-06 13:04:31 +02:00
2015-02-18 14:05:30 +01:00
addTransaction ( T ) ;
2015-03-16 11:25:01 +01:00
return ParseResultTransaction ( T , ParseResult ) ;
2012-09-05 09:37:39 +00:00
}
2017-01-29 14:39:53 -05:00
void IncrementalParser : : commitTransaction ( ParseResultTransaction & PRT ,
bool ClearDiagClient ) {
2015-03-18 09:38:35 +01:00
Transaction * T = PRT . getPointer ( ) ;
2015-05-28 11:44:26 +02:00
if ( ! T ) {
if ( PRT . getInt ( ) ! = kSuccess ) {
// Nothing has been emitted to Codegen, reset the Diags.
DiagnosticsEngine & Diags = getCI ( ) - > getSema ( ) . getDiagnostics ( ) ;
Diags . Reset ( /*soft=*/ true ) ;
2017-01-29 14:39:53 -05:00
if ( ClearDiagClient )
Diags . getClient ( ) - > clear ( ) ;
2015-05-28 11:44:26 +02:00
}
2015-03-18 09:38:35 +01:00
return ;
2015-05-28 11:44:26 +02:00
}
2015-03-18 09:38:35 +01:00
2012-11-14 22:06:58 +00:00
assert ( T - > isCompleted ( ) & & " Transaction not ended!? " ) ;
2013-04-09 15:30:16 +00:00
assert ( T - > getState ( ) ! = Transaction : : kCommitted
& & " Committing an already committed transaction. " ) ;
2015-03-16 20:08:37 +01:00
assert ( ( T - > getIssuedDiags ( ) = = Transaction : : kErrors | | ! T - > empty ( ) )
& & " Valid Transactions must not be empty; " ) ;
2012-09-05 09:37:39 +00:00
2013-06-25 09:19:15 +02:00
// If committing a nested transaction the active one should be its parent
// from now on.
2013-06-25 10:51:58 +02:00
if ( T - > isNestedTransaction ( ) )
2013-06-25 09:19:15 +02:00
m_Consumer - > setTransaction ( T - > getParent ( ) ) ;
2012-09-05 09:37:39 +00:00
// Check for errors...
2012-11-14 22:06:58 +00:00
if ( T - > getIssuedDiags ( ) = = Transaction : : kErrors ) {
2015-03-16 14:49:28 +01:00
// Make module visible to TransactionUnloader.
bool MustStartNewModule = false ;
if ( ! T - > isNestedTransaction ( ) & & hasCodeGenerator ( ) ) {
MustStartNewModule = true ;
std : : unique_ptr < llvm : : Module > M ( getCodeGenerator ( ) - > ReleaseModule ( ) ) ;
if ( M ) {
T - > setModule ( std : : move ( M ) ) ;
}
}
2015-05-28 11:44:26 +02:00
// Module has been released from Codegen, reset the Diags now.
DiagnosticsEngine & Diags = getCI ( ) - > getSema ( ) . getDiagnostics ( ) ;
Diags . Reset ( /*soft=*/ true ) ;
2017-01-29 14:39:53 -05:00
if ( ClearDiagClient )
Diags . getClient ( ) - > clear ( ) ;
2015-05-28 11:44:26 +02:00
2016-06-25 00:14:00 +02:00
PRT . setPointer ( nullptr ) ;
PRT . setInt ( kFailed ) ;
2016-06-23 20:38:57 +03:00
m_Interpreter - > unload ( * T ) ;
2015-03-16 14:49:28 +01:00
if ( MustStartNewModule ) {
// Create a new module.
2016-12-14 15:32:18 -05:00
stdstrstream ModuleName ;
ModuleName < < " cling-module- " < < + + m_ModuleNo ;
getCodeGenerator ( ) - > StartModule ( ModuleName . str ( ) ,
2015-03-17 14:55:07 +01:00
* m_Interpreter - > getLLVMContext ( ) ,
getCI ( ) - > getCodeGenOpts ( ) ) ;
2015-03-16 14:49:28 +01:00
}
2012-09-05 09:37:39 +00:00
return ;
}
2012-11-14 22:06:58 +00:00
if ( T - > hasNestedTransactions ( ) ) {
2015-03-18 09:38:35 +01:00
Transaction * TopmostParent = T - > getTopmostParent ( ) ;
EParseResult PR = kSuccess ;
if ( TopmostParent - > getIssuedDiags ( ) = = Transaction : : kErrors )
PR = kFailed ;
else if ( TopmostParent - > getIssuedDiags ( ) = = Transaction : : kWarnings )
PR = kSuccessWithWarnings ;
2013-07-06 13:03:21 +02:00
for ( Transaction : : const_nested_iterator I = T - > nested_begin ( ) ,
2013-04-08 14:35:35 +00:00
E = T - > nested_end ( ) ; I ! = E ; + + I )
2016-06-25 00:14:00 +02:00
if ( ( * I ) - > getState ( ) ! = Transaction : : kCommitted ) {
ParseResultTransaction PRT ( * I , PR ) ;
commitTransaction ( PRT ) ;
}
2012-11-14 22:06:58 +00:00
}
2013-07-06 13:01:48 +02:00
// If there was an error coming from the transformers.
if ( T - > getIssuedDiags ( ) = = Transaction : : kErrors ) {
2016-06-23 20:38:57 +03:00
m_Interpreter - > unload ( * T ) ;
2013-06-25 10:51:58 +02:00
return ;
2013-07-06 13:01:48 +02:00
}
2012-09-05 09:37:39 +00:00
2013-06-13 14:53:58 +02:00
// Here we expect a template instantiation. We need to open the transaction
// that we are currently work with.
2013-07-06 13:04:31 +02:00
{
2014-04-16 11:59:36 +02:00
Transaction * prevConsumerT = m_Consumer - > getTransaction ( ) ;
m_Consumer - > setTransaction ( T ) ;
2013-07-06 13:04:31 +02:00
Transaction * nestedT = beginTransaction ( CompilationOptions ( ) ) ;
// Pull all template instantiations in that came from the consumers.
getCI ( ) - > getSema ( ) . PerformPendingInstantiations ( ) ;
2015-06-07 20:38:49 +02:00
ParseResultTransaction nestedPRT = endTransaction ( nestedT ) ;
commitTransaction ( nestedPRT ) ;
2014-04-16 11:59:36 +02:00
m_Consumer - > setTransaction ( prevConsumerT ) ;
2013-07-06 13:04:31 +02:00
}
2013-06-13 14:53:58 +02:00
m_Consumer - > HandleTranslationUnit ( getCI ( ) - > getASTContext ( ) ) ;
2013-04-10 15:15:49 +00:00
2013-06-25 10:51:58 +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...
2013-07-06 13:04:31 +02:00
if ( T - > getCompilationOpts ( ) . CodeGeneration & & hasCodeGenerator ( ) ) {
2014-04-16 11:59:36 +02:00
Transaction * prevConsumerT = m_Consumer - > getTransaction ( ) ;
m_Consumer - > setTransaction ( T ) ;
2013-07-06 13:04:31 +02:00
codeGenTransaction ( T ) ;
T - > setState ( Transaction : : kCommitted ) ;
2014-04-16 12:01:44 +02:00
if ( ! T - > getParent ( ) ) {
2015-02-03 21:32:09 +01:00
if ( m_Interpreter - > executeTransaction ( * T )
2014-04-16 12:01:44 +02:00
> = Interpreter : : kExeFirstError ) {
2015-01-09 09:42:00 +01:00
// Roll back on error in initializers
//assert(0 && "Error on inits.");
2016-06-23 20:38:57 +03:00
m_Interpreter - > unload ( * T ) ;
2015-01-09 09:42:00 +01:00
T - > setState ( Transaction : : kRolledBackWithErrors ) ;
2014-04-16 12:01:44 +02:00
return ;
}
2013-07-06 13:04:31 +02:00
}
2014-04-16 11:59:36 +02:00
m_Consumer - > setTransaction ( prevConsumerT ) ;
2013-07-06 13:04:31 +02:00
}
2013-06-25 10:51:58 +02:00
T - > setState ( Transaction : : kCommitted ) ;
2013-04-09 15:30:16 +00:00
2013-08-02 16:20:54 +02:00
if ( InterpreterCallbacks * callbacks = m_Interpreter - > getCallbacks ( ) )
2013-06-25 10:51:58 +02:00
callbacks - > TransactionCommitted ( * T ) ;
2013-08-02 16:20:54 +02:00
2012-09-05 09:37:39 +00:00
}
2015-03-16 11:23:01 +01:00
void IncrementalParser : : emitTransaction ( Transaction * T ) {
for ( auto DI = T - > decls_begin ( ) , DE = T - > decls_end ( ) ; DI ! = DE ; + + DI )
m_Consumer - > HandleTopLevelDecl ( DI - > m_DGR ) ;
}
2013-06-25 10:51:58 +02:00
void IncrementalParser : : codeGenTransaction ( Transaction * T ) {
// codegen the transaction
assert ( T - > getCompilationOpts ( ) . CodeGeneration & & " CodeGen turned off " ) ;
2013-07-31 09:04:30 +02:00
assert ( T - > getState ( ) = = Transaction : : kCompleted & & " Must be completed " ) ;
2013-06-25 10:51:58 +02:00
assert ( hasCodeGenerator ( ) & & " No CodeGen " ) ;
2013-08-04 18:39:41 +02:00
// Could trigger derserialization of decls.
Transaction * deserT = beginTransaction ( CompilationOptions ( ) ) ;
2013-08-07 14:26:15 +02:00
2015-01-07 10:22:52 +01:00
2015-02-04 15:34:59 +01:00
// Commit this transaction first - T might need symbols from it, so
// trigger emission of weak symbols by providing use.
2015-03-16 11:25:01 +01:00
ParseResultTransaction PRT = endTransaction ( deserT ) ;
2015-03-18 09:38:35 +01:00
commitTransaction ( PRT ) ;
2016-06-25 00:14:00 +02:00
deserT = PRT . getPointer ( ) ;
2015-02-04 15:34:59 +01:00
2015-01-07 10:22:52 +01:00
// This llvm::Module is done; finalize it and pass it to the execution
// engine.
if ( ! T - > isNestedTransaction ( ) & & hasCodeGenerator ( ) ) {
2015-03-16 15:43:40 +01:00
// The initializers are emitted to the symbol "_GLOBAL__sub_I_" + filename.
// Make that unique!
ASTContext & Context = getCI ( ) - > getASTContext ( ) ;
SourceManager & SM = Context . getSourceManager ( ) ;
const FileEntry * MainFile = SM . getFileEntryForID ( SM . getMainFileID ( ) ) ;
FileEntry * NcMainFile = const_cast < FileEntry * > ( MainFile ) ;
// Hack to temporarily set the file entry's name to a unique name.
assert ( MainFile - > getName ( ) = = * ( const char * * ) NcMainFile
& & " FileEntry does not start with the name " ) ;
const char * & FileName = * ( const char * * ) NcMainFile ;
const char * OldName = FileName ;
std : : string ModName = getCodeGenerator ( ) - > GetModule ( ) - > getName ( ) . str ( ) ;
FileName = ModName . c_str ( ) ;
deserT = beginTransaction ( CompilationOptions ( ) ) ;
// Reset the module builder to clean up global initializers, c'tors, d'tors
getCodeGenerator ( ) - > HandleTranslationUnit ( Context ) ;
FileName = OldName ;
2016-06-25 00:14:00 +02:00
auto PRT = endTransaction ( deserT ) ;
commitTransaction ( PRT ) ;
deserT = PRT . getPointer ( ) ;
2015-03-16 15:43:40 +01:00
2015-01-15 15:13:14 +01:00
std : : unique_ptr < llvm : : Module > M ( getCodeGenerator ( ) - > ReleaseModule ( ) ) ;
2015-02-03 21:32:09 +01:00
2015-01-15 15:13:14 +01:00
if ( M ) {
2015-02-10 11:13:59 +01:00
m_Interpreter - > addModule ( M . get ( ) ) ;
2015-02-03 21:32:09 +01:00
T - > setModule ( std : : move ( M ) ) ;
2015-01-07 10:22:52 +01:00
}
2015-05-28 11:44:26 +02:00
if ( T - > getIssuedDiags ( ) ! = Transaction : : kNone ) {
// Module has been released from Codegen, reset the Diags now.
DiagnosticsEngine & Diags = getCI ( ) - > getSema ( ) . getDiagnostics ( ) ;
Diags . Reset ( /*soft=*/ true ) ;
Diags . getClient ( ) - > clear ( ) ;
}
2015-01-07 10:22:52 +01:00
// Create a new module.
2016-12-14 15:32:18 -05:00
smallstream ModuleName ;
ModuleName < < " cling-module- " < < + + m_ModuleNo ;
getCodeGenerator ( ) - > StartModule ( ModuleName . str ( ) ,
2015-03-17 14:55:07 +01:00
* m_Interpreter - > getLLVMContext ( ) ,
getCI ( ) - > getCodeGenOpts ( ) ) ;
2015-01-07 10:22:52 +01:00
}
2013-06-25 10:51:58 +02:00
}
2016-06-23 20:38:57 +03:00
void IncrementalParser : : deregisterTransaction ( Transaction & T ) {
2016-09-13 08:46:43 +02:00
if ( & T = = m_Consumer - > getTransaction ( ) )
m_Consumer - > setTransaction ( T . getParent ( ) ) ;
2016-06-25 00:14:00 +02:00
if ( Transaction * Parent = T . getParent ( ) ) {
Parent - > removeNestedTransaction ( & T ) ;
T . setParent ( 0 ) ;
} else {
2015-02-18 12:38:05 +01:00
// Remove from the queue
2016-06-23 20:38:57 +03:00
assert ( & T = = m_Transactions . back ( ) & & " Out of order transaction removal " ) ;
2015-02-18 12:38:05 +01:00
m_Transactions . pop_back ( ) ;
2015-03-20 10:46:50 +01:00
if ( ! m_Transactions . empty ( ) )
m_Transactions . back ( ) - > setNext ( 0 ) ;
2015-02-18 12:38:05 +01:00
}
2016-06-25 00:14:00 +02:00
2016-06-23 20:38:57 +03:00
m_TransactionPool - > releaseTransaction ( & T ) ;
2012-09-05 09:37:39 +00:00
}
2012-11-14 22:06:58 +00:00
std : : vector < const Transaction * > IncrementalParser : : getAllTransactions ( ) {
2013-08-18 10:48:11 +02:00
std : : vector < const Transaction * > result ( m_Transactions . size ( ) ) ;
2012-11-14 22:06:58 +00:00
const cling : : Transaction * T = getFirstTransaction ( ) ;
while ( T ) {
result . push_back ( T ) ;
T = T - > getNext ( ) ;
}
return result ;
}
2012-09-05 09:37:39 +00:00
// Each input line is contained in separate memory buffer. The SourceManager
// assigns sort-of invalid FileID for each buffer, i.e there is no FileEntry
// for the MemoryBuffer's FileID. That in turn is problem because invalid
// SourceLocations are given to the diagnostics. Thus the diagnostics cannot
// order the overloads, for example
//
// Our work-around is creating a virtual file, which doesn't exist on the disk
// with enormous size (no allocation is done). That file has valid FileEntry
// and so on... We use it for generating valid SourceLocations with valid
// offsets so that it doesn't cause any troubles to the diagnostics.
//
// +---------------------+
// | Main memory buffer |
// +---------------------+
// | Virtual file SLoc |
// | address space |<-----------------+
// | ... |<------------+ |
// | ... | | |
// | ... |<----+ | |
// | ... | | | |
// +~~~~~~~~~~~~~~~~~~~~~+ | | |
// | input_line_1 | ....+.......+..--+
// +---------------------+ | |
// | input_line_2 | ....+.....--+
// +---------------------+ |
// | ... | |
// +---------------------+ |
// | input_line_N | ..--+
// +---------------------+
//
2013-08-22 14:11:12 +02:00
void IncrementalParser : : initializeVirtualFile ( ) {
2012-09-05 09:37:39 +00:00
SourceManager & SM = getCI ( ) - > getSourceManager ( ) ;
2013-03-11 13:11:15 +00:00
m_VirtualFileID = SM . getMainFileID ( ) ;
2016-07-02 00:20:48 -04:00
if ( m_VirtualFileID . isInvalid ( ) )
cling : : errs ( ) < < " VirtualFileID could not be created. \n " ;
2012-09-05 09:37:39 +00:00
}
2015-03-16 11:25:01 +01:00
IncrementalParser : : ParseResultTransaction
IncrementalParser : : Compile ( llvm : : StringRef input ,
const CompilationOptions & Opts ) {
2012-11-20 03:26:52 +00:00
Transaction * CurT = beginTransaction ( Opts ) ;
EParseResult ParseRes = ParseInternal ( input ) ;
2012-09-05 09:37:39 +00:00
2015-06-04 11:58:39 +02:00
if ( ParseRes = = kSuccessWithWarnings )
CurT - > setIssuedDiags ( Transaction : : kWarnings ) ;
else if ( ParseRes = = kFailed )
CurT - > setIssuedDiags ( Transaction : : kErrors ) ;
2015-03-16 11:25:01 +01:00
ParseResultTransaction PRT = endTransaction ( CurT ) ;
2015-03-18 09:38:35 +01:00
commitTransaction ( PRT ) ;
2013-06-27 14:29:44 +02:00
2015-03-16 11:25:01 +01:00
return PRT ;
2012-09-05 09:37:39 +00:00
}
// Add the input to the memory buffer, parse it, and add it to the AST.
IncrementalParser : : EParseResult
IncrementalParser : : ParseInternal ( llvm : : StringRef input ) {
if ( input . empty ( ) ) return IncrementalParser : : kSuccess ;
Sema & S = getCI ( ) - > getSema ( ) ;
2013-04-10 07:09:57 +00:00
2015-01-23 10:49:18 +01:00
const CompilationOptions & CO
= m_Consumer - > getTransaction ( ) - > getCompilationOpts ( ) ;
2013-04-10 07:14:37 +00:00
assert ( ! ( S . getLangOpts ( ) . Modules
2015-01-23 10:49:18 +01:00
& & CO . CodeGenerationForModule )
2014-02-20 10:16:47 +01:00
& & " CodeGenerationForModule to be removed once PCMs are available! " ) ;
2013-04-10 07:09:57 +00:00
2012-09-05 09:37:39 +00:00
// Recover resources if we crash before exiting this method.
llvm : : CrashRecoveryContextCleanupRegistrar < Sema > CleanupSema ( & S ) ;
Preprocessor & PP = m_CI - > getPreprocessor ( ) ;
if ( ! PP . getCurrentLexer ( ) ) {
PP . EnterSourceFile ( m_CI - > getSourceManager ( ) . getMainFileID ( ) ,
0 , SourceLocation ( ) ) ;
}
2012-10-04 13:09:52 +00:00
assert ( PP . isIncrementalProcessingEnabled ( ) & & " Not in incremental mode!? " ) ;
2012-09-05 09:37:39 +00:00
PP . enableIncrementalProcessing ( ) ;
std : : ostringstream source_name ;
source_name < < " input_line_ " < < ( m_MemoryBuffers . size ( ) + 1 ) ;
// Create an uninitialized memory buffer, copy code in and append "\n"
size_t InputSize = input . size ( ) ; // don't include trailing 0
// MemBuffer size should *not* include terminating zero
2015-01-15 15:13:14 +01:00
std : : unique_ptr < llvm : : MemoryBuffer >
MB ( llvm : : MemoryBuffer : : getNewUninitMemBuffer ( InputSize + 1 ,
source_name . str ( ) ) ) ;
2012-09-05 09:37:39 +00:00
char * MBStart = const_cast < char * > ( MB - > getBufferStart ( ) ) ;
memcpy ( MBStart , input . data ( ) , InputSize ) ;
memcpy ( MBStart + InputSize , " \n " , 2 ) ;
SourceManager & SM = getCI ( ) - > getSourceManager ( ) ;
// Create SourceLocation, which will allow clang to order the overload
// candidates for example
2014-04-01 16:44:59 +02:00
SourceLocation NewLoc = getLastMemoryBufferEndLoc ( ) . getLocWithOffset ( 1 ) ;
2012-09-05 09:37:39 +00:00
2015-02-10 11:18:15 +01:00
llvm : : MemoryBuffer * MBNonOwn = MB . get ( ) ;
2016-06-23 18:09:06 +02:00
2016-07-07 10:47:58 +02:00
// Create FileID for the current buffer.
FileID FID ;
if ( CO . CodeCompletionOffset = = - 1 )
{
FID = SM . createFileID ( std : : move ( MB ) , SrcMgr : : C_User ,
/*LoadedID*/ 0 ,
/*LoadedOffset*/ 0 , NewLoc ) ;
} else {
// Create FileEntry and FileID for the current buffer.
// Enabling the completion point only works on FileEntries.
const clang : : FileEntry * FE
= SM . getFileManager ( ) . getVirtualFile ( " vfile for " + source_name . str ( ) ,
InputSize , 0 /* mod time*/ ) ;
SM . overrideFileContents ( FE , std : : move ( MB ) ) ;
FID = SM . createFileID ( FE , NewLoc , SrcMgr : : C_User ) ;
// The completion point is set one a 1-based line/column numbering.
// It relies on the implementation to account for the wrapper extra line.
PP . SetCodeCompletionPoint ( FE , 1 /* start point 1-based line*/ ,
CO . CodeCompletionOffset + 1 /* 1-based column*/ ) ;
2016-06-23 18:09:06 +02:00
}
2012-09-05 09:37:39 +00:00
2015-02-10 11:18:15 +01:00
m_MemoryBuffers . push_back ( std : : make_pair ( MBNonOwn , FID ) ) ;
2015-01-23 16:18:01 +01:00
2016-05-18 16:08:14 +02:00
// NewLoc only used for diags.
2012-09-05 09:37:39 +00:00
PP . EnterSourceFile ( FID , /*DirLookup*/ 0 , NewLoc ) ;
2013-12-03 15:31:31 +01:00
m_Consumer - > getTransaction ( ) - > setBufferFID ( FID ) ;
2012-09-05 09:37:39 +00:00
2015-01-23 10:49:18 +01:00
DiagnosticsEngine & Diags = getCI ( ) - > getDiagnostics ( ) ;
2017-01-09 16:32:49 +01:00
FilteringDiagConsumer * PromptDiagClient
= static_cast < FilteringDiagConsumer * > ( Diags . getClient ( ) ) ;
struct PromptDiagClientRAII_t {
FilteringDiagConsumer * fClient ;
~ PromptDiagClientRAII_t ( ) { fClient - > pop ( ) ; }
} PromptDiagClientRAII { PromptDiagClient } ;
PromptDiagClient - > push ( CO . IgnorePromptDiags ) ;
2015-01-23 10:49:18 +01:00
2014-12-02 17:43:18 +01:00
Sema : : SavePendingInstantiationsRAII SavedPendingInstantiations ( S ) ;
2012-09-05 09:37:39 +00:00
Parser : : DeclGroupPtrTy ADecl ;
while ( ! m_Parser - > ParseTopLevelDecl ( ADecl ) ) {
// If we got a null return and something *was* parsed, ignore it. This
// is due to a top-level semicolon, an action override, or a parse error
// skipping something.
2015-03-17 10:54:04 +01:00
if ( Diags . hasErrorOccurred ( ) | | Diags . hasFatalErrorOccurred ( ) )
m_Consumer - > getTransaction ( ) - > setIssuedDiags ( Transaction : : kErrors ) ;
if ( ADecl )
2013-09-23 15:36:32 +02:00
m_Consumer - > HandleTopLevelDecl ( ADecl . get ( ) ) ;
2012-09-05 09:37:39 +00:00
} ;
2016-06-23 18:09:06 +02:00
if ( CO . CodeCompletionOffset ! = - 1 ) {
2016-07-21 15:50:02 +02:00
assert ( ( int ) SM . getFileOffset ( PP . getCodeCompletionLoc ( ) )
= = CO . CodeCompletionOffset
& & " Completion point wrongly set! " ) ;
2016-07-07 10:47:58 +02:00
assert ( PP . isCodeCompletionReached ( )
& & " Code completion set but not reached! " ) ;
2016-07-22 15:53:33 +02:00
// Let's ignore this transaction:
2016-07-22 15:46:38 +02:00
m_Consumer - > getTransaction ( ) - > setIssuedDiags ( Transaction : : kErrors ) ;
2016-06-23 18:09:06 +02:00
return kSuccess ;
}
2015-02-17 09:49:47 +01:00
# ifdef LLVM_ON_WIN32
2015-02-13 16:48:02 +01:00
// Microsoft-specific:
// Late parsed templates can leave unswallowed "macro"-like tokens.
// They will seriously confuse the Parser when entering the next
// source file. So lex until we are EOF.
Token Tok ;
do {
PP . Lex ( Tok ) ;
} while ( Tok . isNot ( tok : : eof ) ) ;
2015-02-17 09:49:47 +01:00
# endif
# ifndef NDEBUG
Token AssertTok ;
PP . Lex ( AssertTok ) ;
assert ( AssertTok . is ( tok : : eof ) & & " Lexer must be EOF when starting incremental parse! " ) ;
# endif
2015-02-13 16:48:02 +01:00
2015-03-17 10:54:04 +01:00
// Process any TopLevelDecls generated by #pragma weak.
for ( llvm : : SmallVector < Decl * , 2 > : : iterator I = S . WeakTopLevelDecls ( ) . begin ( ) ,
E = S . WeakTopLevelDecls ( ) . end ( ) ; I ! = E ; + + I ) {
m_Consumer - > HandleTopLevelDecl ( DeclGroupRef ( * I ) ) ;
2012-09-05 09:37:39 +00:00
}
2015-03-17 10:54:04 +01:00
if ( m_Consumer - > getTransaction ( ) - > getIssuedDiags ( ) = = Transaction : : kErrors )
2015-03-16 20:26:27 +01:00
return kFailed ;
2015-01-23 10:49:18 +01:00
else if ( Diags . getNumWarnings ( ) )
2015-03-16 20:26:27 +01:00
return kSuccessWithWarnings ;
2015-03-16 20:08:08 +01:00
2015-03-16 20:26:27 +01:00
return kSuccess ;
2012-09-05 09:37:39 +00:00
}
2013-07-06 13:02:49 +02:00
void IncrementalParser : : printTransactionStructure ( ) const {
for ( size_t i = 0 , e = m_Transactions . size ( ) ; i < e ; + + i ) {
m_Transactions [ i ] - > printStructureBrief ( ) ;
}
}
2016-04-08 15:11:17 +02:00
void IncrementalParser : : SetTransformers ( bool isChildInterpreter ) {
2016-02-09 18:11:01 +01:00
// Add transformers to the IncrementalParser, which owns them
Sema * TheSema = & m_CI - > getSema ( ) ;
// Register the AST Transformers
typedef std : : unique_ptr < ASTTransformer > ASTTPtr_t ;
std : : vector < ASTTPtr_t > ASTTransformers ;
ASTTransformers . emplace_back ( new AutoSynthesizer ( TheSema ) ) ;
ASTTransformers . emplace_back ( new EvaluateTSynthesizer ( TheSema ) ) ;
2016-04-13 16:21:46 +02:00
if ( hasCodeGenerator ( ) ) {
// Don't protect against crashes if we cannot run anything.
// cling might also be in a PCH-generation mode; don't inject our Sema pointer
// into the PCH.
ASTTransformers . emplace_back ( new NullDerefProtectionTransformer ( m_Interpreter ) ) ;
}
2016-02-09 18:11:01 +01:00
typedef std : : unique_ptr < WrapperTransformer > WTPtr_t ;
std : : vector < WTPtr_t > WrapperTransformers ;
2016-09-10 14:06:04 -04:00
WrapperTransformers . emplace_back ( new ValuePrinterSynthesizer ( TheSema ) ) ;
2016-02-09 18:11:01 +01:00
WrapperTransformers . emplace_back ( new DeclExtractor ( TheSema ) ) ;
WrapperTransformers . emplace_back ( new ValueExtractionSynthesizer ( TheSema , isChildInterpreter ) ) ;
WrapperTransformers . emplace_back ( new CheckEmptyTransactionTransformer ( TheSema ) ) ;
m_Consumer - > SetTransformers ( std : : move ( ASTTransformers ) ,
std : : move ( WrapperTransformers ) ) ;
}
2013-08-26 17:59:33 +02:00
2012-09-05 09:37:39 +00:00
} // namespace cling