2014-08-06 15:05:09 +02:00
//------------------------------------------------------------------------------
// CLING - the C++ LLVM-based InterpreterG :)
// author: Manasij Mukherjee <manasij7479@gmail.com>
// author: Vassil Vassilev <vvasilev@cern.ch>
//
// This file is dual-licensed: you can choose to license it under the University
// of Illinois Open Source License or the GNU Lesser General Public License. See
// LICENSE.TXT for details.
//------------------------------------------------------------------------------
2014-07-12 23:53:55 +05:30
# include "llvm/ADT/SmallVector.h"
# include "llvm/Support/Path.h"
2015-08-11 15:45:40 +02:00
2014-07-12 23:53:55 +05:30
# include "clang/Sema/Sema.h"
# include "clang/Basic/Diagnostic.h"
# include "clang/AST/AST.h"
2014-08-21 12:21:41 +02:00
# include "clang/AST/ASTContext.h" // for operator new[](unsigned long, ASTCtx..)
2014-08-05 21:35:25 +02:00
# include "clang/AST/RecursiveASTVisitor.h"
2014-08-19 01:08:40 +05:30
# include "clang/AST/DeclVisitor.h"
2014-07-12 23:53:55 +05:30
# include "clang/Lex/Preprocessor.h"
2014-07-13 22:13:20 +05:30
# include "clang/Frontend/CompilerInstance.h"
2014-07-12 23:53:55 +05:30
2014-06-08 22:12:58 +05:30
# include "cling/Interpreter/Interpreter.h"
# include "cling/Interpreter/InterpreterCallbacks.h"
2014-06-08 03:22:32 +05:30
# include "cling/Interpreter/AutoloadCallback.h"
2014-07-13 22:13:20 +05:30
# include "cling/Interpreter/Transaction.h"
2016-09-10 15:04:39 -04:00
# include "cling/Utils/Output.h"
2016-06-26 10:06:18 +02:00
# include "DeclUnloader.h"
2014-07-12 23:53:55 +05:30
2016-11-17 16:00:06 -06:00
# include <clang/Lex/HeaderSearch.h>
2014-09-22 15:52:38 +02:00
namespace {
2014-10-02 14:38:33 +02:00
static const char annoTag [ ] = " $clingAutoload$ " ;
static const size_t lenAnnoTag = sizeof ( annoTag ) - 1 ;
2014-09-22 15:52:38 +02:00
}
2014-06-08 03:22:32 +05:30
using namespace clang ;
namespace cling {
2014-07-12 23:53:55 +05:30
2014-09-12 13:53:07 +02:00
void AutoloadCallback : : report ( clang : : SourceLocation l , llvm : : StringRef name ,
llvm : : StringRef header ) {
2014-06-08 03:22:32 +05:30
Sema & sema = m_Interpreter - > getSema ( ) ;
unsigned id
= sema . getDiagnostics ( ) . getCustomDiagID ( DiagnosticsEngine : : Level : : Warning ,
" Note: '%0' can be found in %1 " ) ;
/* unsigned idn //TODO: To be enabled after we have a way to get the full path
= sema . getDiagnostics ( ) . getCustomDiagID ( DiagnosticsEngine : : Level : : Note ,
" Type : %0 , Full Path: %1 " ) */ ;
2014-09-22 15:52:38 +02:00
if ( header . startswith ( llvm : : StringRef ( annoTag , lenAnnoTag ) ) )
sema . Diags . Report ( l , id ) < < name < < header . drop_front ( lenAnnoTag ) ;
2014-06-08 03:22:32 +05:30
}
2014-08-05 05:42:08 +05:30
bool AutoloadCallback : : LookupObject ( TagDecl * t ) {
2014-09-19 13:59:31 +02:00
if ( m_ShowSuggestions & & t - > hasAttr < AnnotateAttr > ( ) )
2014-08-05 05:42:08 +05:30
report ( t - > getLocation ( ) , t - > getNameAsString ( ) , t - > getAttr < AnnotateAttr > ( ) - > getAnnotation ( ) ) ;
return false ;
}
2014-08-19 01:08:40 +05:30
2020-02-13 18:26:48 -06:00
class AutoLoadingVisitor : public RecursiveASTVisitor < AutoLoadingVisitor > {
2014-08-06 00:07:20 +02:00
private :
2015-02-05 10:24:50 -06:00
///\brief Flag determining the visitor's actions. If true, register autoload
/// entries, i.e. remember the connection between filename and the declaration
2015-02-05 15:32:42 +01:00
/// that needs to be updated on #include of the filename.
/// If false, react on an #include by adjusting the forward decls, e.g. by
2018-05-16 08:42:15 +02:00
/// removing the default template arguments (that will now be provided by
/// the definition read from the include).
2014-08-06 00:07:20 +02:00
bool m_IsStoringState ;
2016-11-13 17:48:11 -06:00
bool m_IsAutloadEntry ; // True during the traversal of an explicitly annotated decl.
2014-08-06 00:07:20 +02:00
AutoloadCallback : : FwdDeclsMap * m_Map ;
clang : : Preprocessor * m_PP ;
2015-06-19 16:26:08 +02:00
clang : : Sema * m_Sema ;
2016-11-17 16:00:06 -06:00
std : : pair < const clang : : FileEntry * , const clang : : FileEntry * > m_PrevFE ;
std : : pair < std : : string , std : : string > m_PrevFileName ;
2014-08-06 00:07:20 +02:00
private :
2016-11-08 15:52:58 -06:00
bool IsAutoloadEntry ( Decl * D ) {
2016-11-17 16:00:06 -06:00
for ( auto attr = D - > specific_attr_begin < AnnotateAttr > ( ) ,
end = D - > specific_attr_end < AnnotateAttr > ( ) ;
attr ! = end ;
+ + attr )
{
2016-09-10 15:04:39 -04:00
// cling::errs() << "Annotation: " << c->getAnnotation() << "\n";
2016-11-17 16:00:06 -06:00
if ( ! attr - > isInherited ( ) ) {
llvm : : StringRef annotation = attr - > getAnnotation ( ) ;
assert ( ! annotation . empty ( ) & & " Empty annotation! " ) ;
if ( annotation . startswith ( llvm : : StringRef ( annoTag , lenAnnoTag ) ) ) {
2016-11-08 15:52:58 -06:00
// autoload annotation.
return true ;
2016-11-17 16:00:06 -06:00
}
}
}
return false ;
2016-11-08 15:52:58 -06:00
}
2016-11-17 16:00:06 -06:00
using Annotations_t = std : : pair < llvm : : StringRef , llvm : : StringRef > ;
2014-09-12 13:57:19 +02:00
2020-02-13 18:26:48 -06:00
void InsertIntoAutoLoadingState ( Decl * decl , Annotations_t FileNames ) {
2014-08-06 00:07:20 +02:00
assert ( m_PP ) ;
2016-11-17 16:00:06 -06:00
auto addFile = [ this , decl ] ( llvm : : StringRef FileName , bool warn ) {
if ( FileName . empty ( ) ) return ( const FileEntry * ) nullptr ;
const FileEntry * FE = 0 ;
SourceLocation fileNameLoc ;
2017-12-05 17:10:44 -06:00
// Remember this file wth full path, not "./File.h" (ROOT-8863).
bool isAngled = true ;
2016-11-17 16:00:06 -06:00
const DirectoryLookup * FromDir = 0 ;
const FileEntry * FromFile = 0 ;
const DirectoryLookup * CurDir = 0 ;
bool needCacheUpdate = false ;
if ( FileName . equals ( m_PrevFileName . first ) )
FE = m_PrevFE . first ;
else if ( FileName . equals ( m_PrevFileName . second ) )
FE = m_PrevFE . second ;
else {
FE = m_PP - > LookupFile ( fileNameLoc , FileName , isAngled ,
FromDir , FromFile , CurDir , /*SearchPath*/ 0 ,
/*RelativePath*/ 0 , /*suggestedModule*/ 0 ,
2020-03-11 12:36:32 +02:00
/*IsMapped*/ 0 , /*IsFrameworkFound*/ nullptr ,
/*SkipCache*/ false ,
2017-05-18 15:16:30 +02:00
/*OpenFile*/ false , /*CacheFail*/ true ) ;
2016-11-17 16:00:06 -06:00
needCacheUpdate = true ;
}
2014-08-06 00:07:20 +02:00
2016-11-17 16:00:06 -06:00
if ( FE ) {
auto & Vec = ( * m_Map ) [ FE ] ;
Vec . push_back ( decl ) ;
if ( needCacheUpdate ) return FE ;
else return ( const FileEntry * ) nullptr ;
} else if ( warn ) {
// If the top level header is expected to be findable at run-time,
// the direct header might not because the include path might be
// different enough and only the top-header is guaranteed to be seen
// by the user as an interface header to be available on the
// run-time include path.
2016-09-10 15:04:39 -04:00
cling : : errs ( )
2020-02-13 18:26:48 -06:00
< < " Error in cling::AutoLoadingVisitor::InsertIntoAutoLoadingState: \n "
2015-08-11 15:45:40 +02:00
" Missing FileEntry for " < < FileName < < " \n " ;
2016-11-17 16:00:06 -06:00
if ( NamedDecl * ND = dyn_cast < NamedDecl > ( decl ) ) {
2016-09-10 15:04:39 -04:00
cling : : errs ( ) < < " requested to autoload type " ;
ND - > getNameForDiagnostic ( cling : : errs ( ) ,
2016-11-17 16:00:06 -06:00
ND - > getASTContext ( ) . getPrintingPolicy ( ) ,
true /*qualified*/ ) ;
2016-09-10 15:04:39 -04:00
cling : : errs ( ) < < " \n " ;
2016-11-17 16:00:06 -06:00
}
return ( const FileEntry * ) nullptr ;
} else {
// Case of the direct header that is not a top level header, no
// warning in this case (to likely to be a false positive).
return ( const FileEntry * ) nullptr ;
2015-08-11 15:45:40 +02:00
}
2016-11-17 16:00:06 -06:00
} ;
const FileEntry * cacheUpdate ;
if ( ( cacheUpdate = addFile ( FileNames . first , true ) ) ) {
m_PrevFE . first = cacheUpdate ;
m_PrevFileName . first = FileNames . first ;
}
if ( ( cacheUpdate = addFile ( FileNames . second , false ) ) ) {
m_PrevFE . second = cacheUpdate ;
m_PrevFileName . second = FileNames . second ;
2014-10-02 16:26:08 +02:00
}
2016-11-17 16:00:06 -06:00
2014-08-06 00:07:20 +02:00
}
2014-08-05 21:35:25 +02:00
public :
2020-02-13 18:26:48 -06:00
AutoLoadingVisitor ( ) :
2016-11-13 17:48:11 -06:00
m_IsStoringState ( false ) , m_IsAutloadEntry ( false ) , m_Map ( 0 ) , m_PP ( 0 ) ,
2016-11-17 16:00:06 -06:00
m_Sema ( 0 ) , m_PrevFE ( { nullptr , nullptr } )
2016-11-13 17:48:11 -06:00
{ }
2015-06-19 16:26:08 +02:00
void RemoveDefaultArgsOf ( Decl * D , Sema * S ) {
m_Sema = S ;
2016-11-08 11:21:43 -06:00
2016-11-13 17:45:31 -06:00
auto cursor = D - > getMostRecentDecl ( ) ;
2016-11-13 17:48:11 -06:00
m_IsAutloadEntry = IsAutoloadEntry ( cursor ) ;
2016-11-13 17:45:31 -06:00
TraverseDecl ( cursor ) ;
while ( cursor ! = D & & ( cursor = cursor - > getPreviousDecl ( ) ) ) {
2016-11-13 17:48:11 -06:00
m_IsAutloadEntry = IsAutoloadEntry ( cursor ) ;
2016-11-13 17:45:31 -06:00
TraverseDecl ( cursor ) ;
2016-11-08 11:21:43 -06:00
}
2016-11-13 17:48:11 -06:00
m_IsAutloadEntry = false ;
2014-07-12 23:53:55 +05:30
}
2014-08-05 21:35:25 +02:00
2014-08-06 00:07:20 +02:00
void TrackDefaultArgStateOf ( Decl * D , AutoloadCallback : : FwdDeclsMap & map ,
Preprocessor & PP ) {
m_IsStoringState = true ;
m_Map = & map ;
m_PP = & PP ;
TraverseDecl ( D ) ;
m_PP = 0 ;
m_Map = 0 ;
m_IsStoringState = false ;
}
2014-08-06 01:39:34 +02:00
bool shouldVisitTemplateInstantiations ( ) { return true ; }
2014-08-05 21:35:25 +02:00
2014-08-06 01:39:34 +02:00
bool VisitDecl ( Decl * D ) {
2014-08-13 14:05:29 +02:00
if ( ! m_IsStoringState )
2014-08-06 12:52:20 +02:00
return true ;
2014-08-08 12:44:32 +02:00
if ( ! D - > hasAttr < AnnotateAttr > ( ) )
2014-08-13 14:05:59 +02:00
return true ;
2014-08-08 12:44:32 +02:00
2016-11-17 16:00:06 -06:00
Annotations_t annotations ;
for ( auto attr = D - > specific_attr_begin < AnnotateAttr > ( ) ,
end = D - > specific_attr_end < AnnotateAttr > ( ) ;
attr ! = end ;
+ + attr )
{
if ( ! attr - > isInherited ( ) ) {
auto annot = attr - > getAnnotation ( ) ;
if ( annot . startswith ( llvm : : StringRef ( annoTag , lenAnnoTag ) ) ) {
if ( annotations . first . empty ( ) ) {
annotations . first = annot . drop_front ( lenAnnoTag ) ;
} else {
annotations . second = annot . drop_front ( lenAnnoTag ) ;
}
}
}
}
2020-02-13 18:26:48 -06:00
InsertIntoAutoLoadingState ( D , annotations ) ;
2014-08-06 01:39:34 +02:00
return true ;
}
2014-08-13 14:06:58 +02:00
bool VisitCXXRecordDecl ( CXXRecordDecl * D ) {
2016-11-13 17:49:24 -06:00
// Since we are only interested in fixing forward declaration
// there is no need to continue on when we see a complete definition.
if ( D - > isCompleteDefinition ( ) )
return false ;
2014-08-13 14:06:58 +02:00
if ( ! D - > hasAttr < AnnotateAttr > ( ) )
return true ;
if ( ClassTemplateDecl * TmplD = D - > getDescribedClassTemplate ( ) )
return VisitTemplateDecl ( TmplD ) ;
return true ;
}
bool VisitTemplateTypeParmDecl ( TemplateTypeParmDecl * D ) {
if ( m_IsStoringState )
return true ;
2016-11-13 17:48:11 -06:00
if ( m_IsAutloadEntry ) {
if ( D - > hasDefaultArgument ( ) & & ! D - > defaultArgumentWasInherited ( ) )
D - > removeDefaultArgument ( ) ;
} else {
if ( D - > hasDefaultArgument ( ) & & D - > defaultArgumentWasInherited ( ) )
D - > removeDefaultArgument ( ) ;
}
2014-08-13 14:06:58 +02:00
return true ;
}
bool VisitTemplateDecl ( TemplateDecl * D ) {
2014-08-13 17:02:54 +02:00
if ( D - > getTemplatedDecl ( ) & &
! D - > getTemplatedDecl ( ) - > hasAttr < AnnotateAttr > ( ) )
2014-08-12 09:32:38 +02:00
return true ;
2014-08-13 14:06:58 +02:00
2015-02-05 15:33:28 +01:00
// If we have a definition we might be about to re-#include the
// same header containing definition that was #included previously,
// i.e. we might have multiple fwd decls for the same template.
// DO NOT remove the defaults here; the definition needs to keep it.
// (ROOT-7037)
if ( ClassTemplateDecl * CTD = dyn_cast < ClassTemplateDecl > ( D ) )
if ( CXXRecordDecl * TemplatedD = CTD - > getTemplatedDecl ( ) )
if ( TemplatedD - > getDefinition ( ) )
return true ;
2014-08-08 12:45:02 +02:00
for ( auto P : D - > getTemplateParameters ( ) - > asArray ( ) )
TraverseDecl ( P ) ;
return true ;
}
2014-08-13 17:02:54 +02:00
bool VisitTemplateTemplateParmDecl ( TemplateTemplateParmDecl * D ) {
if ( m_IsStoringState )
return true ;
2016-11-13 17:48:11 -06:00
if ( m_IsAutloadEntry ) {
if ( D - > hasDefaultArgument ( ) & & ! D - > defaultArgumentWasInherited ( ) )
D - > removeDefaultArgument ( ) ;
} else {
if ( D - > hasDefaultArgument ( ) & & D - > defaultArgumentWasInherited ( ) )
D - > removeDefaultArgument ( ) ;
}
2014-08-13 17:02:54 +02:00
return true ;
}
2014-08-13 14:06:58 +02:00
bool VisitNonTypeTemplateParmDecl ( NonTypeTemplateParmDecl * D ) {
2014-08-12 09:32:38 +02:00
if ( m_IsStoringState )
return true ;
2016-11-13 17:48:11 -06:00
if ( m_IsAutloadEntry ) {
if ( D - > hasDefaultArgument ( ) & & ! D - > defaultArgumentWasInherited ( ) )
D - > removeDefaultArgument ( ) ;
} else {
if ( D - > hasDefaultArgument ( ) & & D - > defaultArgumentWasInherited ( ) )
D - > removeDefaultArgument ( ) ;
}
2014-08-05 21:35:25 +02:00
return true ;
2014-07-13 22:45:08 +05:30
}
2014-08-05 21:35:25 +02:00
2014-08-13 14:06:58 +02:00
bool VisitParmVarDecl ( ParmVarDecl * D ) {
2014-08-12 09:32:38 +02:00
if ( m_IsStoringState )
return true ;
2016-11-13 17:48:11 -06:00
if ( m_IsAutloadEntry ) {
if ( D - > hasDefaultArg ( ) & & ! D - > hasInheritedDefaultArg ( ) )
D - > setDefaultArg ( nullptr ) ;
} else {
if ( D - > hasDefaultArg ( ) & & D - > hasInheritedDefaultArg ( ) )
D - > setDefaultArg ( nullptr ) ;
}
2014-08-05 21:35:25 +02:00
return true ;
}
} ;
2014-06-08 03:22:32 +05:30
2020-09-30 15:46:22 +00:00
void AutoloadCallback : : InclusionDirective ( clang : : SourceLocation /*HashLoc*/ ,
const clang : : Token & /*IncludeTok*/ ,
llvm : : StringRef /*FileName*/ ,
bool /*IsAngled*/ ,
clang : : CharSourceRange /*FilenameRange*/ ,
2020-03-17 16:36:37 +02:00
const clang : : FileEntry * File ,
2020-09-30 15:46:22 +00:00
llvm : : StringRef /*SearchPath*/ ,
llvm : : StringRef /*RelativePath*/ ,
const clang : : Module */ * Imported */ ,
clang : : SrcMgr : : CharacteristicKind /*FileType*/ ) {
2014-08-21 15:58:59 +02:00
// If File is 0 this means that the #included file doesn't exist.
if ( ! File )
return ;
2014-07-29 23:14:15 +02:00
2014-08-06 14:56:28 +02:00
auto found = m_Map . find ( File ) ;
if ( found = = m_Map . end ( ) )
return ; // nothing to do, file not referred in any annotation
2014-07-12 23:53:55 +05:30
2020-02-13 18:26:48 -06:00
AutoLoadingVisitor defaultArgsCleaner ;
2014-08-06 14:56:28 +02:00
for ( auto D : found - > second ) {
2015-06-19 16:26:08 +02:00
defaultArgsCleaner . RemoveDefaultArgsOf ( D , & getInterpreter ( ) - > getSema ( ) ) ;
2014-07-12 23:53:55 +05:30
}
2014-08-06 00:07:20 +02:00
// Don't need to keep track of cleaned up decls from file.
2014-08-06 14:56:28 +02:00
m_Map . erase ( found ) ;
2014-07-12 23:53:55 +05:30
}
2014-06-08 03:22:32 +05:30
2014-07-13 22:13:20 +05:30
AutoloadCallback : : ~ AutoloadCallback ( ) {
}
2014-08-06 00:07:20 +02:00
void AutoloadCallback : : TransactionCommitted ( const Transaction & T ) {
2014-08-08 12:45:25 +02:00
if ( T . decls_begin ( ) = = T . decls_end ( ) )
return ;
if ( T . decls_begin ( ) - > m_DGR . isNull ( ) )
return ;
2015-06-22 15:18:05 +02:00
// The first decl must be
2020-02-13 18:26:48 -06:00
// extern int __Cling_AutoLoading_Map;
bool HaveAutoLoadingMapMarker = false ;
2015-06-22 15:18:05 +02:00
for ( auto I = T . decls_begin ( ) , E = T . decls_end ( ) ;
2020-02-13 18:26:48 -06:00
! HaveAutoLoadingMapMarker & & I ! = E ; + + I ) {
2015-06-22 15:18:05 +02:00
if ( I - > m_Call ! = cling : : Transaction : : kCCIHandleTopLevelDecl )
return ;
for ( auto & & D : I - > m_DGR ) {
if ( isa < EmptyDecl > ( D ) )
continue ;
else if ( auto VD = dyn_cast < VarDecl > ( D ) ) {
2020-02-13 18:26:48 -06:00
HaveAutoLoadingMapMarker
2015-06-22 15:18:05 +02:00
= VD - > hasExternalStorage ( ) & & VD - > getIdentifier ( )
2020-02-13 18:26:48 -06:00
& & VD - > getName ( ) . equals ( " __Cling_AutoLoading_Map " ) ;
if ( ! HaveAutoLoadingMapMarker )
2015-06-22 15:18:05 +02:00
return ;
break ;
} else
return ;
2014-07-13 22:13:20 +05:30
}
2015-06-22 15:18:05 +02:00
}
2020-02-13 18:26:48 -06:00
if ( ! HaveAutoLoadingMapMarker )
2015-06-22 15:18:05 +02:00
return ;
2020-02-13 18:26:48 -06:00
AutoLoadingVisitor defaultArgsStateCollector ;
2015-06-22 15:18:05 +02:00
Preprocessor & PP = m_Interpreter - > getCI ( ) - > getPreprocessor ( ) ;
for ( auto I = T . decls_begin ( ) , E = T . decls_end ( ) ; I ! = E ; + + I )
for ( auto & & D : I - > m_DGR )
defaultArgsStateCollector . TrackDefaultArgStateOf ( D , m_Map , PP ) ;
2014-07-13 22:13:20 +05:30
}
2014-08-05 21:48:45 +02:00
} //end namespace cling