2016-04-14 19:51:27 +08:00
|
|
|
//===- unittest/AST/ASTImporterTest.cpp - AST node import test ------------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2016-04-14 19:51:27 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Tests for the correct import of AST nodes from one AST context to another.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2019-05-13 18:06:25 +08:00
|
|
|
#include "llvm/ADT/StringMap.h"
|
2019-02-14 21:07:03 +08:00
|
|
|
|
2018-12-17 20:42:12 +08:00
|
|
|
#include "clang/AST/DeclContextInternals.h"
|
2019-05-13 18:06:25 +08:00
|
|
|
|
|
|
|
#include "ASTImporterFixtures.h"
|
|
|
|
#include "MatchVerifier.h"
|
2016-04-14 19:51:27 +08:00
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace ast_matchers {
|
|
|
|
|
2018-04-24 18:11:53 +08:00
|
|
|
using internal::Matcher;
|
|
|
|
using internal::BindableMatcher;
|
|
|
|
using llvm::StringMap;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
// Base class for those tests which use the family of `testImport` functions.
|
2019-02-08 00:52:48 +08:00
|
|
|
class TestImportBase : public CompilerOptionSpecificTest,
|
|
|
|
public ::testing::WithParamInterface<ArgVector> {
|
2018-04-24 18:11:53 +08:00
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
template <typename NodeType>
|
2019-04-08 21:59:15 +08:00
|
|
|
llvm::Expected<NodeType> importNode(ASTUnit *From, ASTUnit *To,
|
|
|
|
ASTImporter &Importer, NodeType Node) {
|
2018-06-25 21:04:37 +08:00
|
|
|
ASTContext &ToCtx = To->getASTContext();
|
2018-03-31 06:03:29 +08:00
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
// Add 'From' file to virtual file system so importer can 'find' it
|
|
|
|
// while importing SourceLocations. It is safe to add same file multiple
|
|
|
|
// times - it just isn't replaced.
|
|
|
|
StringRef FromFileName = From->getMainFileName();
|
|
|
|
createVirtualFileIfNeeded(To, FromFileName,
|
|
|
|
From->getBufferForFile(FromFileName));
|
2018-04-24 18:11:53 +08:00
|
|
|
|
[ASTImporter] Use llvm::Expected and Error in the importer API
Summary:
This is the final phase of the refactoring towards using llvm::Expected
and llvm::Error in the ASTImporter API.
This involves the following:
- remove old Import functions which returned with a pointer,
- use the Import_New functions (which return with Err or Expected) everywhere
and handle their return value
- rename Import_New functions to Import
This affects both Clang and LLDB.
Reviewers: shafik, teemperor, aprantl, a_sidorin, balazske, a.sidorin
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits, lldb-commits
Tags: #clang, #lldb
Differential Revision: https://reviews.llvm.org/D61438
llvm-svn: 360760
2019-05-15 18:29:48 +08:00
|
|
|
auto Imported = Importer.Import(Node);
|
2016-04-14 19:51:27 +08:00
|
|
|
|
2019-04-08 21:59:15 +08:00
|
|
|
if (Imported) {
|
|
|
|
// This should dump source locations and assert if some source locations
|
|
|
|
// were not imported.
|
|
|
|
SmallString<1024> ImportChecker;
|
|
|
|
llvm::raw_svector_ostream ToNothing(ImportChecker);
|
|
|
|
ToCtx.getTranslationUnitDecl()->print(ToNothing);
|
2018-06-25 21:04:37 +08:00
|
|
|
|
2019-04-08 21:59:15 +08:00
|
|
|
// This traverses the AST to catch certain bugs like poorly or not
|
|
|
|
// implemented subtrees.
|
|
|
|
(*Imported)->dump(ToNothing);
|
|
|
|
}
|
2018-06-25 21:04:37 +08:00
|
|
|
|
|
|
|
return Imported;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename NodeType>
|
|
|
|
testing::AssertionResult
|
|
|
|
testImport(const std::string &FromCode, const ArgVector &FromArgs,
|
|
|
|
const std::string &ToCode, const ArgVector &ToArgs,
|
|
|
|
MatchVerifier<NodeType> &Verifier,
|
|
|
|
const BindableMatcher<NodeType> &SearchMatcher,
|
|
|
|
const BindableMatcher<NodeType> &VerificationMatcher) {
|
|
|
|
const char *const InputFileName = "input.cc";
|
|
|
|
const char *const OutputFileName = "output.cc";
|
|
|
|
|
|
|
|
std::unique_ptr<ASTUnit> FromAST = tooling::buildASTFromCodeWithArgs(
|
|
|
|
FromCode, FromArgs, InputFileName),
|
|
|
|
ToAST = tooling::buildASTFromCodeWithArgs(
|
|
|
|
ToCode, ToArgs, OutputFileName);
|
|
|
|
|
|
|
|
ASTContext &FromCtx = FromAST->getASTContext(),
|
|
|
|
&ToCtx = ToAST->getASTContext();
|
|
|
|
|
|
|
|
ASTImporter Importer(ToCtx, ToAST->getFileManager(), FromCtx,
|
|
|
|
FromAST->getFileManager(), false);
|
|
|
|
|
|
|
|
auto FoundNodes = match(SearchMatcher, FromCtx);
|
|
|
|
if (FoundNodes.size() != 1)
|
|
|
|
return testing::AssertionFailure()
|
|
|
|
<< "Multiple potential nodes were found!";
|
|
|
|
|
|
|
|
auto ToImport = selectFirst<NodeType>(DeclToImportID, FoundNodes);
|
|
|
|
if (!ToImport)
|
|
|
|
return testing::AssertionFailure() << "Node type mismatch!";
|
|
|
|
|
|
|
|
// Sanity check: the node being imported should match in the same way as
|
|
|
|
// the result node.
|
|
|
|
BindableMatcher<NodeType> WrapperMatcher(VerificationMatcher);
|
|
|
|
EXPECT_TRUE(Verifier.match(ToImport, WrapperMatcher));
|
|
|
|
|
|
|
|
auto Imported = importNode(FromAST.get(), ToAST.get(), Importer, ToImport);
|
2019-04-08 21:59:15 +08:00
|
|
|
if (!Imported) {
|
|
|
|
std::string ErrorText;
|
|
|
|
handleAllErrors(
|
|
|
|
Imported.takeError(),
|
|
|
|
[&ErrorText](const ImportError &Err) { ErrorText = Err.message(); });
|
|
|
|
return testing::AssertionFailure()
|
|
|
|
<< "Import failed, error: \"" << ErrorText << "\"!";
|
|
|
|
}
|
2018-11-09 08:41:36 +08:00
|
|
|
|
2019-04-08 21:59:15 +08:00
|
|
|
return Verifier.match(*Imported, WrapperMatcher);
|
2018-06-25 21:04:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename NodeType>
|
|
|
|
testing::AssertionResult
|
|
|
|
testImport(const std::string &FromCode, const ArgVector &FromArgs,
|
|
|
|
const std::string &ToCode, const ArgVector &ToArgs,
|
|
|
|
MatchVerifier<NodeType> &Verifier,
|
|
|
|
const BindableMatcher<NodeType> &VerificationMatcher) {
|
|
|
|
return testImport(
|
|
|
|
FromCode, FromArgs, ToCode, ToArgs, Verifier,
|
|
|
|
translationUnitDecl(
|
|
|
|
has(namedDecl(hasName(DeclToImportID)).bind(DeclToImportID))),
|
|
|
|
VerificationMatcher);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
protected:
|
|
|
|
ArgVector getExtraArgs() const override { return GetParam(); }
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
public:
|
|
|
|
|
|
|
|
/// Test how AST node named "declToImport" located in the translation unit
|
|
|
|
/// of "FromCode" virtual file is imported to "ToCode" virtual file.
|
|
|
|
/// The verification is done by running AMatcher over the imported node.
|
|
|
|
template <typename NodeType, typename MatcherType>
|
|
|
|
void testImport(const std::string &FromCode, Language FromLang,
|
|
|
|
const std::string &ToCode, Language ToLang,
|
|
|
|
MatchVerifier<NodeType> &Verifier,
|
|
|
|
const MatcherType &AMatcher) {
|
|
|
|
ArgVector FromArgs = getArgVectorForLanguage(FromLang),
|
|
|
|
ToArgs = getArgVectorForLanguage(ToLang);
|
|
|
|
EXPECT_TRUE(
|
|
|
|
testImport(FromCode, FromArgs, ToCode, ToArgs, Verifier, AMatcher));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ImportAction {
|
|
|
|
StringRef FromFilename;
|
|
|
|
StringRef ToFilename;
|
|
|
|
// FIXME: Generalize this to support other node kinds.
|
|
|
|
BindableMatcher<Decl> ImportPredicate;
|
|
|
|
|
|
|
|
ImportAction(StringRef FromFilename, StringRef ToFilename,
|
|
|
|
DeclarationMatcher ImportPredicate)
|
|
|
|
: FromFilename(FromFilename), ToFilename(ToFilename),
|
|
|
|
ImportPredicate(ImportPredicate) {}
|
|
|
|
|
|
|
|
ImportAction(StringRef FromFilename, StringRef ToFilename,
|
|
|
|
const std::string &DeclName)
|
|
|
|
: FromFilename(FromFilename), ToFilename(ToFilename),
|
|
|
|
ImportPredicate(namedDecl(hasName(DeclName))) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
using SingleASTUnit = std::unique_ptr<ASTUnit>;
|
|
|
|
using AllASTUnits = StringMap<SingleASTUnit>;
|
|
|
|
|
|
|
|
struct CodeEntry {
|
|
|
|
std::string CodeSample;
|
|
|
|
Language Lang;
|
|
|
|
};
|
|
|
|
|
|
|
|
using CodeFiles = StringMap<CodeEntry>;
|
|
|
|
|
|
|
|
/// Builds an ASTUnit for one potential compile options set.
|
|
|
|
SingleASTUnit createASTUnit(StringRef FileName, const CodeEntry &CE) const {
|
|
|
|
ArgVector Args = getArgVectorForLanguage(CE.Lang);
|
|
|
|
auto AST = tooling::buildASTFromCodeWithArgs(CE.CodeSample, Args, FileName);
|
|
|
|
EXPECT_TRUE(AST.get());
|
|
|
|
return AST;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Test an arbitrary sequence of imports for a set of given in-memory files.
|
|
|
|
/// The verification is done by running VerificationMatcher against a
|
|
|
|
/// specified AST node inside of one of given files.
|
|
|
|
/// \param CodeSamples Map whose key is the file name and the value is the
|
|
|
|
/// file content.
|
|
|
|
/// \param ImportActions Sequence of imports. Each import in sequence
|
|
|
|
/// specifies "from file" and "to file" and a matcher that is used for
|
|
|
|
/// searching a declaration for import in "from file".
|
|
|
|
/// \param FileForFinalCheck Name of virtual file for which the final check is
|
|
|
|
/// applied.
|
|
|
|
/// \param FinalSelectPredicate Matcher that specifies the AST node in the
|
|
|
|
/// FileForFinalCheck for which the verification will be done.
|
|
|
|
/// \param VerificationMatcher Matcher that will be used for verification
|
|
|
|
/// after all imports in sequence are done.
|
|
|
|
void testImportSequence(const CodeFiles &CodeSamples,
|
|
|
|
const std::vector<ImportAction> &ImportActions,
|
|
|
|
StringRef FileForFinalCheck,
|
|
|
|
BindableMatcher<Decl> FinalSelectPredicate,
|
|
|
|
BindableMatcher<Decl> VerificationMatcher) {
|
|
|
|
AllASTUnits AllASTs;
|
|
|
|
using ImporterKey = std::pair<const ASTUnit *, const ASTUnit *>;
|
|
|
|
llvm::DenseMap<ImporterKey, std::unique_ptr<ASTImporter>> Importers;
|
|
|
|
|
|
|
|
auto GenASTsIfNeeded = [this, &AllASTs, &CodeSamples](StringRef Filename) {
|
|
|
|
if (!AllASTs.count(Filename)) {
|
|
|
|
auto Found = CodeSamples.find(Filename);
|
|
|
|
assert(Found != CodeSamples.end() && "Wrong file for import!");
|
|
|
|
AllASTs[Filename] = createASTUnit(Filename, Found->getValue());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const ImportAction &Action : ImportActions) {
|
|
|
|
StringRef FromFile = Action.FromFilename, ToFile = Action.ToFilename;
|
|
|
|
GenASTsIfNeeded(FromFile);
|
|
|
|
GenASTsIfNeeded(ToFile);
|
|
|
|
|
|
|
|
ASTUnit *From = AllASTs[FromFile].get();
|
|
|
|
ASTUnit *To = AllASTs[ToFile].get();
|
|
|
|
|
|
|
|
// Create a new importer if needed.
|
|
|
|
std::unique_ptr<ASTImporter> &ImporterRef = Importers[{From, To}];
|
|
|
|
if (!ImporterRef)
|
|
|
|
ImporterRef.reset(new ASTImporter(
|
|
|
|
To->getASTContext(), To->getFileManager(), From->getASTContext(),
|
|
|
|
From->getFileManager(), false));
|
|
|
|
|
|
|
|
// Find the declaration and import it.
|
|
|
|
auto FoundDecl = match(Action.ImportPredicate.bind(DeclToImportID),
|
|
|
|
From->getASTContext());
|
|
|
|
EXPECT_TRUE(FoundDecl.size() == 1);
|
|
|
|
const Decl *ToImport = selectFirst<Decl>(DeclToImportID, FoundDecl);
|
|
|
|
auto Imported = importNode(From, To, *ImporterRef, ToImport);
|
2019-04-08 21:59:15 +08:00
|
|
|
EXPECT_TRUE(static_cast<bool>(Imported));
|
|
|
|
if (!Imported)
|
|
|
|
llvm::consumeError(Imported.takeError());
|
2018-06-25 21:04:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find the declaration and import it.
|
|
|
|
auto FoundDecl = match(FinalSelectPredicate.bind(DeclToVerifyID),
|
|
|
|
AllASTs[FileForFinalCheck]->getASTContext());
|
|
|
|
EXPECT_TRUE(FoundDecl.size() == 1);
|
|
|
|
const Decl *ToVerify = selectFirst<Decl>(DeclToVerifyID, FoundDecl);
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
EXPECT_TRUE(
|
|
|
|
Verifier.match(ToVerify, BindableMatcher<Decl>(VerificationMatcher)));
|
|
|
|
}
|
|
|
|
};
|
2017-12-22 01:41:06 +08:00
|
|
|
|
2018-07-17 20:06:36 +08:00
|
|
|
template <typename T> RecordDecl *getRecordDecl(T *D) {
|
|
|
|
auto *ET = cast<ElaboratedType>(D->getType().getTypePtr());
|
|
|
|
return cast<RecordType>(ET->getNamedType().getTypePtr())->getDecl();
|
2018-07-17 22:20:02 +08:00
|
|
|
}
|
2018-07-17 20:06:36 +08:00
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
struct ImportExpr : TestImportBase {};
|
|
|
|
struct ImportType : TestImportBase {};
|
|
|
|
struct ImportDecl : TestImportBase {};
|
2018-04-24 18:11:53 +08:00
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct CanonicalRedeclChain : ASTImporterOptionSpecificTestBase {};
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
|
|
|
|
TEST_P(CanonicalRedeclChain, ShouldBeConsequentWithMatchers) {
|
|
|
|
Decl *FromTU = getTuDecl("void f();", Lang_CXX);
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
auto *D0 = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto Redecls = getCanonicalForwardRedeclChain(D0);
|
|
|
|
ASSERT_EQ(Redecls.size(), 1u);
|
|
|
|
EXPECT_EQ(D0, Redecls[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CanonicalRedeclChain, ShouldBeConsequentWithMatchers2) {
|
|
|
|
Decl *FromTU = getTuDecl("void f(); void f(); void f();", Lang_CXX);
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
auto *D0 = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
auto *D2 = LastDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
FunctionDecl *D1 = D2->getPreviousDecl();
|
|
|
|
|
|
|
|
auto Redecls = getCanonicalForwardRedeclChain(D0);
|
|
|
|
ASSERT_EQ(Redecls.size(), 3u);
|
|
|
|
EXPECT_EQ(D0, Redecls[0]);
|
|
|
|
EXPECT_EQ(D1, Redecls[1]);
|
|
|
|
EXPECT_EQ(D2, Redecls[2]);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(CanonicalRedeclChain, ShouldBeSameForAllDeclInTheChain) {
|
|
|
|
Decl *FromTU = getTuDecl("void f(); void f(); void f();", Lang_CXX);
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
auto *D0 = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
auto *D2 = LastDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
FunctionDecl *D1 = D2->getPreviousDecl();
|
|
|
|
|
|
|
|
auto RedeclsD0 = getCanonicalForwardRedeclChain(D0);
|
|
|
|
auto RedeclsD1 = getCanonicalForwardRedeclChain(D1);
|
|
|
|
auto RedeclsD2 = getCanonicalForwardRedeclChain(D2);
|
|
|
|
|
|
|
|
EXPECT_THAT(RedeclsD0, ::testing::ContainerEq(RedeclsD1));
|
|
|
|
EXPECT_THAT(RedeclsD1, ::testing::ContainerEq(RedeclsD2));
|
|
|
|
}
|
|
|
|
|
[ASTImporter] Add an ImportImpl method to allow customizing Import behavior.
Summary:
We are currently implementing support in LLDB that reconstructs the STL templates from
the target program in the expression evaluator. This reconstruction happens during the
import process from our debug info AST into the expression evaluation AST, which means
we need a way to intercept the ASTImporter import process.
This patch adds an protected ImportImpl method that we can overwrite in LLDB to implement
our special importing logic (which is essentially just looking into a C++ module that is attached to
the target context). Because ImportImpl has to call MapImported/AddToLookup for the decls it
creates, this patch also exposes those via a new unified method and checks that we call it when
importing decls.
Reviewers: martong, balazske, a.sidorin, shafik, a_sidorin
Reviewed By: martong, a_sidorin
Subscribers: rnkovacs, cfe-commits, lldb-commits, aprantl
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59485
llvm-svn: 359502
2019-04-30 05:02:35 +08:00
|
|
|
namespace {
|
|
|
|
struct RedirectingImporter : public ASTImporter {
|
|
|
|
using ASTImporter::ASTImporter;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
llvm::Expected<Decl *> ImportImpl(Decl *FromD) override {
|
|
|
|
auto *ND = dyn_cast<NamedDecl>(FromD);
|
|
|
|
if (!ND || ND->getName() != "shouldNotBeImported")
|
|
|
|
return ASTImporter::ImportImpl(FromD);
|
|
|
|
for (Decl *D : getToContext().getTranslationUnitDecl()->decls()) {
|
|
|
|
if (auto *ND = dyn_cast<NamedDecl>(D))
|
|
|
|
if (ND->getName() == "realDecl") {
|
|
|
|
RegisterImportedDecl(FromD, ND);
|
|
|
|
return ND;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ASTImporter::ImportImpl(FromD);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
struct RedirectingImporterTest : ASTImporterOptionSpecificTestBase {
|
|
|
|
RedirectingImporterTest() {
|
|
|
|
Creator = [](ASTContext &ToContext, FileManager &ToFileManager,
|
|
|
|
ASTContext &FromContext, FileManager &FromFileManager,
|
2019-07-01 23:37:07 +08:00
|
|
|
bool MinimalImport,
|
|
|
|
const std::shared_ptr<ASTImporterSharedState> &SharedState) {
|
[ASTImporter] Add an ImportImpl method to allow customizing Import behavior.
Summary:
We are currently implementing support in LLDB that reconstructs the STL templates from
the target program in the expression evaluator. This reconstruction happens during the
import process from our debug info AST into the expression evaluation AST, which means
we need a way to intercept the ASTImporter import process.
This patch adds an protected ImportImpl method that we can overwrite in LLDB to implement
our special importing logic (which is essentially just looking into a C++ module that is attached to
the target context). Because ImportImpl has to call MapImported/AddToLookup for the decls it
creates, this patch also exposes those via a new unified method and checks that we call it when
importing decls.
Reviewers: martong, balazske, a.sidorin, shafik, a_sidorin
Reviewed By: martong, a_sidorin
Subscribers: rnkovacs, cfe-commits, lldb-commits, aprantl
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59485
llvm-svn: 359502
2019-04-30 05:02:35 +08:00
|
|
|
return new RedirectingImporter(ToContext, ToFileManager, FromContext,
|
|
|
|
FromFileManager, MinimalImport,
|
2019-07-01 23:37:07 +08:00
|
|
|
SharedState);
|
[ASTImporter] Add an ImportImpl method to allow customizing Import behavior.
Summary:
We are currently implementing support in LLDB that reconstructs the STL templates from
the target program in the expression evaluator. This reconstruction happens during the
import process from our debug info AST into the expression evaluation AST, which means
we need a way to intercept the ASTImporter import process.
This patch adds an protected ImportImpl method that we can overwrite in LLDB to implement
our special importing logic (which is essentially just looking into a C++ module that is attached to
the target context). Because ImportImpl has to call MapImported/AddToLookup for the decls it
creates, this patch also exposes those via a new unified method and checks that we call it when
importing decls.
Reviewers: martong, balazske, a.sidorin, shafik, a_sidorin
Reviewed By: martong, a_sidorin
Subscribers: rnkovacs, cfe-commits, lldb-commits, aprantl
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59485
llvm-svn: 359502
2019-04-30 05:02:35 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Test that an ASTImporter subclass can intercept an import call.
|
|
|
|
TEST_P(RedirectingImporterTest, InterceptImport) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl("class shouldNotBeImported {};", Lang_CXX,
|
|
|
|
"class realDecl {};", Lang_CXX, "shouldNotBeImported");
|
|
|
|
auto *Imported = cast<CXXRecordDecl>(To);
|
|
|
|
EXPECT_EQ(Imported->getQualifiedNameAsString(), "realDecl");
|
|
|
|
|
|
|
|
// Make sure our importer prevented the importing of the decl.
|
|
|
|
auto *ToTU = Imported->getTranslationUnitDecl();
|
|
|
|
auto Pattern = functionDecl(hasName("shouldNotBeImported"));
|
|
|
|
unsigned count =
|
|
|
|
DeclCounterWithPredicate<CXXRecordDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_EQ(0U, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test that when we indirectly import a declaration the custom ASTImporter
|
|
|
|
// is still intercepting the import.
|
|
|
|
TEST_P(RedirectingImporterTest, InterceptIndirectImport) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl("class shouldNotBeImported {};"
|
|
|
|
"class F { shouldNotBeImported f; };",
|
|
|
|
Lang_CXX, "class realDecl {};", Lang_CXX, "F");
|
|
|
|
|
|
|
|
// Make sure our ASTImporter prevented the importing of the decl.
|
|
|
|
auto *ToTU = To->getTranslationUnitDecl();
|
|
|
|
auto Pattern = functionDecl(hasName("shouldNotBeImported"));
|
|
|
|
unsigned count =
|
|
|
|
DeclCounterWithPredicate<CXXRecordDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_EQ(0U, count);
|
|
|
|
}
|
|
|
|
|
[ASTImporter] Mark erroneous nodes in from ctx
Summary:
During import of a specific Decl D, it may happen that some AST nodes
had already been created before we recognize an error. In this case we
signal back the error to the caller, but the "to" context remains
polluted with those nodes which had been created. Ideally, those nodes
should not had been created, but that time we did not know about the
error, the error happened later. Since the AST is immutable (most of
the cases we can't remove existing nodes) we choose to mark these nodes
as erroneous.
Here are the steps of the algorithm:
1) We keep track of the nodes which we visit during the import of D: See
ImportPathTy.
2) If a Decl is already imported and it is already on the import path
(we have a cycle) then we copy/store the relevant part of the import
path. We store these cycles for each Decl.
3) When we recognize an error during the import of D then we set up this
error to all Decls in the stored cycles for D and we clear the stored
cycles.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62375
llvm-svn: 364771
2019-07-01 22:19:53 +08:00
|
|
|
struct ImportPath : ASTImporterOptionSpecificTestBase {
|
|
|
|
Decl *FromTU;
|
|
|
|
FunctionDecl *D0, *D1, *D2;
|
|
|
|
ImportPath() {
|
|
|
|
FromTU = getTuDecl("void f(); void f(); void f();", Lang_CXX);
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
D0 = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
D2 = LastDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
D1 = D2->getPreviousDecl();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(ImportPath, Push) {
|
|
|
|
ASTImporter::ImportPathTy path;
|
|
|
|
path.push(D0);
|
|
|
|
EXPECT_FALSE(path.hasCycleAtBack());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportPath, SmallCycle) {
|
|
|
|
ASTImporter::ImportPathTy path;
|
|
|
|
path.push(D0);
|
|
|
|
path.push(D0);
|
|
|
|
EXPECT_TRUE(path.hasCycleAtBack());
|
|
|
|
path.pop();
|
|
|
|
EXPECT_FALSE(path.hasCycleAtBack());
|
|
|
|
path.push(D0);
|
|
|
|
EXPECT_TRUE(path.hasCycleAtBack());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportPath, GetSmallCycle) {
|
|
|
|
ASTImporter::ImportPathTy path;
|
|
|
|
path.push(D0);
|
|
|
|
path.push(D0);
|
|
|
|
EXPECT_TRUE(path.hasCycleAtBack());
|
|
|
|
std::array<Decl* ,2> Res;
|
|
|
|
int i = 0;
|
|
|
|
for (Decl *Di : path.getCycleAtBack()) {
|
|
|
|
Res[i++] = Di;
|
|
|
|
}
|
|
|
|
ASSERT_EQ(i, 2);
|
|
|
|
EXPECT_EQ(Res[0], D0);
|
|
|
|
EXPECT_EQ(Res[1], D0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportPath, GetCycle) {
|
|
|
|
ASTImporter::ImportPathTy path;
|
|
|
|
path.push(D0);
|
|
|
|
path.push(D1);
|
|
|
|
path.push(D2);
|
|
|
|
path.push(D0);
|
|
|
|
EXPECT_TRUE(path.hasCycleAtBack());
|
|
|
|
std::array<Decl* ,4> Res;
|
|
|
|
int i = 0;
|
|
|
|
for (Decl *Di : path.getCycleAtBack()) {
|
|
|
|
Res[i++] = Di;
|
|
|
|
}
|
|
|
|
ASSERT_EQ(i, 4);
|
|
|
|
EXPECT_EQ(Res[0], D0);
|
|
|
|
EXPECT_EQ(Res[1], D2);
|
|
|
|
EXPECT_EQ(Res[2], D1);
|
|
|
|
EXPECT_EQ(Res[3], D0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportPath, CycleAfterCycle) {
|
|
|
|
ASTImporter::ImportPathTy path;
|
|
|
|
path.push(D0);
|
|
|
|
path.push(D1);
|
|
|
|
path.push(D0);
|
|
|
|
path.push(D1);
|
|
|
|
path.push(D2);
|
|
|
|
path.push(D0);
|
|
|
|
EXPECT_TRUE(path.hasCycleAtBack());
|
|
|
|
std::array<Decl* ,4> Res;
|
|
|
|
int i = 0;
|
|
|
|
for (Decl *Di : path.getCycleAtBack()) {
|
|
|
|
Res[i++] = Di;
|
|
|
|
}
|
|
|
|
ASSERT_EQ(i, 4);
|
|
|
|
EXPECT_EQ(Res[0], D0);
|
|
|
|
EXPECT_EQ(Res[1], D2);
|
|
|
|
EXPECT_EQ(Res[2], D1);
|
|
|
|
EXPECT_EQ(Res[3], D0);
|
|
|
|
|
|
|
|
path.pop();
|
|
|
|
path.pop();
|
|
|
|
path.pop();
|
|
|
|
EXPECT_TRUE(path.hasCycleAtBack());
|
|
|
|
i = 0;
|
|
|
|
for (Decl *Di : path.getCycleAtBack()) {
|
|
|
|
Res[i++] = Di;
|
|
|
|
}
|
|
|
|
ASSERT_EQ(i, 3);
|
|
|
|
EXPECT_EQ(Res[0], D0);
|
|
|
|
EXPECT_EQ(Res[1], D1);
|
|
|
|
EXPECT_EQ(Res[2], D0);
|
|
|
|
|
|
|
|
path.pop();
|
|
|
|
EXPECT_FALSE(path.hasCycleAtBack());
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportStringLiteral) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)\"foo\"; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
stringLiteral(hasType(asString("const char [4]"))))));
|
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)L\"foo\"; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
stringLiteral(hasType(asString("const wchar_t [4]"))))));
|
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void) \"foo\" \"bar\"; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
stringLiteral(hasType(asString("const char [7]"))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
[ASTImporter] Add support for importing ChooseExpr AST nodes.
Summary:
This allows ASTs to be merged when they contain ChooseExpr (the GNU
__builtin_choose_expr construction). This is needed, for example, for
cross-CTU analysis of C code that makes use of __builtin_choose_expr.
The node is already supported in the AST, but it didn't have a matcher
in ASTMatchers. So, this change adds the matcher and adds support to
ASTImporter.
This was originally reviewed and approved in
https://reviews.llvm.org/D58292 and submitted as r354832. It was
reverted in r354839 due to failures on the Windows CI builds.
This version fixes the test failures on Windows, which were caused by
differences in template expansion between versions of clang on different
OSes. The version of clang built with MSVC and running on Windows never
expands the template in the C++ test in ImportExpr.ImportChooseExpr in
clang/unittests/AST/ASTImporter.cpp, but the version on Linux does for
the empty arguments and -fms-compatibility.
So, this version of the patch drops the C++ test for
__builtin_choose_expr, since that version was written to catch
regressions of the logic for isConditionTrue() in the AST import code
for ChooseExpr, and those regressions are also caught by
ASTImporterOptionSpecificTestBase.ImportChooseExpr, which does work on
Windows.
Reviewers: shafik, a_sidorin, martong, aaron.ballman, rnk, a.sidorin
Subscribers: cfe-commits, jdoerfert, rnkovacs, aaron.ballman
Tags: #clang
Differential Revision: https://reviews.llvm.org/D58663
llvm-svn: 354916
2019-02-27 03:26:41 +08:00
|
|
|
TEST_P(ImportExpr, ImportChooseExpr) {
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
|
|
|
|
// This case tests C code that is not condition-dependent and has a true
|
|
|
|
// condition.
|
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)__builtin_choose_expr(1, 2, 3); }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(chooseExpr())));
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportGNUNullExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)__null; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(gnuNullExpr(hasType(isInteger())))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCXXNullPtrLiteralExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)nullptr; }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionDecl(hasDescendant(cxxNullPtrLiteralExpr())));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportFloatinglLiteralExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)1.0; }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
floatLiteral(equals(1.0), hasType(asString("double"))))));
|
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)1.0e-5f; }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
floatLiteral(equals(1.0e-5f), hasType(asString("float"))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-08-09 20:18:07 +08:00
|
|
|
TEST_P(ImportExpr, ImportImaginaryLiteralExpr) {
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)1.0i; }",
|
|
|
|
Lang_CXX14, "", Lang_CXX14, Verifier,
|
|
|
|
functionDecl(hasDescendant(imaginaryLiteral())));
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCompoundLiteralExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() {"
|
|
|
|
" struct s { int x; long y; unsigned z; }; "
|
|
|
|
" (void)(struct s){ 42, 0L, 1U }; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
compoundLiteralExpr(
|
|
|
|
hasType(asString("struct s")),
|
|
|
|
has(initListExpr(
|
|
|
|
hasType(asString("struct s")),
|
|
|
|
has(integerLiteral(
|
|
|
|
equals(42), hasType(asString("int")))),
|
|
|
|
has(integerLiteral(
|
|
|
|
equals(0), hasType(asString("long")))),
|
|
|
|
has(integerLiteral(
|
|
|
|
equals(1), hasType(asString("unsigned int"))))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCXXThisExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"class declToImport { void f() { (void)this; } };",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
cxxRecordDecl(
|
|
|
|
hasMethod(
|
|
|
|
hasDescendant(
|
|
|
|
cxxThisExpr(
|
|
|
|
hasType(
|
|
|
|
asString("class declToImport *")))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportAtomicExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { int *ptr; __atomic_load_n(ptr, 1); }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
atomicExpr(
|
|
|
|
has(ignoringParenImpCasts(
|
|
|
|
declRefExpr(hasDeclaration(varDecl(hasName("ptr"))),
|
|
|
|
hasType(asString("int *"))))),
|
|
|
|
has(integerLiteral(equals(1), hasType(asString("int"))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportLabelDeclAndAddrLabelExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2017-12-22 01:41:06 +08:00
|
|
|
testImport(
|
2018-06-29 18:25:19 +08:00
|
|
|
"void declToImport() { loop: goto loop; (void)&&loop; }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(
|
|
|
|
hasDescendant(
|
|
|
|
labelStmt(hasDeclaration(labelDecl(hasName("loop"))))),
|
|
|
|
hasDescendant(
|
|
|
|
addrLabelExpr(hasDeclaration(labelDecl(hasName("loop")))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
AST_MATCHER_P(TemplateDecl, hasTemplateDecl,
|
|
|
|
internal::Matcher<NamedDecl>, InnerMatcher) {
|
|
|
|
const NamedDecl *Template = Node.getTemplatedDecl();
|
|
|
|
return Template && InnerMatcher.matches(*Template, Finder, Builder);
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportParenListExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2017-12-22 01:41:06 +08:00
|
|
|
testImport(
|
2017-08-02 21:04:44 +08:00
|
|
|
"template<typename T> class dummy { void f() { dummy X(*this); } };"
|
|
|
|
"typedef dummy<int> declToImport;"
|
|
|
|
"template class dummy<int>;",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
typedefDecl(hasType(templateSpecializationType(
|
|
|
|
hasDeclaration(classTemplateSpecializationDecl(hasSpecializedTemplate(
|
|
|
|
classTemplateDecl(hasTemplateDecl(cxxRecordDecl(hasMethod(allOf(
|
|
|
|
hasName("f"),
|
|
|
|
hasBody(compoundStmt(has(declStmt(hasSingleDecl(
|
|
|
|
varDecl(hasInitializer(parenListExpr(has(unaryOperator(
|
|
|
|
hasOperatorName("*"),
|
2017-12-22 01:41:06 +08:00
|
|
|
hasUnaryOperand(cxxThisExpr())))))))))))))))))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportSwitch) {
|
2017-10-18 17:25:18 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { int b; switch (b) { case 1: break; } }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
switchStmt(has(compoundStmt(has(caseStmt())))))));
|
2017-10-18 17:25:18 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportStmtExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2017-12-22 01:41:06 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { int b; int a = b ?: 1; int C = ({int X=4; X;}); }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
2018-06-29 18:25:19 +08:00
|
|
|
functionDecl(hasDescendant(
|
|
|
|
varDecl(
|
|
|
|
hasName("C"),
|
|
|
|
hasType(asString("int")),
|
|
|
|
hasInitializer(
|
|
|
|
stmtExpr(
|
|
|
|
hasAnySubstatement(declStmt(hasSingleDecl(
|
|
|
|
varDecl(
|
|
|
|
hasName("X"),
|
|
|
|
hasType(asString("int")),
|
|
|
|
hasInitializer(
|
2017-12-22 01:41:06 +08:00
|
|
|
integerLiteral(equals(4))))))),
|
2018-06-29 18:25:19 +08:00
|
|
|
hasDescendant(
|
|
|
|
implicitCastExpr())))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportConditionalOperator) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2017-12-22 01:41:06 +08:00
|
|
|
testImport(
|
2018-06-29 18:25:19 +08:00
|
|
|
"void declToImport() { (void)(true ? 1 : -5); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
conditionalOperator(
|
2017-12-22 01:41:06 +08:00
|
|
|
hasCondition(cxxBoolLiteral(equals(true))),
|
|
|
|
hasTrueExpression(integerLiteral(equals(1))),
|
|
|
|
hasFalseExpression(
|
2018-06-29 18:25:19 +08:00
|
|
|
unaryOperator(hasUnaryOperand(integerLiteral(equals(5))))))
|
|
|
|
)));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportBinaryConditionalOperator) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2017-12-22 01:41:06 +08:00
|
|
|
testImport(
|
2018-06-29 18:25:19 +08:00
|
|
|
"void declToImport() { (void)(1 ?: -5); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
binaryConditionalOperator(
|
2017-12-22 01:41:06 +08:00
|
|
|
hasCondition(
|
2018-06-29 18:25:19 +08:00
|
|
|
implicitCastExpr(
|
|
|
|
hasSourceExpression(opaqueValueExpr(
|
|
|
|
hasSourceExpression(integerLiteral(equals(1))))),
|
|
|
|
hasType(booleanType()))),
|
2017-12-22 01:41:06 +08:00
|
|
|
hasTrueExpression(
|
2018-06-29 18:25:19 +08:00
|
|
|
opaqueValueExpr(
|
|
|
|
hasSourceExpression(integerLiteral(equals(1))))),
|
2017-12-22 01:41:06 +08:00
|
|
|
hasFalseExpression(
|
2018-06-29 18:25:19 +08:00
|
|
|
unaryOperator(
|
|
|
|
hasOperatorName("-"),
|
|
|
|
hasUnaryOperand(integerLiteral(equals(5)))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportDesignatedInitExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() {"
|
|
|
|
" struct point { double x; double y; };"
|
|
|
|
" struct point ptarray[10] = "
|
|
|
|
"{ [2].y = 1.0, [2].x = 2.0, [0].x = 1.0 }; }",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
initListExpr(
|
|
|
|
has(designatedInitExpr(
|
|
|
|
designatorCountIs(2),
|
2018-11-09 08:41:36 +08:00
|
|
|
hasDescendant(floatLiteral(equals(1.0))),
|
|
|
|
hasDescendant(integerLiteral(equals(2))))),
|
2018-06-29 18:25:19 +08:00
|
|
|
has(designatedInitExpr(
|
|
|
|
designatorCountIs(2),
|
2018-11-09 08:41:36 +08:00
|
|
|
hasDescendant(floatLiteral(equals(2.0))),
|
|
|
|
hasDescendant(integerLiteral(equals(2))))),
|
2018-06-29 18:25:19 +08:00
|
|
|
has(designatedInitExpr(
|
|
|
|
designatorCountIs(2),
|
2018-11-09 08:41:36 +08:00
|
|
|
hasDescendant(floatLiteral(equals(1.0))),
|
|
|
|
hasDescendant(integerLiteral(equals(0)))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportPredefinedExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
// __func__ expands as StringLiteral("declToImport")
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { (void)__func__; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
predefinedExpr(
|
|
|
|
hasType(
|
|
|
|
asString("const char [13]")),
|
|
|
|
has(stringLiteral(hasType(
|
|
|
|
asString("const char [13]"))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportInitListExpr) {
|
2016-04-14 19:51:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2017-12-22 01:41:06 +08:00
|
|
|
testImport(
|
2018-06-29 18:25:19 +08:00
|
|
|
"void declToImport() {"
|
|
|
|
" struct point { double x; double y; };"
|
|
|
|
" point ptarray[10] = { [2].y = 1.0, [2].x = 2.0,"
|
|
|
|
" [0].x = 1.0 }; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
initListExpr(
|
|
|
|
has(
|
|
|
|
cxxConstructExpr(
|
|
|
|
requiresZeroInitialization())),
|
|
|
|
has(
|
|
|
|
initListExpr(
|
|
|
|
hasType(asString("struct point")),
|
|
|
|
has(floatLiteral(equals(1.0))),
|
|
|
|
has(implicitValueInitExpr(
|
|
|
|
hasType(asString("double")))))),
|
|
|
|
has(
|
|
|
|
initListExpr(
|
|
|
|
hasType(asString("struct point")),
|
|
|
|
has(floatLiteral(equals(2.0))),
|
|
|
|
has(floatLiteral(equals(1.0)))))))));
|
2016-04-14 19:51:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-28 18:16:56 +08:00
|
|
|
const internal::VariadicDynCastAllOfMatcher<Expr, VAArgExpr> vaArgExpr;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportVAArgExpr) {
|
2016-09-28 18:16:56 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport(__builtin_va_list list, ...) {"
|
|
|
|
" (void)__builtin_va_arg(list, int); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
cStyleCastExpr(hasSourceExpression(vaArgExpr())))));
|
2016-09-28 18:16:56 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, CXXTemporaryObjectExpr) {
|
2018-01-28 00:11:45 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"struct C {};"
|
|
|
|
"void declToImport() { C c = C(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
exprWithCleanups(has(cxxConstructExpr(
|
|
|
|
has(materializeTemporaryExpr(has(implicitCastExpr(
|
|
|
|
has(cxxTemporaryObjectExpr())))))))))));
|
2018-01-28 00:11:45 +08:00
|
|
|
}
|
2016-09-28 18:16:56 +08:00
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportType, ImportAtomicType) {
|
2016-11-23 23:24:23 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { typedef _Atomic(int) a_int; }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionDecl(hasDescendant(typedefDecl(has(atomicType())))));
|
2016-11-23 23:24:23 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportFunctionTemplateDecl) {
|
2017-12-28 01:04:42 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename T> void declToImport() { };",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl());
|
2017-12-28 01:04:42 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCXXDependentScopeMemberExpr) {
|
2017-12-28 01:04:42 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename T> struct C { T t; };"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" C<T> d;"
|
|
|
|
" (void)d.t;"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl(hasDescendant(
|
|
|
|
cStyleCastExpr(has(cxxDependentScopeMemberExpr())))));
|
|
|
|
testImport(
|
|
|
|
"template <typename T> struct C { T t; };"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" C<T> d;"
|
|
|
|
" (void)(&d)->t;"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl(hasDescendant(
|
|
|
|
cStyleCastExpr(has(cxxDependentScopeMemberExpr())))));
|
2017-12-28 01:04:42 +08:00
|
|
|
}
|
2016-11-23 23:24:23 +08:00
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportType, ImportTypeAliasTemplate) {
|
2017-11-14 19:30:38 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-02-14 19:18:00 +08:00
|
|
|
testImport(
|
|
|
|
"template <int K>"
|
|
|
|
"struct dummy { static const int i = K; };"
|
|
|
|
"template <int K> using dummy2 = dummy<K>;"
|
|
|
|
"int declToImport() { return dummy2<3>::i; }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionDecl(
|
2018-06-29 18:25:19 +08:00
|
|
|
hasDescendant(implicitCastExpr(has(declRefExpr()))),
|
2018-02-14 19:18:00 +08:00
|
|
|
unless(hasAncestor(translationUnitDecl(has(typeAliasDecl()))))));
|
|
|
|
}
|
|
|
|
|
|
|
|
const internal::VariadicDynCastAllOfMatcher<Decl, VarTemplateSpecializationDecl>
|
|
|
|
varTemplateSpecializationDecl;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportVarTemplate) {
|
2018-02-14 19:18:00 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport(
|
|
|
|
"template <typename T>"
|
|
|
|
"T pi = T(3.1415926535897932385L);"
|
2018-06-29 18:25:19 +08:00
|
|
|
"void declToImport() { (void)pi<int>; }",
|
|
|
|
Lang_CXX14, "", Lang_CXX14, Verifier,
|
2018-02-14 19:18:00 +08:00
|
|
|
functionDecl(
|
2018-06-29 18:25:19 +08:00
|
|
|
hasDescendant(declRefExpr(to(varTemplateSpecializationDecl()))),
|
2018-02-14 19:18:00 +08:00
|
|
|
unless(hasAncestor(translationUnitDecl(has(varDecl(
|
|
|
|
hasName("pi"), unless(varTemplateSpecializationDecl()))))))));
|
2017-11-14 19:30:38 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportType, ImportPackExpansion) {
|
2017-11-14 19:30:38 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename... Args>"
|
|
|
|
"struct dummy {"
|
|
|
|
" dummy(Args... args) {}"
|
|
|
|
" static const int i = 4;"
|
|
|
|
"};"
|
|
|
|
"int declToImport() { return dummy<int>::i; }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
returnStmt(has(implicitCastExpr(has(declRefExpr())))))));
|
2017-11-14 19:30:38 +08:00
|
|
|
}
|
|
|
|
|
2018-01-28 00:11:45 +08:00
|
|
|
const internal::VariadicDynCastAllOfMatcher<Type,
|
|
|
|
DependentTemplateSpecializationType>
|
|
|
|
dependentTemplateSpecializationType;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportType, ImportDependentTemplateSpecialization) {
|
2018-01-28 00:11:45 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template<typename T>"
|
|
|
|
"struct A;"
|
|
|
|
"template<typename T>"
|
|
|
|
"struct declToImport {"
|
|
|
|
" typename A<T>::template B<T> a;"
|
|
|
|
"};",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
classTemplateDecl(has(cxxRecordDecl(has(
|
|
|
|
fieldDecl(hasType(dependentTemplateSpecializationType())))))));
|
2018-01-28 00:11:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const internal::VariadicDynCastAllOfMatcher<Stmt, SizeOfPackExpr>
|
|
|
|
sizeOfPackExpr;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportSizeOfPackExpr) {
|
2018-01-28 00:11:45 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename... Ts>"
|
|
|
|
"void declToImport() {"
|
|
|
|
" const int i = sizeof...(Ts);"
|
|
|
|
"};"
|
|
|
|
"void g() { declToImport<int>(); }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionTemplateDecl(hasDescendant(sizeOfPackExpr())));
|
2018-01-28 00:11:45 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename... Ts>"
|
|
|
|
"using X = int[sizeof...(Ts)];"
|
|
|
|
"template <typename... Us>"
|
|
|
|
"struct Y {"
|
|
|
|
" X<Us..., int, double, int, Us...> f;"
|
|
|
|
"};"
|
|
|
|
"Y<float, int> declToImport;",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
varDecl(hasType(classTemplateSpecializationDecl(has(fieldDecl(hasType(
|
|
|
|
hasUnqualifiedDesugaredType(constantArrayType(hasSize(7))))))))));
|
|
|
|
}
|
|
|
|
|
2017-11-27 01:04:06 +08:00
|
|
|
/// \brief Matches __builtin_types_compatible_p:
|
|
|
|
/// GNU extension to check equivalent types
|
|
|
|
/// Given
|
|
|
|
/// \code
|
|
|
|
/// __builtin_types_compatible_p(int, int)
|
|
|
|
/// \endcode
|
|
|
|
// will generate TypeTraitExpr <...> 'int'
|
|
|
|
const internal::VariadicDynCastAllOfMatcher<Stmt, TypeTraitExpr> typeTraitExpr;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportTypeTraitExpr) {
|
2017-11-27 01:04:06 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"void declToImport() { "
|
|
|
|
" (void)__builtin_types_compatible_p(int, int);"
|
|
|
|
"}",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasDescendant(typeTraitExpr(hasType(asString("int"))))));
|
2017-11-27 01:04:06 +08:00
|
|
|
}
|
|
|
|
|
2018-01-28 00:11:45 +08:00
|
|
|
const internal::VariadicDynCastAllOfMatcher<Stmt, CXXTypeidExpr> cxxTypeidExpr;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCXXTypeidExpr) {
|
2018-01-28 00:11:45 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport(
|
|
|
|
"namespace std { class type_info {}; }"
|
|
|
|
"void declToImport() {"
|
|
|
|
" int x;"
|
|
|
|
" auto a = typeid(int); auto b = typeid(x);"
|
|
|
|
"}",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionDecl(
|
|
|
|
hasDescendant(varDecl(
|
|
|
|
hasName("a"), hasInitializer(hasDescendant(cxxTypeidExpr())))),
|
|
|
|
hasDescendant(varDecl(
|
|
|
|
hasName("b"), hasInitializer(hasDescendant(cxxTypeidExpr()))))));
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportTypeTraitExprValDep) {
|
2017-11-27 01:04:06 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template<typename T> struct declToImport {"
|
|
|
|
" void m() { (void)__is_pod(T); }"
|
|
|
|
"};"
|
|
|
|
"void f() { declToImport<int>().m(); }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
classTemplateDecl(has(cxxRecordDecl(has(
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
typeTraitExpr(hasType(booleanType())))))))));
|
2017-11-27 01:04:06 +08:00
|
|
|
}
|
2017-11-14 19:30:38 +08:00
|
|
|
|
2018-07-12 19:50:21 +08:00
|
|
|
TEST_P(ImportDecl, ImportRecordDeclInFunc) {
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport("int declToImport() { "
|
|
|
|
" struct data_t {int a;int b;};"
|
|
|
|
" struct data_t d;"
|
|
|
|
" return 0;"
|
|
|
|
"}",
|
|
|
|
Lang_C, "", Lang_C, Verifier,
|
|
|
|
functionDecl(hasBody(compoundStmt(
|
|
|
|
has(declStmt(hasSingleDecl(varDecl(hasName("d")))))))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportRecordTypeInFunc) {
|
2018-07-12 19:50:21 +08:00
|
|
|
Decl *FromTU = getTuDecl("int declToImport() { "
|
|
|
|
" struct data_t {int a;int b;};"
|
|
|
|
" struct data_t d;"
|
|
|
|
" return 0;"
|
|
|
|
"}",
|
|
|
|
Lang_C, "input.c");
|
2018-08-06 22:38:37 +08:00
|
|
|
auto *FromVar =
|
2018-07-12 19:50:21 +08:00
|
|
|
FirstDeclMatcher<VarDecl>().match(FromTU, varDecl(hasName("d")));
|
|
|
|
ASSERT_TRUE(FromVar);
|
|
|
|
auto ToType =
|
|
|
|
ImportType(FromVar->getType().getCanonicalType(), FromVar, Lang_C);
|
|
|
|
EXPECT_FALSE(ToType.isNull());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportRecordDeclInFuncParams) {
|
2018-07-12 19:50:21 +08:00
|
|
|
// This construct is not supported by ASTImporter.
|
2018-08-06 22:38:37 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"int declToImport(struct data_t{int a;int b;} ***d){ return 0; }",
|
|
|
|
Lang_C, "input.c");
|
|
|
|
auto *From = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("declToImport")));
|
2018-07-12 19:50:21 +08:00
|
|
|
ASSERT_TRUE(From);
|
2018-08-06 22:38:37 +08:00
|
|
|
auto *To = Import(From, Lang_C);
|
|
|
|
EXPECT_EQ(To, nullptr);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportRecordDeclInFuncFromMacro) {
|
2018-08-06 22:38:37 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"#define NONAME_SIZEOF(type) sizeof(struct{type *dummy;}) \n"
|
|
|
|
"int declToImport(){ return NONAME_SIZEOF(int); }",
|
|
|
|
Lang_C, "input.c");
|
|
|
|
auto *From = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("declToImport")));
|
|
|
|
ASSERT_TRUE(From);
|
|
|
|
auto *To = Import(From, Lang_C);
|
|
|
|
ASSERT_TRUE(To);
|
|
|
|
EXPECT_TRUE(MatchVerifier<FunctionDecl>().match(
|
|
|
|
To, functionDecl(hasName("declToImport"),
|
|
|
|
hasDescendant(unaryExprOrTypeTraitExpr()))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportRecordDeclInFuncParamsFromMacro) {
|
2018-08-06 22:38:37 +08:00
|
|
|
// This construct is not supported by ASTImporter.
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"#define PAIR_STRUCT(type) struct data_t{type a;type b;} \n"
|
|
|
|
"int declToImport(PAIR_STRUCT(int) ***d){ return 0; }",
|
|
|
|
Lang_C, "input.c");
|
|
|
|
auto *From = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("declToImport")));
|
|
|
|
ASSERT_TRUE(From);
|
|
|
|
auto *To = Import(From, Lang_C);
|
2018-07-12 19:50:21 +08:00
|
|
|
EXPECT_EQ(To, nullptr);
|
|
|
|
}
|
|
|
|
|
2017-11-27 18:30:00 +08:00
|
|
|
const internal::VariadicDynCastAllOfMatcher<Expr, CXXPseudoDestructorExpr>
|
|
|
|
cxxPseudoDestructorExpr;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCXXPseudoDestructorExpr) {
|
2017-11-27 18:30:00 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"typedef int T;"
|
|
|
|
"void declToImport(int *p) {"
|
|
|
|
" T t;"
|
|
|
|
" p->T::~T();"
|
|
|
|
"}",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(
|
|
|
|
callExpr(has(cxxPseudoDestructorExpr())))));
|
2017-11-27 18:30:00 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportUsingDecl) {
|
2017-12-04 00:04:07 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"namespace foo { int bar; }"
|
|
|
|
"void declToImport() { using foo::bar; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionDecl(hasDescendant(usingDecl())));
|
2017-12-04 00:04:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Matches shadow declarations introduced into a scope by a
|
|
|
|
/// (resolved) using declaration.
|
|
|
|
///
|
|
|
|
/// Given
|
|
|
|
/// \code
|
|
|
|
/// namespace n { int f; }
|
|
|
|
/// namespace declToImport { using n::f; }
|
|
|
|
/// \endcode
|
|
|
|
/// usingShadowDecl()
|
|
|
|
/// matches \code f \endcode
|
|
|
|
const internal::VariadicDynCastAllOfMatcher<Decl,
|
|
|
|
UsingShadowDecl> usingShadowDecl;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportUsingShadowDecl) {
|
2017-12-04 00:04:07 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"namespace foo { int bar; }"
|
|
|
|
"namespace declToImport { using foo::bar; }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
namespaceDecl(has(usingShadowDecl())));
|
2017-12-04 00:04:07 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportUnresolvedLookupExpr) {
|
2018-01-10 00:40:40 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template<typename T> int foo();"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" (void)::foo<T>;"
|
|
|
|
" (void)::template foo<T>;"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl(hasDescendant(unresolvedLookupExpr())));
|
2018-01-10 00:40:40 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, ImportCXXUnresolvedConstructExpr) {
|
2018-01-10 00:40:40 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename T> struct C { T t; };"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" C<T> d;"
|
|
|
|
" d.t = T();"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl(hasDescendant(
|
|
|
|
binaryOperator(has(cxxUnresolvedConstructExpr())))));
|
|
|
|
testImport(
|
|
|
|
"template <typename T> struct C { T t; };"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" C<T> d;"
|
|
|
|
" (&d)->t = T();"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl(hasDescendant(
|
|
|
|
binaryOperator(has(cxxUnresolvedConstructExpr())))));
|
2018-01-10 00:40:40 +08:00
|
|
|
}
|
|
|
|
|
2018-01-26 19:36:54 +08:00
|
|
|
/// Check that function "declToImport()" (which is the templated function
|
|
|
|
/// for corresponding FunctionTemplateDecl) is not added into DeclContext.
|
|
|
|
/// Same for class template declarations.
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportTemplatedDeclForTemplate) {
|
2018-01-26 19:36:54 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template <typename T> void declToImport() { T a = 1; }"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
functionTemplateDecl(hasAncestor(translationUnitDecl(
|
|
|
|
unless(has(functionDecl(hasName("declToImport"))))))));
|
|
|
|
testImport(
|
|
|
|
"template <typename T> struct declToImport { T t; };"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
classTemplateDecl(hasAncestor(translationUnitDecl(
|
|
|
|
unless(has(cxxRecordDecl(hasName("declToImport"))))))));
|
2018-01-26 19:36:54 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportClassTemplatePartialSpecialization) {
|
2018-06-25 19:38:43 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
auto Code =
|
|
|
|
R"s(
|
|
|
|
struct declToImport {
|
|
|
|
template <typename T0> struct X;
|
|
|
|
template <typename T0> struct X<T0 *> {};
|
|
|
|
};
|
|
|
|
)s";
|
|
|
|
testImport(Code, Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
recordDecl(has(classTemplateDecl()),
|
|
|
|
has(classTemplateSpecializationDecl())));
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, CXXOperatorCallExpr) {
|
2018-01-28 00:11:45 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"class declToImport {"
|
|
|
|
" void f() { *this = declToImport(); }"
|
|
|
|
"};",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
cxxRecordDecl(has(cxxMethodDecl(hasDescendant(
|
|
|
|
cxxOperatorCallExpr())))));
|
2018-01-28 00:11:45 +08:00
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, DependentSizedArrayType) {
|
2018-01-28 00:11:45 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
2018-06-29 18:25:19 +08:00
|
|
|
testImport(
|
|
|
|
"template<typename T, int Size> class declToImport {"
|
|
|
|
" T data[Size];"
|
|
|
|
"};",
|
|
|
|
Lang_CXX, "", Lang_CXX, Verifier,
|
|
|
|
classTemplateDecl(has(cxxRecordDecl(
|
|
|
|
has(fieldDecl(hasType(dependentSizedArrayType())))))));
|
2018-01-28 00:11:45 +08:00
|
|
|
}
|
|
|
|
|
2019-03-14 22:20:23 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportBeginLocOfDeclRefExpr) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"class A { public: static int X; }; void f() { (void)A::X; }", Lang_CXX);
|
|
|
|
auto From = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
ASSERT_TRUE(From);
|
|
|
|
ASSERT_TRUE(
|
|
|
|
cast<CStyleCastExpr>(cast<CompoundStmt>(From->getBody())->body_front())
|
|
|
|
->getSubExpr()
|
|
|
|
->getBeginLoc()
|
|
|
|
.isValid());
|
|
|
|
FunctionDecl *To = Import(From, Lang_CXX);
|
|
|
|
ASSERT_TRUE(To);
|
|
|
|
ASSERT_TRUE(
|
|
|
|
cast<CStyleCastExpr>(cast<CompoundStmt>(To->getBody())->body_front())
|
|
|
|
->getSubExpr()
|
|
|
|
->getBeginLoc()
|
|
|
|
.isValid());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportOfTemplatedDeclOfClassTemplateDecl) {
|
2018-05-30 17:19:26 +08:00
|
|
|
Decl *FromTU = getTuDecl("template<class X> struct S{};", Lang_CXX);
|
|
|
|
auto From =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(FromTU, classTemplateDecl());
|
|
|
|
ASSERT_TRUE(From);
|
|
|
|
auto To = cast<ClassTemplateDecl>(Import(From, Lang_CXX));
|
|
|
|
ASSERT_TRUE(To);
|
|
|
|
Decl *ToTemplated = To->getTemplatedDecl();
|
|
|
|
Decl *ToTemplated1 = Import(From->getTemplatedDecl(), Lang_CXX);
|
|
|
|
EXPECT_TRUE(ToTemplated1);
|
|
|
|
EXPECT_EQ(ToTemplated1, ToTemplated);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportOfTemplatedDeclOfFunctionTemplateDecl) {
|
2018-06-25 19:38:43 +08:00
|
|
|
Decl *FromTU = getTuDecl("template<class X> void f(){}", Lang_CXX);
|
|
|
|
auto From = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU, functionTemplateDecl());
|
|
|
|
ASSERT_TRUE(From);
|
|
|
|
auto To = cast<FunctionTemplateDecl>(Import(From, Lang_CXX));
|
|
|
|
ASSERT_TRUE(To);
|
|
|
|
Decl *ToTemplated = To->getTemplatedDecl();
|
|
|
|
Decl *ToTemplated1 = Import(From->getTemplatedDecl(), Lang_CXX);
|
|
|
|
EXPECT_TRUE(ToTemplated1);
|
|
|
|
EXPECT_EQ(ToTemplated1, ToTemplated);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-06-25 19:38:43 +08:00
|
|
|
ImportOfTemplatedDeclShouldImportTheClassTemplateDecl) {
|
|
|
|
Decl *FromTU = getTuDecl("template<class X> struct S{};", Lang_CXX);
|
|
|
|
auto FromFT =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(FromTU, classTemplateDecl());
|
|
|
|
ASSERT_TRUE(FromFT);
|
|
|
|
|
|
|
|
auto ToTemplated =
|
|
|
|
cast<CXXRecordDecl>(Import(FromFT->getTemplatedDecl(), Lang_CXX));
|
|
|
|
EXPECT_TRUE(ToTemplated);
|
|
|
|
auto ToTU = ToTemplated->getTranslationUnitDecl();
|
|
|
|
auto ToFT =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(ToTU, classTemplateDecl());
|
|
|
|
EXPECT_TRUE(ToFT);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-07-17 17:52:41 +08:00
|
|
|
ImportOfTemplatedDeclShouldImportTheFunctionTemplateDecl) {
|
2018-06-25 19:38:43 +08:00
|
|
|
Decl *FromTU = getTuDecl("template<class X> void f(){}", Lang_CXX);
|
|
|
|
auto FromFT = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU, functionTemplateDecl());
|
|
|
|
ASSERT_TRUE(FromFT);
|
|
|
|
|
|
|
|
auto ToTemplated =
|
|
|
|
cast<FunctionDecl>(Import(FromFT->getTemplatedDecl(), Lang_CXX));
|
|
|
|
EXPECT_TRUE(ToTemplated);
|
|
|
|
auto ToTU = ToTemplated->getTranslationUnitDecl();
|
|
|
|
auto ToFT = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
ToTU, functionTemplateDecl());
|
|
|
|
EXPECT_TRUE(ToFT);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportCorrectTemplatedDecl) {
|
2018-05-30 17:19:26 +08:00
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
namespace x {
|
|
|
|
template<class X> struct S1{};
|
|
|
|
template<class X> struct S2{};
|
|
|
|
template<class X> struct S3{};
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX);
|
|
|
|
auto FromNs =
|
|
|
|
FirstDeclMatcher<NamespaceDecl>().match(FromTU, namespaceDecl());
|
|
|
|
auto ToNs = cast<NamespaceDecl>(Import(FromNs, Lang_CXX));
|
|
|
|
ASSERT_TRUE(ToNs);
|
|
|
|
auto From =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(FromTU,
|
|
|
|
classTemplateDecl(
|
|
|
|
hasName("S2")));
|
|
|
|
auto To =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(ToNs,
|
|
|
|
classTemplateDecl(
|
|
|
|
hasName("S2")));
|
|
|
|
ASSERT_TRUE(From);
|
|
|
|
ASSERT_TRUE(To);
|
|
|
|
auto ToTemplated = To->getTemplatedDecl();
|
|
|
|
auto ToTemplated1 =
|
|
|
|
cast<CXXRecordDecl>(Import(From->getTemplatedDecl(), Lang_CXX));
|
|
|
|
EXPECT_TRUE(ToTemplated1);
|
|
|
|
ASSERT_EQ(ToTemplated1, ToTemplated);
|
|
|
|
}
|
|
|
|
|
[ASTImporter] Add support for importing ChooseExpr AST nodes.
Summary:
This allows ASTs to be merged when they contain ChooseExpr (the GNU
__builtin_choose_expr construction). This is needed, for example, for
cross-CTU analysis of C code that makes use of __builtin_choose_expr.
The node is already supported in the AST, but it didn't have a matcher
in ASTMatchers. So, this change adds the matcher and adds support to
ASTImporter.
This was originally reviewed and approved in
https://reviews.llvm.org/D58292 and submitted as r354832. It was
reverted in r354839 due to failures on the Windows CI builds.
This version fixes the test failures on Windows, which were caused by
differences in template expansion between versions of clang on different
OSes. The version of clang built with MSVC and running on Windows never
expands the template in the C++ test in ImportExpr.ImportChooseExpr in
clang/unittests/AST/ASTImporter.cpp, but the version on Linux does for
the empty arguments and -fms-compatibility.
So, this version of the patch drops the C++ test for
__builtin_choose_expr, since that version was written to catch
regressions of the logic for isConditionTrue() in the AST import code
for ChooseExpr, and those regressions are also caught by
ASTImporterOptionSpecificTestBase.ImportChooseExpr, which does work on
Windows.
Reviewers: shafik, a_sidorin, martong, aaron.ballman, rnk, a.sidorin
Subscribers: cfe-commits, jdoerfert, rnkovacs, aaron.ballman
Tags: #clang
Differential Revision: https://reviews.llvm.org/D58663
llvm-svn: 354916
2019-02-27 03:26:41 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportChooseExpr) {
|
|
|
|
// This tests the import of isConditionTrue directly to make sure the importer
|
|
|
|
// gets it right.
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
"void declToImport() { (void)__builtin_choose_expr(1, 0, 1); }",
|
|
|
|
Lang_C, "", Lang_C);
|
|
|
|
|
|
|
|
auto ToResults = match(chooseExpr().bind("choose"), To->getASTContext());
|
|
|
|
auto FromResults = match(chooseExpr().bind("choose"), From->getASTContext());
|
|
|
|
|
|
|
|
const ChooseExpr *FromChooseExpr =
|
|
|
|
selectFirst<ChooseExpr>("choose", FromResults);
|
|
|
|
ASSERT_TRUE(FromChooseExpr);
|
|
|
|
|
|
|
|
const ChooseExpr *ToChooseExpr = selectFirst<ChooseExpr>("choose", ToResults);
|
|
|
|
ASSERT_TRUE(ToChooseExpr);
|
|
|
|
|
|
|
|
EXPECT_EQ(FromChooseExpr->isConditionTrue(), ToChooseExpr->isConditionTrue());
|
|
|
|
EXPECT_EQ(FromChooseExpr->isConditionDependent(),
|
|
|
|
ToChooseExpr->isConditionDependent());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportFunctionWithBackReferringParameter) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T> struct X {};
|
|
|
|
|
|
|
|
void declToImport(int y, X<int> &x) {}
|
|
|
|
|
|
|
|
template <> struct X<int> {
|
|
|
|
void g() {
|
|
|
|
X<int> x;
|
|
|
|
declToImport(0, x);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
auto Matcher = functionDecl(hasName("declToImport"),
|
|
|
|
parameterCountIs(2),
|
|
|
|
hasParameter(0, hasName("y")),
|
|
|
|
hasParameter(1, hasName("x")),
|
|
|
|
hasParameter(1, hasType(asString("X<int> &"))));
|
|
|
|
ASSERT_TRUE(Verifier.match(From, Matcher));
|
|
|
|
EXPECT_TRUE(Verifier.match(To, Matcher));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-03-31 06:03:29 +08:00
|
|
|
TUshouldNotContainTemplatedDeclOfFunctionTemplates) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl("template <typename T> void declToImport() { T a = 1; }"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
auto Check = [](Decl *D) -> bool {
|
|
|
|
auto TU = D->getTranslationUnitDecl();
|
|
|
|
for (auto Child : TU->decls()) {
|
|
|
|
if (auto *FD = dyn_cast<FunctionDecl>(Child)) {
|
|
|
|
if (FD->getNameAsString() == "declToImport") {
|
|
|
|
GTEST_NONFATAL_FAILURE_(
|
|
|
|
"TU should not contain any FunctionDecl with name declToImport");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
ASSERT_TRUE(Check(From));
|
|
|
|
EXPECT_TRUE(Check(To));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
TUshouldNotContainTemplatedDeclOfClassTemplates) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl("template <typename T> struct declToImport { T t; };"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
auto Check = [](Decl *D) -> bool {
|
|
|
|
auto TU = D->getTranslationUnitDecl();
|
|
|
|
for (auto Child : TU->decls()) {
|
|
|
|
if (auto *RD = dyn_cast<CXXRecordDecl>(Child)) {
|
|
|
|
if (RD->getNameAsString() == "declToImport") {
|
|
|
|
GTEST_NONFATAL_FAILURE_(
|
|
|
|
"TU should not contain any CXXRecordDecl with name declToImport");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
ASSERT_TRUE(Check(From));
|
|
|
|
EXPECT_TRUE(Check(To));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
TUshouldNotContainTemplatedDeclOfTypeAlias) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl(
|
|
|
|
"template <typename T> struct X {};"
|
|
|
|
"template <typename T> using declToImport = X<T>;"
|
|
|
|
"void instantiate() { declToImport<int> a; }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11);
|
|
|
|
|
|
|
|
auto Check = [](Decl *D) -> bool {
|
|
|
|
auto TU = D->getTranslationUnitDecl();
|
|
|
|
for (auto Child : TU->decls()) {
|
|
|
|
if (auto *AD = dyn_cast<TypeAliasDecl>(Child)) {
|
|
|
|
if (AD->getNameAsString() == "declToImport") {
|
|
|
|
GTEST_NONFATAL_FAILURE_(
|
|
|
|
"TU should not contain any TypeAliasDecl with name declToImport");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
ASSERT_TRUE(Check(From));
|
|
|
|
EXPECT_TRUE(Check(To));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
TUshouldNotContainClassTemplateSpecializationOfImplicitInstantiation) {
|
2018-03-31 06:03:29 +08:00
|
|
|
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
class Base {};
|
|
|
|
class declToImport : public Base<declToImport> {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
// Check that the ClassTemplateSpecializationDecl is NOT the child of the TU.
|
|
|
|
auto Pattern =
|
|
|
|
translationUnitDecl(unless(has(classTemplateSpecializationDecl())));
|
|
|
|
ASSERT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(From->getTranslationUnitDecl(), Pattern));
|
|
|
|
EXPECT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(To->getTranslationUnitDecl(), Pattern));
|
|
|
|
|
|
|
|
// Check that the ClassTemplateSpecializationDecl is the child of the
|
|
|
|
// ClassTemplateDecl.
|
|
|
|
Pattern = translationUnitDecl(has(classTemplateDecl(
|
|
|
|
hasName("Base"), has(classTemplateSpecializationDecl()))));
|
|
|
|
ASSERT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(From->getTranslationUnitDecl(), Pattern));
|
|
|
|
EXPECT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(To->getTranslationUnitDecl(), Pattern));
|
|
|
|
}
|
|
|
|
|
2018-04-24 18:11:53 +08:00
|
|
|
AST_MATCHER_P(RecordDecl, hasFieldOrder, std::vector<StringRef>, Order) {
|
|
|
|
size_t Index = 0;
|
2019-09-02 15:17:01 +08:00
|
|
|
for (Decl *D : Node.decls()) {
|
|
|
|
if (isa<FieldDecl>(D) || isa<IndirectFieldDecl>(D)) {
|
|
|
|
auto *ND = cast<NamedDecl>(D);
|
|
|
|
if (Index == Order.size())
|
|
|
|
return false;
|
|
|
|
if (ND->getName() != Order[Index])
|
|
|
|
return false;
|
|
|
|
++Index;
|
|
|
|
}
|
2018-04-24 18:11:53 +08:00
|
|
|
}
|
|
|
|
return Index == Order.size();
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-03-31 06:03:29 +08:00
|
|
|
TUshouldContainClassTemplateSpecializationOfExplicitInstantiation) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
namespace NS {
|
|
|
|
template<class T>
|
|
|
|
class X {};
|
|
|
|
template class X<int>;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX, "NS");
|
|
|
|
|
|
|
|
// Check that the ClassTemplateSpecializationDecl is NOT the child of the
|
|
|
|
// ClassTemplateDecl.
|
|
|
|
auto Pattern = namespaceDecl(has(classTemplateDecl(
|
|
|
|
hasName("X"), unless(has(classTemplateSpecializationDecl())))));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(From, Pattern));
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(To, Pattern));
|
|
|
|
|
|
|
|
// Check that the ClassTemplateSpecializationDecl is the child of the
|
|
|
|
// NamespaceDecl.
|
|
|
|
Pattern = namespaceDecl(has(classTemplateSpecializationDecl(hasName("X"))));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(From, Pattern));
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(To, Pattern));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
CXXRecordDeclFieldsShouldBeInCorrectOrder) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl(
|
|
|
|
"struct declToImport { int a; int b; };",
|
|
|
|
Lang_CXX11, "", Lang_CXX11);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
ASSERT_TRUE(Verifier.match(From, cxxRecordDecl(hasFieldOrder({"a", "b"}))));
|
|
|
|
EXPECT_TRUE(Verifier.match(To, cxxRecordDecl(hasFieldOrder({"a", "b"}))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2019-07-25 17:07:17 +08:00
|
|
|
CXXRecordDeclFieldOrderShouldNotDependOnImportOrder) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
// The original recursive algorithm of ASTImporter first imports 'c' then
|
|
|
|
// 'b' and lastly 'a'. Therefore we must restore the order somehow.
|
|
|
|
R"s(
|
|
|
|
struct declToImport {
|
|
|
|
int a = c + b;
|
|
|
|
int b = 1;
|
|
|
|
int c = 2;
|
|
|
|
};
|
|
|
|
)s",
|
|
|
|
Lang_CXX11, "", Lang_CXX11);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
ASSERT_TRUE(
|
|
|
|
Verifier.match(From, cxxRecordDecl(hasFieldOrder({"a", "b", "c"}))));
|
|
|
|
EXPECT_TRUE(
|
|
|
|
Verifier.match(To, cxxRecordDecl(hasFieldOrder({"a", "b", "c"}))));
|
|
|
|
}
|
|
|
|
|
2019-09-02 15:17:01 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
CXXRecordDeclFieldAndIndirectFieldOrder) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
// First field is "a", then the field for unnamed union, then "b" and "c"
|
|
|
|
// from it (indirect fields), then "d".
|
|
|
|
R"s(
|
|
|
|
struct declToImport {
|
|
|
|
int a = d;
|
|
|
|
union {
|
|
|
|
int b;
|
|
|
|
int c;
|
|
|
|
};
|
|
|
|
int d;
|
|
|
|
};
|
|
|
|
)s",
|
|
|
|
Lang_CXX11, "", Lang_CXX11);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
ASSERT_TRUE(Verifier.match(
|
|
|
|
From, cxxRecordDecl(hasFieldOrder({"a", "", "b", "c", "d"}))));
|
|
|
|
EXPECT_TRUE(Verifier.match(
|
|
|
|
To, cxxRecordDecl(hasFieldOrder({"a", "", "b", "c", "d"}))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ShouldImportImplicitCXXRecordDecl) {
|
2018-05-23 22:24:02 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
struct declToImport {
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
// Match the implicit Decl.
|
|
|
|
auto Matcher = cxxRecordDecl(has(cxxRecordDecl()));
|
|
|
|
ASSERT_TRUE(Verifier.match(From, Matcher));
|
|
|
|
EXPECT_TRUE(Verifier.match(To, Matcher));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ShouldImportImplicitCXXRecordDeclOfClassTemplate) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
template <typename U>
|
|
|
|
struct declToImport {
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
// Match the implicit Decl.
|
|
|
|
auto Matcher = classTemplateDecl(has(cxxRecordDecl(has(cxxRecordDecl()))));
|
|
|
|
ASSERT_TRUE(Verifier.match(From, Matcher));
|
|
|
|
EXPECT_TRUE(Verifier.match(To, Matcher));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ShouldImportImplicitCXXRecordDeclOfClassTemplateSpecializationDecl) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
class Base {};
|
|
|
|
class declToImport : public Base<declToImport> {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
auto hasImplicitClass = has(cxxRecordDecl());
|
|
|
|
auto Pattern = translationUnitDecl(has(classTemplateDecl(
|
|
|
|
hasName("Base"),
|
|
|
|
has(classTemplateSpecializationDecl(hasImplicitClass)))));
|
|
|
|
ASSERT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(From->getTranslationUnitDecl(), Pattern));
|
|
|
|
EXPECT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(To->getTranslationUnitDecl(), Pattern));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, IDNSOrdinary) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl("void declToImport() {}", Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
auto Matcher = functionDecl();
|
|
|
|
ASSERT_TRUE(Verifier.match(From, Matcher));
|
|
|
|
EXPECT_TRUE(Verifier.match(To, Matcher));
|
|
|
|
EXPECT_EQ(From->getIdentifierNamespace(), To->getIdentifierNamespace());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, IDNSOfNonmemberOperator) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X {};
|
|
|
|
void operator<<(int, X);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
Decl *From = LastDeclMatcher<Decl>{}.match(FromTU, functionDecl());
|
|
|
|
const Decl *To = Import(From, Lang_CXX);
|
|
|
|
EXPECT_EQ(From->getIdentifierNamespace(), To->getIdentifierNamespace());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-03-31 06:03:29 +08:00
|
|
|
ShouldImportMembersOfClassTemplateSpecializationDecl) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) = getImportedDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
class Base { int a; };
|
|
|
|
class declToImport : Base<declToImport> {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "", Lang_CXX);
|
|
|
|
|
|
|
|
auto Pattern = translationUnitDecl(has(classTemplateDecl(
|
|
|
|
hasName("Base"),
|
|
|
|
has(classTemplateSpecializationDecl(has(fieldDecl(hasName("a"))))))));
|
|
|
|
ASSERT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(From->getTranslationUnitDecl(), Pattern));
|
|
|
|
EXPECT_TRUE(
|
|
|
|
MatchVerifier<Decl>{}.match(To->getTranslationUnitDecl(), Pattern));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportDefinitionOfClassTemplateAfterFwdDecl) {
|
2018-05-15 19:09:07 +08:00
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B;
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("B")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B {
|
|
|
|
void f();
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
FunctionDecl *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
auto *FromCTD = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("B")));
|
|
|
|
auto *ToCTD = cast<ClassTemplateDecl>(Import(FromCTD, Lang_CXX));
|
|
|
|
EXPECT_TRUE(ToCTD->isThisDeclarationADefinition());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-05-23 21:53:36 +08:00
|
|
|
ImportDefinitionOfClassTemplateIfThereIsAnExistingFwdDeclAndDefinition) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B {
|
|
|
|
void f();
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
struct B;
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
ASSERT_EQ(1u, DeclCounterWithPredicate<ClassTemplateDecl>(
|
|
|
|
[](const ClassTemplateDecl *T) {
|
|
|
|
return T->isThisDeclarationADefinition();
|
|
|
|
})
|
|
|
|
.match(ToTU, classTemplateDecl()));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B {
|
|
|
|
void f();
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
ClassTemplateDecl *FromD = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("B")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
// We should have only one definition.
|
|
|
|
EXPECT_EQ(1u, DeclCounterWithPredicate<ClassTemplateDecl>(
|
|
|
|
[](const ClassTemplateDecl *T) {
|
|
|
|
return T->isThisDeclarationADefinition();
|
|
|
|
})
|
|
|
|
.match(ToTU, classTemplateDecl()));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-05-23 21:53:36 +08:00
|
|
|
ImportDefinitionOfClassIfThereIsAnExistingFwdDeclAndDefinition) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
struct B {
|
|
|
|
void f();
|
|
|
|
};
|
|
|
|
|
|
|
|
struct B;
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
ASSERT_EQ(2u, DeclCounter<CXXRecordDecl>().match(
|
2018-05-24 18:49:38 +08:00
|
|
|
ToTU, cxxRecordDecl(unless(isImplicit()))));
|
2018-05-23 21:53:36 +08:00
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct B {
|
|
|
|
void f();
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("B")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
EXPECT_EQ(2u, DeclCounter<CXXRecordDecl>().match(
|
2018-05-24 18:49:38 +08:00
|
|
|
ToTU, cxxRecordDecl(unless(isImplicit()))));
|
2018-05-23 21:53:36 +08:00
|
|
|
}
|
|
|
|
|
2018-07-04 21:34:05 +08:00
|
|
|
static void CompareSourceLocs(FullSourceLoc Loc1, FullSourceLoc Loc2) {
|
|
|
|
EXPECT_EQ(Loc1.getExpansionLineNumber(), Loc2.getExpansionLineNumber());
|
|
|
|
EXPECT_EQ(Loc1.getExpansionColumnNumber(), Loc2.getExpansionColumnNumber());
|
|
|
|
EXPECT_EQ(Loc1.getSpellingLineNumber(), Loc2.getSpellingLineNumber());
|
|
|
|
EXPECT_EQ(Loc1.getSpellingColumnNumber(), Loc2.getSpellingColumnNumber());
|
|
|
|
}
|
|
|
|
static void CompareSourceRanges(SourceRange Range1, SourceRange Range2,
|
|
|
|
SourceManager &SM1, SourceManager &SM2) {
|
|
|
|
CompareSourceLocs(FullSourceLoc{ Range1.getBegin(), SM1 },
|
|
|
|
FullSourceLoc{ Range2.getBegin(), SM2 });
|
|
|
|
CompareSourceLocs(FullSourceLoc{ Range1.getEnd(), SM1 },
|
|
|
|
FullSourceLoc{ Range2.getEnd(), SM2 });
|
|
|
|
}
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportSourceLocs) {
|
2018-07-04 21:34:05 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
#define MFOO(arg) arg = arg + 1
|
|
|
|
|
|
|
|
void foo() {
|
|
|
|
int a = 5;
|
|
|
|
MFOO(a);
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, functionDecl());
|
|
|
|
auto ToD = Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
auto ToLHS = LastDeclMatcher<DeclRefExpr>().match(ToD, declRefExpr());
|
|
|
|
auto FromLHS = LastDeclMatcher<DeclRefExpr>().match(FromTU, declRefExpr());
|
|
|
|
auto ToRHS = LastDeclMatcher<IntegerLiteral>().match(ToD, integerLiteral());
|
|
|
|
auto FromRHS =
|
|
|
|
LastDeclMatcher<IntegerLiteral>().match(FromTU, integerLiteral());
|
|
|
|
|
|
|
|
SourceManager &ToSM = ToAST->getASTContext().getSourceManager();
|
|
|
|
SourceManager &FromSM = FromD->getASTContext().getSourceManager();
|
|
|
|
CompareSourceRanges(ToD->getSourceRange(), FromD->getSourceRange(), ToSM,
|
|
|
|
FromSM);
|
|
|
|
CompareSourceRanges(ToLHS->getSourceRange(), FromLHS->getSourceRange(), ToSM,
|
|
|
|
FromSM);
|
|
|
|
CompareSourceRanges(ToRHS->getSourceRange(), FromRHS->getSourceRange(), ToSM,
|
|
|
|
FromSM);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportNestedMacro) {
|
2018-07-04 21:34:05 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
#define FUNC_INT void declToImport
|
|
|
|
#define FUNC FUNC_INT
|
|
|
|
FUNC(int a);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, functionDecl());
|
|
|
|
auto ToD = Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
SourceManager &ToSM = ToAST->getASTContext().getSourceManager();
|
|
|
|
SourceManager &FromSM = FromD->getASTContext().getSourceManager();
|
|
|
|
CompareSourceRanges(ToD->getSourceRange(), FromD->getSourceRange(), ToSM,
|
|
|
|
FromSM);
|
|
|
|
}
|
|
|
|
|
2018-05-23 21:53:36 +08:00
|
|
|
TEST_P(
|
2019-02-08 00:52:48 +08:00
|
|
|
ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportDefinitionOfClassTemplateSpecIfThereIsAnExistingFwdDeclAndDefinition) {
|
2018-05-23 21:53:36 +08:00
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B;
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct B<int> {};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct B<int>;
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
// We should have only one definition.
|
|
|
|
ASSERT_EQ(1u, DeclCounterWithPredicate<ClassTemplateSpecializationDecl>(
|
|
|
|
[](const ClassTemplateSpecializationDecl *T) {
|
|
|
|
return T->isThisDeclarationADefinition();
|
|
|
|
})
|
|
|
|
.match(ToTU, classTemplateSpecializationDecl()));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B;
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct B<int> {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl(hasName("B")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
// We should have only one definition.
|
|
|
|
EXPECT_EQ(1u, DeclCounterWithPredicate<ClassTemplateSpecializationDecl>(
|
|
|
|
[](const ClassTemplateSpecializationDecl *T) {
|
|
|
|
return T->isThisDeclarationADefinition();
|
|
|
|
})
|
|
|
|
.match(ToTU, classTemplateSpecializationDecl()));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ObjectsWithUnnamedStructType) {
|
2018-07-05 17:51:13 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct { int a; int b; } object0 = { 2, 3 };
|
|
|
|
struct { int x; int y; int z; } object1;
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
auto *Obj0 =
|
|
|
|
FirstDeclMatcher<VarDecl>().match(FromTU, varDecl(hasName("object0")));
|
|
|
|
auto *From0 = getRecordDecl(Obj0);
|
|
|
|
auto *Obj1 =
|
|
|
|
FirstDeclMatcher<VarDecl>().match(FromTU, varDecl(hasName("object1")));
|
|
|
|
auto *From1 = getRecordDecl(Obj1);
|
|
|
|
|
|
|
|
auto *To0 = Import(From0, Lang_CXX);
|
|
|
|
auto *To1 = Import(From1, Lang_CXX);
|
|
|
|
|
|
|
|
EXPECT_TRUE(To0);
|
|
|
|
EXPECT_TRUE(To1);
|
|
|
|
EXPECT_NE(To0, To1);
|
|
|
|
EXPECT_NE(To0->getCanonicalDecl(), To1->getCanonicalDecl());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, AnonymousRecords) {
|
2018-12-17 20:42:12 +08:00
|
|
|
auto *Code =
|
|
|
|
R"(
|
|
|
|
struct X {
|
|
|
|
struct { int a; };
|
|
|
|
struct { int b; };
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *FromTU0 = getTuDecl(Code, Lang_C, "input0.c");
|
|
|
|
|
|
|
|
Decl *FromTU1 = getTuDecl(Code, Lang_C, "input1.c");
|
|
|
|
|
|
|
|
auto *X0 =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(FromTU0, recordDecl(hasName("X")));
|
|
|
|
auto *X1 =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(FromTU1, recordDecl(hasName("X")));
|
|
|
|
Import(X0, Lang_C);
|
|
|
|
Import(X1, Lang_C);
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
// We expect no (ODR) warning during the import.
|
|
|
|
EXPECT_EQ(0u, ToTU->getASTContext().getDiagnostics().getNumWarnings());
|
|
|
|
EXPECT_EQ(1u,
|
|
|
|
DeclCounter<RecordDecl>().match(ToTU, recordDecl(hasName("X"))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, AnonymousRecordsReversed) {
|
2018-12-17 20:42:12 +08:00
|
|
|
Decl *FromTU0 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X {
|
|
|
|
struct { int a; };
|
|
|
|
struct { int b; };
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_C, "input0.c");
|
|
|
|
|
|
|
|
Decl *FromTU1 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X { // reversed order
|
|
|
|
struct { int b; };
|
|
|
|
struct { int a; };
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_C, "input1.c");
|
|
|
|
|
|
|
|
auto *X0 =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(FromTU0, recordDecl(hasName("X")));
|
|
|
|
auto *X1 =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(FromTU1, recordDecl(hasName("X")));
|
|
|
|
Import(X0, Lang_C);
|
|
|
|
Import(X1, Lang_C);
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
// We expect one (ODR) warning during the import.
|
|
|
|
EXPECT_EQ(1u, ToTU->getASTContext().getDiagnostics().getNumWarnings());
|
2019-08-27 19:36:10 +08:00
|
|
|
EXPECT_EQ(1u,
|
2018-12-17 20:42:12 +08:00
|
|
|
DeclCounter<RecordDecl>().match(ToTU, recordDecl(hasName("X"))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportDoesUpdateUsedFlag) {
|
[ASTImporter] Refactor Decl creation
Summary:
Generalize the creation of Decl nodes during Import. With this patch we do the
same things after and before a new AST node is created (::Create) The import
logic should be really simple, we create the node, then we mark that as
imported, then we recursively import the parts for that node and then set them
on that node. However, the AST is actually a graph, so we have to handle
circles. If we mark something as imported (`MapImported()`) then we return with
the corresponding `To` decl whenever we want to import that node again, this way
circles are handled. In order to make this algorithm work we must ensure
things, which are handled in the generic CreateDecl<> template:
* There are no `Import()` calls in between any node creation (::Create)
and the `MapImported()` call.
* Before actually creating an AST node (::Create), we must check if
the Node had been imported already, if yes then return with that one.
One very important case for this is connected to templates: we may
start an import both from the templated decl of a template and from
the template itself.
Now, the virtual `Imported` function is called in `ASTImporter::Impor(Decl *)`,
but only once, when the `Decl` is imported. One point of this refactor is to
separate responsibilities. The original `Imported()` had 3 responsibilities:
- notify subclasses when an import happened
- register the decl into `ImportedDecls`
- initialise the Decl (set attributes, etc)
Now all of these are in separate functions:
- `Imported`
- `MapImported`
- `InitializeImportedDecl`
I tried to check all the clients, I executed tests for `ExternalASTMerger.cpp`
and some unittests for lldb.
Reviewers: a.sidorin, balazske, xazax.hun, r.stahl
Subscribers: rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47632
llvm-svn: 336896
2018-07-12 17:42:05 +08:00
|
|
|
auto Pattern = varDecl(hasName("x"));
|
|
|
|
VarDecl *Imported1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl("extern int x;", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<VarDecl>().match(FromTU, Pattern);
|
|
|
|
Imported1 = cast<VarDecl>(Import(FromD, Lang_CXX));
|
|
|
|
}
|
|
|
|
VarDecl *Imported2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl("int x;", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<VarDecl>().match(FromTU, Pattern);
|
|
|
|
Imported2 = cast<VarDecl>(Import(FromD, Lang_CXX));
|
|
|
|
}
|
|
|
|
EXPECT_EQ(Imported1->getCanonicalDecl(), Imported2->getCanonicalDecl());
|
|
|
|
EXPECT_FALSE(Imported2->isUsed(false));
|
|
|
|
{
|
|
|
|
Decl *FromTU =
|
|
|
|
getTuDecl("extern int x; int f() { return x; }", Lang_CXX, "input2.cc");
|
2018-09-17 20:04:52 +08:00
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
[ASTImporter] Refactor Decl creation
Summary:
Generalize the creation of Decl nodes during Import. With this patch we do the
same things after and before a new AST node is created (::Create) The import
logic should be really simple, we create the node, then we mark that as
imported, then we recursively import the parts for that node and then set them
on that node. However, the AST is actually a graph, so we have to handle
circles. If we mark something as imported (`MapImported()`) then we return with
the corresponding `To` decl whenever we want to import that node again, this way
circles are handled. In order to make this algorithm work we must ensure
things, which are handled in the generic CreateDecl<> template:
* There are no `Import()` calls in between any node creation (::Create)
and the `MapImported()` call.
* Before actually creating an AST node (::Create), we must check if
the Node had been imported already, if yes then return with that one.
One very important case for this is connected to templates: we may
start an import both from the templated decl of a template and from
the template itself.
Now, the virtual `Imported` function is called in `ASTImporter::Impor(Decl *)`,
but only once, when the `Decl` is imported. One point of this refactor is to
separate responsibilities. The original `Imported()` had 3 responsibilities:
- notify subclasses when an import happened
- register the decl into `ImportedDecls`
- initialise the Decl (set attributes, etc)
Now all of these are in separate functions:
- `Imported`
- `MapImported`
- `InitializeImportedDecl`
I tried to check all the clients, I executed tests for `ExternalASTMerger.cpp`
and some unittests for lldb.
Reviewers: a.sidorin, balazske, xazax.hun, r.stahl
Subscribers: rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47632
llvm-svn: 336896
2018-07-12 17:42:05 +08:00
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_TRUE(Imported2->isUsed(false));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportDoesUpdateUsedFlag2) {
|
2018-09-17 20:04:52 +08:00
|
|
|
auto Pattern = varDecl(hasName("x"));
|
|
|
|
VarDecl *ExistingD;
|
|
|
|
{
|
|
|
|
Decl *ToTU = getToTuDecl("int x = 1;", Lang_CXX);
|
|
|
|
ExistingD = FirstDeclMatcher<VarDecl>().match(ToTU, Pattern);
|
|
|
|
}
|
|
|
|
EXPECT_FALSE(ExistingD->isUsed(false));
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"int x = 1; int f() { return x; }", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_TRUE(ExistingD->isUsed(false));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportDoesUpdateUsedFlag3) {
|
2018-09-17 20:04:52 +08:00
|
|
|
auto Pattern = varDecl(hasName("a"));
|
|
|
|
VarDecl *ExistingD;
|
|
|
|
{
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
static const int a = 1;
|
|
|
|
};
|
|
|
|
)", Lang_CXX);
|
|
|
|
ExistingD = FirstDeclMatcher<VarDecl>().match(ToTU, Pattern);
|
|
|
|
}
|
|
|
|
EXPECT_FALSE(ExistingD->isUsed(false));
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
static const int a = 1;
|
|
|
|
};
|
|
|
|
const int *f() { return &A::a; } // requires storage,
|
|
|
|
// thus used flag will be set
|
|
|
|
)", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromFunD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
auto *FromD = FirstDeclMatcher<VarDecl>().match(FromTU, Pattern);
|
|
|
|
ASSERT_TRUE(FromD->isUsed(false));
|
|
|
|
Import(FromFunD, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_TRUE(ExistingD->isUsed(false));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ReimportWithUsedFlag) {
|
[ASTImporter] Refactor Decl creation
Summary:
Generalize the creation of Decl nodes during Import. With this patch we do the
same things after and before a new AST node is created (::Create) The import
logic should be really simple, we create the node, then we mark that as
imported, then we recursively import the parts for that node and then set them
on that node. However, the AST is actually a graph, so we have to handle
circles. If we mark something as imported (`MapImported()`) then we return with
the corresponding `To` decl whenever we want to import that node again, this way
circles are handled. In order to make this algorithm work we must ensure
things, which are handled in the generic CreateDecl<> template:
* There are no `Import()` calls in between any node creation (::Create)
and the `MapImported()` call.
* Before actually creating an AST node (::Create), we must check if
the Node had been imported already, if yes then return with that one.
One very important case for this is connected to templates: we may
start an import both from the templated decl of a template and from
the template itself.
Now, the virtual `Imported` function is called in `ASTImporter::Impor(Decl *)`,
but only once, when the `Decl` is imported. One point of this refactor is to
separate responsibilities. The original `Imported()` had 3 responsibilities:
- notify subclasses when an import happened
- register the decl into `ImportedDecls`
- initialise the Decl (set attributes, etc)
Now all of these are in separate functions:
- `Imported`
- `MapImported`
- `InitializeImportedDecl`
I tried to check all the clients, I executed tests for `ExternalASTMerger.cpp`
and some unittests for lldb.
Reviewers: a.sidorin, balazske, xazax.hun, r.stahl
Subscribers: rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47632
llvm-svn: 336896
2018-07-12 17:42:05 +08:00
|
|
|
auto Pattern = varDecl(hasName("x"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl("int x;", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<VarDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *Imported1 = cast<VarDecl>(Import(FromD, Lang_CXX));
|
|
|
|
|
|
|
|
ASSERT_FALSE(Imported1->isUsed(false));
|
|
|
|
|
|
|
|
FromD->setIsUsed();
|
|
|
|
auto *Imported2 = cast<VarDecl>(Import(FromD, Lang_CXX));
|
|
|
|
|
|
|
|
EXPECT_EQ(Imported1, Imported2);
|
|
|
|
EXPECT_TRUE(Imported2->isUsed(false));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct ImportFunctions : ASTImporterOptionSpecificTestBase {};
|
2018-03-31 06:03:29 +08:00
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
TEST_P(ImportFunctions, ImportPrototypeOfRecursiveFunction) {
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *FromTU = getTuDecl("void f(); void f() { f(); }", Lang_CXX);
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
auto *From =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern); // Proto
|
2018-03-31 06:03:29 +08:00
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
Decl *ImportedD = Import(From, Lang_CXX);
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *ToTU = ImportedD->getTranslationUnitDecl();
|
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
auto *To0 = FirstDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
auto *To1 = LastDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_TRUE(ImportedD == To0);
|
|
|
|
EXPECT_FALSE(To0->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_TRUE(To1->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(To1->getPreviousDecl(), To0);
|
2018-03-31 06:03:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, ImportDefinitionOfRecursiveFunction) {
|
|
|
|
Decl *FromTU = getTuDecl("void f(); void f() { f(); }", Lang_CXX);
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
auto *From =
|
|
|
|
LastDeclMatcher<FunctionDecl>().match(FromTU, Pattern); // Def
|
2018-03-31 06:03:29 +08:00
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
Decl *ImportedD = Import(From, Lang_CXX);
|
2018-03-31 06:03:29 +08:00
|
|
|
Decl *ToTU = ImportedD->getTranslationUnitDecl();
|
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
auto *To0 = FirstDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
auto *To1 = LastDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_TRUE(ImportedD == To1);
|
|
|
|
EXPECT_FALSE(To0->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_TRUE(To1->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(To1->getPreviousDecl(), To0);
|
2018-03-31 06:03:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, OverriddenMethodsShouldBeImported) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
void B::f() {}
|
|
|
|
struct D : B { void f(); };
|
|
|
|
)";
|
|
|
|
auto Pattern =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("D"))));
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX);
|
|
|
|
CXXMethodDecl *Proto =
|
|
|
|
FirstDeclMatcher<CXXMethodDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
ASSERT_EQ(Proto->size_overridden_methods(), 1u);
|
|
|
|
CXXMethodDecl *To = cast<CXXMethodDecl>(Import(Proto, Lang_CXX));
|
|
|
|
EXPECT_EQ(To->size_overridden_methods(), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, VirtualFlagShouldBePreservedWhenImportingPrototype) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
void B::f() {}
|
|
|
|
)";
|
|
|
|
auto Pattern =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("B"))));
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX);
|
|
|
|
CXXMethodDecl *Proto =
|
|
|
|
FirstDeclMatcher<CXXMethodDecl>().match(FromTU, Pattern);
|
|
|
|
CXXMethodDecl *Def = LastDeclMatcher<CXXMethodDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
ASSERT_TRUE(Proto->isVirtual());
|
|
|
|
ASSERT_TRUE(Def->isVirtual());
|
|
|
|
CXXMethodDecl *To = cast<CXXMethodDecl>(Import(Proto, Lang_CXX));
|
|
|
|
EXPECT_TRUE(To->isVirtual());
|
|
|
|
}
|
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
TEST_P(ImportFunctions,
|
|
|
|
ImportDefinitionIfThereIsAnExistingDefinitionAndFwdDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
void f() {}
|
|
|
|
void f();
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
ASSERT_EQ(1u,
|
|
|
|
DeclCounterWithPredicate<FunctionDecl>([](const FunctionDecl *FD) {
|
|
|
|
return FD->doesThisDeclarationHaveABody();
|
|
|
|
}).match(ToTU, functionDecl()));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl("void f() {}", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, functionDecl());
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
EXPECT_EQ(1u,
|
|
|
|
DeclCounterWithPredicate<FunctionDecl>([](const FunctionDecl *FD) {
|
|
|
|
return FD->doesThisDeclarationHaveABody();
|
|
|
|
}).match(ToTU, functionDecl()));
|
|
|
|
}
|
|
|
|
|
2019-01-29 05:55:33 +08:00
|
|
|
TEST_P(ImportFunctions, ImportOverriddenMethodTwice) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
struct D:B { void f(); };
|
|
|
|
)";
|
|
|
|
auto BFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("B"))));
|
|
|
|
auto DFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("D"))));
|
|
|
|
|
|
|
|
Decl *FromTU0 = getTuDecl(Code, Lang_CXX);
|
|
|
|
auto *DF = FirstDeclMatcher<CXXMethodDecl>().match(FromTU0, DFP);
|
|
|
|
Import(DF, Lang_CXX);
|
|
|
|
|
|
|
|
Decl *FromTU1 = getTuDecl(Code, Lang_CXX, "input1.cc");
|
|
|
|
auto *BF = FirstDeclMatcher<CXXMethodDecl>().match(FromTU1, BFP);
|
|
|
|
Import(BF, Lang_CXX);
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, DFP), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, ImportOverriddenMethodTwiceDefinitionFirst) {
|
|
|
|
auto CodeWithoutDef =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
struct D:B { void f(); };
|
|
|
|
)";
|
|
|
|
auto CodeWithDef =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(){}; };
|
|
|
|
struct D:B { void f(){}; };
|
|
|
|
)";
|
|
|
|
auto BFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("B"))));
|
|
|
|
auto DFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("D"))));
|
|
|
|
auto BFDefP = cxxMethodDecl(
|
|
|
|
hasName("f"), hasParent(cxxRecordDecl(hasName("B"))), isDefinition());
|
|
|
|
auto DFDefP = cxxMethodDecl(
|
|
|
|
hasName("f"), hasParent(cxxRecordDecl(hasName("D"))), isDefinition());
|
|
|
|
auto FDefAllP = cxxMethodDecl(hasName("f"), isDefinition());
|
|
|
|
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(CodeWithDef, Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<CXXMethodDecl>().match(FromTU, DFP);
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(CodeWithoutDef, Lang_CXX, "input1.cc");
|
|
|
|
auto *FromB = FirstDeclMatcher<CXXMethodDecl>().match(FromTU, BFP);
|
|
|
|
Import(FromB, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, DFP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFDefP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, DFDefP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, FDefAllP), 2u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, ImportOverriddenMethodTwiceOutOfClassDef) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
struct D:B { void f(); };
|
|
|
|
void B::f(){};
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto BFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("B"))));
|
|
|
|
auto BFDefP = cxxMethodDecl(
|
|
|
|
hasName("f"), hasParent(cxxRecordDecl(hasName("B"))), isDefinition());
|
|
|
|
auto DFP = cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("D"))),
|
|
|
|
unless(isDefinition()));
|
|
|
|
|
|
|
|
Decl *FromTU0 = getTuDecl(Code, Lang_CXX);
|
|
|
|
auto *D = FirstDeclMatcher<CXXMethodDecl>().match(FromTU0, DFP);
|
|
|
|
Import(D, Lang_CXX);
|
|
|
|
|
|
|
|
Decl *FromTU1 = getTuDecl(Code, Lang_CXX, "input1.cc");
|
|
|
|
auto *B = FirstDeclMatcher<CXXMethodDecl>().match(FromTU1, BFP);
|
|
|
|
Import(B, Lang_CXX);
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFDefP), 0u);
|
|
|
|
|
|
|
|
auto *ToB = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("B")));
|
|
|
|
auto *ToBFInClass = FirstDeclMatcher<CXXMethodDecl>().match(ToTU, BFP);
|
|
|
|
auto *ToBFOutOfClass = FirstDeclMatcher<CXXMethodDecl>().match(
|
|
|
|
ToTU, cxxMethodDecl(hasName("f"), isDefinition()));
|
|
|
|
|
|
|
|
// The definition should be out-of-class.
|
|
|
|
EXPECT_NE(ToBFInClass, ToBFOutOfClass);
|
|
|
|
EXPECT_NE(ToBFInClass->getLexicalDeclContext(),
|
|
|
|
ToBFOutOfClass->getLexicalDeclContext());
|
|
|
|
EXPECT_EQ(ToBFOutOfClass->getDeclContext(), ToB);
|
|
|
|
EXPECT_EQ(ToBFOutOfClass->getLexicalDeclContext(), ToTU);
|
|
|
|
|
|
|
|
// Check that the redecl chain is intact.
|
|
|
|
EXPECT_EQ(ToBFOutOfClass->getPreviousDecl(), ToBFInClass);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions,
|
|
|
|
ImportOverriddenMethodTwiceOutOfClassDefInSeparateCode) {
|
|
|
|
auto CodeTU0 =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
struct D:B { void f(); };
|
|
|
|
)";
|
|
|
|
auto CodeTU1 =
|
|
|
|
R"(
|
|
|
|
struct B { virtual void f(); };
|
|
|
|
struct D:B { void f(); };
|
|
|
|
void B::f(){}
|
|
|
|
void D::f(){}
|
|
|
|
void foo(B &b, D &d) { b.f(); d.f(); }
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto BFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("B"))));
|
|
|
|
auto BFDefP = cxxMethodDecl(
|
|
|
|
hasName("f"), hasParent(cxxRecordDecl(hasName("B"))), isDefinition());
|
|
|
|
auto DFP =
|
|
|
|
cxxMethodDecl(hasName("f"), hasParent(cxxRecordDecl(hasName("D"))));
|
|
|
|
auto DFDefP = cxxMethodDecl(
|
|
|
|
hasName("f"), hasParent(cxxRecordDecl(hasName("D"))), isDefinition());
|
|
|
|
auto FooDef = functionDecl(hasName("foo"));
|
|
|
|
|
|
|
|
{
|
|
|
|
Decl *FromTU0 = getTuDecl(CodeTU0, Lang_CXX, "input0.cc");
|
|
|
|
auto *D = FirstDeclMatcher<CXXMethodDecl>().match(FromTU0, DFP);
|
|
|
|
Import(D, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Decl *FromTU1 = getTuDecl(CodeTU1, Lang_CXX, "input1.cc");
|
|
|
|
auto *Foo = FirstDeclMatcher<FunctionDecl>().match(FromTU1, FooDef);
|
|
|
|
Import(Foo, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, DFP), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, BFDefP), 0u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, DFDefP), 0u);
|
|
|
|
|
|
|
|
auto *ToB = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("B")));
|
|
|
|
auto *ToD = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("D")));
|
|
|
|
auto *ToBFInClass = FirstDeclMatcher<CXXMethodDecl>().match(ToTU, BFP);
|
|
|
|
auto *ToBFOutOfClass = FirstDeclMatcher<CXXMethodDecl>().match(
|
|
|
|
ToTU, cxxMethodDecl(hasName("f"), isDefinition()));
|
|
|
|
auto *ToDFInClass = FirstDeclMatcher<CXXMethodDecl>().match(ToTU, DFP);
|
|
|
|
auto *ToDFOutOfClass = LastDeclMatcher<CXXMethodDecl>().match(
|
|
|
|
ToTU, cxxMethodDecl(hasName("f"), isDefinition()));
|
|
|
|
|
|
|
|
// The definition should be out-of-class.
|
|
|
|
EXPECT_NE(ToBFInClass, ToBFOutOfClass);
|
|
|
|
EXPECT_NE(ToBFInClass->getLexicalDeclContext(),
|
|
|
|
ToBFOutOfClass->getLexicalDeclContext());
|
|
|
|
EXPECT_EQ(ToBFOutOfClass->getDeclContext(), ToB);
|
|
|
|
EXPECT_EQ(ToBFOutOfClass->getLexicalDeclContext(), ToTU);
|
|
|
|
|
|
|
|
EXPECT_NE(ToDFInClass, ToDFOutOfClass);
|
|
|
|
EXPECT_NE(ToDFInClass->getLexicalDeclContext(),
|
|
|
|
ToDFOutOfClass->getLexicalDeclContext());
|
|
|
|
EXPECT_EQ(ToDFOutOfClass->getDeclContext(), ToD);
|
|
|
|
EXPECT_EQ(ToDFOutOfClass->getLexicalDeclContext(), ToTU);
|
|
|
|
|
|
|
|
// Check that the redecl chain is intact.
|
|
|
|
EXPECT_EQ(ToBFOutOfClass->getPreviousDecl(), ToBFInClass);
|
|
|
|
EXPECT_EQ(ToDFOutOfClass->getPreviousDecl(), ToDFInClass);
|
|
|
|
}
|
|
|
|
|
2019-02-14 21:07:03 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportVariableChainInC) {
|
|
|
|
std::string Code = "static int v; static int v = 0;";
|
|
|
|
auto Pattern = varDecl(hasName("v"));
|
|
|
|
|
|
|
|
TranslationUnitDecl *FromTu = getTuDecl(Code, Lang_C, "input0.c");
|
|
|
|
|
|
|
|
auto *From0 = FirstDeclMatcher<VarDecl>().match(FromTu, Pattern);
|
|
|
|
auto *From1 = LastDeclMatcher<VarDecl>().match(FromTu, Pattern);
|
|
|
|
|
|
|
|
auto *To0 = Import(From0, Lang_C);
|
|
|
|
auto *To1 = Import(From1, Lang_C);
|
|
|
|
|
|
|
|
EXPECT_TRUE(To0);
|
|
|
|
ASSERT_TRUE(To1);
|
|
|
|
EXPECT_NE(To0, To1);
|
|
|
|
EXPECT_EQ(To1->getPreviousDecl(), To0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, ImportFromDifferentScopedAnonNamespace) {
|
|
|
|
TranslationUnitDecl *FromTu = getTuDecl(
|
|
|
|
"namespace NS0 { namespace { void f(); } }"
|
|
|
|
"namespace NS1 { namespace { void f(); } }",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
auto *FromF0 = FirstDeclMatcher<FunctionDecl>().match(FromTu, Pattern);
|
|
|
|
auto *FromF1 = LastDeclMatcher<FunctionDecl>().match(FromTu, Pattern);
|
|
|
|
|
|
|
|
auto *ToF0 = Import(FromF0, Lang_CXX);
|
|
|
|
auto *ToF1 = Import(FromF1, Lang_CXX);
|
|
|
|
|
|
|
|
EXPECT_TRUE(ToF0);
|
|
|
|
ASSERT_TRUE(ToF1);
|
|
|
|
EXPECT_NE(ToF0, ToF1);
|
|
|
|
EXPECT_FALSE(ToF1->getPreviousDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions, ImportFunctionFromUnnamedNamespace) {
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl("namespace { void f() {} } void g0() { f(); }",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("g0")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
Decl *FromTU =
|
|
|
|
getTuDecl("namespace { void f() { int a; } } void g1() { f(); }",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("g1")));
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, functionDecl(hasName("f"))),
|
|
|
|
2u);
|
|
|
|
}
|
|
|
|
|
2019-02-15 20:04:05 +08:00
|
|
|
TEST_P(ImportFunctions, ImportImplicitFunctionsInLambda) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void foo() {
|
|
|
|
(void)[]() { ; };
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("foo")));
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ToD);
|
|
|
|
CXXRecordDecl *LambdaRec =
|
|
|
|
cast<LambdaExpr>(cast<CStyleCastExpr>(
|
|
|
|
*cast<CompoundStmt>(ToD->getBody())->body_begin())
|
|
|
|
->getSubExpr())
|
|
|
|
->getLambdaClass();
|
|
|
|
EXPECT_TRUE(LambdaRec->getDestructor());
|
|
|
|
}
|
|
|
|
|
2019-03-07 21:38:20 +08:00
|
|
|
TEST_P(ImportFunctions,
|
|
|
|
CallExprOfMemberFunctionTemplateWithExplicitTemplateArgs) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X {
|
|
|
|
template <typename T>
|
|
|
|
void foo(){}
|
|
|
|
};
|
|
|
|
void f() {
|
|
|
|
X x;
|
|
|
|
x.foo<int>();
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ToD);
|
|
|
|
EXPECT_TRUE(MatchVerifier<FunctionDecl>().match(
|
|
|
|
ToD, functionDecl(hasName("f"), hasDescendant(declRefExpr()))));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctions,
|
|
|
|
DependentCallExprOfMemberFunctionTemplateWithExplicitTemplateArgs) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X {
|
|
|
|
template <typename T>
|
|
|
|
void foo(){}
|
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
void f() {
|
|
|
|
X x;
|
|
|
|
x.foo<T>();
|
|
|
|
}
|
|
|
|
void g() {
|
|
|
|
f<int>();
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("g")));
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ToD);
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_TRUE(MatchVerifier<TranslationUnitDecl>().match(
|
|
|
|
ToTU, translationUnitDecl(hasDescendant(
|
|
|
|
functionDecl(hasName("f"), hasDescendant(declRefExpr()))))));
|
|
|
|
}
|
|
|
|
|
2019-08-07 20:40:17 +08:00
|
|
|
struct ImportFunctionTemplates : ASTImporterOptionSpecificTestBase {};
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplates, ImportFunctionTemplateInRecordDeclTwice) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
template <class T>
|
|
|
|
void f(T t);
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *FromTU1 = getTuDecl(Code, Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD1 = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU1, functionTemplateDecl(hasName("f")));
|
|
|
|
auto *ToD1 = Import(FromD1, Lang_CXX);
|
|
|
|
Decl *FromTU2 = getTuDecl(Code, Lang_CXX, "input2.cc");
|
|
|
|
auto *FromD2 = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU2, functionTemplateDecl(hasName("f")));
|
|
|
|
auto *ToD2 = Import(FromD2, Lang_CXX);
|
|
|
|
EXPECT_EQ(ToD1, ToD2);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplates,
|
|
|
|
ImportFunctionTemplateWithDefInRecordDeclTwice) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
template <class T>
|
|
|
|
void f(T t);
|
|
|
|
};
|
|
|
|
template <class T>
|
|
|
|
void X::f(T t) {};
|
|
|
|
)";
|
|
|
|
Decl *FromTU1 = getTuDecl(Code, Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD1 = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU1, functionTemplateDecl(hasName("f")));
|
|
|
|
auto *ToD1 = Import(FromD1, Lang_CXX);
|
|
|
|
Decl *FromTU2 = getTuDecl(Code, Lang_CXX, "input2.cc");
|
|
|
|
auto *FromD2 = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU2, functionTemplateDecl(hasName("f")));
|
|
|
|
auto *ToD2 = Import(FromD2, Lang_CXX);
|
|
|
|
EXPECT_EQ(ToD1, ToD2);
|
|
|
|
}
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
TEST_P(ImportFunctionTemplates,
|
|
|
|
ImportFunctionWhenThereIsAFunTemplateWithSameName) {
|
|
|
|
getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
void foo(T) {}
|
|
|
|
void foo();
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
Decl *FromTU = getTuDecl("void foo();", Lang_CXX);
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("foo")));
|
|
|
|
auto *ImportedD = Import(FromD, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ImportedD);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplates,
|
|
|
|
ImportConstructorWhenThereIsAFunTemplateWithSameName) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
struct Foo {
|
|
|
|
template <typename T>
|
|
|
|
Foo(T) {}
|
|
|
|
Foo();
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
getToTuDecl(Code, Lang_CXX);
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX);
|
|
|
|
auto *FromD =
|
|
|
|
LastDeclMatcher<CXXConstructorDecl>().match(FromTU, cxxConstructorDecl());
|
|
|
|
auto *ImportedD = Import(FromD, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ImportedD);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplates,
|
|
|
|
ImportOperatorWhenThereIsAFunTemplateWithSameName) {
|
|
|
|
getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
void operator<(T,T) {}
|
|
|
|
struct X{};
|
|
|
|
void operator<(X, X);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X{};
|
|
|
|
void operator<(X, X);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromD = LastDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasOverloadedOperatorName("<")));
|
|
|
|
auto *ImportedD = Import(FromD, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ImportedD);
|
|
|
|
}
|
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
struct ImportFriendFunctions : ImportFunctions {};
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, ImportFriendFunctionRedeclChainProto) {
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl("struct X { friend void f(); };"
|
|
|
|
"void f();",
|
|
|
|
Lang_CXX,
|
|
|
|
"input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_FALSE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
auto *ToFD = LastDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_FALSE(ToFD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(ToFD->getPreviousDecl(), ImportedD);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions,
|
|
|
|
ImportFriendFunctionRedeclChainProto_OutOfClassProtoFirst) {
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl("void f();"
|
|
|
|
"struct X { friend void f(); };",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_FALSE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
auto *ToFD = LastDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_FALSE(ToFD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(ToFD->getPreviousDecl(), ImportedD);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, ImportFriendFunctionRedeclChainDef) {
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl("struct X { friend void f(){} };"
|
|
|
|
"void f();",
|
|
|
|
Lang_CXX,
|
|
|
|
"input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_TRUE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
auto *ToFD = LastDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_FALSE(ToFD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(ToFD->getPreviousDecl(), ImportedD);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions,
|
|
|
|
ImportFriendFunctionRedeclChainDef_OutOfClassDef) {
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl("struct X { friend void f(); };"
|
|
|
|
"void f(){}",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_FALSE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
auto *ToFD = LastDeclMatcher<FunctionDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_TRUE(ToFD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(ToFD->getPreviousDecl(), ImportedD);
|
|
|
|
}
|
|
|
|
|
2019-05-20 18:38:14 +08:00
|
|
|
TEST_P(ImportFriendFunctions, ImportFriendFunctionRedeclChainDefWithClass) {
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X;
|
|
|
|
void f(X *x){}
|
|
|
|
class X{
|
|
|
|
friend void f(X *x);
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_TRUE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
auto *InClassFD = cast<FunctionDecl>(FirstDeclMatcher<FriendDecl>()
|
|
|
|
.match(ToTU, friendDecl())
|
|
|
|
->getFriendDecl());
|
|
|
|
EXPECT_FALSE(InClassFD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(InClassFD->getPreviousDecl(), ImportedD);
|
|
|
|
// The parameters must refer the same type
|
|
|
|
EXPECT_EQ((*InClassFD->param_begin())->getOriginalType(),
|
|
|
|
(*ImportedD->param_begin())->getOriginalType());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions,
|
2019-05-20 18:38:14 +08:00
|
|
|
ImportFriendFunctionRedeclChainDefWithClass_ImportTheProto) {
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X;
|
|
|
|
void f(X *x){}
|
|
|
|
class X{
|
|
|
|
friend void f(X *x);
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = LastDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
auto *ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_FALSE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
auto *OutOfClassFD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
ToTU, functionDecl(unless(hasParent(friendDecl()))));
|
|
|
|
|
|
|
|
EXPECT_TRUE(OutOfClassFD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(ImportedD->getPreviousDecl(), OutOfClassFD);
|
|
|
|
// The parameters must refer the same type
|
|
|
|
EXPECT_EQ((*OutOfClassFD->param_begin())->getOriginalType(),
|
|
|
|
(*ImportedD->param_begin())->getOriginalType());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, ImportFriendFunctionFromMultipleTU) {
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
FunctionDecl *ImportedD;
|
|
|
|
{
|
|
|
|
Decl *FromTU =
|
|
|
|
getTuDecl("struct X { friend void f(){} };", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
ImportedD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
}
|
|
|
|
FunctionDecl *ImportedD1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl("void f();", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
ImportedD1 = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
}
|
|
|
|
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
EXPECT_TRUE(ImportedD->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_FALSE(ImportedD1->doesThisDeclarationHaveABody());
|
|
|
|
EXPECT_EQ(ImportedD1->getPreviousDecl(), ImportedD);
|
|
|
|
}
|
|
|
|
|
2018-08-21 22:32:21 +08:00
|
|
|
TEST_P(ImportFriendFunctions, Lookup) {
|
|
|
|
auto FunctionPattern = functionDecl(hasName("f"));
|
|
|
|
auto ClassPattern = cxxRecordDecl(hasName("X"));
|
|
|
|
|
|
|
|
TranslationUnitDecl *FromTU =
|
|
|
|
getTuDecl("struct X { friend void f(); };", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU, FunctionPattern);
|
|
|
|
ASSERT_TRUE(FromD->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
ASSERT_FALSE(FromD->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
{
|
|
|
|
auto FromName = FromD->getDeclName();
|
|
|
|
auto *Class = FirstDeclMatcher<CXXRecordDecl>().match(FromTU, ClassPattern);
|
|
|
|
auto LookupRes = Class->noload_lookup(FromName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 0u);
|
|
|
|
LookupRes = FromTU->noload_lookup(FromName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto *ToD = cast<FunctionDecl>(Import(FromD, Lang_CXX));
|
|
|
|
auto ToName = ToD->getDeclName();
|
|
|
|
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
auto *Class = FirstDeclMatcher<CXXRecordDecl>().match(ToTU, ClassPattern);
|
|
|
|
auto LookupRes = Class->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 0u);
|
|
|
|
LookupRes = ToTU->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 1u);
|
|
|
|
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, FunctionPattern), 1u);
|
|
|
|
auto *To0 = FirstDeclMatcher<FunctionDecl>().match(ToTU, FunctionPattern);
|
|
|
|
EXPECT_TRUE(To0->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
EXPECT_FALSE(To0->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, DISABLED_LookupWithProtoAfter) {
|
|
|
|
auto FunctionPattern = functionDecl(hasName("f"));
|
|
|
|
auto ClassPattern = cxxRecordDecl(hasName("X"));
|
|
|
|
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
"struct X { friend void f(); };"
|
|
|
|
// This proto decl makes f available to normal
|
|
|
|
// lookup, otherwise it is hidden.
|
|
|
|
// Normal C++ lookup (implemented in
|
|
|
|
// `clang::Sema::CppLookupName()` and in `LookupDirect()`)
|
|
|
|
// returns the found `NamedDecl` only if the set IDNS is matched
|
|
|
|
"void f();",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromFriend =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(FromTU, FunctionPattern);
|
|
|
|
auto *FromNormal =
|
|
|
|
LastDeclMatcher<FunctionDecl>().match(FromTU, FunctionPattern);
|
|
|
|
ASSERT_TRUE(FromFriend->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
ASSERT_FALSE(FromFriend->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
ASSERT_FALSE(FromNormal->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
ASSERT_TRUE(FromNormal->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
|
|
|
|
auto FromName = FromFriend->getDeclName();
|
|
|
|
auto *FromClass =
|
|
|
|
FirstDeclMatcher<CXXRecordDecl>().match(FromTU, ClassPattern);
|
|
|
|
auto LookupRes = FromClass->noload_lookup(FromName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 0u);
|
|
|
|
LookupRes = FromTU->noload_lookup(FromName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 1u);
|
|
|
|
|
|
|
|
auto *ToFriend = cast<FunctionDecl>(Import(FromFriend, Lang_CXX));
|
|
|
|
auto ToName = ToFriend->getDeclName();
|
|
|
|
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
auto *ToClass = FirstDeclMatcher<CXXRecordDecl>().match(ToTU, ClassPattern);
|
|
|
|
LookupRes = ToClass->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 0u);
|
|
|
|
LookupRes = ToTU->noload_lookup(ToName);
|
|
|
|
// Test is disabled because this result is 2.
|
|
|
|
EXPECT_EQ(LookupRes.size(), 1u);
|
|
|
|
|
|
|
|
ASSERT_EQ(DeclCounter<FunctionDecl>().match(ToTU, FunctionPattern), 2u);
|
|
|
|
ToFriend = FirstDeclMatcher<FunctionDecl>().match(ToTU, FunctionPattern);
|
|
|
|
auto *ToNormal = LastDeclMatcher<FunctionDecl>().match(ToTU, FunctionPattern);
|
|
|
|
EXPECT_TRUE(ToFriend->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
EXPECT_FALSE(ToFriend->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
EXPECT_FALSE(ToNormal->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
EXPECT_TRUE(ToNormal->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, LookupWithProtoBefore) {
|
|
|
|
auto FunctionPattern = functionDecl(hasName("f"));
|
|
|
|
auto ClassPattern = cxxRecordDecl(hasName("X"));
|
|
|
|
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
"void f();"
|
|
|
|
"struct X { friend void f(); };",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromNormal =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(FromTU, FunctionPattern);
|
|
|
|
auto *FromFriend =
|
|
|
|
LastDeclMatcher<FunctionDecl>().match(FromTU, FunctionPattern);
|
|
|
|
ASSERT_FALSE(FromNormal->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
ASSERT_TRUE(FromNormal->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
ASSERT_TRUE(FromFriend->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
ASSERT_TRUE(FromFriend->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
|
|
|
|
auto FromName = FromNormal->getDeclName();
|
|
|
|
auto *FromClass =
|
|
|
|
FirstDeclMatcher<CXXRecordDecl>().match(FromTU, ClassPattern);
|
|
|
|
auto LookupRes = FromClass->noload_lookup(FromName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 0u);
|
|
|
|
LookupRes = FromTU->noload_lookup(FromName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 1u);
|
|
|
|
|
|
|
|
auto *ToNormal = cast<FunctionDecl>(Import(FromNormal, Lang_CXX));
|
|
|
|
auto ToName = ToNormal->getDeclName();
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
|
|
|
|
auto *ToClass = FirstDeclMatcher<CXXRecordDecl>().match(ToTU, ClassPattern);
|
|
|
|
LookupRes = ToClass->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 0u);
|
|
|
|
LookupRes = ToTU->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 1u);
|
|
|
|
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, FunctionPattern), 2u);
|
|
|
|
ToNormal = FirstDeclMatcher<FunctionDecl>().match(ToTU, FunctionPattern);
|
|
|
|
auto *ToFriend = LastDeclMatcher<FunctionDecl>().match(ToTU, FunctionPattern);
|
|
|
|
EXPECT_FALSE(ToNormal->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
EXPECT_TRUE(ToNormal->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
EXPECT_TRUE(ToFriend->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
EXPECT_TRUE(ToFriend->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, ImportFriendChangesLookup) {
|
|
|
|
auto Pattern = functionDecl(hasName("f"));
|
|
|
|
|
|
|
|
TranslationUnitDecl *FromNormalTU =
|
|
|
|
getTuDecl("void f();", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromNormalF =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(FromNormalTU, Pattern);
|
|
|
|
TranslationUnitDecl *FromFriendTU =
|
|
|
|
getTuDecl("class X { friend void f(); };", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromFriendF =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(FromFriendTU, Pattern);
|
|
|
|
auto FromNormalName = FromNormalF->getDeclName();
|
|
|
|
auto FromFriendName = FromFriendF->getDeclName();
|
|
|
|
|
|
|
|
ASSERT_TRUE(FromNormalF->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
ASSERT_FALSE(FromNormalF->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
ASSERT_FALSE(FromFriendF->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
ASSERT_TRUE(FromFriendF->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
auto LookupRes = FromNormalTU->noload_lookup(FromNormalName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 1u);
|
|
|
|
LookupRes = FromFriendTU->noload_lookup(FromFriendName);
|
|
|
|
ASSERT_EQ(LookupRes.size(), 1u);
|
|
|
|
|
|
|
|
auto *ToNormalF = cast<FunctionDecl>(Import(FromNormalF, Lang_CXX));
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
auto ToName = ToNormalF->getDeclName();
|
|
|
|
EXPECT_TRUE(ToNormalF->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
EXPECT_FALSE(ToNormalF->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
LookupRes = ToTU->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 1u);
|
2018-10-10 21:27:25 +08:00
|
|
|
|
2018-08-21 22:32:21 +08:00
|
|
|
auto *ToFriendF = cast<FunctionDecl>(Import(FromFriendF, Lang_CXX));
|
|
|
|
LookupRes = ToTU->noload_lookup(ToName);
|
|
|
|
EXPECT_EQ(LookupRes.size(), 1u);
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
|
|
|
|
EXPECT_TRUE(ToNormalF->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
EXPECT_FALSE(ToNormalF->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
|
|
|
|
EXPECT_TRUE(ToFriendF->isInIdentifierNamespace(Decl::IDNS_Ordinary));
|
|
|
|
EXPECT_TRUE(ToFriendF->isInIdentifierNamespace(Decl::IDNS_OrdinaryFriend));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctions, ImportFriendList) {
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
"struct X { friend void f(); };"
|
|
|
|
"void f();",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromFriendF = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
|
|
|
|
auto *FromClass = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X")));
|
|
|
|
auto *FromFriend = FirstDeclMatcher<FriendDecl>().match(FromTU, friendDecl());
|
|
|
|
auto FromFriends = FromClass->friends();
|
|
|
|
unsigned int FrN = 0;
|
|
|
|
for (auto Fr : FromFriends) {
|
|
|
|
ASSERT_EQ(Fr, FromFriend);
|
|
|
|
++FrN;
|
|
|
|
}
|
|
|
|
ASSERT_EQ(FrN, 1u);
|
|
|
|
|
|
|
|
Import(FromFriendF, Lang_CXX);
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
auto *ToClass = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("X")));
|
|
|
|
auto *ToFriend = FirstDeclMatcher<FriendDecl>().match(ToTU, friendDecl());
|
|
|
|
auto ToFriends = ToClass->friends();
|
|
|
|
FrN = 0;
|
|
|
|
for (auto Fr : ToFriends) {
|
|
|
|
EXPECT_EQ(Fr, ToFriend);
|
|
|
|
++FrN;
|
|
|
|
}
|
|
|
|
EXPECT_EQ(FrN, 1u);
|
|
|
|
}
|
|
|
|
|
2018-04-24 18:11:53 +08:00
|
|
|
AST_MATCHER_P(TagDecl, hasTypedefForAnonDecl, Matcher<TypedefNameDecl>,
|
|
|
|
InnerMatcher) {
|
|
|
|
if (auto *Typedef = Node.getTypedefNameForAnonDecl())
|
|
|
|
return InnerMatcher.matches(*Typedef, Finder, Builder);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportDecl, ImportEnumSequential) {
|
2018-04-24 18:11:53 +08:00
|
|
|
CodeFiles Samples{{"main.c",
|
|
|
|
{"void foo();"
|
|
|
|
"void moo();"
|
|
|
|
"int main() { foo(); moo(); }",
|
|
|
|
Lang_C}},
|
|
|
|
|
|
|
|
{"foo.c",
|
|
|
|
{"typedef enum { THING_VALUE } thing_t;"
|
|
|
|
"void conflict(thing_t type);"
|
|
|
|
"void foo() { (void)THING_VALUE; }"
|
|
|
|
"void conflict(thing_t type) {}",
|
|
|
|
Lang_C}},
|
|
|
|
|
|
|
|
{"moo.c",
|
|
|
|
{"typedef enum { THING_VALUE } thing_t;"
|
|
|
|
"void conflict(thing_t type);"
|
|
|
|
"void moo() { conflict(THING_VALUE); }",
|
|
|
|
Lang_C}}};
|
|
|
|
|
|
|
|
auto VerificationMatcher =
|
|
|
|
enumDecl(has(enumConstantDecl(hasName("THING_VALUE"))),
|
|
|
|
hasTypedefForAnonDecl(hasName("thing_t")));
|
|
|
|
|
|
|
|
ImportAction ImportFoo{"foo.c", "main.c", functionDecl(hasName("foo"))},
|
|
|
|
ImportMoo{"moo.c", "main.c", functionDecl(hasName("moo"))};
|
|
|
|
|
|
|
|
testImportSequence(
|
|
|
|
Samples, {ImportFoo, ImportMoo}, // "foo", them "moo".
|
|
|
|
// Just check that there is only one enum decl in the result AST.
|
|
|
|
"main.c", enumDecl(), VerificationMatcher);
|
|
|
|
|
|
|
|
// For different import order, result should be the same.
|
|
|
|
testImportSequence(
|
|
|
|
Samples, {ImportMoo, ImportFoo}, // "moo", them "foo".
|
|
|
|
// Check that there is only one enum decl in the result AST.
|
|
|
|
"main.c", enumDecl(), VerificationMatcher);
|
|
|
|
}
|
|
|
|
|
2019-07-25 17:07:17 +08:00
|
|
|
TEST_P(ImportDecl, ImportFieldOrder) {
|
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport("struct declToImport {"
|
|
|
|
" int b = a + 2;"
|
|
|
|
" int a = 5;"
|
|
|
|
"};",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
recordDecl(hasFieldOrder({"b", "a"})));
|
|
|
|
}
|
|
|
|
|
2018-05-07 20:08:27 +08:00
|
|
|
const internal::VariadicDynCastAllOfMatcher<Expr, DependentScopeDeclRefExpr>
|
|
|
|
dependentScopeDeclRefExpr;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, DependentScopeDeclRefExpr) {
|
2018-05-07 20:08:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport("template <typename T> struct S { static T foo; };"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" (void) S<T>::foo;"
|
|
|
|
"}"
|
2018-06-29 18:25:19 +08:00
|
|
|
"void instantiate() { declToImport<int>(); }"
|
|
|
|
"template <typename T> T S<T>::foo;",
|
2018-05-07 20:08:27 +08:00
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionTemplateDecl(has(functionDecl(has(compoundStmt(
|
|
|
|
has(cStyleCastExpr(has(dependentScopeDeclRefExpr())))))))));
|
|
|
|
|
|
|
|
testImport("template <typename T> struct S {"
|
|
|
|
"template<typename S> static void foo(){};"
|
|
|
|
"};"
|
|
|
|
"template <typename T> void declToImport() {"
|
|
|
|
" S<T>::template foo<T>();"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionTemplateDecl(has(functionDecl(has(compoundStmt(
|
|
|
|
has(callExpr(has(dependentScopeDeclRefExpr())))))))));
|
|
|
|
}
|
|
|
|
|
|
|
|
const internal::VariadicDynCastAllOfMatcher<Type, DependentNameType>
|
|
|
|
dependentNameType;
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, DependentNameType) {
|
2018-05-07 20:08:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport("template <typename T> struct declToImport {"
|
|
|
|
" typedef typename T::type dependent_name;"
|
|
|
|
"};",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
classTemplateDecl(has(
|
|
|
|
cxxRecordDecl(has(typedefDecl(has(dependentNameType())))))));
|
|
|
|
}
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
TEST_P(ImportExpr, UnresolvedMemberExpr) {
|
2018-05-07 20:08:27 +08:00
|
|
|
MatchVerifier<Decl> Verifier;
|
|
|
|
testImport("struct S { template <typename T> void mem(); };"
|
|
|
|
"template <typename U> void declToImport() {"
|
|
|
|
" S s;"
|
|
|
|
" s.mem<U>();"
|
|
|
|
"}"
|
|
|
|
"void instantiate() { declToImport<int>(); }",
|
|
|
|
Lang_CXX11, "", Lang_CXX11, Verifier,
|
|
|
|
functionTemplateDecl(has(functionDecl(has(
|
|
|
|
compoundStmt(has(callExpr(has(unresolvedMemberExpr())))))))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
class ImportImplicitMethods : public ASTImporterOptionSpecificTestBase {
|
2018-07-16 20:16:39 +08:00
|
|
|
public:
|
|
|
|
static constexpr auto DefaultCode = R"(
|
|
|
|
struct A { int x; };
|
|
|
|
void f() {
|
|
|
|
A a;
|
|
|
|
A a1(a);
|
|
|
|
A a2(A{});
|
|
|
|
a = a1;
|
|
|
|
a = A{};
|
|
|
|
a.~A();
|
|
|
|
})";
|
|
|
|
|
|
|
|
template <typename MatcherType>
|
|
|
|
void testImportOf(
|
|
|
|
const MatcherType &MethodMatcher, const char *Code = DefaultCode) {
|
|
|
|
test(MethodMatcher, Code, /*ExpectedCount=*/1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename MatcherType>
|
|
|
|
void testNoImportOf(
|
|
|
|
const MatcherType &MethodMatcher, const char *Code = DefaultCode) {
|
|
|
|
test(MethodMatcher, Code, /*ExpectedCount=*/0u);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
template <typename MatcherType>
|
|
|
|
void test(const MatcherType &MethodMatcher,
|
|
|
|
const char *Code, unsigned int ExpectedCount) {
|
|
|
|
auto ClassMatcher = cxxRecordDecl(unless(isImplicit()));
|
|
|
|
|
|
|
|
Decl *ToTU = getToTuDecl(Code, Lang_CXX11);
|
|
|
|
auto *ToClass = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, ClassMatcher);
|
|
|
|
|
2018-07-16 22:05:18 +08:00
|
|
|
ASSERT_EQ(DeclCounter<CXXMethodDecl>().match(ToClass, MethodMatcher), 1u);
|
2018-07-16 20:16:39 +08:00
|
|
|
|
|
|
|
{
|
|
|
|
CXXMethodDecl *Method =
|
|
|
|
FirstDeclMatcher<CXXMethodDecl>().match(ToClass, MethodMatcher);
|
|
|
|
ToClass->removeDecl(Method);
|
2019-07-01 23:37:07 +08:00
|
|
|
SharedStatePtr->getLookupTable()->remove(Method);
|
2018-07-16 20:16:39 +08:00
|
|
|
}
|
|
|
|
|
2018-07-16 22:05:18 +08:00
|
|
|
ASSERT_EQ(DeclCounter<CXXMethodDecl>().match(ToClass, MethodMatcher), 0u);
|
2018-07-16 20:16:39 +08:00
|
|
|
|
|
|
|
Decl *ImportedClass = nullptr;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX11, "input1.cc");
|
|
|
|
auto *FromClass = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, ClassMatcher);
|
|
|
|
ImportedClass = Import(FromClass, Lang_CXX11);
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPECT_EQ(ToClass, ImportedClass);
|
|
|
|
EXPECT_EQ(DeclCounter<CXXMethodDecl>().match(ToClass, MethodMatcher),
|
|
|
|
ExpectedCount);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, DefaultConstructor) {
|
|
|
|
testImportOf(cxxConstructorDecl(isDefaultConstructor()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, CopyConstructor) {
|
|
|
|
testImportOf(cxxConstructorDecl(isCopyConstructor()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, MoveConstructor) {
|
|
|
|
testImportOf(cxxConstructorDecl(isMoveConstructor()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, Destructor) {
|
|
|
|
testImportOf(cxxDestructorDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, CopyAssignment) {
|
|
|
|
testImportOf(cxxMethodDecl(isCopyAssignmentOperator()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, MoveAssignment) {
|
|
|
|
testImportOf(cxxMethodDecl(isMoveAssignmentOperator()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, DoNotImportUserProvided) {
|
|
|
|
auto Code = R"(
|
|
|
|
struct A { A() { int x; } };
|
|
|
|
)";
|
|
|
|
testNoImportOf(cxxConstructorDecl(isDefaultConstructor()), Code);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, DoNotImportDefault) {
|
|
|
|
auto Code = R"(
|
|
|
|
struct A { A() = default; };
|
|
|
|
)";
|
|
|
|
testNoImportOf(cxxConstructorDecl(isDefaultConstructor()), Code);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, DoNotImportDeleted) {
|
|
|
|
auto Code = R"(
|
|
|
|
struct A { A() = delete; };
|
|
|
|
)";
|
|
|
|
testNoImportOf(cxxConstructorDecl(isDefaultConstructor()), Code);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportImplicitMethods, DoNotImportOtherMethod) {
|
|
|
|
auto Code = R"(
|
|
|
|
struct A { void f() { } };
|
|
|
|
)";
|
|
|
|
testNoImportOf(cxxMethodDecl(hasName("f")), Code);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfEquivalentRecord) {
|
2018-07-11 17:37:24 +08:00
|
|
|
Decl *ToR1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { };", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromR = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("A")));
|
|
|
|
|
|
|
|
ToR1 = Import(FromR, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
Decl *ToR2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { };", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromR = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("A")));
|
|
|
|
|
|
|
|
ToR2 = Import(FromR, Lang_CXX);
|
|
|
|
}
|
2018-10-10 21:27:25 +08:00
|
|
|
|
2018-07-11 17:37:24 +08:00
|
|
|
EXPECT_EQ(ToR1, ToR2);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfNonEquivalentRecord) {
|
2018-07-11 17:37:24 +08:00
|
|
|
Decl *ToR1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { int x; };", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromR = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("A")));
|
|
|
|
ToR1 = Import(FromR, Lang_CXX);
|
|
|
|
}
|
|
|
|
Decl *ToR2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { unsigned x; };", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromR = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("A")));
|
|
|
|
ToR2 = Import(FromR, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_NE(ToR1, ToR2);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfEquivalentField) {
|
2018-07-11 17:37:24 +08:00
|
|
|
Decl *ToF1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { int x; };", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromF = FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
FromTU, fieldDecl(hasName("x")));
|
|
|
|
ToF1 = Import(FromF, Lang_CXX);
|
|
|
|
}
|
|
|
|
Decl *ToF2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { int x; };", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromF = FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
FromTU, fieldDecl(hasName("x")));
|
|
|
|
ToF2 = Import(FromF, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_EQ(ToF1, ToF2);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfNonEquivalentField) {
|
2018-07-11 17:37:24 +08:00
|
|
|
Decl *ToF1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { int x; };", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromF = FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
FromTU, fieldDecl(hasName("x")));
|
|
|
|
ToF1 = Import(FromF, Lang_CXX);
|
|
|
|
}
|
|
|
|
Decl *ToF2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { unsigned x; };", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromF = FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
FromTU, fieldDecl(hasName("x")));
|
|
|
|
ToF2 = Import(FromF, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_NE(ToF1, ToF2);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfEquivalentMethod) {
|
2018-07-11 17:37:24 +08:00
|
|
|
Decl *ToM1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { void x(); }; void A::x() { }", Lang_CXX, "input0.cc");
|
|
|
|
auto *FromM = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("x"), isDefinition()));
|
|
|
|
ToM1 = Import(FromM, Lang_CXX);
|
|
|
|
}
|
|
|
|
Decl *ToM2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { void x(); }; void A::x() { }", Lang_CXX, "input1.cc");
|
|
|
|
auto *FromM = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("x"), isDefinition()));
|
|
|
|
ToM2 = Import(FromM, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_EQ(ToM1, ToM2);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfNonEquivalentMethod) {
|
2018-07-11 17:37:24 +08:00
|
|
|
Decl *ToM1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { void x(); }; void A::x() { }",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromM = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("x"), isDefinition()));
|
|
|
|
ToM1 = Import(FromM, Lang_CXX);
|
|
|
|
}
|
|
|
|
Decl *ToM2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
"struct A { void x() const; }; void A::x() const { }",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *FromM = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("x"), isDefinition()));
|
|
|
|
ToM2 = Import(FromM, Lang_CXX);
|
|
|
|
}
|
|
|
|
EXPECT_NE(ToM1, ToM2);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportUnnamedStructsWithRecursingField) {
|
2018-07-17 20:06:36 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
struct {
|
|
|
|
struct A *next;
|
|
|
|
} entry0;
|
|
|
|
struct {
|
|
|
|
struct A *next;
|
|
|
|
} entry1;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_C, "input0.cc");
|
|
|
|
auto *From =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(FromTU, recordDecl(hasName("A")));
|
|
|
|
|
|
|
|
Import(From, Lang_C);
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
auto *Entry0 =
|
|
|
|
FirstDeclMatcher<FieldDecl>().match(ToTU, fieldDecl(hasName("entry0")));
|
|
|
|
auto *Entry1 =
|
|
|
|
FirstDeclMatcher<FieldDecl>().match(ToTU, fieldDecl(hasName("entry1")));
|
|
|
|
auto *R0 = getRecordDecl(Entry0);
|
|
|
|
auto *R1 = getRecordDecl(Entry1);
|
|
|
|
EXPECT_NE(R0, R1);
|
|
|
|
EXPECT_TRUE(MatchVerifier<RecordDecl>().match(
|
|
|
|
R0, recordDecl(has(fieldDecl(hasName("next"))))));
|
|
|
|
EXPECT_TRUE(MatchVerifier<RecordDecl>().match(
|
|
|
|
R1, recordDecl(has(fieldDecl(hasName("next"))))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportUnnamedFieldsInCorrectOrder) {
|
2018-08-08 17:40:57 +08:00
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void f(int X, int Y, bool Z) {
|
|
|
|
(void)[X, Y, Z] { (void)Z; };
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto *FromF = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
auto *ToF = cast_or_null<FunctionDecl>(Import(FromF, Lang_CXX11));
|
|
|
|
EXPECT_TRUE(ToF);
|
|
|
|
|
|
|
|
CXXRecordDecl *FromLambda =
|
|
|
|
cast<LambdaExpr>(cast<CStyleCastExpr>(cast<CompoundStmt>(
|
|
|
|
FromF->getBody())->body_front())->getSubExpr())->getLambdaClass();
|
|
|
|
|
|
|
|
auto *ToLambda = cast_or_null<CXXRecordDecl>(Import(FromLambda, Lang_CXX11));
|
|
|
|
EXPECT_TRUE(ToLambda);
|
|
|
|
|
|
|
|
// Check if the fields of the lambda class are imported in correct order.
|
|
|
|
unsigned FromIndex = 0u;
|
|
|
|
for (auto *FromField : FromLambda->fields()) {
|
|
|
|
ASSERT_FALSE(FromField->getDeclName());
|
|
|
|
auto *ToField = cast_or_null<FieldDecl>(Import(FromField, Lang_CXX11));
|
|
|
|
EXPECT_TRUE(ToField);
|
2018-10-19 21:32:20 +08:00
|
|
|
Optional<unsigned> ToIndex = ASTImporter::getFieldIndex(ToField);
|
|
|
|
EXPECT_TRUE(ToIndex);
|
|
|
|
EXPECT_EQ(*ToIndex, FromIndex);
|
2018-08-08 17:40:57 +08:00
|
|
|
++FromIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPECT_EQ(FromIndex, 3u);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
MergeFieldDeclsOfClassTemplateSpecialization) {
|
2018-08-22 19:52:14 +08:00
|
|
|
std::string ClassTemplate =
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct X {
|
|
|
|
int a{0}; // FieldDecl with InitListExpr
|
|
|
|
X(char) : a(3) {} // (1)
|
|
|
|
X(int) {} // (2)
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
void foo() {
|
|
|
|
// ClassTemplateSpec with ctor (1): FieldDecl without InitlistExpr
|
|
|
|
X<char> xc('c');
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
auto *ToSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
// FieldDecl without InitlistExpr:
|
|
|
|
auto *ToField = *ToSpec->field_begin();
|
|
|
|
ASSERT_TRUE(ToField);
|
|
|
|
ASSERT_FALSE(ToField->getInClassInitializer());
|
|
|
|
Decl *FromTU = getTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
void bar() {
|
|
|
|
// ClassTemplateSpec with ctor (2): FieldDecl WITH InitlistExpr
|
|
|
|
X<char> xc(1);
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
// FieldDecl with InitlistExpr:
|
|
|
|
auto *FromField = *FromSpec->field_begin();
|
|
|
|
ASSERT_TRUE(FromField);
|
|
|
|
ASSERT_TRUE(FromField->getInClassInitializer());
|
|
|
|
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
ASSERT_TRUE(ImportedSpec);
|
|
|
|
EXPECT_EQ(ImportedSpec, ToSpec);
|
|
|
|
// After the import, the FieldDecl has to be merged, thus it should have the
|
|
|
|
// InitListExpr.
|
|
|
|
EXPECT_TRUE(ToField->getInClassInitializer());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
MergeFunctionOfClassTemplateSpecialization) {
|
2018-08-22 19:52:14 +08:00
|
|
|
std::string ClassTemplate =
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct X {
|
|
|
|
void f() {}
|
|
|
|
void g() {}
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
void foo() {
|
|
|
|
X<char> x;
|
|
|
|
x.f();
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
void bar() {
|
|
|
|
X<char> x;
|
|
|
|
x.g();
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
auto FunPattern = functionDecl(hasName("g"),
|
|
|
|
hasParent(classTemplateSpecializationDecl()));
|
|
|
|
auto *FromFun =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(FromTU, FunPattern);
|
|
|
|
auto *ToFun =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(ToTU, FunPattern);
|
|
|
|
ASSERT_TRUE(FromFun->hasBody());
|
|
|
|
ASSERT_FALSE(ToFun->hasBody());
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
ASSERT_TRUE(ImportedSpec);
|
|
|
|
auto *ToSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
EXPECT_EQ(ImportedSpec, ToSpec);
|
|
|
|
EXPECT_TRUE(ToFun->hasBody());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-08-22 19:52:14 +08:00
|
|
|
ODRViolationOfClassTemplateSpecializationsShouldBeReported) {
|
|
|
|
std::string ClassTemplate =
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct X {};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
template <>
|
|
|
|
struct X<char> {
|
|
|
|
int a;
|
|
|
|
};
|
|
|
|
void foo() {
|
|
|
|
X<char> x;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
template <>
|
|
|
|
struct X<char> {
|
|
|
|
int b;
|
|
|
|
};
|
|
|
|
void foo() {
|
|
|
|
X<char> x;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
|
|
|
|
// We expect one (ODR) warning during the import.
|
|
|
|
EXPECT_EQ(1u, ToTU->getASTContext().getDiagnostics().getNumWarnings());
|
|
|
|
|
|
|
|
// The second specialization is different from the first, thus it violates
|
|
|
|
// ODR, consequently we expect to keep the first specialization only, which is
|
|
|
|
// already in the "To" context.
|
2019-03-19 21:34:10 +08:00
|
|
|
EXPECT_FALSE(ImportedSpec);
|
|
|
|
EXPECT_EQ(1u,
|
|
|
|
DeclCounter<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(hasName("X"))));
|
2018-08-22 19:52:14 +08:00
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
MergeCtorOfClassTemplateSpecialization) {
|
2018-08-22 19:52:14 +08:00
|
|
|
std::string ClassTemplate =
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct X {
|
|
|
|
X(char) {}
|
|
|
|
X(int) {}
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
void foo() {
|
|
|
|
X<char> x('c');
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(ClassTemplate +
|
|
|
|
R"(
|
|
|
|
void bar() {
|
|
|
|
X<char> x(1);
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
// Match the void(int) ctor.
|
|
|
|
auto CtorPattern =
|
|
|
|
cxxConstructorDecl(hasParameter(0, varDecl(hasType(asString("int")))),
|
|
|
|
hasParent(classTemplateSpecializationDecl()));
|
|
|
|
auto *FromCtor =
|
|
|
|
FirstDeclMatcher<CXXConstructorDecl>().match(FromTU, CtorPattern);
|
|
|
|
auto *ToCtor =
|
|
|
|
FirstDeclMatcher<CXXConstructorDecl>().match(ToTU, CtorPattern);
|
|
|
|
ASSERT_TRUE(FromCtor->hasBody());
|
|
|
|
ASSERT_FALSE(ToCtor->hasBody());
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
ASSERT_TRUE(ImportedSpec);
|
|
|
|
auto *ToSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
EXPECT_EQ(ImportedSpec, ToSpec);
|
|
|
|
EXPECT_TRUE(ToCtor->hasBody());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-08-22 19:52:14 +08:00
|
|
|
ClassTemplatePartialSpecializationsShouldNotBeDuplicated) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
// primary template
|
|
|
|
template<class T1, class T2, int I>
|
|
|
|
class A {};
|
|
|
|
|
|
|
|
// partial specialization
|
|
|
|
template<class T, int I>
|
|
|
|
class A<T, T*, I> {};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(Code, Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX11);
|
|
|
|
auto *FromSpec =
|
|
|
|
FirstDeclMatcher<ClassTemplatePartialSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplatePartialSpecializationDecl());
|
|
|
|
auto *ToSpec =
|
|
|
|
FirstDeclMatcher<ClassTemplatePartialSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplatePartialSpecializationDecl());
|
|
|
|
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
EXPECT_EQ(ImportedSpec, ToSpec);
|
|
|
|
EXPECT_EQ(1u, DeclCounter<ClassTemplatePartialSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplatePartialSpecializationDecl()));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ClassTemplateSpecializationsShouldNotBeDuplicated) {
|
2018-08-22 19:52:14 +08:00
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
// primary template
|
|
|
|
template<class T1, class T2, int I>
|
|
|
|
class A {};
|
|
|
|
|
|
|
|
// full specialization
|
|
|
|
template<>
|
|
|
|
class A<int, int, 1> {};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(Code, Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX11);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl());
|
|
|
|
auto *ToSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl());
|
|
|
|
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
EXPECT_EQ(ImportedSpec, ToSpec);
|
|
|
|
EXPECT_EQ(1u, DeclCounter<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl()));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ClassTemplateFullAndPartialSpecsShouldNotBeMixed) {
|
2018-08-22 19:52:14 +08:00
|
|
|
std::string PrimaryTemplate =
|
|
|
|
R"(
|
|
|
|
template<class T1, class T2, int I>
|
|
|
|
class A {};
|
|
|
|
)";
|
|
|
|
auto PartialSpec =
|
|
|
|
R"(
|
|
|
|
template<class T, int I>
|
|
|
|
class A<T, T*, I> {};
|
|
|
|
)";
|
|
|
|
auto FullSpec =
|
|
|
|
R"(
|
|
|
|
template<>
|
|
|
|
class A<int, int, 1> {};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(PrimaryTemplate + FullSpec, Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(PrimaryTemplate + PartialSpec, Lang_CXX11);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl());
|
|
|
|
|
|
|
|
auto *ImportedSpec = Import(FromSpec, Lang_CXX11);
|
|
|
|
EXPECT_TRUE(ImportedSpec);
|
|
|
|
// Check the number of partial specializations.
|
|
|
|
EXPECT_EQ(1u, DeclCounter<ClassTemplatePartialSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplatePartialSpecializationDecl()));
|
|
|
|
// Check the number of full specializations.
|
|
|
|
EXPECT_EQ(1u, DeclCounter<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(
|
|
|
|
unless(classTemplatePartialSpecializationDecl()))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
InitListExprValueKindShouldBeImported) {
|
2018-09-03 21:10:53 +08:00
|
|
|
Decl *TU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
const int &init();
|
|
|
|
void foo() { const int &a{init()}; }
|
|
|
|
)", Lang_CXX11, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<VarDecl>().match(TU, varDecl(hasName("a")));
|
|
|
|
ASSERT_TRUE(FromD->getAnyInitializer());
|
|
|
|
auto *InitExpr = FromD->getAnyInitializer();
|
|
|
|
ASSERT_TRUE(InitExpr);
|
|
|
|
ASSERT_TRUE(InitExpr->isGLValue());
|
|
|
|
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX11);
|
|
|
|
EXPECT_TRUE(ToD);
|
|
|
|
auto *ToInitExpr = cast<VarDecl>(ToD)->getAnyInitializer();
|
|
|
|
EXPECT_TRUE(ToInitExpr);
|
|
|
|
EXPECT_TRUE(ToInitExpr->isGLValue());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct ImportVariables : ASTImporterOptionSpecificTestBase {};
|
2018-09-17 20:04:52 +08:00
|
|
|
|
|
|
|
TEST_P(ImportVariables, ImportOfOneDeclBringsInTheWholeChain) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
static const int a = 1 + 2;
|
|
|
|
};
|
|
|
|
const int A::a;
|
|
|
|
)", Lang_CXX, "input1.cc");
|
|
|
|
|
|
|
|
auto *FromDWithInit = FirstDeclMatcher<VarDecl>().match(
|
|
|
|
FromTU, varDecl(hasName("a"))); // Decl with init
|
|
|
|
auto *FromDWithDef = LastDeclMatcher<VarDecl>().match(
|
|
|
|
FromTU, varDecl(hasName("a"))); // Decl with definition
|
|
|
|
ASSERT_NE(FromDWithInit, FromDWithDef);
|
|
|
|
ASSERT_EQ(FromDWithDef->getPreviousDecl(), FromDWithInit);
|
|
|
|
|
|
|
|
auto *ToD0 = cast<VarDecl>(Import(FromDWithInit, Lang_CXX11));
|
|
|
|
auto *ToD1 = cast<VarDecl>(Import(FromDWithDef, Lang_CXX11));
|
|
|
|
ASSERT_TRUE(ToD0);
|
|
|
|
ASSERT_TRUE(ToD1);
|
|
|
|
EXPECT_NE(ToD0, ToD1);
|
|
|
|
EXPECT_EQ(ToD1->getPreviousDecl(), ToD0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportVariables, InitAndDefinitionAreInDifferentTUs) {
|
|
|
|
auto StructA =
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
static const int a = 1 + 2;
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(StructA, Lang_CXX);
|
|
|
|
Decl *FromTU = getTuDecl(std::string(StructA) + "const int A::a;", Lang_CXX,
|
|
|
|
"input1.cc");
|
|
|
|
|
|
|
|
auto *FromDWithInit = FirstDeclMatcher<VarDecl>().match(
|
|
|
|
FromTU, varDecl(hasName("a"))); // Decl with init
|
|
|
|
auto *FromDWithDef = LastDeclMatcher<VarDecl>().match(
|
|
|
|
FromTU, varDecl(hasName("a"))); // Decl with definition
|
|
|
|
ASSERT_EQ(FromDWithInit, FromDWithDef->getPreviousDecl());
|
|
|
|
ASSERT_TRUE(FromDWithInit->getInit());
|
|
|
|
ASSERT_FALSE(FromDWithInit->isThisDeclarationADefinition());
|
|
|
|
ASSERT_TRUE(FromDWithDef->isThisDeclarationADefinition());
|
|
|
|
ASSERT_FALSE(FromDWithDef->getInit());
|
|
|
|
|
|
|
|
auto *ToD = FirstDeclMatcher<VarDecl>().match(
|
|
|
|
ToTU, varDecl(hasName("a"))); // Decl with init
|
|
|
|
ASSERT_TRUE(ToD->getInit());
|
|
|
|
ASSERT_FALSE(ToD->getDefinition());
|
|
|
|
|
|
|
|
auto *ImportedD = cast<VarDecl>(Import(FromDWithDef, Lang_CXX11));
|
|
|
|
EXPECT_TRUE(ImportedD->getAnyInitializer());
|
|
|
|
EXPECT_TRUE(ImportedD->getDefinition());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportVariables, InitAndDefinitionAreInTheFromContext) {
|
|
|
|
auto StructA =
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
static const int a;
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(StructA, Lang_CXX);
|
|
|
|
Decl *FromTU = getTuDecl(std::string(StructA) + "const int A::a = 1 + 2;",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
|
|
|
|
auto *FromDDeclarationOnly = FirstDeclMatcher<VarDecl>().match(
|
|
|
|
FromTU, varDecl(hasName("a")));
|
|
|
|
auto *FromDWithDef = LastDeclMatcher<VarDecl>().match(
|
|
|
|
FromTU, varDecl(hasName("a"))); // Decl with definition and with init.
|
|
|
|
ASSERT_EQ(FromDDeclarationOnly, FromDWithDef->getPreviousDecl());
|
|
|
|
ASSERT_FALSE(FromDDeclarationOnly->getInit());
|
|
|
|
ASSERT_FALSE(FromDDeclarationOnly->isThisDeclarationADefinition());
|
|
|
|
ASSERT_TRUE(FromDWithDef->isThisDeclarationADefinition());
|
|
|
|
ASSERT_TRUE(FromDWithDef->getInit());
|
|
|
|
|
|
|
|
auto *ToD = FirstDeclMatcher<VarDecl>().match(
|
|
|
|
ToTU, varDecl(hasName("a")));
|
|
|
|
ASSERT_FALSE(ToD->getInit());
|
|
|
|
ASSERT_FALSE(ToD->getDefinition());
|
|
|
|
|
|
|
|
auto *ImportedD = cast<VarDecl>(Import(FromDWithDef, Lang_CXX11));
|
|
|
|
EXPECT_TRUE(ImportedD->getAnyInitializer());
|
|
|
|
EXPECT_TRUE(ImportedD->getDefinition());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct ImportClasses : ASTImporterOptionSpecificTestBase {};
|
2018-12-17 20:42:12 +08:00
|
|
|
|
2018-12-17 21:53:12 +08:00
|
|
|
TEST_P(ImportClasses, ImportDefinitionWhenProtoIsInNestedToContext) {
|
|
|
|
Decl *ToTU = getToTuDecl("struct A { struct X *Xp; };", Lang_C);
|
|
|
|
Decl *FromTU1 = getTuDecl("struct X {};", Lang_C, "input1.cc");
|
|
|
|
auto Pattern = recordDecl(hasName("X"), unless(isImplicit()));
|
|
|
|
auto ToProto = FirstDeclMatcher<RecordDecl>().match(ToTU, Pattern);
|
|
|
|
auto FromDef = FirstDeclMatcher<RecordDecl>().match(FromTU1, Pattern);
|
|
|
|
|
|
|
|
Decl *ImportedDef = Import(FromDef, Lang_C);
|
|
|
|
|
|
|
|
EXPECT_NE(ImportedDef, ToProto);
|
|
|
|
EXPECT_EQ(DeclCounter<RecordDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
auto ToDef = LastDeclMatcher<RecordDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_TRUE(ImportedDef == ToDef);
|
|
|
|
EXPECT_TRUE(ToDef->isThisDeclarationADefinition());
|
|
|
|
EXPECT_FALSE(ToProto->isThisDeclarationADefinition());
|
|
|
|
EXPECT_EQ(ToDef->getPreviousDecl(), ToProto);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportClasses, ImportDefinitionWhenProtoIsInNestedToContextCXX) {
|
|
|
|
Decl *ToTU = getToTuDecl("struct A { struct X *Xp; };", Lang_CXX);
|
|
|
|
Decl *FromTU1 = getTuDecl("struct X {};", Lang_CXX, "input1.cc");
|
|
|
|
auto Pattern = recordDecl(hasName("X"), unless(isImplicit()));
|
|
|
|
auto ToProto = FirstDeclMatcher<RecordDecl>().match(ToTU, Pattern);
|
|
|
|
auto FromDef = FirstDeclMatcher<RecordDecl>().match(FromTU1, Pattern);
|
|
|
|
|
|
|
|
Decl *ImportedDef = Import(FromDef, Lang_CXX);
|
|
|
|
|
|
|
|
EXPECT_NE(ImportedDef, ToProto);
|
|
|
|
EXPECT_EQ(DeclCounter<RecordDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
auto ToDef = LastDeclMatcher<RecordDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_TRUE(ImportedDef == ToDef);
|
|
|
|
EXPECT_TRUE(ToDef->isThisDeclarationADefinition());
|
|
|
|
EXPECT_FALSE(ToProto->isThisDeclarationADefinition());
|
|
|
|
EXPECT_EQ(ToDef->getPreviousDecl(), ToProto);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportClasses, ImportNestedPrototypeThenDefinition) {
|
|
|
|
Decl *FromTU0 = getTuDecl("struct A { struct X *Xp; };", Lang_C, "input0.cc");
|
|
|
|
Decl *FromTU1 = getTuDecl("struct X {};", Lang_C, "input1.cc");
|
|
|
|
auto Pattern = recordDecl(hasName("X"), unless(isImplicit()));
|
|
|
|
auto FromProto = FirstDeclMatcher<RecordDecl>().match(FromTU0, Pattern);
|
|
|
|
auto FromDef = FirstDeclMatcher<RecordDecl>().match(FromTU1, Pattern);
|
|
|
|
|
|
|
|
Decl *ImportedProto = Import(FromProto, Lang_C);
|
|
|
|
Decl *ImportedDef = Import(FromDef, Lang_C);
|
|
|
|
Decl *ToTU = ImportedDef->getTranslationUnitDecl();
|
|
|
|
|
|
|
|
EXPECT_NE(ImportedDef, ImportedProto);
|
|
|
|
EXPECT_EQ(DeclCounter<RecordDecl>().match(ToTU, Pattern), 2u);
|
|
|
|
auto ToProto = FirstDeclMatcher<RecordDecl>().match(ToTU, Pattern);
|
|
|
|
auto ToDef = LastDeclMatcher<RecordDecl>().match(ToTU, Pattern);
|
|
|
|
EXPECT_TRUE(ImportedDef == ToDef);
|
|
|
|
EXPECT_TRUE(ImportedProto == ToProto);
|
|
|
|
EXPECT_TRUE(ToDef->isThisDeclarationADefinition());
|
|
|
|
EXPECT_FALSE(ToProto->isThisDeclarationADefinition());
|
|
|
|
EXPECT_EQ(ToDef->getPreviousDecl(), ToProto);
|
|
|
|
}
|
|
|
|
|
2018-12-17 20:42:12 +08:00
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct ImportFriendClasses : ASTImporterOptionSpecificTestBase {};
|
2018-12-17 20:42:12 +08:00
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, ImportOfFriendRecordDoesNotMergeDefinition) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class A {
|
|
|
|
template <int I> class F {};
|
|
|
|
class X {
|
|
|
|
template <int I> friend class F;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
auto *FromClass = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("F"), isDefinition()));
|
|
|
|
auto *FromFriendClass = LastDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("F")));
|
|
|
|
|
|
|
|
ASSERT_TRUE(FromClass);
|
|
|
|
ASSERT_TRUE(FromFriendClass);
|
|
|
|
ASSERT_NE(FromClass, FromFriendClass);
|
|
|
|
ASSERT_EQ(FromFriendClass->getDefinition(), FromClass);
|
|
|
|
ASSERT_EQ(FromFriendClass->getPreviousDecl(), FromClass);
|
|
|
|
ASSERT_EQ(FromFriendClass->getDescribedClassTemplate()->getPreviousDecl(),
|
|
|
|
FromClass->getDescribedClassTemplate());
|
|
|
|
|
|
|
|
auto *ToClass = cast<CXXRecordDecl>(Import(FromClass, Lang_CXX));
|
|
|
|
auto *ToFriendClass = cast<CXXRecordDecl>(Import(FromFriendClass, Lang_CXX));
|
|
|
|
|
|
|
|
EXPECT_TRUE(ToClass);
|
|
|
|
EXPECT_TRUE(ToFriendClass);
|
|
|
|
EXPECT_NE(ToClass, ToFriendClass);
|
|
|
|
EXPECT_EQ(ToFriendClass->getDefinition(), ToClass);
|
|
|
|
EXPECT_EQ(ToFriendClass->getPreviousDecl(), ToClass);
|
|
|
|
EXPECT_EQ(ToFriendClass->getDescribedClassTemplate()->getPreviousDecl(),
|
|
|
|
ToClass->getDescribedClassTemplate());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, ImportOfRecursiveFriendClass) {
|
|
|
|
Decl *FromTu = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class declToImport {
|
|
|
|
friend class declToImport;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input.cc");
|
|
|
|
|
|
|
|
auto *FromD = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTu, cxxRecordDecl(hasName("declToImport")));
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX);
|
|
|
|
auto Pattern = cxxRecordDecl(has(friendDecl()));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(FromD, Pattern));
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(ToD, Pattern));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, ImportOfRecursiveFriendClassTemplate) {
|
|
|
|
Decl *FromTu = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template<class A> class declToImport {
|
|
|
|
template<class A1> friend class declToImport;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input.cc");
|
|
|
|
|
|
|
|
auto *FromD =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(FromTu, classTemplateDecl());
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX);
|
|
|
|
|
|
|
|
auto Pattern = classTemplateDecl(
|
|
|
|
has(cxxRecordDecl(has(friendDecl(has(classTemplateDecl()))))));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(FromD, Pattern));
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(ToD, Pattern));
|
|
|
|
|
|
|
|
auto *Class =
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(ToD, classTemplateDecl());
|
|
|
|
auto *Friend = FirstDeclMatcher<FriendDecl>().match(ToD, friendDecl());
|
|
|
|
EXPECT_NE(Friend->getFriendDecl(), Class);
|
|
|
|
EXPECT_EQ(Friend->getFriendDecl()->getPreviousDecl(), Class);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, ProperPrevDeclForClassTemplateDecls) {
|
|
|
|
auto Pattern = classTemplateSpecializationDecl(hasName("X"));
|
|
|
|
|
|
|
|
ClassTemplateSpecializationDecl *Imported1;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl("template<class T> class X;"
|
|
|
|
"struct Y { friend class X<int>; };",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, Pattern);
|
|
|
|
|
|
|
|
Imported1 = cast<ClassTemplateSpecializationDecl>(Import(FromD, Lang_CXX));
|
|
|
|
}
|
|
|
|
ClassTemplateSpecializationDecl *Imported2;
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl("template<class T> class X;"
|
|
|
|
"template<> class X<int>{};"
|
|
|
|
"struct Z { friend class X<int>; };",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, Pattern);
|
|
|
|
|
|
|
|
Imported2 = cast<ClassTemplateSpecializationDecl>(Import(FromD, Lang_CXX));
|
|
|
|
}
|
|
|
|
|
|
|
|
Decl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_EQ(DeclCounter<ClassTemplateSpecializationDecl>().match(ToTU, Pattern),
|
|
|
|
2u);
|
|
|
|
ASSERT_TRUE(Imported2->getPreviousDecl());
|
|
|
|
EXPECT_EQ(Imported2->getPreviousDecl(), Imported1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, TypeForDeclShouldBeSetInTemplated) {
|
|
|
|
Decl *FromTU0 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
class Y;
|
|
|
|
};
|
|
|
|
class X::Y {
|
|
|
|
template <typename T>
|
|
|
|
friend class F; // The decl context of F is the global namespace.
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *Fwd = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU0, classTemplateDecl(hasName("F")));
|
|
|
|
auto *Imported0 = cast<ClassTemplateDecl>(Import(Fwd, Lang_CXX));
|
|
|
|
Decl *FromTU1 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class F {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *Definition = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU1, classTemplateDecl(hasName("F")));
|
|
|
|
auto *Imported1 = cast<ClassTemplateDecl>(Import(Definition, Lang_CXX));
|
|
|
|
EXPECT_EQ(Imported0->getTemplatedDecl()->getTypeForDecl(),
|
|
|
|
Imported1->getTemplatedDecl()->getTypeForDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, DeclsFromFriendsShouldBeInRedeclChains) {
|
|
|
|
Decl *From, *To;
|
|
|
|
std::tie(From, To) =
|
|
|
|
getImportedDecl("class declToImport {};", Lang_CXX,
|
|
|
|
"class Y { friend class declToImport; };", Lang_CXX);
|
|
|
|
auto *Imported = cast<CXXRecordDecl>(To);
|
|
|
|
|
|
|
|
EXPECT_TRUE(Imported->getPreviousDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses,
|
|
|
|
ImportOfClassTemplateDefinitionShouldConnectToFwdFriend) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
class Y;
|
|
|
|
};
|
|
|
|
class X::Y {
|
|
|
|
template <typename T>
|
|
|
|
friend class F; // The decl context of F is the global namespace.
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *ToDecl = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("F")));
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class F {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *Definition = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("F")));
|
|
|
|
auto *ImportedDef = cast<ClassTemplateDecl>(Import(Definition, Lang_CXX));
|
|
|
|
EXPECT_TRUE(ImportedDef->getPreviousDecl());
|
|
|
|
EXPECT_EQ(ToDecl, ImportedDef->getPreviousDecl());
|
|
|
|
EXPECT_EQ(ToDecl->getTemplatedDecl(),
|
|
|
|
ImportedDef->getTemplatedDecl()->getPreviousDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses,
|
|
|
|
ImportOfClassTemplateDefinitionAndFwdFriendShouldBeLinked) {
|
|
|
|
Decl *FromTU0 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
class Y;
|
|
|
|
};
|
|
|
|
class X::Y {
|
|
|
|
template <typename T>
|
|
|
|
friend class F; // The decl context of F is the global namespace.
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *Fwd = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU0, classTemplateDecl(hasName("F")));
|
|
|
|
auto *ImportedFwd = cast<ClassTemplateDecl>(Import(Fwd, Lang_CXX));
|
|
|
|
Decl *FromTU1 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class F {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *Definition = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU1, classTemplateDecl(hasName("F")));
|
|
|
|
auto *ImportedDef = cast<ClassTemplateDecl>(Import(Definition, Lang_CXX));
|
|
|
|
EXPECT_TRUE(ImportedDef->getPreviousDecl());
|
|
|
|
EXPECT_EQ(ImportedFwd, ImportedDef->getPreviousDecl());
|
|
|
|
EXPECT_EQ(ImportedFwd->getTemplatedDecl(),
|
|
|
|
ImportedDef->getTemplatedDecl()->getPreviousDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFriendClasses, ImportOfClassDefinitionAndFwdFriendShouldBeLinked) {
|
|
|
|
Decl *FromTU0 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
class Y;
|
|
|
|
};
|
|
|
|
class X::Y {
|
|
|
|
friend class F; // The decl context of F is the global namespace.
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *Friend = FirstDeclMatcher<FriendDecl>().match(FromTU0, friendDecl());
|
|
|
|
QualType FT = Friend->getFriendType()->getType();
|
|
|
|
FT = FromTU0->getASTContext().getCanonicalType(FT);
|
|
|
|
auto *Fwd = cast<TagType>(FT)->getDecl();
|
|
|
|
auto *ImportedFwd = Import(Fwd, Lang_CXX);
|
|
|
|
Decl *FromTU1 = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class F {};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
auto *Definition = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU1, cxxRecordDecl(hasName("F")));
|
|
|
|
auto *ImportedDef = Import(Definition, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ImportedDef->getPreviousDecl());
|
|
|
|
EXPECT_EQ(ImportedFwd, ImportedDef->getPreviousDecl());
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, FriendFunInClassTemplate) {
|
2018-12-17 21:53:12 +08:00
|
|
|
auto *Code = R"(
|
|
|
|
template <class T>
|
|
|
|
struct X {
|
|
|
|
friend void foo(){}
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(Code, Lang_CXX);
|
|
|
|
auto *ToFoo = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
ToTU, functionDecl(hasName("foo")));
|
|
|
|
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(Code, Lang_CXX, "input.cc");
|
|
|
|
auto *FromFoo = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("foo")));
|
|
|
|
auto *ImportedFoo = Import(FromFoo, Lang_CXX);
|
|
|
|
EXPECT_EQ(ImportedFoo, ToFoo);
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct DeclContextTest : ASTImporterOptionSpecificTestBase {};
|
2018-05-18 17:08:47 +08:00
|
|
|
|
|
|
|
TEST_P(DeclContextTest, removeDeclOfClassTemplateSpecialization) {
|
|
|
|
Decl *TU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
namespace NS {
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
struct S {};
|
|
|
|
template struct S<int>;
|
|
|
|
|
|
|
|
inline namespace INS {
|
|
|
|
template <typename T>
|
|
|
|
struct S {};
|
|
|
|
template struct S<int>;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
)", Lang_CXX11, "input0.cc");
|
|
|
|
auto *NS = FirstDeclMatcher<NamespaceDecl>().match(
|
|
|
|
TU, namespaceDecl());
|
|
|
|
auto *Spec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
TU, classTemplateSpecializationDecl());
|
|
|
|
ASSERT_TRUE(NS->containsDecl(Spec));
|
|
|
|
|
|
|
|
NS->removeDecl(Spec);
|
|
|
|
EXPECT_FALSE(NS->containsDecl(Spec));
|
|
|
|
}
|
|
|
|
|
2018-12-17 20:42:12 +08:00
|
|
|
TEST_P(DeclContextTest,
|
|
|
|
removeDeclShouldNotFailEvenIfWeHaveExternalVisibleStorage) {
|
|
|
|
Decl *TU = getTuDecl("extern int A; int A;", Lang_CXX);
|
|
|
|
auto *A0 = FirstDeclMatcher<VarDecl>().match(TU, varDecl(hasName("A")));
|
|
|
|
auto *A1 = LastDeclMatcher<VarDecl>().match(TU, varDecl(hasName("A")));
|
|
|
|
|
|
|
|
// Investigate the list.
|
|
|
|
auto *DC = A0->getDeclContext();
|
|
|
|
ASSERT_TRUE(DC->containsDecl(A0));
|
|
|
|
ASSERT_TRUE(DC->containsDecl(A1));
|
|
|
|
|
|
|
|
// Investigate the lookup table.
|
|
|
|
auto *Map = DC->getLookupPtr();
|
|
|
|
ASSERT_TRUE(Map);
|
|
|
|
auto I = Map->find(A0->getDeclName());
|
|
|
|
ASSERT_NE(I, Map->end());
|
|
|
|
StoredDeclsList &L = I->second;
|
|
|
|
// The lookup table contains the most recent decl of A.
|
|
|
|
ASSERT_NE(L.getAsDecl(), A0);
|
|
|
|
ASSERT_EQ(L.getAsDecl(), A1);
|
|
|
|
|
|
|
|
ASSERT_TRUE(L.getAsDecl());
|
|
|
|
// Simulate the private function DeclContext::reconcileExternalVisibleStorage.
|
|
|
|
// The point here is to have a Vec with only one element, which is not the
|
|
|
|
// one we are going to delete from the DC later.
|
|
|
|
L.setHasExternalDecls();
|
|
|
|
ASSERT_TRUE(L.getAsVector());
|
|
|
|
ASSERT_EQ(1u, L.getAsVector()->size());
|
|
|
|
|
|
|
|
// This asserts in the old implementation.
|
|
|
|
DC->removeDecl(A0);
|
|
|
|
EXPECT_FALSE(DC->containsDecl(A0));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct ImportFunctionTemplateSpecializations
|
|
|
|
: ASTImporterOptionSpecificTestBase {};
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplateSpecializations,
|
|
|
|
TUshouldNotContainFunctionTemplateImplicitInstantiation) {
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
int f() { return 0; }
|
|
|
|
void foo() { f<int>(); }
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
// Check that the function template instantiation is NOT the child of the TU.
|
|
|
|
auto Pattern = translationUnitDecl(
|
|
|
|
unless(has(functionDecl(hasName("f"), isTemplateInstantiation()))));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(FromTU, Pattern));
|
|
|
|
|
|
|
|
auto *Foo = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("foo")));
|
|
|
|
ASSERT_TRUE(Import(Foo, Lang_CXX));
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(ToTU, Pattern));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplateSpecializations,
|
|
|
|
TUshouldNotContainFunctionTemplateExplicitInstantiation) {
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
int f() { return 0; }
|
|
|
|
template int f<int>();
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
// Check that the function template instantiation is NOT the child of the TU.
|
|
|
|
auto Instantiation = functionDecl(hasName("f"), isTemplateInstantiation());
|
|
|
|
auto Pattern = translationUnitDecl(unless(has(Instantiation)));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(FromTU, Pattern));
|
|
|
|
|
|
|
|
ASSERT_TRUE(
|
|
|
|
Import(FirstDeclMatcher<Decl>().match(FromTU, Instantiation), Lang_CXX));
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(ToTU, Pattern));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplateSpecializations,
|
|
|
|
TUshouldContainFunctionTemplateSpecialization) {
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
int f() { return 0; }
|
|
|
|
template <> int f<int>() { return 4; }
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
// Check that the function template specialization is the child of the TU.
|
|
|
|
auto Specialization =
|
|
|
|
functionDecl(hasName("f"), isExplicitTemplateSpecialization());
|
|
|
|
auto Pattern = translationUnitDecl(has(Specialization));
|
|
|
|
ASSERT_TRUE(MatchVerifier<Decl>{}.match(FromTU, Pattern));
|
|
|
|
|
|
|
|
ASSERT_TRUE(
|
|
|
|
Import(FirstDeclMatcher<Decl>().match(FromTU, Specialization), Lang_CXX));
|
|
|
|
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_TRUE(MatchVerifier<Decl>{}.match(ToTU, Pattern));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplateSpecializations,
|
|
|
|
FunctionTemplateSpecializationRedeclChain) {
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
int f() { return 0; }
|
|
|
|
template <> int f<int>() { return 4; }
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
auto Spec = functionDecl(hasName("f"), isExplicitTemplateSpecialization(),
|
|
|
|
hasParent(translationUnitDecl()));
|
|
|
|
auto *FromSpecD = FirstDeclMatcher<Decl>().match(FromTU, Spec);
|
|
|
|
{
|
|
|
|
auto *TU = FromTU;
|
|
|
|
auto *SpecD = FromSpecD;
|
|
|
|
auto *TemplateD = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
TU, functionTemplateDecl());
|
|
|
|
auto *FirstSpecD = *(TemplateD->spec_begin());
|
|
|
|
ASSERT_EQ(SpecD, FirstSpecD);
|
|
|
|
ASSERT_TRUE(SpecD->getPreviousDecl());
|
|
|
|
ASSERT_FALSE(cast<FunctionDecl>(SpecD->getPreviousDecl())
|
|
|
|
->doesThisDeclarationHaveABody());
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT_TRUE(Import(FromSpecD, Lang_CXX));
|
|
|
|
|
|
|
|
{
|
|
|
|
auto *TU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
auto *SpecD = FirstDeclMatcher<Decl>().match(TU, Spec);
|
|
|
|
auto *TemplateD = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
TU, functionTemplateDecl());
|
|
|
|
auto *FirstSpecD = *(TemplateD->spec_begin());
|
|
|
|
EXPECT_EQ(SpecD, FirstSpecD);
|
|
|
|
ASSERT_TRUE(SpecD->getPreviousDecl());
|
|
|
|
EXPECT_FALSE(cast<FunctionDecl>(SpecD->getPreviousDecl())
|
|
|
|
->doesThisDeclarationHaveABody());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ImportFunctionTemplateSpecializations,
|
|
|
|
MatchNumberOfFunctionTemplateSpecializations) {
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T> constexpr int f() { return 0; }
|
|
|
|
template <> constexpr int f<int>() { return 4; }
|
|
|
|
void foo() {
|
|
|
|
static_assert(f<char>() == 0, "");
|
|
|
|
static_assert(f<int>() == 4, "");
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("foo")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX11);
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_EQ(
|
|
|
|
DeclCounter<FunctionDecl>().match(FromTU, functionDecl(hasName("f"))),
|
|
|
|
DeclCounter<FunctionDecl>().match(ToTU, functionDecl(hasName("f"))));
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
2018-11-26 23:54:08 +08:00
|
|
|
ImportShouldNotReportFalseODRErrorWhenRecordIsBeingDefined) {
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B;
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
auto *FromD = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("B")));
|
|
|
|
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct B {
|
|
|
|
void f();
|
|
|
|
B* b;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
FunctionDecl *FromD = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
Import(FromD, Lang_CXX);
|
|
|
|
auto *FromCTD = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("B")));
|
|
|
|
auto *ToCTD = cast<ClassTemplateDecl>(Import(FromCTD, Lang_CXX));
|
|
|
|
EXPECT_TRUE(ToCTD->isThisDeclarationADefinition());
|
|
|
|
|
|
|
|
// We expect no (ODR) warning during the import.
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_EQ(0u, ToTU->getASTContext().getDiagnostics().getNumWarnings());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportingTypedefShouldImportTheCompleteType) {
|
2018-11-27 17:51:36 +08:00
|
|
|
// We already have an incomplete underlying type in the "To" context.
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct S {
|
|
|
|
void foo();
|
|
|
|
};
|
|
|
|
using U = S<int>;
|
|
|
|
)";
|
|
|
|
Decl *ToTU = getToTuDecl(Code, Lang_CXX11);
|
|
|
|
auto *ToD = FirstDeclMatcher<TypedefNameDecl>().match(ToTU,
|
|
|
|
typedefNameDecl(hasName("U")));
|
|
|
|
ASSERT_TRUE(ToD->getUnderlyingType()->isIncompleteType());
|
|
|
|
|
|
|
|
// The "From" context has the same typedef, but the underlying type is
|
|
|
|
// complete this time.
|
|
|
|
Decl *FromTU = getTuDecl(std::string(Code) +
|
|
|
|
R"(
|
|
|
|
void foo(U* u) {
|
|
|
|
u->foo();
|
|
|
|
}
|
|
|
|
)", Lang_CXX11);
|
|
|
|
auto *FromD = FirstDeclMatcher<TypedefNameDecl>().match(FromTU,
|
|
|
|
typedefNameDecl(hasName("U")));
|
|
|
|
ASSERT_FALSE(FromD->getUnderlyingType()->isIncompleteType());
|
|
|
|
|
|
|
|
// The imported type should be complete.
|
|
|
|
auto *ImportedD = cast<TypedefNameDecl>(Import(FromD, Lang_CXX11));
|
|
|
|
EXPECT_FALSE(ImportedD->getUnderlyingType()->isIncompleteType());
|
|
|
|
}
|
|
|
|
|
2019-05-07 18:55:11 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportTemplateParameterLists) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
template<class T>
|
|
|
|
int f() { return 0; }
|
|
|
|
template <> int f<int>() { return 4; }
|
|
|
|
)";
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(Code, Lang_CXX);
|
|
|
|
auto *FromD = FirstDeclMatcher<FunctionDecl>().match(FromTU,
|
|
|
|
functionDecl(hasName("f"), isExplicitTemplateSpecialization()));
|
2019-05-07 22:53:04 +08:00
|
|
|
ASSERT_EQ(FromD->getNumTemplateParameterLists(), 1u);
|
2019-05-07 18:55:11 +08:00
|
|
|
|
|
|
|
auto *ToD = Import(FromD, Lang_CXX);
|
|
|
|
// The template parameter list should exist.
|
2019-05-07 22:53:04 +08:00
|
|
|
EXPECT_EQ(ToD->getNumTemplateParameterLists(), 1u);
|
2019-05-07 18:55:11 +08:00
|
|
|
}
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
struct ASTImporterLookupTableTest : ASTImporterOptionSpecificTestBase {};
|
2018-12-17 21:53:12 +08:00
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, OneDecl) {
|
|
|
|
auto *ToTU = getToTuDecl("int a;", Lang_CXX);
|
|
|
|
auto *D = FirstDeclMatcher<VarDecl>().match(ToTU, varDecl(hasName("a")));
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto Res = LT.lookup(ToTU, D->getDeclName());
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), D);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Decl *findInDeclListOfDC(DeclContext *DC, DeclarationName Name) {
|
|
|
|
for (Decl *D : DC->decls()) {
|
|
|
|
if (auto *ND = dyn_cast<NamedDecl>(D))
|
|
|
|
if (ND->getDeclName() == Name)
|
|
|
|
return ND;
|
|
|
|
}
|
|
|
|
return nullptr;
|
2018-12-18 16:36:16 +08:00
|
|
|
}
|
2018-12-17 21:53:12 +08:00
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
FriendWhichIsnotFoundByNormalLookupShouldBeFoundByImporterSpecificLookup) {
|
|
|
|
auto *Code = R"(
|
|
|
|
template <class T>
|
|
|
|
struct X {
|
|
|
|
friend void foo(){}
|
|
|
|
};
|
|
|
|
)";
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(Code, Lang_CXX);
|
|
|
|
auto *X = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("X")));
|
|
|
|
auto *Foo = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
ToTU, functionDecl(hasName("foo")));
|
|
|
|
DeclContext *FooDC = Foo->getDeclContext();
|
|
|
|
DeclContext *FooLexicalDC = Foo->getLexicalDeclContext();
|
|
|
|
ASSERT_EQ(cast<Decl>(FooLexicalDC), X->getTemplatedDecl());
|
|
|
|
ASSERT_EQ(cast<Decl>(FooDC), ToTU);
|
|
|
|
DeclarationName FooName = Foo->getDeclName();
|
|
|
|
|
|
|
|
// Cannot find in the LookupTable of its DC (TUDecl)
|
|
|
|
SmallVector<NamedDecl *, 2> FoundDecls;
|
|
|
|
FooDC->getRedeclContext()->localUncachedLookup(FooName, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 0u);
|
|
|
|
|
|
|
|
// Cannot find in the LookupTable of its LexicalDC (X)
|
|
|
|
FooLexicalDC->getRedeclContext()->localUncachedLookup(FooName, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 0u);
|
|
|
|
|
|
|
|
// Can't find in the list of Decls of the DC.
|
|
|
|
EXPECT_EQ(findInDeclListOfDC(FooDC, FooName), nullptr);
|
|
|
|
|
|
|
|
// Can't find in the list of Decls of the LexicalDC
|
|
|
|
EXPECT_EQ(findInDeclListOfDC(FooLexicalDC, FooName), nullptr);
|
|
|
|
|
|
|
|
// ASTImporter specific lookup finds it.
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto Res = LT.lookup(FooDC, Foo->getDeclName());
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), Foo);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
FwdDeclStructShouldBeFoundByImporterSpecificLookup) {
|
|
|
|
TranslationUnitDecl *ToTU =
|
|
|
|
getToTuDecl("struct A { struct Foo *p; };", Lang_C);
|
|
|
|
auto *Foo =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(ToTU, recordDecl(hasName("Foo")));
|
|
|
|
auto *A =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(ToTU, recordDecl(hasName("A")));
|
|
|
|
DeclContext *FooDC = Foo->getDeclContext();
|
|
|
|
DeclContext *FooLexicalDC = Foo->getLexicalDeclContext();
|
|
|
|
ASSERT_EQ(cast<Decl>(FooLexicalDC), A);
|
|
|
|
ASSERT_EQ(cast<Decl>(FooDC), ToTU);
|
|
|
|
DeclarationName FooName = Foo->getDeclName();
|
|
|
|
|
|
|
|
// Cannot find in the LookupTable of its DC (TUDecl).
|
|
|
|
SmallVector<NamedDecl *, 2> FoundDecls;
|
|
|
|
FooDC->getRedeclContext()->localUncachedLookup(FooName, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 0u);
|
|
|
|
|
|
|
|
// Cannot find in the LookupTable of its LexicalDC (A).
|
|
|
|
FooLexicalDC->getRedeclContext()->localUncachedLookup(FooName, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 0u);
|
|
|
|
|
|
|
|
// Can't find in the list of Decls of the DC.
|
|
|
|
EXPECT_EQ(findInDeclListOfDC(FooDC, FooName), nullptr);
|
|
|
|
|
|
|
|
// Can find in the list of Decls of the LexicalDC.
|
|
|
|
EXPECT_EQ(findInDeclListOfDC(FooLexicalDC, FooName), Foo);
|
|
|
|
|
|
|
|
// ASTImporter specific lookup finds it.
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto Res = LT.lookup(FooDC, Foo->getDeclName());
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), Foo);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupFindsNamesInDifferentDC) {
|
|
|
|
TranslationUnitDecl *ToTU =
|
|
|
|
getToTuDecl("int V; struct A { int V; }; struct B { int V; };", Lang_C);
|
|
|
|
DeclarationName VName = FirstDeclMatcher<VarDecl>()
|
|
|
|
.match(ToTU, varDecl(hasName("V")))
|
|
|
|
->getDeclName();
|
|
|
|
auto *A =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(ToTU, recordDecl(hasName("A")));
|
|
|
|
auto *B =
|
|
|
|
FirstDeclMatcher<RecordDecl>().match(ToTU, recordDecl(hasName("B")));
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
|
|
|
|
auto Res = LT.lookup(cast<DeclContext>(A), VName);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
ToTU, fieldDecl(hasName("V"),
|
|
|
|
hasParent(recordDecl(hasName("A"))))));
|
|
|
|
Res = LT.lookup(cast<DeclContext>(B), VName);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
ToTU, fieldDecl(hasName("V"),
|
|
|
|
hasParent(recordDecl(hasName("B"))))));
|
|
|
|
Res = LT.lookup(ToTU, VName);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), FirstDeclMatcher<VarDecl>().match(
|
|
|
|
ToTU, varDecl(hasName("V"),
|
|
|
|
hasParent(translationUnitDecl()))));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupFindsOverloadedNames) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
void foo();
|
|
|
|
void foo(int);
|
|
|
|
void foo(int, int);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *F0 = FirstDeclMatcher<FunctionDecl>().match(ToTU, functionDecl());
|
|
|
|
auto *F2 = LastDeclMatcher<FunctionDecl>().match(ToTU, functionDecl());
|
|
|
|
DeclarationName Name = F0->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 3u);
|
|
|
|
EXPECT_EQ(Res.count(F0), 1u);
|
|
|
|
EXPECT_EQ(Res.count(F2), 1u);
|
|
|
|
}
|
|
|
|
|
2019-02-08 17:19:34 +08:00
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
DifferentOperatorsShouldHaveDifferentResultSet) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X{};
|
|
|
|
void operator+(X, X);
|
|
|
|
void operator-(X, X);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *FPlus = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
ToTU, functionDecl(hasOverloadedOperatorName("+")));
|
|
|
|
auto *FMinus = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
ToTU, functionDecl(hasOverloadedOperatorName("-")));
|
|
|
|
DeclarationName NamePlus = FPlus->getDeclName();
|
|
|
|
auto ResPlus = LT.lookup(ToTU, NamePlus);
|
|
|
|
EXPECT_EQ(ResPlus.size(), 1u);
|
|
|
|
EXPECT_EQ(ResPlus.count(FPlus), 1u);
|
|
|
|
EXPECT_EQ(ResPlus.count(FMinus), 0u);
|
|
|
|
DeclarationName NameMinus = FMinus->getDeclName();
|
|
|
|
auto ResMinus = LT.lookup(ToTU, NameMinus);
|
|
|
|
EXPECT_EQ(ResMinus.size(), 1u);
|
|
|
|
EXPECT_EQ(ResMinus.count(FMinus), 1u);
|
|
|
|
EXPECT_EQ(ResMinus.count(FPlus), 0u);
|
|
|
|
EXPECT_NE(*ResMinus.begin(), *ResPlus.begin());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupDeclNamesFromDifferentTUs) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X {};
|
|
|
|
void operator+(X, X);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *ToPlus = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
ToTU, functionDecl(hasOverloadedOperatorName("+")));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X {};
|
|
|
|
void operator+(X, X);
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromPlus = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasOverloadedOperatorName("+")));
|
|
|
|
|
|
|
|
// FromPlus have a different TU, thus its DeclarationName is different too.
|
|
|
|
ASSERT_NE(ToPlus->getDeclName(), FromPlus->getDeclName());
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto Res = LT.lookup(ToTU, ToPlus->getDeclName());
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), ToPlus);
|
|
|
|
|
|
|
|
// FromPlus have a different TU, thus its DeclarationName is different too.
|
|
|
|
Res = LT.lookup(ToTU, FromPlus->getDeclName());
|
|
|
|
ASSERT_EQ(Res.size(), 0u);
|
|
|
|
}
|
|
|
|
|
[ASTImporter] Fix unhandled cases in ASTImporterLookupTable
Summary:
In most cases the FriendDecl contains the declaration of the befriended
class as a child node, so it is discovered during the recursive
visitation. However, there are cases when the befriended class is not a
child, thus it must be fetched explicitly from the FriendDecl, and only
then can we add it to the lookup table.
(Note, this does affect only CTU and does not affect LLDB, because we
cannot and do not use the ASTImporterLookupTable in LLDB.)
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62064
llvm-svn: 363062
2019-06-11 21:35:25 +08:00
|
|
|
static const RecordDecl *getRecordDeclOfFriend(FriendDecl *FD) {
|
|
|
|
QualType Ty = FD->getFriendType()->getType().getCanonicalType();
|
|
|
|
return cast<RecordType>(Ty)->getDecl();
|
2018-12-17 21:53:12 +08:00
|
|
|
}
|
|
|
|
|
[ASTImporter] Fix unhandled cases in ASTImporterLookupTable
Summary:
In most cases the FriendDecl contains the declaration of the befriended
class as a child node, so it is discovered during the recursive
visitation. However, there are cases when the befriended class is not a
child, thus it must be fetched explicitly from the FriendDecl, and only
then can we add it to the lookup table.
(Note, this does affect only CTU and does not affect LLDB, because we
cannot and do not use the ASTImporterLookupTable in LLDB.)
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62064
llvm-svn: 363062
2019-06-11 21:35:25 +08:00
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
LookupFindsFwdFriendClassDeclWithElaboratedType) {
|
2018-12-17 21:53:12 +08:00
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class Y { friend class F; };
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
// In this case, the CXXRecordDecl is hidden, the FriendDecl is not a parent.
|
|
|
|
// So we must dig up the underlying CXXRecordDecl.
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *FriendD = FirstDeclMatcher<FriendDecl>().match(ToTU, friendDecl());
|
|
|
|
const RecordDecl *RD = getRecordDeclOfFriend(FriendD);
|
|
|
|
auto *Y = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("Y")));
|
|
|
|
|
|
|
|
DeclarationName Name = RD->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), RD);
|
|
|
|
|
|
|
|
Res = LT.lookup(Y, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 0u);
|
|
|
|
}
|
|
|
|
|
[ASTImporter] Fix unhandled cases in ASTImporterLookupTable
Summary:
In most cases the FriendDecl contains the declaration of the befriended
class as a child node, so it is discovered during the recursive
visitation. However, there are cases when the befriended class is not a
child, thus it must be fetched explicitly from the FriendDecl, and only
then can we add it to the lookup table.
(Note, this does affect only CTU and does not affect LLDB, because we
cannot and do not use the ASTImporterLookupTable in LLDB.)
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62064
llvm-svn: 363062
2019-06-11 21:35:25 +08:00
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
LookupFindsFwdFriendClassDeclWithUnelaboratedType) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class F;
|
|
|
|
class Y { friend F; };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
|
|
|
|
// In this case, the CXXRecordDecl is hidden, the FriendDecl is not a parent.
|
|
|
|
// So we must dig up the underlying CXXRecordDecl.
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *FriendD = FirstDeclMatcher<FriendDecl>().match(ToTU, friendDecl());
|
|
|
|
const RecordDecl *RD = getRecordDeclOfFriend(FriendD);
|
|
|
|
auto *Y = FirstDeclMatcher<CXXRecordDecl>().match(ToTU, cxxRecordDecl(hasName("Y")));
|
|
|
|
|
|
|
|
DeclarationName Name = RD->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), RD);
|
|
|
|
|
|
|
|
Res = LT.lookup(Y, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 0u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
LookupFindsFriendClassDeclWithTypeAliasDoesNotAssert) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class F;
|
|
|
|
using alias_of_f = F;
|
|
|
|
class Y { friend alias_of_f; };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
|
|
|
|
// ASTImporterLookupTable constructor handles using declarations correctly,
|
|
|
|
// no assert is expected.
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
|
|
|
|
auto *Alias = FirstDeclMatcher<TypeAliasDecl>().match(
|
|
|
|
ToTU, typeAliasDecl(hasName("alias_of_f")));
|
|
|
|
DeclarationName Name = Alias->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.count(Alias), 1u);
|
|
|
|
}
|
|
|
|
|
2018-12-17 21:53:12 +08:00
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupFindsFwdFriendClassTemplateDecl) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class Y { template <class T> friend class F; };
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *F = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("F")));
|
|
|
|
DeclarationName Name = F->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 2u);
|
|
|
|
EXPECT_EQ(Res.count(F), 1u);
|
|
|
|
EXPECT_EQ(Res.count(F->getTemplatedDecl()), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, DependentFriendClass) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class F;
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class Y {
|
|
|
|
friend class F<T>;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *F = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("F")));
|
|
|
|
DeclarationName Name = F->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 2u);
|
|
|
|
EXPECT_EQ(Res.count(F), 1u);
|
|
|
|
EXPECT_EQ(Res.count(F->getTemplatedDecl()), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, FriendClassTemplateSpecialization) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class F;
|
|
|
|
|
|
|
|
class Y {
|
|
|
|
friend class F<int>;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *F = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("F")));
|
|
|
|
DeclarationName Name = F->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
ASSERT_EQ(Res.size(), 3u);
|
|
|
|
EXPECT_EQ(Res.count(F), 1u);
|
|
|
|
EXPECT_EQ(Res.count(F->getTemplatedDecl()), 1u);
|
|
|
|
EXPECT_EQ(Res.count(*F->spec_begin()), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupFindsFwdFriendFunctionDecl) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class Y { friend void F(); };
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *F =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(ToTU, functionDecl(hasName("F")));
|
|
|
|
DeclarationName Name = F->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), F);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest,
|
|
|
|
LookupFindsDeclsInClassTemplateSpecialization) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
struct X {
|
|
|
|
int F;
|
|
|
|
};
|
|
|
|
void foo() {
|
|
|
|
X<char> xc;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
|
|
|
|
auto *Template = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("X")));
|
|
|
|
auto *FieldInTemplate = FirstDeclMatcher<FieldDecl>().match(
|
|
|
|
ToTU,
|
|
|
|
fieldDecl(hasParent(cxxRecordDecl(hasParent(classTemplateDecl())))));
|
|
|
|
|
|
|
|
auto *Spec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
FieldDecl *FieldInSpec = *Spec->field_begin();
|
|
|
|
ASSERT_TRUE(FieldInSpec);
|
|
|
|
|
|
|
|
DeclarationName Name = FieldInSpec->getDeclName();
|
|
|
|
auto TemplateDC = cast<DeclContext>(Template->getTemplatedDecl());
|
|
|
|
|
|
|
|
SmallVector<NamedDecl *, 2> FoundDecls;
|
|
|
|
TemplateDC->getRedeclContext()->localUncachedLookup(Name, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 1u);
|
|
|
|
EXPECT_EQ(FoundDecls[0], FieldInTemplate);
|
|
|
|
|
|
|
|
auto Res = LT.lookup(TemplateDC, Name);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), FieldInTemplate);
|
|
|
|
|
|
|
|
cast<DeclContext>(Spec)->getRedeclContext()->localUncachedLookup(Name,
|
|
|
|
FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 1u);
|
|
|
|
EXPECT_EQ(FoundDecls[0], FieldInSpec);
|
|
|
|
|
|
|
|
Res = LT.lookup(cast<DeclContext>(Spec), Name);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), FieldInSpec);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupFindsFwdFriendFunctionTemplateDecl) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class Y { template <class T> friend void F(); };
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *F = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
ToTU, functionTemplateDecl(hasName("F")));
|
|
|
|
DeclarationName Name = F->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 2u);
|
|
|
|
EXPECT_EQ(Res.count(F), 1u);
|
|
|
|
EXPECT_EQ(Res.count(F->getTemplatedDecl()), 1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, MultipleBefriendingClasses) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X;
|
|
|
|
struct A {
|
|
|
|
friend struct X;
|
|
|
|
};
|
|
|
|
struct B {
|
|
|
|
friend struct X;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *X = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("X")));
|
|
|
|
auto *FriendD0 = FirstDeclMatcher<FriendDecl>().match(ToTU, friendDecl());
|
|
|
|
auto *FriendD1 = LastDeclMatcher<FriendDecl>().match(ToTU, friendDecl());
|
|
|
|
const RecordDecl *RD0 = getRecordDeclOfFriend(FriendD0);
|
|
|
|
const RecordDecl *RD1 = getRecordDeclOfFriend(FriendD1);
|
|
|
|
ASSERT_EQ(RD0, RD1);
|
|
|
|
ASSERT_EQ(RD1, X);
|
|
|
|
|
|
|
|
DeclarationName Name = X->getDeclName();
|
|
|
|
auto Res = LT.lookup(ToTU, Name);
|
|
|
|
EXPECT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), X);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, EnumConstantDecl) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
enum E {
|
|
|
|
A,
|
|
|
|
B
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_C);
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto *E = FirstDeclMatcher<EnumDecl>().match(ToTU, enumDecl(hasName("E")));
|
|
|
|
auto *A = FirstDeclMatcher<EnumConstantDecl>().match(
|
|
|
|
ToTU, enumConstantDecl(hasName("A")));
|
|
|
|
|
|
|
|
DeclarationName Name = A->getDeclName();
|
|
|
|
// Redecl context is the TU.
|
|
|
|
ASSERT_EQ(E->getRedeclContext(), ToTU);
|
|
|
|
|
|
|
|
SmallVector<NamedDecl *, 2> FoundDecls;
|
|
|
|
// Normal lookup finds in the DC.
|
|
|
|
E->localUncachedLookup(Name, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 1u);
|
|
|
|
|
|
|
|
// Normal lookup finds in the Redecl context.
|
|
|
|
ToTU->localUncachedLookup(Name, FoundDecls);
|
|
|
|
EXPECT_EQ(FoundDecls.size(), 1u);
|
|
|
|
|
|
|
|
// Import specific lookup finds in the DC.
|
|
|
|
auto Res = LT.lookup(E, Name);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), A);
|
|
|
|
|
|
|
|
// Import specific lookup finds in the Redecl context.
|
|
|
|
Res = LT.lookup(ToTU, Name);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), A);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterLookupTableTest, LookupSearchesInTheWholeRedeclChain) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
namespace N {
|
|
|
|
int A;
|
|
|
|
}
|
|
|
|
namespace N {
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *N1 =
|
|
|
|
LastDeclMatcher<NamespaceDecl>().match(ToTU, namespaceDecl(hasName("N")));
|
|
|
|
auto *A = FirstDeclMatcher<VarDecl>().match(ToTU, varDecl(hasName("A")));
|
|
|
|
DeclarationName Name = A->getDeclName();
|
|
|
|
|
|
|
|
ASTImporterLookupTable LT(*ToTU);
|
|
|
|
auto Res = LT.lookup(N1, Name);
|
|
|
|
ASSERT_EQ(Res.size(), 1u);
|
|
|
|
EXPECT_EQ(*Res.begin(), A);
|
|
|
|
}
|
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
|
2019-02-11 18:27:58 +08:00
|
|
|
// FIXME This test is disabled currently, upcoming patches will make it
|
|
|
|
// possible to enable.
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
DISABLED_RedeclChainShouldBeCorrectAmongstNamespaces) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
namespace NS {
|
|
|
|
struct X;
|
|
|
|
struct Y {
|
|
|
|
static const int I = 3;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
namespace NS {
|
|
|
|
struct X { // <--- To be imported
|
|
|
|
void method(int i = Y::I) {}
|
|
|
|
int f;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromFwd = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X"), unless(isImplicit())));
|
|
|
|
auto *FromDef = LastDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU,
|
|
|
|
cxxRecordDecl(hasName("X"), isDefinition(), unless(isImplicit())));
|
|
|
|
ASSERT_NE(FromFwd, FromDef);
|
|
|
|
ASSERT_FALSE(FromFwd->isThisDeclarationADefinition());
|
|
|
|
ASSERT_TRUE(FromDef->isThisDeclarationADefinition());
|
|
|
|
ASSERT_EQ(FromFwd->getCanonicalDecl(), FromDef->getCanonicalDecl());
|
|
|
|
|
|
|
|
auto *ToDef = cast_or_null<CXXRecordDecl>(Import(FromDef, Lang_CXX));
|
|
|
|
auto *ToFwd = cast_or_null<CXXRecordDecl>(Import(FromFwd, Lang_CXX));
|
|
|
|
EXPECT_NE(ToFwd, ToDef);
|
|
|
|
EXPECT_FALSE(ToFwd->isThisDeclarationADefinition());
|
|
|
|
EXPECT_TRUE(ToDef->isThisDeclarationADefinition());
|
|
|
|
EXPECT_EQ(ToFwd->getCanonicalDecl(), ToDef->getCanonicalDecl());
|
|
|
|
auto *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
// We expect no (ODR) warning during the import.
|
|
|
|
EXPECT_EQ(0u, ToTU->getASTContext().getDiagnostics().getNumWarnings());
|
|
|
|
}
|
|
|
|
|
2019-02-18 21:09:27 +08:00
|
|
|
struct ImportFriendFunctionTemplates : ASTImporterOptionSpecificTestBase {};
|
|
|
|
|
|
|
|
TEST_P(ImportFriendFunctionTemplates, LookupShouldFindPreviousFriend) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class X {
|
|
|
|
template <typename T> friend void foo();
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *Friend = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
ToTU, functionTemplateDecl(hasName("foo")));
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T> void foo();
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromFoo = FirstDeclMatcher<FunctionTemplateDecl>().match(
|
|
|
|
FromTU, functionTemplateDecl(hasName("foo")));
|
|
|
|
auto *Imported = Import(FromFoo, Lang_CXX);
|
|
|
|
|
2019-03-07 21:01:51 +08:00
|
|
|
EXPECT_EQ(Imported->getPreviousDecl(), Friend);
|
2019-02-18 21:09:27 +08:00
|
|
|
}
|
|
|
|
|
2019-06-25 16:00:51 +08:00
|
|
|
struct ASTImporterWithFakeErrors : ASTImporter {
|
|
|
|
using ASTImporter::ASTImporter;
|
|
|
|
bool returnWithErrorInTest() override { return true; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ErrorHandlingTest : ASTImporterOptionSpecificTestBase {
|
|
|
|
ErrorHandlingTest() {
|
|
|
|
Creator = [](ASTContext &ToContext, FileManager &ToFileManager,
|
|
|
|
ASTContext &FromContext, FileManager &FromFileManager,
|
2019-07-01 23:37:07 +08:00
|
|
|
bool MinimalImport,
|
|
|
|
const std::shared_ptr<ASTImporterSharedState> &SharedState) {
|
2019-06-25 16:00:51 +08:00
|
|
|
return new ASTImporterWithFakeErrors(ToContext, ToFileManager,
|
|
|
|
FromContext, FromFileManager,
|
2019-07-01 23:37:07 +08:00
|
|
|
MinimalImport, SharedState);
|
2019-06-25 16:00:51 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
// In this test we purposely report an error (UnsupportedConstruct) when
|
|
|
|
// importing the below stmt.
|
|
|
|
static constexpr auto* ErroneousStmt = R"( asm(""); )";
|
|
|
|
};
|
|
|
|
|
|
|
|
// Check a case when no new AST node is created in the AST before encountering
|
|
|
|
// the error.
|
|
|
|
TEST_P(ErrorHandlingTest, ErrorHappensBeforeCreatingANewNode) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class X {};
|
|
|
|
template <>
|
|
|
|
class X<int> { int a; };
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename T>
|
|
|
|
class X {};
|
|
|
|
template <>
|
|
|
|
class X<int> { double b; };
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromSpec = FirstDeclMatcher<ClassTemplateSpecializationDecl>().match(
|
|
|
|
FromTU, classTemplateSpecializationDecl(hasName("X")));
|
|
|
|
ClassTemplateSpecializationDecl *ImportedSpec = Import(FromSpec, Lang_CXX);
|
|
|
|
EXPECT_FALSE(ImportedSpec);
|
|
|
|
|
|
|
|
// The original Decl is kept, no new decl is created.
|
|
|
|
EXPECT_EQ(DeclCounter<ClassTemplateSpecializationDecl>().match(
|
|
|
|
ToTU, classTemplateSpecializationDecl(hasName("X"))),
|
|
|
|
1u);
|
|
|
|
|
|
|
|
// But an error is set to the counterpart in the "from" context.
|
|
|
|
ASTImporter *Importer = findFromTU(FromSpec)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromSpec);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::NameConflict);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check a case when a new AST node is created but not linked to the AST before
|
|
|
|
// encountering the error.
|
|
|
|
TEST_P(ErrorHandlingTest,
|
|
|
|
ErrorHappensAfterCreatingTheNodeButBeforeLinkingThatToTheAST) {
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
std::string("void foo() { ") + ErroneousStmt + " }",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromFoo = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("foo")));
|
|
|
|
|
|
|
|
FunctionDecl *ImportedFoo = Import(FromFoo, Lang_CXX);
|
|
|
|
EXPECT_FALSE(ImportedFoo);
|
|
|
|
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
// Created, but not linked.
|
|
|
|
EXPECT_EQ(
|
|
|
|
DeclCounter<FunctionDecl>().match(ToTU, functionDecl(hasName("foo"))),
|
|
|
|
0u);
|
|
|
|
|
|
|
|
ASTImporter *Importer = findFromTU(FromFoo)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromFoo);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check a case when a new AST node is created and linked to the AST before
|
|
|
|
// encountering the error. The error is set for the counterpart of the nodes in
|
|
|
|
// the "from" context.
|
|
|
|
TEST_P(ErrorHandlingTest, ErrorHappensAfterNodeIsCreatedAndLinked) {
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
std::string(R"(
|
|
|
|
void f();
|
|
|
|
void f() { )") + ErroneousStmt + R"( }
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromProto = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
auto *FromDef =
|
|
|
|
LastDeclMatcher<FunctionDecl>().match(FromTU, functionDecl(hasName("f")));
|
|
|
|
FunctionDecl *ImportedProto = Import(FromProto, Lang_CXX);
|
|
|
|
EXPECT_FALSE(ImportedProto); // Could not import.
|
|
|
|
// However, we created two nodes in the AST. 1) the fwd decl 2) the
|
|
|
|
// definition. The definition is not added to its DC, but the fwd decl is
|
|
|
|
// there.
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
EXPECT_EQ(DeclCounter<FunctionDecl>().match(ToTU, functionDecl(hasName("f"))),
|
|
|
|
1u);
|
|
|
|
// Match the fwd decl.
|
|
|
|
auto *ToProto =
|
|
|
|
FirstDeclMatcher<FunctionDecl>().match(ToTU, functionDecl(hasName("f")));
|
|
|
|
EXPECT_TRUE(ToProto);
|
|
|
|
// An error is set to the counterpart in the "from" context both for the fwd
|
|
|
|
// decl and the definition.
|
|
|
|
ASTImporter *Importer = findFromTU(FromProto)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromProto);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromDef);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
}
|
|
|
|
|
2019-07-01 20:44:39 +08:00
|
|
|
// An error should be set for a class if we cannot import one member.
|
|
|
|
TEST_P(ErrorHandlingTest, ErrorIsPropagatedFromMemberToClass) {
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
std::string(R"(
|
|
|
|
class X {
|
|
|
|
void f() { )") + ErroneousStmt + R"( } // This member has the error
|
|
|
|
// during import.
|
|
|
|
void ok(); // The error should not prevent importing this.
|
|
|
|
}; // An error will be set for X too.
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromX = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X")));
|
|
|
|
CXXRecordDecl *ImportedX = Import(FromX, Lang_CXX);
|
|
|
|
|
|
|
|
// An error is set for X.
|
|
|
|
EXPECT_FALSE(ImportedX);
|
|
|
|
ASTImporter *Importer = findFromTU(FromX)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromX);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
|
|
|
|
// An error is set for f().
|
|
|
|
auto *FromF = FirstDeclMatcher<CXXMethodDecl>().match(
|
|
|
|
FromTU, cxxMethodDecl(hasName("f")));
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromF);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
// And any subsequent import should fail.
|
|
|
|
CXXMethodDecl *ImportedF = Import(FromF, Lang_CXX);
|
|
|
|
EXPECT_FALSE(ImportedF);
|
|
|
|
|
[ASTImporter] Mark erroneous nodes in from ctx
Summary:
During import of a specific Decl D, it may happen that some AST nodes
had already been created before we recognize an error. In this case we
signal back the error to the caller, but the "to" context remains
polluted with those nodes which had been created. Ideally, those nodes
should not had been created, but that time we did not know about the
error, the error happened later. Since the AST is immutable (most of
the cases we can't remove existing nodes) we choose to mark these nodes
as erroneous.
Here are the steps of the algorithm:
1) We keep track of the nodes which we visit during the import of D: See
ImportPathTy.
2) If a Decl is already imported and it is already on the import path
(we have a cycle) then we copy/store the relevant part of the import
path. We store these cycles for each Decl.
3) When we recognize an error during the import of D then we set up this
error to all Decls in the stored cycles for D and we clear the stored
cycles.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62375
llvm-svn: 364771
2019-07-01 22:19:53 +08:00
|
|
|
// There is an error set for the other member too.
|
2019-07-01 20:44:39 +08:00
|
|
|
auto *FromOK = FirstDeclMatcher<CXXMethodDecl>().match(
|
|
|
|
FromTU, cxxMethodDecl(hasName("ok")));
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromOK);
|
[ASTImporter] Mark erroneous nodes in from ctx
Summary:
During import of a specific Decl D, it may happen that some AST nodes
had already been created before we recognize an error. In this case we
signal back the error to the caller, but the "to" context remains
polluted with those nodes which had been created. Ideally, those nodes
should not had been created, but that time we did not know about the
error, the error happened later. Since the AST is immutable (most of
the cases we can't remove existing nodes) we choose to mark these nodes
as erroneous.
Here are the steps of the algorithm:
1) We keep track of the nodes which we visit during the import of D: See
ImportPathTy.
2) If a Decl is already imported and it is already on the import path
(we have a cycle) then we copy/store the relevant part of the import
path. We store these cycles for each Decl.
3) When we recognize an error during the import of D then we set up this
error to all Decls in the stored cycles for D and we clear the stored
cycles.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62375
llvm-svn: 364771
2019-07-01 22:19:53 +08:00
|
|
|
EXPECT_TRUE(OptErr);
|
|
|
|
// Cannot import the other member.
|
2019-07-01 20:44:39 +08:00
|
|
|
CXXMethodDecl *ImportedOK = Import(FromOK, Lang_CXX);
|
[ASTImporter] Mark erroneous nodes in from ctx
Summary:
During import of a specific Decl D, it may happen that some AST nodes
had already been created before we recognize an error. In this case we
signal back the error to the caller, but the "to" context remains
polluted with those nodes which had been created. Ideally, those nodes
should not had been created, but that time we did not know about the
error, the error happened later. Since the AST is immutable (most of
the cases we can't remove existing nodes) we choose to mark these nodes
as erroneous.
Here are the steps of the algorithm:
1) We keep track of the nodes which we visit during the import of D: See
ImportPathTy.
2) If a Decl is already imported and it is already on the import path
(we have a cycle) then we copy/store the relevant part of the import
path. We store these cycles for each Decl.
3) When we recognize an error during the import of D then we set up this
error to all Decls in the stored cycles for D and we clear the stored
cycles.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62375
llvm-svn: 364771
2019-07-01 22:19:53 +08:00
|
|
|
EXPECT_FALSE(ImportedOK);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that an error propagates to the dependent AST nodes.
|
|
|
|
// In the below code it means that an error in X should propagate to A.
|
|
|
|
// And even to F since the containing A is erroneous.
|
|
|
|
// And to all AST nodes which we visit during the import process which finally
|
|
|
|
// ends up in a failure (in the error() function).
|
|
|
|
TEST_P(ErrorHandlingTest, ErrorPropagatesThroughImportCycles) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
std::string(R"(
|
|
|
|
namespace NS {
|
|
|
|
class A {
|
|
|
|
template <int I> class F {};
|
|
|
|
class X {
|
|
|
|
template <int I> friend class F;
|
|
|
|
void error() { )") + ErroneousStmt + R"( }
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
class B {};
|
|
|
|
} // NS
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input0.cc");
|
|
|
|
|
|
|
|
auto *FromFRD = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("F"), isDefinition()));
|
|
|
|
auto *FromA = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("A"), isDefinition()));
|
|
|
|
auto *FromB = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("B"), isDefinition()));
|
|
|
|
auto *FromNS = FirstDeclMatcher<NamespaceDecl>().match(
|
|
|
|
FromTU, namespaceDecl(hasName("NS")));
|
|
|
|
|
|
|
|
// Start by importing the templated CXXRecordDecl of F.
|
|
|
|
// Import fails for that.
|
|
|
|
EXPECT_FALSE(Import(FromFRD, Lang_CXX));
|
|
|
|
// Import fails for A.
|
|
|
|
EXPECT_FALSE(Import(FromA, Lang_CXX));
|
|
|
|
// But we should be able to import the independent B.
|
|
|
|
EXPECT_TRUE(Import(FromB, Lang_CXX));
|
|
|
|
// And the namespace.
|
|
|
|
EXPECT_TRUE(Import(FromNS, Lang_CXX));
|
|
|
|
|
|
|
|
// An error is set to the templated CXXRecordDecl of F.
|
|
|
|
ASTImporter *Importer = findFromTU(FromFRD)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromFRD);
|
|
|
|
EXPECT_TRUE(OptErr);
|
|
|
|
|
|
|
|
// An error is set to A.
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromA);
|
|
|
|
EXPECT_TRUE(OptErr);
|
|
|
|
|
|
|
|
// There is no error set to B.
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromB);
|
|
|
|
EXPECT_FALSE(OptErr);
|
2019-07-01 20:44:39 +08:00
|
|
|
|
[ASTImporter] Mark erroneous nodes in from ctx
Summary:
During import of a specific Decl D, it may happen that some AST nodes
had already been created before we recognize an error. In this case we
signal back the error to the caller, but the "to" context remains
polluted with those nodes which had been created. Ideally, those nodes
should not had been created, but that time we did not know about the
error, the error happened later. Since the AST is immutable (most of
the cases we can't remove existing nodes) we choose to mark these nodes
as erroneous.
Here are the steps of the algorithm:
1) We keep track of the nodes which we visit during the import of D: See
ImportPathTy.
2) If a Decl is already imported and it is already on the import path
(we have a cycle) then we copy/store the relevant part of the import
path. We store these cycles for each Decl.
3) When we recognize an error during the import of D then we set up this
error to all Decls in the stored cycles for D and we clear the stored
cycles.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62375
llvm-svn: 364771
2019-07-01 22:19:53 +08:00
|
|
|
// There is no error set to NS.
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromNS);
|
|
|
|
EXPECT_FALSE(OptErr);
|
|
|
|
|
|
|
|
// Check some of those decls whose ancestor is X, they all should have an
|
|
|
|
// error set if we visited them during an import process which finally failed.
|
|
|
|
// These decls are part of a cycle in an ImportPath.
|
|
|
|
// There would not be any error set for these decls if we hadn't follow the
|
|
|
|
// ImportPaths and the cycles.
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(
|
|
|
|
FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("F"))));
|
|
|
|
// An error is set to the 'F' ClassTemplateDecl.
|
|
|
|
EXPECT_TRUE(OptErr);
|
|
|
|
// An error is set to the FriendDecl.
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(
|
|
|
|
FirstDeclMatcher<FriendDecl>().match(
|
|
|
|
FromTU, friendDecl()));
|
|
|
|
EXPECT_TRUE(OptErr);
|
|
|
|
// An error is set to the implicit class of A.
|
|
|
|
OptErr =
|
|
|
|
Importer->getImportDeclErrorIfAny(FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("A"), isImplicit())));
|
|
|
|
EXPECT_TRUE(OptErr);
|
|
|
|
// An error is set to the implicit class of X.
|
|
|
|
OptErr =
|
|
|
|
Importer->getImportDeclErrorIfAny(FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X"), isImplicit())));
|
|
|
|
EXPECT_TRUE(OptErr);
|
2019-07-01 20:44:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ErrorHandlingTest, ErrorIsNotPropagatedFromMemberToNamespace) {
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(
|
|
|
|
std::string(R"(
|
|
|
|
namespace X {
|
|
|
|
void f() { )") + ErroneousStmt + R"( } // This member has the error
|
|
|
|
// during import.
|
|
|
|
void ok(); // The error should not prevent importing this.
|
|
|
|
}; // An error will be set for X too.
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromX = FirstDeclMatcher<NamespaceDecl>().match(
|
|
|
|
FromTU, namespaceDecl(hasName("X")));
|
|
|
|
NamespaceDecl *ImportedX = Import(FromX, Lang_CXX);
|
|
|
|
|
|
|
|
// There is no error set for X.
|
|
|
|
EXPECT_TRUE(ImportedX);
|
|
|
|
ASTImporter *Importer = findFromTU(FromX)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromX);
|
|
|
|
ASSERT_FALSE(OptErr);
|
|
|
|
|
|
|
|
// An error is set for f().
|
|
|
|
auto *FromF = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromF);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
// And any subsequent import should fail.
|
|
|
|
FunctionDecl *ImportedF = Import(FromF, Lang_CXX);
|
|
|
|
EXPECT_FALSE(ImportedF);
|
|
|
|
|
|
|
|
// There is no error set for ok().
|
|
|
|
auto *FromOK = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("ok")));
|
|
|
|
OptErr = Importer->getImportDeclErrorIfAny(FromOK);
|
|
|
|
EXPECT_FALSE(OptErr);
|
|
|
|
// And we should be able to import.
|
|
|
|
FunctionDecl *ImportedOK = Import(FromOK, Lang_CXX);
|
|
|
|
EXPECT_TRUE(ImportedOK);
|
|
|
|
}
|
|
|
|
|
2019-07-01 23:37:07 +08:00
|
|
|
// An error should be set for a class if it had a previous import with an error
|
|
|
|
// from another TU.
|
|
|
|
TEST_P(ErrorHandlingTest,
|
|
|
|
ImportedDeclWithErrorShouldFailTheImportOfDeclWhichMapToIt) {
|
|
|
|
// We already have a fwd decl.
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
"class X;", Lang_CXX);
|
|
|
|
// Then we import a definition.
|
|
|
|
{
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(std::string(R"(
|
|
|
|
class X {
|
|
|
|
void f() { )") + ErroneousStmt + R"( }
|
|
|
|
void ok();
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromX = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X")));
|
|
|
|
CXXRecordDecl *ImportedX = Import(FromX, Lang_CXX);
|
|
|
|
|
|
|
|
// An error is set for X ...
|
|
|
|
EXPECT_FALSE(ImportedX);
|
|
|
|
ASTImporter *Importer = findFromTU(FromX)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromX);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
}
|
|
|
|
// ... but the node had been created.
|
|
|
|
auto *ToXDef = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("X"), isDefinition()));
|
|
|
|
// An error is set for "ToXDef" in the shared state.
|
|
|
|
Optional<ImportError> OptErr =
|
|
|
|
SharedStatePtr->getImportDeclErrorIfAny(ToXDef);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
|
|
|
|
auto *ToXFwd = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("X"), unless(isDefinition())));
|
|
|
|
// An error is NOT set for the fwd Decl of X in the shared state.
|
|
|
|
OptErr = SharedStatePtr->getImportDeclErrorIfAny(ToXFwd);
|
|
|
|
ASSERT_FALSE(OptErr);
|
|
|
|
|
|
|
|
// Try to import X again but from another TU.
|
|
|
|
{
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(std::string(R"(
|
|
|
|
class X {
|
|
|
|
void f() { )") + ErroneousStmt + R"( }
|
|
|
|
void ok();
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX, "input1.cc");
|
|
|
|
|
|
|
|
auto *FromX = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X")));
|
|
|
|
CXXRecordDecl *ImportedX = Import(FromX, Lang_CXX);
|
|
|
|
|
|
|
|
// If we did not save the errors for the "to" context then the below checks
|
|
|
|
// would fail, because the lookup finds the fwd Decl of the existing
|
|
|
|
// definition in the "to" context. We can reach the existing definition via
|
|
|
|
// the found fwd Decl. That existing definition is structurally equivalent
|
|
|
|
// (we check only the fields) with this one we want to import, so we return
|
|
|
|
// with the existing definition, which is erroneous (one method is missing).
|
|
|
|
|
|
|
|
// The import should fail.
|
|
|
|
EXPECT_FALSE(ImportedX);
|
|
|
|
ASTImporter *Importer = findFromTU(FromX)->Importer.get();
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromX);
|
|
|
|
// And an error is set for this new X in the "from" ctx.
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-30 18:12:14 +08:00
|
|
|
TEST_P(ErrorHandlingTest, ImportOfOverriddenMethods) {
|
|
|
|
auto MatchFooA =
|
|
|
|
functionDecl(hasName("foo"), hasAncestor(cxxRecordDecl(hasName("A"))));
|
|
|
|
auto MatchFooB =
|
|
|
|
functionDecl(hasName("foo"), hasAncestor(cxxRecordDecl(hasName("B"))));
|
|
|
|
auto MatchFooC =
|
|
|
|
functionDecl(hasName("foo"), hasAncestor(cxxRecordDecl(hasName("C"))));
|
|
|
|
|
|
|
|
// Provoke import of a method that has overridden methods with import error.
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(std::string(R"(
|
|
|
|
struct C;
|
|
|
|
struct A {
|
|
|
|
virtual void foo();
|
|
|
|
void f1(C *);
|
|
|
|
};
|
|
|
|
void A::foo() {
|
|
|
|
)") + ErroneousStmt + R"(
|
|
|
|
}
|
|
|
|
struct B : public A {
|
|
|
|
void foo() override;
|
|
|
|
};
|
|
|
|
struct C : public B {
|
|
|
|
void foo() override;
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto *FromFooA = FirstDeclMatcher<FunctionDecl>().match(FromTU, MatchFooA);
|
|
|
|
auto *FromFooB = FirstDeclMatcher<FunctionDecl>().match(FromTU, MatchFooB);
|
|
|
|
auto *FromFooC = FirstDeclMatcher<FunctionDecl>().match(FromTU, MatchFooC);
|
|
|
|
|
|
|
|
EXPECT_FALSE(Import(FromFooA, Lang_CXX11));
|
|
|
|
ASTImporter *Importer = findFromTU(FromFooA)->Importer.get();
|
|
|
|
auto CheckError = [&Importer](Decl *FromD) {
|
|
|
|
Optional<ImportError> OptErr = Importer->getImportDeclErrorIfAny(FromD);
|
|
|
|
ASSERT_TRUE(OptErr);
|
|
|
|
EXPECT_EQ(OptErr->Error, ImportError::UnsupportedConstruct);
|
|
|
|
};
|
|
|
|
CheckError(FromFooA);
|
|
|
|
EXPECT_FALSE(Import(FromFooB, Lang_CXX11));
|
|
|
|
CheckError(FromFooB);
|
|
|
|
EXPECT_FALSE(Import(FromFooC, Lang_CXX11));
|
|
|
|
CheckError(FromFooC);
|
|
|
|
}
|
|
|
|
|
2019-07-08 20:49:13 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, LambdaInFunctionBody) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void f() {
|
|
|
|
auto L = [](){};
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto Pattern = lambdaExpr();
|
|
|
|
CXXRecordDecl *FromL =
|
|
|
|
FirstDeclMatcher<LambdaExpr>().match(FromTU, Pattern)->getLambdaClass();
|
|
|
|
|
|
|
|
auto ToL = Import(FromL, Lang_CXX11);
|
|
|
|
unsigned ToLSize = std::distance(ToL->decls().begin(), ToL->decls().end());
|
|
|
|
unsigned FromLSize =
|
|
|
|
std::distance(FromL->decls().begin(), FromL->decls().end());
|
|
|
|
EXPECT_NE(ToLSize, 0u);
|
|
|
|
EXPECT_EQ(ToLSize, FromLSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, LambdaInFunctionParam) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename F>
|
|
|
|
void f(F L = [](){}) {}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto Pattern = lambdaExpr();
|
|
|
|
CXXRecordDecl *FromL =
|
|
|
|
FirstDeclMatcher<LambdaExpr>().match(FromTU, Pattern)->getLambdaClass();
|
|
|
|
|
|
|
|
auto ToL = Import(FromL, Lang_CXX11);
|
|
|
|
unsigned ToLSize = std::distance(ToL->decls().begin(), ToL->decls().end());
|
|
|
|
unsigned FromLSize =
|
|
|
|
std::distance(FromL->decls().begin(), FromL->decls().end());
|
|
|
|
EXPECT_NE(ToLSize, 0u);
|
|
|
|
EXPECT_EQ(ToLSize, FromLSize);
|
|
|
|
}
|
|
|
|
|
2019-07-17 22:40:09 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, LambdaInGlobalScope) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
auto l1 = [](unsigned lp) { return 1; };
|
|
|
|
auto l2 = [](int lp) { return 2; };
|
|
|
|
int f(int p) {
|
|
|
|
return l1(p) + l2(p);
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
FunctionDecl *FromF = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
FunctionDecl *ToF = Import(FromF, Lang_CXX11);
|
|
|
|
EXPECT_TRUE(ToF);
|
|
|
|
}
|
|
|
|
|
2019-08-12 18:07:38 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportExistingFriendClassTemplateDef) {
|
|
|
|
auto Code =
|
|
|
|
R"(
|
|
|
|
template <class T1, class T2>
|
|
|
|
struct Base {
|
|
|
|
template <class U1, class U2>
|
|
|
|
friend struct Class;
|
|
|
|
};
|
|
|
|
template <class T1, class T2>
|
|
|
|
struct Class { };
|
|
|
|
)";
|
|
|
|
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(Code, Lang_CXX);
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl(Code, Lang_CXX, "input.cc");
|
|
|
|
|
|
|
|
auto *ToClassProto = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("Class")));
|
|
|
|
auto *ToClassDef = LastDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
ToTU, classTemplateDecl(hasName("Class")));
|
|
|
|
ASSERT_FALSE(ToClassProto->isThisDeclarationADefinition());
|
|
|
|
ASSERT_TRUE(ToClassDef->isThisDeclarationADefinition());
|
|
|
|
// Previous friend decl is not linked to it!
|
|
|
|
ASSERT_FALSE(ToClassDef->getPreviousDecl());
|
|
|
|
ASSERT_EQ(ToClassDef->getMostRecentDecl(), ToClassDef);
|
|
|
|
ASSERT_EQ(ToClassProto->getMostRecentDecl(), ToClassProto);
|
|
|
|
|
|
|
|
auto *FromClassProto = FirstDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("Class")));
|
|
|
|
auto *FromClassDef = LastDeclMatcher<ClassTemplateDecl>().match(
|
|
|
|
FromTU, classTemplateDecl(hasName("Class")));
|
|
|
|
ASSERT_FALSE(FromClassProto->isThisDeclarationADefinition());
|
|
|
|
ASSERT_TRUE(FromClassDef->isThisDeclarationADefinition());
|
|
|
|
ASSERT_FALSE(FromClassDef->getPreviousDecl());
|
|
|
|
ASSERT_EQ(FromClassDef->getMostRecentDecl(), FromClassDef);
|
|
|
|
ASSERT_EQ(FromClassProto->getMostRecentDecl(), FromClassProto);
|
|
|
|
|
|
|
|
auto *ImportedDef = Import(FromClassDef, Lang_CXX);
|
|
|
|
// At import we should find the definition for 'Class' even if the
|
|
|
|
// prototype (inside 'friend') for it comes first in the AST and is not
|
|
|
|
// linked to the definition.
|
|
|
|
EXPECT_EQ(ImportedDef, ToClassDef);
|
2019-08-27 19:36:10 +08:00
|
|
|
}
|
|
|
|
|
[ASTImporter] Fix LLDB lookup in transparent ctx and with ext src
Summary:
With LLDB we use localUncachedLookup(), however, that fails to find
Decls when a transparent context is involved and the given DC has
external lexical storage. The solution is to use noload_lookup, which
works well with transparent contexts. But, we cannot use only the
noload_lookup since the slow case of localUncachedLookup is still needed
in some other cases.
These other cases are handled in ASTImporterLookupTable, but we cannot
use that with LLDB since that traverses through the AST which initiates
the load of external decls again via DC::decls().
We must avoid loading external decls during the import becuase
ExternalASTSource is implemented with ASTImporter, so external loads
during import results in uncontrolled and faulty import.
Reviewers: shafik, teemperor, jingham, clayborg, a_sidorin, a.sidorin
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits, lldb-commits
Tags: #clang, #lldb
Differential Revision: https://reviews.llvm.org/D61333
llvm-svn: 366325
2019-07-17 21:47:46 +08:00
|
|
|
struct LLDBLookupTest : ASTImporterOptionSpecificTestBase {
|
|
|
|
LLDBLookupTest() {
|
|
|
|
Creator = [](ASTContext &ToContext, FileManager &ToFileManager,
|
|
|
|
ASTContext &FromContext, FileManager &FromFileManager,
|
|
|
|
bool MinimalImport,
|
|
|
|
const std::shared_ptr<ASTImporterSharedState> &SharedState) {
|
|
|
|
return new ASTImporter(ToContext, ToFileManager, FromContext,
|
|
|
|
FromFileManager, MinimalImport,
|
|
|
|
// We use the regular lookup.
|
|
|
|
/*SharedState=*/nullptr);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(LLDBLookupTest, ImporterShouldFindInTransparentContext) {
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
extern "C" {
|
|
|
|
class X{};
|
|
|
|
};
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *ToX = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
ToTU, cxxRecordDecl(hasName("X")));
|
|
|
|
|
|
|
|
// Set up a stub external storage.
|
|
|
|
ToTU->setHasExternalLexicalStorage(true);
|
|
|
|
// Set up DeclContextBits.HasLazyExternalLexicalLookups to true.
|
|
|
|
ToTU->setMustBuildLookupTable();
|
|
|
|
struct TestExternalASTSource : ExternalASTSource {};
|
|
|
|
ToTU->getASTContext().setExternalSource(new TestExternalASTSource());
|
|
|
|
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X;
|
|
|
|
)",
|
|
|
|
Lang_CXX);
|
|
|
|
auto *FromX = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X")));
|
|
|
|
auto *ImportedX = Import(FromX, Lang_CXX);
|
|
|
|
// The lookup must find the existing class definition in the LinkageSpecDecl.
|
|
|
|
// Then the importer renders the existing and the new decl into one chain.
|
|
|
|
EXPECT_EQ(ImportedX->getCanonicalDecl(), ToX->getCanonicalDecl());
|
|
|
|
}
|
|
|
|
|
2019-08-09 16:52:54 +08:00
|
|
|
struct SVEBuiltins : ASTImporterOptionSpecificTestBase {};
|
|
|
|
|
|
|
|
TEST_P(SVEBuiltins, ImportTypes) {
|
|
|
|
static const char *const TypeNames[] = {
|
|
|
|
"__SVInt8_t",
|
|
|
|
"__SVInt16_t",
|
|
|
|
"__SVInt32_t",
|
|
|
|
"__SVInt64_t",
|
|
|
|
"__SVUint8_t",
|
|
|
|
"__SVUint16_t",
|
|
|
|
"__SVUint32_t",
|
|
|
|
"__SVUint64_t",
|
|
|
|
"__SVFloat16_t",
|
|
|
|
"__SVFloat32_t",
|
|
|
|
"__SVFloat64_t",
|
|
|
|
"__SVBool_t"
|
|
|
|
};
|
|
|
|
|
|
|
|
TranslationUnitDecl *ToTU = getToTuDecl("", Lang_CXX);
|
|
|
|
TranslationUnitDecl *FromTU = getTuDecl("", Lang_CXX, "input.cc");
|
|
|
|
for (auto *TypeName : TypeNames) {
|
|
|
|
auto *ToTypedef = FirstDeclMatcher<TypedefDecl>().match(
|
|
|
|
ToTU, typedefDecl(hasName(TypeName)));
|
|
|
|
QualType ToType = ToTypedef->getUnderlyingType();
|
|
|
|
|
|
|
|
auto *FromTypedef = FirstDeclMatcher<TypedefDecl>().match(
|
|
|
|
FromTU, typedefDecl(hasName(TypeName)));
|
|
|
|
QualType FromType = FromTypedef->getUnderlyingType();
|
|
|
|
|
|
|
|
QualType ImportedType = ImportType(FromType, FromTypedef, Lang_CXX);
|
|
|
|
EXPECT_EQ(ImportedType, ToType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-13 16:04:06 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportOfDefaultImplicitFunctions) {
|
|
|
|
// Test that import of implicit functions works and the functions
|
|
|
|
// are merged into one chain.
|
|
|
|
auto GetDeclToImport = [this](StringRef File) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X { };
|
|
|
|
// Force generating some implicit operator definitions for X.
|
|
|
|
void f() { X x1, x2; x1 = x2; X *x3 = new X; delete x3; }
|
|
|
|
)",
|
|
|
|
Lang_CXX11, File);
|
|
|
|
auto *FromD = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X"), unless(isImplicit())));
|
|
|
|
// Destructor is picked as one example of implicit function.
|
|
|
|
return FromD->getDestructor();
|
|
|
|
};
|
|
|
|
|
|
|
|
auto *ToD1 = Import(GetDeclToImport("input1.cc"), Lang_CXX11);
|
|
|
|
ASSERT_TRUE(ToD1);
|
|
|
|
|
|
|
|
auto *ToD2 = Import(GetDeclToImport("input2.cc"), Lang_CXX11);
|
|
|
|
ASSERT_TRUE(ToD2);
|
|
|
|
|
|
|
|
EXPECT_EQ(ToD1->getCanonicalDecl(), ToD2->getCanonicalDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
ImportOfExplicitlyDefaultedOrDeleted) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
struct X { X() = default; X(const X&) = delete; };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto *FromX = FirstDeclMatcher<CXXRecordDecl>().match(
|
|
|
|
FromTU, cxxRecordDecl(hasName("X")));
|
|
|
|
auto *ImportedX = Import(FromX, Lang_CXX11);
|
|
|
|
auto *Constr1 = FirstDeclMatcher<CXXConstructorDecl>().match(
|
|
|
|
ImportedX, cxxConstructorDecl(hasName("X"), unless(isImplicit())));
|
|
|
|
auto *Constr2 = LastDeclMatcher<CXXConstructorDecl>().match(
|
|
|
|
ImportedX, cxxConstructorDecl(hasName("X"), unless(isImplicit())));
|
|
|
|
|
|
|
|
ASSERT_TRUE(ImportedX);
|
|
|
|
EXPECT_TRUE(Constr1->isDefaulted());
|
|
|
|
EXPECT_TRUE(Constr1->isExplicitlyDefaulted());
|
|
|
|
EXPECT_TRUE(Constr2->isDeletedAsWritten());
|
|
|
|
EXPECT_EQ(ImportedX->isAggregate(), FromX->isAggregate());
|
|
|
|
}
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
struct ConflictingDeclsWithLiberalStrategy : ASTImporterOptionSpecificTestBase {
|
|
|
|
ConflictingDeclsWithLiberalStrategy() {
|
|
|
|
this->ODRHandling = ASTImporter::ODRHandlingType::Liberal;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Check that a Decl has been successfully imported into a standalone redecl
|
|
|
|
// chain.
|
|
|
|
template <typename DeclTy, typename PatternTy>
|
|
|
|
static void CheckImportedAsNew(llvm::Expected<Decl *> &Result, Decl *ToTU,
|
|
|
|
PatternTy Pattern) {
|
|
|
|
ASSERT_TRUE(isSuccess(Result));
|
|
|
|
Decl *ImportedD = *Result;
|
|
|
|
ASSERT_TRUE(ImportedD);
|
|
|
|
auto *ToD = FirstDeclMatcher<DeclTy>().match(ToTU, Pattern);
|
|
|
|
EXPECT_NE(ImportedD, ToD);
|
|
|
|
EXPECT_FALSE(ImportedD->getPreviousDecl());
|
|
|
|
EXPECT_EQ(DeclCounter<DeclTy>().match(ToTU, Pattern), 2u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, Typedef) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
typedef int X;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
typedef double X;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = typedefNameDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<TypedefNameDecl>().match(FromTU, Pattern);
|
|
|
|
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<TypedefNameDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, TypeAlias) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
using X = int;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
using X = double;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = typedefNameDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<TypedefNameDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<TypedefNameDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, EnumDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
enum X { a, b };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
enum X { a, b, c };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = enumDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<EnumDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<EnumDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, EnumConstantDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
enum E { X = 0 };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
enum E { X = 1 };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = enumConstantDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<EnumConstantDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<EnumConstantDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, RecordDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
class X { int a; };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
class X { int b; };
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = cxxRecordDecl(hasName("X"), unless(isImplicit()));
|
|
|
|
auto *FromX = FirstDeclMatcher<RecordDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<RecordDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, VarDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
int X;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
double X;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = varDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<VarDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<VarDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, FunctionDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
void X(int);
|
|
|
|
)",
|
|
|
|
Lang_C); // C, no overloading!
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void X(double);
|
|
|
|
)",
|
|
|
|
Lang_C);
|
|
|
|
auto Pattern = functionDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<FunctionDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<FunctionDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, ClassTemplateDecl) {
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <class>
|
|
|
|
struct X;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <int>
|
|
|
|
struct X;
|
|
|
|
)",
|
|
|
|
Lang_CXX11);
|
|
|
|
auto Pattern = classTemplateDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<ClassTemplateDecl>().match(FromTU, Pattern);
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<ClassTemplateDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ConflictingDeclsWithLiberalStrategy, DISABLED_VarTemplateDecl) {
|
|
|
|
const internal::VariadicDynCastAllOfMatcher<Decl, VarTemplateDecl>
|
|
|
|
varTemplateDecl;
|
|
|
|
Decl *ToTU = getToTuDecl(
|
|
|
|
R"(
|
|
|
|
template <class T>
|
|
|
|
constexpr T X;
|
|
|
|
)",
|
|
|
|
Lang_CXX14);
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <int>
|
|
|
|
constexpr int X = 0;
|
|
|
|
)",
|
|
|
|
Lang_CXX14);
|
|
|
|
auto Pattern = varTemplateDecl(hasName("X"));
|
|
|
|
auto *FromX = FirstDeclMatcher<VarTemplateDecl>().match(
|
|
|
|
FromTU, varTemplateDecl(hasName("X")));
|
|
|
|
Expected<Decl *> Result = importOrError(FromX, Lang_CXX11);
|
|
|
|
CheckImportedAsNew<VarTemplateDecl>(Result, ToTU, Pattern);
|
|
|
|
}
|
|
|
|
|
2019-08-09 16:52:54 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, SVEBuiltins,
|
|
|
|
::testing::Values(ArgVector{"-target",
|
|
|
|
"aarch64-linux-gnu"}), );
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, DeclContextTest,
|
|
|
|
::testing::Values(ArgVector()), );
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, CanonicalRedeclChain,
|
|
|
|
::testing::Values(ArgVector()), );
|
|
|
|
|
[ASTImporter] Do not look up lambda classes
Summary:
Consider this code:
```
void f() {
auto L0 = [](){};
auto L1 = [](){};
}
```
First we import `L0` then `L1`. Currently we end up having only one
CXXRecordDecl for the two different lambdas. And that is a problem if
the body of their op() is different. This happens because when we import
`L1` then lookup finds the existing `L0` and since they are structurally
equivalent we just map the imported L0 to be the counterpart of L1.
We have the same problem in this case:
```
template <typename F0, typename F1>
void f(F0 L0 = [](){}, F1 L1 = [](){}) {}
```
In StructuralEquivalenceContext we could distinquish lambdas only by
their source location in these cases. But we the lambdas are actually
structrually equivalent they differn only by the source location.
Thus, the solution is to disable lookup completely if the decl in
the "from" context is a lambda.
However, that could have other problems: what if the lambda is defined
in a header file and included in several TUs? I think we'd have as many
duplicates as many includes we have. I think we could live with that,
because the lambda classes are TU local anyway, we cannot just access
them from another TU.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D66348
llvm-svn: 370461
2019-08-30 18:55:41 +08:00
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, LambdasAreDifferentiated) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void f() {
|
|
|
|
auto L0 = [](){};
|
|
|
|
auto L1 = [](){};
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto Pattern = lambdaExpr();
|
|
|
|
CXXRecordDecl *FromL0 =
|
|
|
|
FirstDeclMatcher<LambdaExpr>().match(FromTU, Pattern)->getLambdaClass();
|
|
|
|
CXXRecordDecl *FromL1 =
|
|
|
|
LastDeclMatcher<LambdaExpr>().match(FromTU, Pattern)->getLambdaClass();
|
|
|
|
ASSERT_NE(FromL0, FromL1);
|
|
|
|
|
|
|
|
CXXRecordDecl *ToL0 = Import(FromL0, Lang_CXX11);
|
|
|
|
CXXRecordDecl *ToL1 = Import(FromL1, Lang_CXX11);
|
|
|
|
EXPECT_NE(ToL0, ToL1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
LambdasInFunctionParamsAreDifferentiated) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
template <typename F0, typename F1>
|
|
|
|
void f(F0 L0 = [](){}, F1 L1 = [](){}) {}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto Pattern = cxxRecordDecl(isLambda());
|
|
|
|
CXXRecordDecl *FromL0 =
|
|
|
|
FirstDeclMatcher<CXXRecordDecl>().match(FromTU, Pattern);
|
|
|
|
CXXRecordDecl *FromL1 =
|
|
|
|
LastDeclMatcher<CXXRecordDecl>().match(FromTU, Pattern);
|
|
|
|
ASSERT_NE(FromL0, FromL1);
|
|
|
|
|
|
|
|
CXXRecordDecl *ToL0 = Import(FromL0, Lang_CXX11);
|
|
|
|
CXXRecordDecl *ToL1 = Import(FromL1, Lang_CXX11);
|
|
|
|
ASSERT_NE(ToL0, ToL1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase,
|
|
|
|
LambdasInFunctionParamsAreDifferentiatedWhenMacroIsUsed) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
#define LAMBDA [](){}
|
|
|
|
template <typename F0, typename F1>
|
|
|
|
void f(F0 L0 = LAMBDA, F1 L1 = LAMBDA) {}
|
|
|
|
)",
|
|
|
|
Lang_CXX11, "input0.cc");
|
|
|
|
auto Pattern = cxxRecordDecl(isLambda());
|
|
|
|
CXXRecordDecl *FromL0 =
|
|
|
|
FirstDeclMatcher<CXXRecordDecl>().match(FromTU, Pattern);
|
|
|
|
CXXRecordDecl *FromL1 =
|
|
|
|
LastDeclMatcher<CXXRecordDecl>().match(FromTU, Pattern);
|
|
|
|
ASSERT_NE(FromL0, FromL1);
|
|
|
|
|
|
|
|
Import(FromL0, Lang_CXX11);
|
|
|
|
Import(FromL1, Lang_CXX11);
|
|
|
|
CXXRecordDecl *ToL0 = Import(FromL0, Lang_CXX11);
|
|
|
|
CXXRecordDecl *ToL1 = Import(FromL1, Lang_CXX11);
|
|
|
|
ASSERT_NE(ToL0, ToL1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportAssignedLambda) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void f() {
|
|
|
|
auto x = []{} = {}; auto x2 = x;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX2a, "input0.cc");
|
|
|
|
auto FromF = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
// We have only one lambda class.
|
|
|
|
ASSERT_EQ(
|
|
|
|
DeclCounter<CXXRecordDecl>().match(FromTU, cxxRecordDecl(isLambda())),
|
|
|
|
1u);
|
|
|
|
|
|
|
|
FunctionDecl *ToF = Import(FromF, Lang_CXX2a);
|
|
|
|
EXPECT_TRUE(ToF);
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
// We have only one lambda class after the import.
|
|
|
|
EXPECT_EQ(DeclCounter<CXXRecordDecl>().match(ToTU, cxxRecordDecl(isLambda())),
|
|
|
|
1u);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(ASTImporterOptionSpecificTestBase, ImportDefaultConstructibleLambdas) {
|
|
|
|
Decl *FromTU = getTuDecl(
|
|
|
|
R"(
|
|
|
|
void f() {
|
|
|
|
auto x = []{} = {};
|
|
|
|
auto xb = []{} = {};
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
Lang_CXX2a, "input0.cc");
|
|
|
|
auto FromF = FirstDeclMatcher<FunctionDecl>().match(
|
|
|
|
FromTU, functionDecl(hasName("f")));
|
|
|
|
// We have two lambda classes.
|
|
|
|
ASSERT_EQ(
|
|
|
|
DeclCounter<CXXRecordDecl>().match(FromTU, cxxRecordDecl(isLambda())),
|
|
|
|
2u);
|
|
|
|
|
|
|
|
FunctionDecl *ToF = Import(FromF, Lang_CXX2a);
|
|
|
|
EXPECT_TRUE(ToF);
|
|
|
|
TranslationUnitDecl *ToTU = ToAST->getASTContext().getTranslationUnitDecl();
|
|
|
|
// We have two lambda classes after the import.
|
|
|
|
EXPECT_EQ(DeclCounter<CXXRecordDecl>().match(ToTU, cxxRecordDecl(isLambda())),
|
|
|
|
2u);
|
|
|
|
}
|
|
|
|
|
2018-12-17 21:53:12 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ASTImporterLookupTableTest,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
[ASTImporter] Mark erroneous nodes in from ctx
Summary:
During import of a specific Decl D, it may happen that some AST nodes
had already been created before we recognize an error. In this case we
signal back the error to the caller, but the "to" context remains
polluted with those nodes which had been created. Ideally, those nodes
should not had been created, but that time we did not know about the
error, the error happened later. Since the AST is immutable (most of
the cases we can't remove existing nodes) we choose to mark these nodes
as erroneous.
Here are the steps of the algorithm:
1) We keep track of the nodes which we visit during the import of D: See
ImportPathTy.
2) If a Decl is already imported and it is already on the import path
(we have a cycle) then we copy/store the relevant part of the import
path. We store these cycles for each Decl.
3) When we recognize an error during the import of D then we set up this
error to all Decls in the stored cycles for D and we clear the stored
cycles.
Reviewers: a_sidorin, a.sidorin, shafik
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits
Tags: #clang
Differential Revision: https://reviews.llvm.org/D62375
llvm-svn: 364771
2019-07-01 22:19:53 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportPath,
|
|
|
|
::testing::Values(ArgVector()), );
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportExpr,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportType,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportDecl,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2019-02-08 00:52:48 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ASTImporterOptionSpecificTestBase,
|
2018-06-25 21:04:37 +08:00
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ErrorHandlingTest,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
[ASTImporter] Add an ImportImpl method to allow customizing Import behavior.
Summary:
We are currently implementing support in LLDB that reconstructs the STL templates from
the target program in the expression evaluator. This reconstruction happens during the
import process from our debug info AST into the expression evaluation AST, which means
we need a way to intercept the ASTImporter import process.
This patch adds an protected ImportImpl method that we can overwrite in LLDB to implement
our special importing logic (which is essentially just looking into a C++ module that is attached to
the target context). Because ImportImpl has to call MapImported/AddToLookup for the decls it
creates, this patch also exposes those via a new unified method and checks that we call it when
importing decls.
Reviewers: martong, balazske, a.sidorin, shafik, a_sidorin
Reviewed By: martong, a_sidorin
Subscribers: rnkovacs, cfe-commits, lldb-commits, aprantl
Tags: #clang
Differential Revision: https://reviews.llvm.org/D59485
llvm-svn: 359502
2019-04-30 05:02:35 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, RedirectingImporterTest,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2018-06-25 21:04:37 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportFunctions,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
2018-05-18 17:08:47 +08:00
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportFunctionTemplates,
|
2019-02-18 21:09:27 +08:00
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportFriendFunctionTemplates,
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportClasses,
|
2019-08-07 20:40:17 +08:00
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2018-12-17 21:53:12 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportFriendFunctions,
|
2018-12-17 20:42:12 +08:00
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportFriendClasses,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
Re-apply: [ASTImporter] Import the whole redecl chain of functions
Summary:
With this patch when any `FunctionDecl` of a redeclaration chain is imported
then we bring in the whole declaration chain. This involves functions and
function template specializations. Also friend functions are affected. The
chain is imported as it is in the "from" tu, the order of the redeclarations
are kept. I also changed the lookup logic in order to find friends, but first
making them visible in their declaration context. We may have long
redeclaration chains if all TU contains the same prototype, but our
measurements shows no degradation in time of CTU analysis (Tmux, Xerces,
Bitcoin, Protobuf). Also, as further work we could squash redundant
prototypes, but first ensure that functionality is working properly; then
should we optimize.
This may seem like a huge patch, sorry about that. But, most of the changes are
new tests, changes in the production code is not that much. I also tried to
create a smaller patch which does not affect specializations, but that patch
failed to pass some of the `clang-import-test`s because there we import
function specializations. Also very importantly, we can't just change the
import of `FunctionDecl`s without changing the import of function template
specializations because they are handled as `FunctionDecl`s.
Reviewers: a.sidorin, r.stahl, xazax.hun, balazske, a_sidorin
Reviewed By: a_sidorin
Subscribers: labath, aprantl, a_sidorin, rnkovacs, dkrupp, cfe-commits
Differential Revision: https://reviews.llvm.org/D47532
Re-apply commit rC335480
llvm-svn: 335731
2018-06-27 21:32:50 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests,
|
|
|
|
ImportFunctionTemplateSpecializations,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2018-09-17 20:04:52 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportImplicitMethods,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ImportVariables,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
[ASTImporter] Fix LLDB lookup in transparent ctx and with ext src
Summary:
With LLDB we use localUncachedLookup(), however, that fails to find
Decls when a transparent context is involved and the given DC has
external lexical storage. The solution is to use noload_lookup, which
works well with transparent contexts. But, we cannot use only the
noload_lookup since the slow case of localUncachedLookup is still needed
in some other cases.
These other cases are handled in ASTImporterLookupTable, but we cannot
use that with LLDB since that traverses through the AST which initiates
the load of external decls again via DC::decls().
We must avoid loading external decls during the import becuase
ExternalASTSource is implemented with ASTImporter, so external loads
during import results in uncontrolled and faulty import.
Reviewers: shafik, teemperor, jingham, clayborg, a_sidorin, a.sidorin
Subscribers: rnkovacs, dkrupp, Szelethus, gamesh411, cfe-commits, lldb-commits
Tags: #clang, #lldb
Differential Revision: https://reviews.llvm.org/D61333
llvm-svn: 366325
2019-07-17 21:47:46 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, LLDBLookupTest,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
|
|
|
|
2019-08-27 19:36:10 +08:00
|
|
|
INSTANTIATE_TEST_CASE_P(ParameterizedTests, ConflictingDeclsWithLiberalStrategy,
|
|
|
|
DefaultTestValuesForRunOptions, );
|
2019-08-13 16:04:06 +08:00
|
|
|
|
2016-04-14 19:51:27 +08:00
|
|
|
} // end namespace ast_matchers
|
|
|
|
} // end namespace clang
|