166032069e
Now that we can easily revert Transactions' IR (by unloading their module) we do not need to queue the Decls before emitting them. This enables EndOfTU actions to be emitted seeminglessly, without extra transactions, or explicit post-EndOfTU-emission: each Transaction will have its TU finalized exactly once. This in turn allows us to turn on Debug emission - which relies on being invoked exactly once per Module. Transformers must be invoked before emitting Decls. They are now invoked for each Decl; WrapperTransformers are invoked when seeing a wrapper decl. They do not see the "full transaction AST" anymore - luckily none of our transformers requires this and actually become simpler because of this change. This also fixes a bug where the relative sequencing of parsed and deserialized Decls was lost (parsed was emitted before deserialized). Remove unused IRTransactions; they should really be llvm::Pass-es. We don't have them anyway. Disable a few transformations if the Decl isFromASTFile. When reverting a Transaction, also revert its nested ones. In the ValueExtractionSynthesizer, pass the ValuePrinter option as int, instead of the transaction pointer - that might have changed (at least its options) by the time we invoke the wrapper. # Please enter the commit message for your changes. Lines starting # with '#' will be ignored, and an empty message aborts the commit. # Explicit paths specified without -i or -o; assuming --only paths... # rebase in progress; onto b3d9f92 # You are currently splitting a commit while rebasing branch 'declcollector-emits-v2' on 'b3d9f92'. # # Changes to be committed: # modified: interpreter/cling/include/cling/Interpreter/RuntimeUniverse.h # modified: interpreter/cling/lib/Interpreter/AutoSynthesizer.cpp # modified: interpreter/cling/lib/Interpreter/AutoSynthesizer.h # modified: interpreter/cling/lib/Interpreter/CheckEmptyTransactionTransformer.cpp # modified: interpreter/cling/lib/Interpreter/CheckEmptyTransactionTransformer.h # modified: interpreter/cling/lib/Interpreter/DeclCollector.cpp # modified: interpreter/cling/lib/Interpreter/DeclCollector.h # modified: interpreter/cling/lib/Interpreter/DeclExtractor.cpp # modified: interpreter/cling/lib/Interpreter/DeclExtractor.h # modified: interpreter/cling/lib/Interpreter/DynamicLookup.cpp # modified: interpreter/cling/lib/Interpreter/DynamicLookup.h # modified: interpreter/cling/lib/Interpreter/IncrementalParser.cpp # modified: interpreter/cling/lib/Interpreter/IncrementalParser.h # modified: interpreter/cling/lib/Interpreter/NullDerefProtectionTransformer.cpp # modified: interpreter/cling/lib/Interpreter/NullDerefProtectionTransformer.h # modified: interpreter/cling/lib/Interpreter/TransactionTransformer.cpp # modified: interpreter/cling/lib/Interpreter/TransactionTransformer.h # modified: interpreter/cling/lib/Interpreter/TransactionUnloader.cpp # modified: interpreter/cling/lib/Interpreter/ValueExtractionSynthesizer.cpp # modified: interpreter/cling/lib/Interpreter/ValueExtractionSynthesizer.h # modified: interpreter/cling/lib/Interpreter/ValuePrinterSynthesizer.cpp # modified: interpreter/cling/lib/Interpreter/ValuePrinterSynthesizer.h # # Untracked files: # .idea/ # 0001-Fix-llvm-merge-issue.patch # 0002-Fix-warnings.patch # 0003-Keep-weak-symbols-around-subsequent-transactions-mig.patch # T.cxx # T.cxx~ # TMVA.root # a.out # boost.root # config/Makefile.depend~ # core/textinput/src/textinput/TerminalDisplayWin.cpp.orig # ct.root # data.root # interpreter/cling/76fc2055249da7b03148a7d4197a279e9943f012.patch # interpreter/cling/lib/Interpreter/CIFactory.cpp~ # interpreter/cling/lib/Interpreter/DeclCollector.h~ # interpreter/cling/lib/Interpreter/DynamicLibraryManager.cpp~ # modulemap-one-header-per-file.txt # osrm-routed # tiles.tar.bz2 # tmva_class_example.root # tutorials/v3.root # weights/ #
87 lines
3.1 KiB
C++
87 lines
3.1 KiB
C++
//------------------------------------------------------------------------------
|
|
// CLING - the C++ LLVM-based InterpreterG :)
|
|
// author: Vassil Vassilev <vasil.georgiev.vasilev@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.
|
|
//------------------------------------------------------------------------------
|
|
|
|
#include "AutoSynthesizer.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/AST/RecursiveASTVisitor.h"
|
|
#include "clang/Sema/Sema.h"
|
|
|
|
using namespace clang;
|
|
|
|
namespace cling {
|
|
class AutoFixer : public RecursiveASTVisitor<AutoFixer> {
|
|
private:
|
|
Sema* m_Sema;
|
|
DeclRefExpr* m_FoundDRE;
|
|
llvm::DenseSet<NamedDecl*> m_HandledDecls;
|
|
private:
|
|
public:
|
|
AutoFixer(Sema* S) : m_Sema(S), m_FoundDRE(0) {}
|
|
|
|
void Fix(CompoundStmt* CS) {
|
|
if (!CS->size())
|
|
return;
|
|
typedef llvm::SmallVector<Stmt*, 32> Statements;
|
|
Statements Stmts;
|
|
Stmts.append(CS->body_begin(), CS->body_end());
|
|
for (Statements::iterator I = Stmts.begin(); I != Stmts.end(); ++I) {
|
|
if (!TraverseStmt(*I) && !m_HandledDecls.count(m_FoundDRE->getDecl())) {
|
|
Sema::DeclGroupPtrTy VDPtrTy
|
|
= m_Sema->ConvertDeclToDeclGroup(m_FoundDRE->getDecl());
|
|
StmtResult DS = m_Sema->ActOnDeclStmt(VDPtrTy,
|
|
m_FoundDRE->getLocStart(),
|
|
m_FoundDRE->getLocEnd());
|
|
assert(!DS.isInvalid() && "Invalid DeclStmt.");
|
|
I = Stmts.insert(I, DS.get());
|
|
m_HandledDecls.insert(m_FoundDRE->getDecl());
|
|
}
|
|
}
|
|
CS->setStmts(m_Sema->getASTContext(), Stmts.data(), Stmts.size());
|
|
}
|
|
|
|
bool VisitDeclRefExpr(DeclRefExpr* DRE) {
|
|
const Decl* D = DRE->getDecl();
|
|
if (const AnnotateAttr* A = D->getAttr<AnnotateAttr>())
|
|
if (A->getAnnotation().equals("__Auto")) {
|
|
m_FoundDRE = DRE;
|
|
return false; // we abort on the first found candidate.
|
|
}
|
|
return true; // returning false will abort the in-depth traversal.
|
|
}
|
|
};
|
|
} // end namespace cling
|
|
|
|
namespace cling {
|
|
AutoSynthesizer::AutoSynthesizer(clang::Sema* S)
|
|
: ASTTransformer(S) {
|
|
// TODO: We would like to keep that local without keeping track of all
|
|
// decls that were handled in the AutoFixer. This can be done by removing
|
|
// the __Auto attribute, but for now I am still hesitant to do it. Having
|
|
// the __Auto attribute is very useful for debugging because it localize the
|
|
// the problem if exists.
|
|
m_AutoFixer.reset(new AutoFixer(S));
|
|
}
|
|
|
|
// pin the vtable here.
|
|
AutoSynthesizer::~AutoSynthesizer()
|
|
{ }
|
|
|
|
ASTTransformer::Result AutoSynthesizer::Transform(Decl* D) {
|
|
if (FunctionDecl* FD = dyn_cast<FunctionDecl>(D)) {
|
|
// getBody() might return nullptr even though hasBody() is true for
|
|
// late template parsed functions. We simply don't do auto auto on
|
|
// those.
|
|
if (CompoundStmt* CS = cast_or_null<CompoundStmt>(FD->getBody()))
|
|
m_AutoFixer->Fix(CS);
|
|
}
|
|
return Result(D, true);
|
|
}
|
|
} // end namespace cling
|