2018-04-10 06:14:10 +08:00
|
|
|
//===- ASTStructuralEquivalence.cpp ---------------------------------------===//
|
2017-04-28 08:31:30 +08:00
|
|
|
//
|
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
|
2017-04-28 08:31:30 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implement StructuralEquivalenceContext class and helper functions
|
|
|
|
// for layout matching.
|
|
|
|
//
|
2018-07-17 20:39:27 +08:00
|
|
|
// The structural equivalence check could have been implemented as a parallel
|
|
|
|
// BFS on a pair of graphs. That must have been the original approach at the
|
|
|
|
// beginning.
|
|
|
|
// Let's consider this simple BFS algorithm from the `s` source:
|
|
|
|
// ```
|
|
|
|
// void bfs(Graph G, int s)
|
|
|
|
// {
|
|
|
|
// Queue<Integer> queue = new Queue<Integer>();
|
|
|
|
// marked[s] = true; // Mark the source
|
|
|
|
// queue.enqueue(s); // and put it on the queue.
|
|
|
|
// while (!q.isEmpty()) {
|
|
|
|
// int v = queue.dequeue(); // Remove next vertex from the queue.
|
|
|
|
// for (int w : G.adj(v))
|
|
|
|
// if (!marked[w]) // For every unmarked adjacent vertex,
|
|
|
|
// {
|
|
|
|
// marked[w] = true;
|
|
|
|
// queue.enqueue(w);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// ```
|
|
|
|
// Indeed, it has it's queue, which holds pairs of nodes, one from each graph,
|
2020-02-21 22:54:58 +08:00
|
|
|
// this is the `DeclsToCheck` member. `VisitedDecls` plays the role of the
|
|
|
|
// marking (`marked`) functionality above, we use it to check whether we've
|
|
|
|
// already seen a pair of nodes.
|
2018-07-17 20:39:27 +08:00
|
|
|
//
|
|
|
|
// We put in the elements into the queue only in the toplevel decl check
|
|
|
|
// function:
|
|
|
|
// ```
|
|
|
|
// static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
// Decl *D1, Decl *D2);
|
|
|
|
// ```
|
|
|
|
// The `while` loop where we iterate over the children is implemented in
|
|
|
|
// `Finish()`. And `Finish` is called only from the two **member** functions
|
|
|
|
// which check the equivalency of two Decls or two Types. ASTImporter (and
|
|
|
|
// other clients) call only these functions.
|
|
|
|
//
|
|
|
|
// The `static` implementation functions are called from `Finish`, these push
|
|
|
|
// the children nodes to the queue via `static bool
|
|
|
|
// IsStructurallyEquivalent(StructuralEquivalenceContext &Context, Decl *D1,
|
|
|
|
// Decl *D2)`. So far so good, this is almost like the BFS. However, if we
|
|
|
|
// let a static implementation function to call `Finish` via another **member**
|
|
|
|
// function that means we end up with two nested while loops each of them
|
|
|
|
// working on the same queue. This is wrong and nobody can reason about it's
|
|
|
|
// doing. Thus, static implementation functions must not call the **member**
|
|
|
|
// functions.
|
|
|
|
//
|
2017-04-28 08:31:30 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/AST/ASTStructuralEquivalence.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
|
|
#include "clang/AST/ASTDiagnostic.h"
|
2018-04-10 06:14:10 +08:00
|
|
|
#include "clang/AST/Decl.h"
|
|
|
|
#include "clang/AST/DeclBase.h"
|
2017-04-28 08:31:30 +08:00
|
|
|
#include "clang/AST/DeclCXX.h"
|
2018-04-26 01:28:03 +08:00
|
|
|
#include "clang/AST/DeclFriend.h"
|
2017-04-28 08:31:30 +08:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
2018-04-10 06:14:10 +08:00
|
|
|
#include "clang/AST/DeclTemplate.h"
|
2019-07-02 15:36:39 +08:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2018-04-10 06:14:10 +08:00
|
|
|
#include "clang/AST/NestedNameSpecifier.h"
|
|
|
|
#include "clang/AST/TemplateBase.h"
|
|
|
|
#include "clang/AST/TemplateName.h"
|
|
|
|
#include "clang/AST/Type.h"
|
|
|
|
#include "clang/Basic/ExceptionSpecificationType.h"
|
|
|
|
#include "clang/Basic/IdentifierTable.h"
|
|
|
|
#include "clang/Basic/LLVM.h"
|
|
|
|
#include "clang/Basic/SourceLocation.h"
|
|
|
|
#include "llvm/ADT/APInt.h"
|
|
|
|
#include "llvm/ADT/APSInt.h"
|
|
|
|
#include "llvm/ADT/None.h"
|
|
|
|
#include "llvm/ADT/Optional.h"
|
|
|
|
#include "llvm/Support/Casting.h"
|
|
|
|
#include "llvm/Support/Compiler.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include <cassert>
|
|
|
|
#include <utility>
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
QualType T1, QualType T2);
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
Decl *D1, Decl *D2);
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
const TemplateArgument &Arg1,
|
|
|
|
const TemplateArgument &Arg2);
|
2019-07-02 15:36:39 +08:00
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
NestedNameSpecifier *NNS1,
|
|
|
|
NestedNameSpecifier *NNS2);
|
|
|
|
static bool IsStructurallyEquivalent(const IdentifierInfo *Name1,
|
|
|
|
const IdentifierInfo *Name2);
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
const DeclarationName Name1,
|
|
|
|
const DeclarationName Name2) {
|
|
|
|
if (Name1.getNameKind() != Name2.getNameKind())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (Name1.getNameKind()) {
|
|
|
|
|
|
|
|
case DeclarationName::Identifier:
|
|
|
|
return IsStructurallyEquivalent(Name1.getAsIdentifierInfo(),
|
|
|
|
Name2.getAsIdentifierInfo());
|
|
|
|
|
|
|
|
case DeclarationName::CXXConstructorName:
|
|
|
|
case DeclarationName::CXXDestructorName:
|
|
|
|
case DeclarationName::CXXConversionFunctionName:
|
|
|
|
return IsStructurallyEquivalent(Context, Name1.getCXXNameType(),
|
|
|
|
Name2.getCXXNameType());
|
|
|
|
|
|
|
|
case DeclarationName::CXXDeductionGuideName: {
|
|
|
|
if (!IsStructurallyEquivalent(
|
|
|
|
Context, Name1.getCXXDeductionGuideTemplate()->getDeclName(),
|
|
|
|
Name2.getCXXDeductionGuideTemplate()->getDeclName()))
|
|
|
|
return false;
|
|
|
|
return IsStructurallyEquivalent(Context,
|
|
|
|
Name1.getCXXDeductionGuideTemplate(),
|
|
|
|
Name2.getCXXDeductionGuideTemplate());
|
|
|
|
}
|
|
|
|
|
|
|
|
case DeclarationName::CXXOperatorName:
|
|
|
|
return Name1.getCXXOverloadedOperator() == Name2.getCXXOverloadedOperator();
|
|
|
|
|
|
|
|
case DeclarationName::CXXLiteralOperatorName:
|
|
|
|
return IsStructurallyEquivalent(Name1.getCXXLiteralIdentifier(),
|
|
|
|
Name2.getCXXLiteralIdentifier());
|
|
|
|
|
|
|
|
case DeclarationName::CXXUsingDirective:
|
|
|
|
return true; // FIXME When do we consider two using directives equal?
|
|
|
|
|
|
|
|
case DeclarationName::ObjCZeroArgSelector:
|
|
|
|
case DeclarationName::ObjCOneArgSelector:
|
|
|
|
case DeclarationName::ObjCMultiArgSelector:
|
|
|
|
return true; // FIXME
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("Unhandled kind of DeclarationName");
|
|
|
|
return true;
|
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
/// Determine structural equivalence of two expressions.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
2018-07-14 03:46:04 +08:00
|
|
|
const Expr *E1, const Expr *E2) {
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!E1 || !E2)
|
|
|
|
return E1 == E2;
|
|
|
|
|
2019-07-02 15:36:39 +08:00
|
|
|
if (auto *DE1 = dyn_cast<DependentScopeDeclRefExpr>(E1)) {
|
|
|
|
auto *DE2 = dyn_cast<DependentScopeDeclRefExpr>(E2);
|
|
|
|
if (!DE2)
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, DE1->getDeclName(),
|
|
|
|
DE2->getDeclName()))
|
|
|
|
return false;
|
|
|
|
return IsStructurallyEquivalent(Context, DE1->getQualifier(),
|
|
|
|
DE2->getQualifier());
|
|
|
|
} else if (auto CastE1 = dyn_cast<ImplicitCastExpr>(E1)) {
|
|
|
|
auto *CastE2 = dyn_cast<ImplicitCastExpr>(E2);
|
|
|
|
if (!CastE2)
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, CastE1->getType(),
|
|
|
|
CastE2->getType()))
|
|
|
|
return false;
|
|
|
|
return IsStructurallyEquivalent(Context, CastE1->getSubExpr(),
|
|
|
|
CastE2->getSubExpr());
|
|
|
|
}
|
|
|
|
// FIXME: Handle other kind of expressions!
|
2017-04-28 08:31:30 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine whether two identifiers are equivalent.
|
|
|
|
static bool IsStructurallyEquivalent(const IdentifierInfo *Name1,
|
|
|
|
const IdentifierInfo *Name2) {
|
|
|
|
if (!Name1 || !Name2)
|
|
|
|
return Name1 == Name2;
|
|
|
|
|
|
|
|
return Name1->getName() == Name2->getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine whether two nested-name-specifiers are equivalent.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
NestedNameSpecifier *NNS1,
|
|
|
|
NestedNameSpecifier *NNS2) {
|
|
|
|
if (NNS1->getKind() != NNS2->getKind())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
NestedNameSpecifier *Prefix1 = NNS1->getPrefix(),
|
|
|
|
*Prefix2 = NNS2->getPrefix();
|
|
|
|
if ((bool)Prefix1 != (bool)Prefix2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (Prefix1)
|
|
|
|
if (!IsStructurallyEquivalent(Context, Prefix1, Prefix2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (NNS1->getKind()) {
|
|
|
|
case NestedNameSpecifier::Identifier:
|
|
|
|
return IsStructurallyEquivalent(NNS1->getAsIdentifier(),
|
|
|
|
NNS2->getAsIdentifier());
|
|
|
|
case NestedNameSpecifier::Namespace:
|
|
|
|
return IsStructurallyEquivalent(Context, NNS1->getAsNamespace(),
|
|
|
|
NNS2->getAsNamespace());
|
|
|
|
case NestedNameSpecifier::NamespaceAlias:
|
|
|
|
return IsStructurallyEquivalent(Context, NNS1->getAsNamespaceAlias(),
|
|
|
|
NNS2->getAsNamespaceAlias());
|
|
|
|
case NestedNameSpecifier::TypeSpec:
|
|
|
|
case NestedNameSpecifier::TypeSpecWithTemplate:
|
|
|
|
return IsStructurallyEquivalent(Context, QualType(NNS1->getAsType(), 0),
|
|
|
|
QualType(NNS2->getAsType(), 0));
|
|
|
|
case NestedNameSpecifier::Global:
|
|
|
|
return true;
|
|
|
|
case NestedNameSpecifier::Super:
|
|
|
|
return IsStructurallyEquivalent(Context, NNS1->getAsRecordDecl(),
|
|
|
|
NNS2->getAsRecordDecl());
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
const TemplateName &N1,
|
|
|
|
const TemplateName &N2) {
|
2019-07-23 23:46:38 +08:00
|
|
|
TemplateDecl *TemplateDeclN1 = N1.getAsTemplateDecl();
|
|
|
|
TemplateDecl *TemplateDeclN2 = N2.getAsTemplateDecl();
|
|
|
|
if (TemplateDeclN1 && TemplateDeclN2) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, TemplateDeclN1, TemplateDeclN2))
|
|
|
|
return false;
|
|
|
|
// If the kind is different we compare only the template decl.
|
|
|
|
if (N1.getKind() != N2.getKind())
|
|
|
|
return true;
|
|
|
|
} else if (TemplateDeclN1 || TemplateDeclN2)
|
|
|
|
return false;
|
|
|
|
else if (N1.getKind() != N2.getKind())
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
2019-07-23 23:46:38 +08:00
|
|
|
|
|
|
|
// Check for special case incompatibilities.
|
2017-04-28 08:31:30 +08:00
|
|
|
switch (N1.getKind()) {
|
|
|
|
|
|
|
|
case TemplateName::OverloadedTemplate: {
|
|
|
|
OverloadedTemplateStorage *OS1 = N1.getAsOverloadedTemplate(),
|
|
|
|
*OS2 = N2.getAsOverloadedTemplate();
|
|
|
|
OverloadedTemplateStorage::iterator I1 = OS1->begin(), I2 = OS2->begin(),
|
|
|
|
E1 = OS1->end(), E2 = OS2->end();
|
|
|
|
for (; I1 != E1 && I2 != E2; ++I1, ++I2)
|
|
|
|
if (!IsStructurallyEquivalent(Context, *I1, *I2))
|
|
|
|
return false;
|
|
|
|
return I1 == E1 && I2 == E2;
|
|
|
|
}
|
|
|
|
|
2019-05-09 11:31:27 +08:00
|
|
|
case TemplateName::AssumedTemplate: {
|
|
|
|
AssumedTemplateStorage *TN1 = N1.getAsAssumedTemplateName(),
|
|
|
|
*TN2 = N1.getAsAssumedTemplateName();
|
|
|
|
return TN1->getDeclName() == TN2->getDeclName();
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
case TemplateName::DependentTemplate: {
|
|
|
|
DependentTemplateName *DN1 = N1.getAsDependentTemplateName(),
|
|
|
|
*DN2 = N2.getAsDependentTemplateName();
|
|
|
|
if (!IsStructurallyEquivalent(Context, DN1->getQualifier(),
|
|
|
|
DN2->getQualifier()))
|
|
|
|
return false;
|
|
|
|
if (DN1->isIdentifier() && DN2->isIdentifier())
|
|
|
|
return IsStructurallyEquivalent(DN1->getIdentifier(),
|
|
|
|
DN2->getIdentifier());
|
|
|
|
else if (DN1->isOverloadedOperator() && DN2->isOverloadedOperator())
|
|
|
|
return DN1->getOperator() == DN2->getOperator();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
case TemplateName::SubstTemplateTemplateParmPack: {
|
|
|
|
SubstTemplateTemplateParmPackStorage
|
|
|
|
*P1 = N1.getAsSubstTemplateTemplateParmPack(),
|
|
|
|
*P2 = N2.getAsSubstTemplateTemplateParmPack();
|
|
|
|
return IsStructurallyEquivalent(Context, P1->getArgumentPack(),
|
|
|
|
P2->getArgumentPack()) &&
|
|
|
|
IsStructurallyEquivalent(Context, P1->getParameterPack(),
|
|
|
|
P2->getParameterPack());
|
|
|
|
}
|
2019-07-23 23:46:38 +08:00
|
|
|
|
|
|
|
case TemplateName::Template:
|
|
|
|
case TemplateName::QualifiedTemplate:
|
|
|
|
case TemplateName::SubstTemplateTemplateParm:
|
|
|
|
// It is sufficient to check value of getAsTemplateDecl.
|
|
|
|
break;
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
}
|
2019-07-23 23:46:38 +08:00
|
|
|
|
|
|
|
return true;
|
2017-04-28 08:31:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine whether two template arguments are equivalent.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
const TemplateArgument &Arg1,
|
|
|
|
const TemplateArgument &Arg2) {
|
|
|
|
if (Arg1.getKind() != Arg2.getKind())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (Arg1.getKind()) {
|
|
|
|
case TemplateArgument::Null:
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case TemplateArgument::Type:
|
2018-07-17 20:39:27 +08:00
|
|
|
return IsStructurallyEquivalent(Context, Arg1.getAsType(), Arg2.getAsType());
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
case TemplateArgument::Integral:
|
2018-07-17 20:39:27 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Arg1.getIntegralType(),
|
2017-04-28 08:31:30 +08:00
|
|
|
Arg2.getIntegralType()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return llvm::APSInt::isSameValue(Arg1.getAsIntegral(),
|
|
|
|
Arg2.getAsIntegral());
|
|
|
|
|
|
|
|
case TemplateArgument::Declaration:
|
2018-07-17 20:39:27 +08:00
|
|
|
return IsStructurallyEquivalent(Context, Arg1.getAsDecl(), Arg2.getAsDecl());
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
case TemplateArgument::NullPtr:
|
|
|
|
return true; // FIXME: Is this correct?
|
|
|
|
|
|
|
|
case TemplateArgument::Template:
|
|
|
|
return IsStructurallyEquivalent(Context, Arg1.getAsTemplate(),
|
|
|
|
Arg2.getAsTemplate());
|
|
|
|
|
|
|
|
case TemplateArgument::TemplateExpansion:
|
|
|
|
return IsStructurallyEquivalent(Context,
|
|
|
|
Arg1.getAsTemplateOrTemplatePattern(),
|
|
|
|
Arg2.getAsTemplateOrTemplatePattern());
|
|
|
|
|
|
|
|
case TemplateArgument::Expression:
|
|
|
|
return IsStructurallyEquivalent(Context, Arg1.getAsExpr(),
|
|
|
|
Arg2.getAsExpr());
|
|
|
|
|
|
|
|
case TemplateArgument::Pack:
|
|
|
|
if (Arg1.pack_size() != Arg2.pack_size())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = Arg1.pack_size(); I != N; ++I)
|
|
|
|
if (!IsStructurallyEquivalent(Context, Arg1.pack_begin()[I],
|
|
|
|
Arg2.pack_begin()[I]))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm_unreachable("Invalid template argument kind");
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine structural equivalence for the common part of array
|
|
|
|
/// types.
|
|
|
|
static bool IsArrayStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
const ArrayType *Array1,
|
|
|
|
const ArrayType *Array2) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Array1->getElementType(),
|
|
|
|
Array2->getElementType()))
|
|
|
|
return false;
|
|
|
|
if (Array1->getSizeModifier() != Array2->getSizeModifier())
|
|
|
|
return false;
|
|
|
|
if (Array1->getIndexTypeQualifiers() != Array2->getIndexTypeQualifiers())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-01-24 22:47:44 +08:00
|
|
|
/// Determine structural equivalence based on the ExtInfo of functions. This
|
|
|
|
/// is inspired by ASTContext::mergeFunctionTypes(), we compare calling
|
|
|
|
/// conventions bits but must not compare some other bits.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
FunctionType::ExtInfo EI1,
|
|
|
|
FunctionType::ExtInfo EI2) {
|
|
|
|
// Compatible functions must have compatible calling conventions.
|
|
|
|
if (EI1.getCC() != EI2.getCC())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Regparm is part of the calling convention.
|
|
|
|
if (EI1.getHasRegParm() != EI2.getHasRegParm())
|
|
|
|
return false;
|
|
|
|
if (EI1.getRegParm() != EI2.getRegParm())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (EI1.getProducesResult() != EI2.getProducesResult())
|
|
|
|
return false;
|
|
|
|
if (EI1.getNoCallerSavedRegs() != EI2.getNoCallerSavedRegs())
|
|
|
|
return false;
|
|
|
|
if (EI1.getNoCfCheck() != EI2.getNoCfCheck())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-05-08 23:23:48 +08:00
|
|
|
/// Check the equivalence of exception specifications.
|
|
|
|
static bool IsEquivalentExceptionSpec(StructuralEquivalenceContext &Context,
|
|
|
|
const FunctionProtoType *Proto1,
|
|
|
|
const FunctionProtoType *Proto2) {
|
|
|
|
|
|
|
|
auto Spec1 = Proto1->getExceptionSpecType();
|
|
|
|
auto Spec2 = Proto2->getExceptionSpecType();
|
|
|
|
|
|
|
|
if (isUnresolvedExceptionSpec(Spec1) || isUnresolvedExceptionSpec(Spec2))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (Spec1 != Spec2)
|
|
|
|
return false;
|
|
|
|
if (Spec1 == EST_Dynamic) {
|
|
|
|
if (Proto1->getNumExceptions() != Proto2->getNumExceptions())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Proto1->getNumExceptions(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Proto1->getExceptionType(I),
|
|
|
|
Proto2->getExceptionType(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (isComputedNoexcept(Spec1)) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Proto1->getNoexceptExpr(),
|
|
|
|
Proto2->getNoexceptExpr()))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
/// Determine structural equivalence of two types.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
QualType T1, QualType T2) {
|
|
|
|
if (T1.isNull() || T2.isNull())
|
|
|
|
return T1.isNull() && T2.isNull();
|
|
|
|
|
2018-07-11 17:37:24 +08:00
|
|
|
QualType OrigT1 = T1;
|
|
|
|
QualType OrigT2 = T2;
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!Context.StrictTypeSpelling) {
|
|
|
|
// We aren't being strict about token-to-token equivalence of types,
|
|
|
|
// so map down to the canonical type.
|
|
|
|
T1 = Context.FromCtx.getCanonicalType(T1);
|
|
|
|
T2 = Context.ToCtx.getCanonicalType(T2);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (T1.getQualifiers() != T2.getQualifiers())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Type::TypeClass TC = T1->getTypeClass();
|
|
|
|
|
|
|
|
if (T1->getTypeClass() != T2->getTypeClass()) {
|
|
|
|
// Compare function types with prototypes vs. without prototypes as if
|
|
|
|
// both did not have prototypes.
|
|
|
|
if (T1->getTypeClass() == Type::FunctionProto &&
|
|
|
|
T2->getTypeClass() == Type::FunctionNoProto)
|
|
|
|
TC = Type::FunctionNoProto;
|
|
|
|
else if (T1->getTypeClass() == Type::FunctionNoProto &&
|
|
|
|
T2->getTypeClass() == Type::FunctionProto)
|
|
|
|
TC = Type::FunctionNoProto;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (TC) {
|
|
|
|
case Type::Builtin:
|
|
|
|
// FIXME: Deal with Char_S/Char_U.
|
|
|
|
if (cast<BuiltinType>(T1)->getKind() != cast<BuiltinType>(T2)->getKind())
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::Complex:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<ComplexType>(T1)->getElementType(),
|
|
|
|
cast<ComplexType>(T2)->getElementType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::Adjusted:
|
|
|
|
case Type::Decayed:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<AdjustedType>(T1)->getOriginalType(),
|
|
|
|
cast<AdjustedType>(T2)->getOriginalType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::Pointer:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<PointerType>(T1)->getPointeeType(),
|
|
|
|
cast<PointerType>(T2)->getPointeeType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::BlockPointer:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<BlockPointerType>(T1)->getPointeeType(),
|
|
|
|
cast<BlockPointerType>(T2)->getPointeeType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::LValueReference:
|
|
|
|
case Type::RValueReference: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Ref1 = cast<ReferenceType>(T1);
|
|
|
|
const auto *Ref2 = cast<ReferenceType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (Ref1->isSpelledAsLValue() != Ref2->isSpelledAsLValue())
|
|
|
|
return false;
|
|
|
|
if (Ref1->isInnerRef() != Ref2->isInnerRef())
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Ref1->getPointeeTypeAsWritten(),
|
|
|
|
Ref2->getPointeeTypeAsWritten()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::MemberPointer: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *MemPtr1 = cast<MemberPointerType>(T1);
|
|
|
|
const auto *MemPtr2 = cast<MemberPointerType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, MemPtr1->getPointeeType(),
|
|
|
|
MemPtr2->getPointeeType()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, QualType(MemPtr1->getClass(), 0),
|
|
|
|
QualType(MemPtr2->getClass(), 0)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::ConstantArray: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Array1 = cast<ConstantArrayType>(T1);
|
|
|
|
const auto *Array2 = cast<ConstantArrayType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!llvm::APInt::isSameValue(Array1->getSize(), Array2->getSize()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!IsArrayStructurallyEquivalent(Context, Array1, Array2))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::IncompleteArray:
|
|
|
|
if (!IsArrayStructurallyEquivalent(Context, cast<ArrayType>(T1),
|
|
|
|
cast<ArrayType>(T2)))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::VariableArray: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Array1 = cast<VariableArrayType>(T1);
|
|
|
|
const auto *Array2 = cast<VariableArrayType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Array1->getSizeExpr(),
|
|
|
|
Array2->getSizeExpr()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!IsArrayStructurallyEquivalent(Context, Array1, Array2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::DependentSizedArray: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Array1 = cast<DependentSizedArrayType>(T1);
|
|
|
|
const auto *Array2 = cast<DependentSizedArrayType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Array1->getSizeExpr(),
|
|
|
|
Array2->getSizeExpr()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!IsArrayStructurallyEquivalent(Context, Array1, Array2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-10-02 14:25:51 +08:00
|
|
|
case Type::DependentAddressSpace: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *DepAddressSpace1 = cast<DependentAddressSpaceType>(T1);
|
|
|
|
const auto *DepAddressSpace2 = cast<DependentAddressSpaceType>(T2);
|
2017-10-02 14:25:51 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, DepAddressSpace1->getAddrSpaceExpr(),
|
|
|
|
DepAddressSpace2->getAddrSpaceExpr()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, DepAddressSpace1->getPointeeType(),
|
|
|
|
DepAddressSpace2->getPointeeType()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
case Type::DependentSizedExtVector: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Vec1 = cast<DependentSizedExtVectorType>(T1);
|
|
|
|
const auto *Vec2 = cast<DependentSizedExtVectorType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Vec1->getSizeExpr(),
|
|
|
|
Vec2->getSizeExpr()))
|
|
|
|
return false;
|
2018-07-14 03:46:04 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Vec1->getElementType(),
|
|
|
|
Vec2->getElementType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::DependentVector: {
|
|
|
|
const auto *Vec1 = cast<DependentVectorType>(T1);
|
|
|
|
const auto *Vec2 = cast<DependentVectorType>(T2);
|
|
|
|
if (Vec1->getVectorKind() != Vec2->getVectorKind())
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Vec1->getSizeExpr(),
|
|
|
|
Vec2->getSizeExpr()))
|
|
|
|
return false;
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Vec1->getElementType(),
|
|
|
|
Vec2->getElementType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::Vector:
|
|
|
|
case Type::ExtVector: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Vec1 = cast<VectorType>(T1);
|
|
|
|
const auto *Vec2 = cast<VectorType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Vec1->getElementType(),
|
|
|
|
Vec2->getElementType()))
|
|
|
|
return false;
|
|
|
|
if (Vec1->getNumElements() != Vec2->getNumElements())
|
|
|
|
return false;
|
|
|
|
if (Vec1->getVectorKind() != Vec2->getVectorKind())
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::FunctionProto: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Proto1 = cast<FunctionProtoType>(T1);
|
|
|
|
const auto *Proto2 = cast<FunctionProtoType>(T2);
|
2018-07-11 17:37:24 +08:00
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
if (Proto1->getNumParams() != Proto2->getNumParams())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Proto1->getNumParams(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Proto1->getParamType(I),
|
|
|
|
Proto2->getParamType(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (Proto1->isVariadic() != Proto2->isVariadic())
|
|
|
|
return false;
|
2018-07-11 17:37:24 +08:00
|
|
|
|
2019-01-28 19:37:49 +08:00
|
|
|
if (Proto1->getMethodQuals() != Proto2->getMethodQuals())
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2018-07-11 17:37:24 +08:00
|
|
|
// Check exceptions, this information is lost in canonical type.
|
|
|
|
const auto *OrigProto1 =
|
|
|
|
cast<FunctionProtoType>(OrigT1.getDesugaredType(Context.FromCtx));
|
|
|
|
const auto *OrigProto2 =
|
|
|
|
cast<FunctionProtoType>(OrigT2.getDesugaredType(Context.ToCtx));
|
2019-05-08 23:23:48 +08:00
|
|
|
if (!IsEquivalentExceptionSpec(Context, OrigProto1, OrigProto2))
|
2018-07-11 17:37:24 +08:00
|
|
|
return false;
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
// Fall through to check the bits common with FunctionNoProtoType.
|
2017-06-03 14:31:42 +08:00
|
|
|
LLVM_FALLTHROUGH;
|
2017-04-28 08:31:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
case Type::FunctionNoProto: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Function1 = cast<FunctionType>(T1);
|
|
|
|
const auto *Function2 = cast<FunctionType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Function1->getReturnType(),
|
|
|
|
Function2->getReturnType()))
|
|
|
|
return false;
|
2019-01-24 22:47:44 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Function1->getExtInfo(),
|
|
|
|
Function2->getExtInfo()))
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::UnresolvedUsing:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<UnresolvedUsingType>(T1)->getDecl(),
|
|
|
|
cast<UnresolvedUsingType>(T2)->getDecl()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::Attributed:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<AttributedType>(T1)->getModifiedType(),
|
|
|
|
cast<AttributedType>(T2)->getModifiedType()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(
|
|
|
|
Context, cast<AttributedType>(T1)->getEquivalentType(),
|
|
|
|
cast<AttributedType>(T2)->getEquivalentType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::Paren:
|
|
|
|
if (!IsStructurallyEquivalent(Context, cast<ParenType>(T1)->getInnerType(),
|
|
|
|
cast<ParenType>(T2)->getInnerType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
2019-05-07 11:20:17 +08:00
|
|
|
case Type::MacroQualified:
|
|
|
|
if (!IsStructurallyEquivalent(
|
|
|
|
Context, cast<MacroQualifiedType>(T1)->getUnderlyingType(),
|
|
|
|
cast<MacroQualifiedType>(T2)->getUnderlyingType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
case Type::Typedef:
|
|
|
|
if (!IsStructurallyEquivalent(Context, cast<TypedefType>(T1)->getDecl(),
|
|
|
|
cast<TypedefType>(T2)->getDecl()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::TypeOfExpr:
|
|
|
|
if (!IsStructurallyEquivalent(
|
|
|
|
Context, cast<TypeOfExprType>(T1)->getUnderlyingExpr(),
|
|
|
|
cast<TypeOfExprType>(T2)->getUnderlyingExpr()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::TypeOf:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<TypeOfType>(T1)->getUnderlyingType(),
|
|
|
|
cast<TypeOfType>(T2)->getUnderlyingType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::UnaryTransform:
|
|
|
|
if (!IsStructurallyEquivalent(
|
|
|
|
Context, cast<UnaryTransformType>(T1)->getUnderlyingType(),
|
2018-04-04 14:31:21 +08:00
|
|
|
cast<UnaryTransformType>(T2)->getUnderlyingType()))
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::Decltype:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<DecltypeType>(T1)->getUnderlyingExpr(),
|
|
|
|
cast<DecltypeType>(T2)->getUnderlyingExpr()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
2020-01-22 08:03:05 +08:00
|
|
|
case Type::Auto: {
|
|
|
|
auto *Auto1 = cast<AutoType>(T1);
|
|
|
|
auto *Auto2 = cast<AutoType>(T2);
|
|
|
|
if (!IsStructurallyEquivalent(Context, Auto1->getDeducedType(),
|
|
|
|
Auto2->getDeducedType()))
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
2020-01-22 08:03:05 +08:00
|
|
|
if (Auto1->isConstrained() != Auto2->isConstrained())
|
|
|
|
return false;
|
|
|
|
if (Auto1->isConstrained()) {
|
|
|
|
if (Auto1->getTypeConstraintConcept() !=
|
|
|
|
Auto2->getTypeConstraintConcept())
|
|
|
|
return false;
|
|
|
|
ArrayRef<TemplateArgument> Auto1Args =
|
|
|
|
Auto1->getTypeConstraintArguments();
|
|
|
|
ArrayRef<TemplateArgument> Auto2Args =
|
|
|
|
Auto2->getTypeConstraintArguments();
|
|
|
|
if (Auto1Args.size() != Auto2Args.size())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Auto1Args.size(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Auto1Args[I], Auto2Args[I]))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
break;
|
2020-01-22 08:03:05 +08:00
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
case Type::DeducedTemplateSpecialization: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *DT1 = cast<DeducedTemplateSpecializationType>(T1);
|
|
|
|
const auto *DT2 = cast<DeducedTemplateSpecializationType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, DT1->getTemplateName(),
|
|
|
|
DT2->getTemplateName()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, DT1->getDeducedType(),
|
|
|
|
DT2->getDeducedType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::Record:
|
|
|
|
case Type::Enum:
|
|
|
|
if (!IsStructurallyEquivalent(Context, cast<TagType>(T1)->getDecl(),
|
|
|
|
cast<TagType>(T2)->getDecl()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::TemplateTypeParm: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Parm1 = cast<TemplateTypeParmType>(T1);
|
|
|
|
const auto *Parm2 = cast<TemplateTypeParmType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (Parm1->getDepth() != Parm2->getDepth())
|
|
|
|
return false;
|
|
|
|
if (Parm1->getIndex() != Parm2->getIndex())
|
|
|
|
return false;
|
|
|
|
if (Parm1->isParameterPack() != Parm2->isParameterPack())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Names of template type parameters are never significant.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::SubstTemplateTypeParm: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Subst1 = cast<SubstTemplateTypeParmType>(T1);
|
|
|
|
const auto *Subst2 = cast<SubstTemplateTypeParmType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
QualType(Subst1->getReplacedParameter(), 0),
|
|
|
|
QualType(Subst2->getReplacedParameter(), 0)))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Subst1->getReplacementType(),
|
|
|
|
Subst2->getReplacementType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::SubstTemplateTypeParmPack: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Subst1 = cast<SubstTemplateTypeParmPackType>(T1);
|
|
|
|
const auto *Subst2 = cast<SubstTemplateTypeParmPackType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
QualType(Subst1->getReplacedParameter(), 0),
|
|
|
|
QualType(Subst2->getReplacedParameter(), 0)))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Subst1->getArgumentPack(),
|
|
|
|
Subst2->getArgumentPack()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
2018-04-10 06:14:10 +08:00
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
case Type::TemplateSpecialization: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Spec1 = cast<TemplateSpecializationType>(T1);
|
|
|
|
const auto *Spec2 = cast<TemplateSpecializationType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Spec1->getTemplateName(),
|
|
|
|
Spec2->getTemplateName()))
|
|
|
|
return false;
|
|
|
|
if (Spec1->getNumArgs() != Spec2->getNumArgs())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Spec1->getNumArgs(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Spec1->getArg(I),
|
|
|
|
Spec2->getArg(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::Elaborated: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Elab1 = cast<ElaboratedType>(T1);
|
|
|
|
const auto *Elab2 = cast<ElaboratedType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
// CHECKME: what if a keyword is ETK_None or ETK_typename ?
|
|
|
|
if (Elab1->getKeyword() != Elab2->getKeyword())
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Elab1->getQualifier(),
|
|
|
|
Elab2->getQualifier()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Elab1->getNamedType(),
|
|
|
|
Elab2->getNamedType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::InjectedClassName: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Inj1 = cast<InjectedClassNameType>(T1);
|
|
|
|
const auto *Inj2 = cast<InjectedClassNameType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
Inj1->getInjectedSpecializationType(),
|
|
|
|
Inj2->getInjectedSpecializationType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::DependentName: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Typename1 = cast<DependentNameType>(T1);
|
|
|
|
const auto *Typename2 = cast<DependentNameType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Typename1->getQualifier(),
|
|
|
|
Typename2->getQualifier()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Typename1->getIdentifier(),
|
|
|
|
Typename2->getIdentifier()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::DependentTemplateSpecialization: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Spec1 = cast<DependentTemplateSpecializationType>(T1);
|
|
|
|
const auto *Spec2 = cast<DependentTemplateSpecializationType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Spec1->getQualifier(),
|
|
|
|
Spec2->getQualifier()))
|
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Spec1->getIdentifier(),
|
|
|
|
Spec2->getIdentifier()))
|
|
|
|
return false;
|
|
|
|
if (Spec1->getNumArgs() != Spec2->getNumArgs())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Spec1->getNumArgs(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Spec1->getArg(I),
|
|
|
|
Spec2->getArg(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::PackExpansion:
|
|
|
|
if (!IsStructurallyEquivalent(Context,
|
|
|
|
cast<PackExpansionType>(T1)->getPattern(),
|
|
|
|
cast<PackExpansionType>(T2)->getPattern()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type::ObjCInterface: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Iface1 = cast<ObjCInterfaceType>(T1);
|
|
|
|
const auto *Iface2 = cast<ObjCInterfaceType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Iface1->getDecl(),
|
|
|
|
Iface2->getDecl()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::ObjCTypeParam: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Obj1 = cast<ObjCTypeParamType>(T1);
|
|
|
|
const auto *Obj2 = cast<ObjCTypeParamType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Obj1->getDecl(), Obj2->getDecl()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (Obj1->getNumProtocols() != Obj2->getNumProtocols())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Obj1->getNumProtocols(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Obj1->getProtocol(I),
|
|
|
|
Obj2->getProtocol(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2018-04-10 06:14:10 +08:00
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
case Type::ObjCObject: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Obj1 = cast<ObjCObjectType>(T1);
|
|
|
|
const auto *Obj2 = cast<ObjCObjectType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Obj1->getBaseType(),
|
|
|
|
Obj2->getBaseType()))
|
|
|
|
return false;
|
|
|
|
if (Obj1->getNumProtocols() != Obj2->getNumProtocols())
|
|
|
|
return false;
|
|
|
|
for (unsigned I = 0, N = Obj1->getNumProtocols(); I != N; ++I) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Obj1->getProtocol(I),
|
|
|
|
Obj2->getProtocol(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Type::ObjCObjectPointer: {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Ptr1 = cast<ObjCObjectPointerType>(T1);
|
|
|
|
const auto *Ptr2 = cast<ObjCObjectPointerType>(T2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Ptr1->getPointeeType(),
|
|
|
|
Ptr2->getPointeeType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-04-10 06:14:10 +08:00
|
|
|
case Type::Atomic:
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, cast<AtomicType>(T1)->getValueType(),
|
|
|
|
cast<AtomicType>(T2)->getValueType()))
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
2018-04-10 06:14:10 +08:00
|
|
|
case Type::Pipe:
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, cast<PipeType>(T1)->getElementType(),
|
|
|
|
cast<PipeType>(T2)->getElementType()))
|
|
|
|
return false;
|
|
|
|
break;
|
2020-04-18 01:44:19 +08:00
|
|
|
case Type::ExtInt: {
|
|
|
|
const auto *Int1 = cast<ExtIntType>(T1);
|
|
|
|
const auto *Int2 = cast<ExtIntType>(T2);
|
|
|
|
|
|
|
|
if (Int1->isUnsigned() != Int2->isUnsigned() ||
|
|
|
|
Int1->getNumBits() != Int2->getNumBits())
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Type::DependentExtInt: {
|
|
|
|
const auto *Int1 = cast<DependentExtIntType>(T1);
|
|
|
|
const auto *Int2 = cast<DependentExtIntType>(T2);
|
|
|
|
|
|
|
|
if (Int1->isUnsigned() != Int2->isUnsigned() ||
|
|
|
|
!IsStructurallyEquivalent(Context, Int1->getNumBitsExpr(),
|
|
|
|
Int2->getNumBitsExpr()))
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
} // end switch
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine structural equivalence of two fields.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
FieldDecl *Field1, FieldDecl *Field2) {
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Owner2 = cast<RecordDecl>(Field2->getDeclContext());
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
// For anonymous structs/unions, match up the anonymous struct/union type
|
|
|
|
// declarations directly, so that we don't go off searching for anonymous
|
|
|
|
// types
|
|
|
|
if (Field1->isAnonymousStructOrUnion() &&
|
|
|
|
Field2->isAnonymousStructOrUnion()) {
|
|
|
|
RecordDecl *D1 = Field1->getType()->castAs<RecordType>()->getDecl();
|
|
|
|
RecordDecl *D2 = Field2->getType()->castAs<RecordType>()->getDecl();
|
|
|
|
return IsStructurallyEquivalent(Context, D1, D2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for equivalent field names.
|
|
|
|
IdentifierInfo *Name1 = Field1->getIdentifier();
|
|
|
|
IdentifierInfo *Name2 = Field2->getIdentifier();
|
2017-07-01 08:06:47 +08:00
|
|
|
if (!::IsStructurallyEquivalent(Name1, Name2)) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(
|
|
|
|
Owner2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(diag::err_odr_tag_type_inconsistent))
|
2017-07-01 08:06:47 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(Owner2);
|
|
|
|
Context.Diag2(Field2->getLocation(), diag::note_odr_field_name)
|
|
|
|
<< Field2->getDeclName();
|
|
|
|
Context.Diag1(Field1->getLocation(), diag::note_odr_field_name)
|
|
|
|
<< Field1->getDeclName();
|
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
2017-07-01 08:06:47 +08:00
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
if (!IsStructurallyEquivalent(Context, Field1->getType(),
|
|
|
|
Field2->getType())) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(
|
|
|
|
Owner2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(Owner2);
|
|
|
|
Context.Diag2(Field2->getLocation(), diag::note_odr_field)
|
|
|
|
<< Field2->getDeclName() << Field2->getType();
|
|
|
|
Context.Diag1(Field1->getLocation(), diag::note_odr_field)
|
|
|
|
<< Field1->getDeclName() << Field1->getType();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Field1->isBitField() != Field2->isBitField()) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(
|
|
|
|
Owner2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(Owner2);
|
|
|
|
if (Field1->isBitField()) {
|
|
|
|
Context.Diag1(Field1->getLocation(), diag::note_odr_bit_field)
|
|
|
|
<< Field1->getDeclName() << Field1->getType()
|
|
|
|
<< Field1->getBitWidthValue(Context.FromCtx);
|
|
|
|
Context.Diag2(Field2->getLocation(), diag::note_odr_not_bit_field)
|
|
|
|
<< Field2->getDeclName();
|
|
|
|
} else {
|
|
|
|
Context.Diag2(Field2->getLocation(), diag::note_odr_bit_field)
|
|
|
|
<< Field2->getDeclName() << Field2->getType()
|
|
|
|
<< Field2->getBitWidthValue(Context.ToCtx);
|
|
|
|
Context.Diag1(Field1->getLocation(), diag::note_odr_not_bit_field)
|
|
|
|
<< Field1->getDeclName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Field1->isBitField()) {
|
|
|
|
// Make sure that the bit-fields are the same length.
|
|
|
|
unsigned Bits1 = Field1->getBitWidthValue(Context.FromCtx);
|
|
|
|
unsigned Bits2 = Field2->getBitWidthValue(Context.ToCtx);
|
|
|
|
|
|
|
|
if (Bits1 != Bits2) {
|
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(Owner2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(Owner2);
|
|
|
|
Context.Diag2(Field2->getLocation(), diag::note_odr_bit_field)
|
|
|
|
<< Field2->getDeclName() << Field2->getType() << Bits2;
|
|
|
|
Context.Diag1(Field1->getLocation(), diag::note_odr_bit_field)
|
|
|
|
<< Field1->getDeclName() << Field1->getType() << Bits1;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
Misc typos fixes in ./lib folder
Summary: Found via `codespell -q 3 -I ../clang-whitelist.txt -L uint,importd,crasher,gonna,cant,ue,ons,orign,ned`
Reviewers: teemperor
Reviewed By: teemperor
Subscribers: teemperor, jholewinski, jvesely, nhaehnle, whisperity, jfb, cfe-commits
Differential Revision: https://reviews.llvm.org/D55475
llvm-svn: 348755
2018-12-10 20:37:46 +08:00
|
|
|
/// Determine structural equivalence of two methods.
|
2018-07-11 17:37:24 +08:00
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
CXXMethodDecl *Method1,
|
|
|
|
CXXMethodDecl *Method2) {
|
|
|
|
bool PropertiesEqual =
|
|
|
|
Method1->getDeclKind() == Method2->getDeclKind() &&
|
|
|
|
Method1->getRefQualifier() == Method2->getRefQualifier() &&
|
|
|
|
Method1->getAccess() == Method2->getAccess() &&
|
|
|
|
Method1->getOverloadedOperator() == Method2->getOverloadedOperator() &&
|
|
|
|
Method1->isStatic() == Method2->isStatic() &&
|
|
|
|
Method1->isConst() == Method2->isConst() &&
|
|
|
|
Method1->isVolatile() == Method2->isVolatile() &&
|
|
|
|
Method1->isVirtual() == Method2->isVirtual() &&
|
|
|
|
Method1->isPure() == Method2->isPure() &&
|
|
|
|
Method1->isDefaulted() == Method2->isDefaulted() &&
|
|
|
|
Method1->isDeleted() == Method2->isDeleted();
|
|
|
|
if (!PropertiesEqual)
|
|
|
|
return false;
|
|
|
|
// FIXME: Check for 'final'.
|
|
|
|
|
|
|
|
if (auto *Constructor1 = dyn_cast<CXXConstructorDecl>(Method1)) {
|
|
|
|
auto *Constructor2 = cast<CXXConstructorDecl>(Method2);
|
2019-05-09 11:59:21 +08:00
|
|
|
if (!Constructor1->getExplicitSpecifier().isEquivalent(
|
|
|
|
Constructor2->getExplicitSpecifier()))
|
2018-07-11 17:37:24 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *Conversion1 = dyn_cast<CXXConversionDecl>(Method1)) {
|
|
|
|
auto *Conversion2 = cast<CXXConversionDecl>(Method2);
|
2019-05-09 11:59:21 +08:00
|
|
|
if (!Conversion1->getExplicitSpecifier().isEquivalent(
|
|
|
|
Conversion2->getExplicitSpecifier()))
|
2018-07-11 17:37:24 +08:00
|
|
|
return false;
|
|
|
|
if (!IsStructurallyEquivalent(Context, Conversion1->getConversionType(),
|
|
|
|
Conversion2->getConversionType()))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const IdentifierInfo *Name1 = Method1->getIdentifier();
|
|
|
|
const IdentifierInfo *Name2 = Method2->getIdentifier();
|
|
|
|
if (!::IsStructurallyEquivalent(Name1, Name2)) {
|
|
|
|
return false;
|
|
|
|
// TODO: Names do not match, add warning like at check for FieldDecl.
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the prototypes.
|
|
|
|
if (!::IsStructurallyEquivalent(Context,
|
|
|
|
Method1->getType(), Method2->getType()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-07-17 22:40:09 +08:00
|
|
|
/// Determine structural equivalence of two lambda classes.
|
|
|
|
static bool
|
|
|
|
IsStructurallyEquivalentLambdas(StructuralEquivalenceContext &Context,
|
|
|
|
CXXRecordDecl *D1, CXXRecordDecl *D2) {
|
|
|
|
assert(D1->isLambda() && D2->isLambda() &&
|
|
|
|
"Must be called on lambda classes");
|
|
|
|
if (!IsStructurallyEquivalent(Context, D1->getLambdaCallOperator(),
|
|
|
|
D2->getLambdaCallOperator()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
/// Determine structural equivalence of two records.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
RecordDecl *D1, RecordDecl *D2) {
|
|
|
|
if (D1->isUnion() != D2->isUnion()) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(), Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_tag_kind_here)
|
|
|
|
<< D1->getDeclName() << (unsigned)D1->getTagKind();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-07-17 20:06:36 +08:00
|
|
|
if (!D1->getDeclName() && !D2->getDeclName()) {
|
2017-04-28 08:31:30 +08:00
|
|
|
// If both anonymous structs/unions are in a record context, make sure
|
|
|
|
// they occur in the same location in the context records.
|
|
|
|
if (Optional<unsigned> Index1 =
|
|
|
|
StructuralEquivalenceContext::findUntaggedStructOrUnionIndex(D1)) {
|
|
|
|
if (Optional<unsigned> Index2 =
|
|
|
|
StructuralEquivalenceContext::findUntaggedStructOrUnionIndex(
|
|
|
|
D2)) {
|
|
|
|
if (*Index1 != *Index2)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If both declarations are class template specializations, we know
|
|
|
|
// the ODR applies, so check the template and template arguments.
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Spec1 = dyn_cast<ClassTemplateSpecializationDecl>(D1);
|
|
|
|
const auto *Spec2 = dyn_cast<ClassTemplateSpecializationDecl>(D2);
|
2017-04-28 08:31:30 +08:00
|
|
|
if (Spec1 && Spec2) {
|
|
|
|
// Check that the specialized templates are the same.
|
|
|
|
if (!IsStructurallyEquivalent(Context, Spec1->getSpecializedTemplate(),
|
|
|
|
Spec2->getSpecializedTemplate()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check that the template arguments are the same.
|
|
|
|
if (Spec1->getTemplateArgs().size() != Spec2->getTemplateArgs().size())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = Spec1->getTemplateArgs().size(); I != N; ++I)
|
|
|
|
if (!IsStructurallyEquivalent(Context, Spec1->getTemplateArgs().get(I),
|
|
|
|
Spec2->getTemplateArgs().get(I)))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// If one is a class template specialization and the other is not, these
|
|
|
|
// structures are different.
|
|
|
|
else if (Spec1 || Spec2)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Compare the definitions of these two records. If either or both are
|
2018-11-26 23:54:08 +08:00
|
|
|
// incomplete (i.e. it is a forward decl), we assume that they are
|
|
|
|
// equivalent.
|
2017-04-28 08:31:30 +08:00
|
|
|
D1 = D1->getDefinition();
|
|
|
|
D2 = D2->getDefinition();
|
|
|
|
if (!D1 || !D2)
|
|
|
|
return true;
|
|
|
|
|
[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
|
|
|
// If any of the records has external storage and we do a minimal check (or
|
2018-08-08 23:04:27 +08:00
|
|
|
// AST import) we assume they are equivalent. (If we didn't have this
|
[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
|
|
|
// assumption then `RecordDecl::LoadFieldsFromExternalStorage` could trigger
|
|
|
|
// another AST import which in turn would call the structural equivalency
|
|
|
|
// check again and finally we'd have an improper result.)
|
|
|
|
if (Context.EqKind == StructuralEquivalenceKind::Minimal)
|
|
|
|
if (D1->hasExternalLexicalStorage() || D2->hasExternalLexicalStorage())
|
|
|
|
return true;
|
|
|
|
|
2018-11-26 23:54:08 +08:00
|
|
|
// If one definition is currently being defined, we do not compare for
|
|
|
|
// equality and we assume that the decls are equal.
|
|
|
|
if (D1->isBeingDefined() || D2->isBeingDefined())
|
|
|
|
return true;
|
|
|
|
|
2018-04-10 06:14:10 +08:00
|
|
|
if (auto *D1CXX = dyn_cast<CXXRecordDecl>(D1)) {
|
|
|
|
if (auto *D2CXX = dyn_cast<CXXRecordDecl>(D2)) {
|
2017-05-13 08:46:33 +08:00
|
|
|
if (D1CXX->hasExternalLexicalStorage() &&
|
|
|
|
!D1CXX->isCompleteDefinition()) {
|
|
|
|
D1CXX->getASTContext().getExternalSource()->CompleteType(D1CXX);
|
|
|
|
}
|
|
|
|
|
2019-07-17 22:40:09 +08:00
|
|
|
if (D1CXX->isLambda() != D2CXX->isLambda())
|
|
|
|
return false;
|
|
|
|
if (D1CXX->isLambda()) {
|
|
|
|
if (!IsStructurallyEquivalentLambdas(Context, D1CXX, D2CXX))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
if (D1CXX->getNumBases() != D2CXX->getNumBases()) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag2(D2->getLocation(), diag::note_odr_number_of_bases)
|
|
|
|
<< D2CXX->getNumBases();
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_number_of_bases)
|
|
|
|
<< D1CXX->getNumBases();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the base classes.
|
|
|
|
for (CXXRecordDecl::base_class_iterator Base1 = D1CXX->bases_begin(),
|
|
|
|
BaseEnd1 = D1CXX->bases_end(),
|
|
|
|
Base2 = D2CXX->bases_begin();
|
|
|
|
Base1 != BaseEnd1; ++Base1, ++Base2) {
|
|
|
|
if (!IsStructurallyEquivalent(Context, Base1->getType(),
|
|
|
|
Base2->getType())) {
|
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
2018-08-10 05:08:08 +08:00
|
|
|
Context.Diag2(Base2->getBeginLoc(), diag::note_odr_base)
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Base2->getType() << Base2->getSourceRange();
|
2018-08-10 05:08:08 +08:00
|
|
|
Context.Diag1(Base1->getBeginLoc(), diag::note_odr_base)
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Base1->getType() << Base1->getSourceRange();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check virtual vs. non-virtual inheritance mismatch.
|
|
|
|
if (Base1->isVirtual() != Base2->isVirtual()) {
|
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
2018-08-10 05:08:08 +08:00
|
|
|
Context.Diag2(Base2->getBeginLoc(), diag::note_odr_virtual_base)
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Base2->isVirtual() << Base2->getSourceRange();
|
2018-08-10 05:08:08 +08:00
|
|
|
Context.Diag1(Base1->getBeginLoc(), diag::note_odr_base)
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Base1->isVirtual() << Base1->getSourceRange();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2018-04-26 01:28:03 +08:00
|
|
|
|
|
|
|
// Check the friends for consistency.
|
|
|
|
CXXRecordDecl::friend_iterator Friend2 = D2CXX->friend_begin(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Friend2End = D2CXX->friend_end();
|
2018-04-26 01:28:03 +08:00
|
|
|
for (CXXRecordDecl::friend_iterator Friend1 = D1CXX->friend_begin(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Friend1End = D1CXX->friend_end();
|
2018-04-26 01:28:03 +08:00
|
|
|
Friend1 != Friend1End; ++Friend1, ++Friend2) {
|
|
|
|
if (Friend2 == Friend2End) {
|
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
|
|
|
<< Context.ToCtx.getTypeDeclType(D2CXX);
|
2018-04-26 01:28:03 +08:00
|
|
|
Context.Diag1((*Friend1)->getFriendLoc(), diag::note_odr_friend);
|
|
|
|
Context.Diag2(D2->getLocation(), diag::note_odr_missing_friend);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsStructurallyEquivalent(Context, *Friend1, *Friend2)) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
|
|
|
<< Context.ToCtx.getTypeDeclType(D2CXX);
|
2018-04-26 01:28:03 +08:00
|
|
|
Context.Diag1((*Friend1)->getFriendLoc(), diag::note_odr_friend);
|
|
|
|
Context.Diag2((*Friend2)->getFriendLoc(), diag::note_odr_friend);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Friend2 != Friend2End) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
2018-04-26 01:28:03 +08:00
|
|
|
Context.Diag2((*Friend2)->getFriendLoc(), diag::note_odr_friend);
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_missing_friend);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-28 08:31:30 +08:00
|
|
|
} else if (D1CXX->getNumBases() > 0) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
const CXXBaseSpecifier *Base1 = D1CXX->bases_begin();
|
2018-08-10 05:08:08 +08:00
|
|
|
Context.Diag1(Base1->getBeginLoc(), diag::note_odr_base)
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Base1->getType() << Base1->getSourceRange();
|
|
|
|
Context.Diag2(D2->getLocation(), diag::note_odr_missing_base);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the fields for consistency.
|
|
|
|
RecordDecl::field_iterator Field2 = D2->field_begin(),
|
|
|
|
Field2End = D2->field_end();
|
|
|
|
for (RecordDecl::field_iterator Field1 = D1->field_begin(),
|
|
|
|
Field1End = D1->field_end();
|
|
|
|
Field1 != Field1End; ++Field1, ++Field2) {
|
|
|
|
if (Field2 == Field2End) {
|
|
|
|
if (Context.Complain) {
|
2017-07-01 08:06:47 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag1(Field1->getLocation(), diag::note_odr_field)
|
|
|
|
<< Field1->getDeclName() << Field1->getType();
|
|
|
|
Context.Diag2(D2->getLocation(), diag::note_odr_missing_field);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsStructurallyEquivalent(Context, *Field1, *Field2))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Field2 != Field2End) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(), Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag2(Field2->getLocation(), diag::note_odr_field)
|
|
|
|
<< Field2->getDeclName() << Field2->getType();
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_missing_field);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Determine structural equivalence of two enums.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
EnumDecl *D1, EnumDecl *D2) {
|
2018-08-09 20:36:25 +08:00
|
|
|
|
|
|
|
// Compare the definitions of these two enums. If either or both are
|
|
|
|
// incomplete (i.e. forward declared), we assume that they are equivalent.
|
|
|
|
D1 = D1->getDefinition();
|
|
|
|
D2 = D2->getDefinition();
|
|
|
|
if (!D1 || !D2)
|
|
|
|
return true;
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
EnumDecl::enumerator_iterator EC2 = D2->enumerator_begin(),
|
|
|
|
EC2End = D2->enumerator_end();
|
|
|
|
for (EnumDecl::enumerator_iterator EC1 = D1->enumerator_begin(),
|
|
|
|
EC1End = D1->enumerator_end();
|
|
|
|
EC1 != EC1End; ++EC1, ++EC2) {
|
|
|
|
if (EC2 == EC2End) {
|
|
|
|
if (Context.Complain) {
|
2017-07-01 08:06:47 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag1(EC1->getLocation(), diag::note_odr_enumerator)
|
|
|
|
<< EC1->getDeclName() << EC1->getInitVal().toString(10);
|
|
|
|
Context.Diag2(D2->getLocation(), diag::note_odr_missing_enumerator);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::APSInt Val1 = EC1->getInitVal();
|
|
|
|
llvm::APSInt Val2 = EC2->getInitVal();
|
|
|
|
if (!llvm::APSInt::isSameValue(Val1, Val2) ||
|
|
|
|
!IsStructurallyEquivalent(EC1->getIdentifier(), EC2->getIdentifier())) {
|
|
|
|
if (Context.Complain) {
|
2017-07-01 08:06:47 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag2(EC2->getLocation(), diag::note_odr_enumerator)
|
|
|
|
<< EC2->getDeclName() << EC2->getInitVal().toString(10);
|
|
|
|
Context.Diag1(EC1->getLocation(), diag::note_odr_enumerator)
|
|
|
|
<< EC1->getDeclName() << EC1->getInitVal().toString(10);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EC2 != EC2End) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(), Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_tag_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Context.ToCtx.getTypeDeclType(D2);
|
|
|
|
Context.Diag2(EC2->getLocation(), diag::note_odr_enumerator)
|
|
|
|
<< EC2->getDeclName() << EC2->getInitVal().toString(10);
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_missing_enumerator);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
TemplateParameterList *Params1,
|
|
|
|
TemplateParameterList *Params2) {
|
|
|
|
if (Params1->size() != Params2->size()) {
|
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(Params2->getTemplateLoc(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_different_num_template_parameters))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< Params1->size() << Params2->size();
|
|
|
|
Context.Diag1(Params1->getTemplateLoc(),
|
|
|
|
diag::note_odr_template_parameter_list);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = Params1->size(); I != N; ++I) {
|
|
|
|
if (Params1->getParam(I)->getKind() != Params2->getParam(I)->getKind()) {
|
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(Params2->getParam(I)->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_different_template_parameter_kind));
|
2017-04-28 08:31:30 +08:00
|
|
|
Context.Diag1(Params1->getParam(I)->getLocation(),
|
|
|
|
diag::note_odr_template_parameter_here);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-07-17 20:39:27 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, Params1->getParam(I),
|
|
|
|
Params2->getParam(I)))
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
TemplateTypeParmDecl *D1,
|
|
|
|
TemplateTypeParmDecl *D2) {
|
|
|
|
if (D1->isParameterPack() != D2->isParameterPack()) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_parameter_pack_non_pack))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< D2->isParameterPack();
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_parameter_pack_non_pack)
|
|
|
|
<< D1->isParameterPack();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
NonTypeTemplateParmDecl *D1,
|
|
|
|
NonTypeTemplateParmDecl *D2) {
|
|
|
|
if (D1->isParameterPack() != D2->isParameterPack()) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_parameter_pack_non_pack))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< D2->isParameterPack();
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_parameter_pack_non_pack)
|
|
|
|
<< D1->isParameterPack();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check types.
|
2018-07-17 20:39:27 +08:00
|
|
|
if (!IsStructurallyEquivalent(Context, D1->getType(), D2->getType())) {
|
2017-04-28 08:31:30 +08:00
|
|
|
if (Context.Complain) {
|
|
|
|
Context.Diag2(D2->getLocation(),
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_non_type_parameter_type_inconsistent))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< D2->getType() << D1->getType();
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_value_here)
|
|
|
|
<< D1->getType();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
TemplateTemplateParmDecl *D1,
|
|
|
|
TemplateTemplateParmDecl *D2) {
|
|
|
|
if (D1->isParameterPack() != D2->isParameterPack()) {
|
|
|
|
if (Context.Complain) {
|
2019-04-01 22:46:53 +08:00
|
|
|
Context.Diag2(D2->getLocation(),
|
|
|
|
Context.getApplicableDiagnostic(
|
|
|
|
diag::err_odr_parameter_pack_non_pack))
|
2017-04-28 08:31:30 +08:00
|
|
|
<< D2->isParameterPack();
|
|
|
|
Context.Diag1(D1->getLocation(), diag::note_odr_parameter_pack_non_pack)
|
|
|
|
<< D1->isParameterPack();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check template parameter lists.
|
|
|
|
return IsStructurallyEquivalent(Context, D1->getTemplateParameters(),
|
|
|
|
D2->getTemplateParameters());
|
|
|
|
}
|
|
|
|
|
2018-01-26 19:36:54 +08:00
|
|
|
static bool IsTemplateDeclCommonStructurallyEquivalent(
|
|
|
|
StructuralEquivalenceContext &Ctx, TemplateDecl *D1, TemplateDecl *D2) {
|
|
|
|
if (!IsStructurallyEquivalent(D1->getIdentifier(), D2->getIdentifier()))
|
|
|
|
return false;
|
|
|
|
if (!D1->getIdentifier()) // Special name
|
|
|
|
if (D1->getNameAsString() != D2->getNameAsString())
|
|
|
|
return false;
|
|
|
|
return IsStructurallyEquivalent(Ctx, D1->getTemplateParameters(),
|
|
|
|
D2->getTemplateParameters());
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
ClassTemplateDecl *D1,
|
|
|
|
ClassTemplateDecl *D2) {
|
|
|
|
// Check template parameters.
|
2018-01-26 19:36:54 +08:00
|
|
|
if (!IsTemplateDeclCommonStructurallyEquivalent(Context, D1, D2))
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check the templated declaration.
|
2018-07-17 20:39:27 +08:00
|
|
|
return IsStructurallyEquivalent(Context, D1->getTemplatedDecl(),
|
|
|
|
D2->getTemplatedDecl());
|
2017-04-28 08:31:30 +08:00
|
|
|
}
|
|
|
|
|
2018-01-26 19:36:54 +08:00
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
FunctionTemplateDecl *D1,
|
|
|
|
FunctionTemplateDecl *D2) {
|
|
|
|
// Check template parameters.
|
|
|
|
if (!IsTemplateDeclCommonStructurallyEquivalent(Context, D1, D2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check the templated declaration.
|
2018-07-17 20:39:27 +08:00
|
|
|
return IsStructurallyEquivalent(Context, D1->getTemplatedDecl()->getType(),
|
|
|
|
D2->getTemplatedDecl()->getType());
|
2018-01-26 19:36:54 +08:00
|
|
|
}
|
|
|
|
|
2019-07-11 05:25:49 +08:00
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
ConceptDecl *D1,
|
|
|
|
ConceptDecl *D2) {
|
|
|
|
// Check template parameters.
|
|
|
|
if (!IsTemplateDeclCommonStructurallyEquivalent(Context, D1, D2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check the constraint expression.
|
|
|
|
return IsStructurallyEquivalent(Context, D1->getConstraintExpr(),
|
|
|
|
D2->getConstraintExpr());
|
|
|
|
}
|
|
|
|
|
2018-04-26 01:28:03 +08:00
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
FriendDecl *D1, FriendDecl *D2) {
|
|
|
|
if ((D1->getFriendType() && D2->getFriendDecl()) ||
|
|
|
|
(D1->getFriendDecl() && D2->getFriendType())) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (D1->getFriendType() && D2->getFriendType())
|
|
|
|
return IsStructurallyEquivalent(Context,
|
|
|
|
D1->getFriendType()->getType(),
|
|
|
|
D2->getFriendType()->getType());
|
|
|
|
if (D1->getFriendDecl() && D2->getFriendDecl())
|
|
|
|
return IsStructurallyEquivalent(Context, D1->getFriendDecl(),
|
|
|
|
D2->getFriendDecl());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
FunctionDecl *D1, FunctionDecl *D2) {
|
|
|
|
// FIXME: Consider checking for function attributes as well.
|
|
|
|
if (!IsStructurallyEquivalent(Context, D1->getType(), D2->getType()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
/// Determine structural equivalence of two declarations.
|
|
|
|
static bool IsStructurallyEquivalent(StructuralEquivalenceContext &Context,
|
|
|
|
Decl *D1, Decl *D2) {
|
|
|
|
// FIXME: Check for known structural equivalences via a callback of some sort.
|
|
|
|
|
2019-09-02 19:01:09 +08:00
|
|
|
D1 = D1->getCanonicalDecl();
|
|
|
|
D2 = D2->getCanonicalDecl();
|
|
|
|
std::pair<Decl *, Decl *> P{D1, D2};
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
// Check whether we already know that these two declarations are not
|
|
|
|
// structurally equivalent.
|
2019-09-02 19:01:09 +08:00
|
|
|
if (Context.NonEquivalentDecls.count(P))
|
2017-04-28 08:31:30 +08:00
|
|
|
return false;
|
|
|
|
|
2019-09-02 19:01:09 +08:00
|
|
|
// Check if a check for these declarations is already pending.
|
|
|
|
// If yes D1 and D2 will be checked later (from DeclsToCheck),
|
|
|
|
// or these are already checked (and equivalent).
|
|
|
|
bool Inserted = Context.VisitedDecls.insert(P).second;
|
|
|
|
if (!Inserted)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
Context.DeclsToCheck.push(P);
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DiagnosticBuilder StructuralEquivalenceContext::Diag1(SourceLocation Loc,
|
|
|
|
unsigned DiagID) {
|
|
|
|
assert(Complain && "Not allowed to complain");
|
|
|
|
if (LastDiagFromC2)
|
|
|
|
FromCtx.getDiagnostics().notePriorDiagnosticFrom(ToCtx.getDiagnostics());
|
|
|
|
LastDiagFromC2 = false;
|
|
|
|
return FromCtx.getDiagnostics().Report(Loc, DiagID);
|
|
|
|
}
|
|
|
|
|
|
|
|
DiagnosticBuilder StructuralEquivalenceContext::Diag2(SourceLocation Loc,
|
|
|
|
unsigned DiagID) {
|
|
|
|
assert(Complain && "Not allowed to complain");
|
|
|
|
if (!LastDiagFromC2)
|
|
|
|
ToCtx.getDiagnostics().notePriorDiagnosticFrom(FromCtx.getDiagnostics());
|
|
|
|
LastDiagFromC2 = true;
|
|
|
|
return ToCtx.getDiagnostics().Report(Loc, DiagID);
|
|
|
|
}
|
|
|
|
|
|
|
|
Optional<unsigned>
|
|
|
|
StructuralEquivalenceContext::findUntaggedStructOrUnionIndex(RecordDecl *Anon) {
|
|
|
|
ASTContext &Context = Anon->getASTContext();
|
|
|
|
QualType AnonTy = Context.getRecordType(Anon);
|
|
|
|
|
2018-04-10 06:14:10 +08:00
|
|
|
const auto *Owner = dyn_cast<RecordDecl>(Anon->getDeclContext());
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!Owner)
|
|
|
|
return None;
|
|
|
|
|
|
|
|
unsigned Index = 0;
|
|
|
|
for (const auto *D : Owner->noload_decls()) {
|
|
|
|
const auto *F = dyn_cast<FieldDecl>(D);
|
|
|
|
if (!F)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (F->isAnonymousStructOrUnion()) {
|
|
|
|
if (Context.hasSameType(F->getType(), AnonTy))
|
|
|
|
break;
|
|
|
|
++Index;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the field looks like this:
|
|
|
|
// struct { ... } A;
|
|
|
|
QualType FieldType = F->getType();
|
2018-04-05 23:31:49 +08:00
|
|
|
// In case of nested structs.
|
|
|
|
while (const auto *ElabType = dyn_cast<ElaboratedType>(FieldType))
|
|
|
|
FieldType = ElabType->getNamedType();
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
if (const auto *RecType = dyn_cast<RecordType>(FieldType)) {
|
|
|
|
const RecordDecl *RecDecl = RecType->getDecl();
|
|
|
|
if (RecDecl->getDeclContext() == Owner && !RecDecl->getIdentifier()) {
|
|
|
|
if (Context.hasSameType(FieldType, AnonTy))
|
|
|
|
break;
|
|
|
|
++Index;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
|
2019-04-01 22:46:53 +08:00
|
|
|
unsigned StructuralEquivalenceContext::getApplicableDiagnostic(
|
|
|
|
unsigned ErrorDiagnostic) {
|
|
|
|
if (ErrorOnTagTypeMismatch)
|
|
|
|
return ErrorDiagnostic;
|
|
|
|
|
|
|
|
switch (ErrorDiagnostic) {
|
|
|
|
case diag::err_odr_variable_type_inconsistent:
|
|
|
|
return diag::warn_odr_variable_type_inconsistent;
|
|
|
|
case diag::err_odr_variable_multiple_def:
|
|
|
|
return diag::warn_odr_variable_multiple_def;
|
|
|
|
case diag::err_odr_function_type_inconsistent:
|
|
|
|
return diag::warn_odr_function_type_inconsistent;
|
|
|
|
case diag::err_odr_tag_type_inconsistent:
|
|
|
|
return diag::warn_odr_tag_type_inconsistent;
|
|
|
|
case diag::err_odr_field_type_inconsistent:
|
|
|
|
return diag::warn_odr_field_type_inconsistent;
|
|
|
|
case diag::err_odr_ivar_type_inconsistent:
|
|
|
|
return diag::warn_odr_ivar_type_inconsistent;
|
|
|
|
case diag::err_odr_objc_superclass_inconsistent:
|
|
|
|
return diag::warn_odr_objc_superclass_inconsistent;
|
|
|
|
case diag::err_odr_objc_method_result_type_inconsistent:
|
|
|
|
return diag::warn_odr_objc_method_result_type_inconsistent;
|
|
|
|
case diag::err_odr_objc_method_num_params_inconsistent:
|
|
|
|
return diag::warn_odr_objc_method_num_params_inconsistent;
|
|
|
|
case diag::err_odr_objc_method_param_type_inconsistent:
|
|
|
|
return diag::warn_odr_objc_method_param_type_inconsistent;
|
|
|
|
case diag::err_odr_objc_method_variadic_inconsistent:
|
|
|
|
return diag::warn_odr_objc_method_variadic_inconsistent;
|
|
|
|
case diag::err_odr_objc_property_type_inconsistent:
|
|
|
|
return diag::warn_odr_objc_property_type_inconsistent;
|
|
|
|
case diag::err_odr_objc_property_impl_kind_inconsistent:
|
|
|
|
return diag::warn_odr_objc_property_impl_kind_inconsistent;
|
|
|
|
case diag::err_odr_objc_synthesize_ivar_inconsistent:
|
|
|
|
return diag::warn_odr_objc_synthesize_ivar_inconsistent;
|
|
|
|
case diag::err_odr_different_num_template_parameters:
|
|
|
|
return diag::warn_odr_different_num_template_parameters;
|
|
|
|
case diag::err_odr_different_template_parameter_kind:
|
|
|
|
return diag::warn_odr_different_template_parameter_kind;
|
|
|
|
case diag::err_odr_parameter_pack_non_pack:
|
|
|
|
return diag::warn_odr_parameter_pack_non_pack;
|
|
|
|
case diag::err_odr_non_type_parameter_type_inconsistent:
|
|
|
|
return diag::warn_odr_non_type_parameter_type_inconsistent;
|
|
|
|
}
|
2019-04-01 23:48:29 +08:00
|
|
|
llvm_unreachable("Diagnostic kind not handled in preceding switch");
|
2019-04-01 22:46:53 +08:00
|
|
|
}
|
|
|
|
|
2018-07-17 20:39:27 +08:00
|
|
|
bool StructuralEquivalenceContext::IsEquivalent(Decl *D1, Decl *D2) {
|
|
|
|
|
|
|
|
// Ensure that the implementation functions (all static functions in this TU)
|
|
|
|
// never call the public ASTStructuralEquivalence::IsEquivalent() functions,
|
|
|
|
// because that will wreak havoc the internal state (DeclsToCheck and
|
2019-09-02 19:01:09 +08:00
|
|
|
// VisitedDecls members) and can cause faulty behaviour.
|
|
|
|
// In other words: Do not start a graph search from a new node with the
|
|
|
|
// internal data of another search in progress.
|
|
|
|
// FIXME: Better encapsulation and separation of internal and public
|
|
|
|
// functionality.
|
2018-07-17 20:39:27 +08:00
|
|
|
assert(DeclsToCheck.empty());
|
2019-09-02 19:01:09 +08:00
|
|
|
assert(VisitedDecls.empty());
|
2018-07-17 20:39:27 +08:00
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!::IsStructurallyEquivalent(*this, D1, D2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return !Finish();
|
|
|
|
}
|
|
|
|
|
2018-07-17 20:39:27 +08:00
|
|
|
bool StructuralEquivalenceContext::IsEquivalent(QualType T1, QualType T2) {
|
|
|
|
assert(DeclsToCheck.empty());
|
2019-09-02 19:01:09 +08:00
|
|
|
assert(VisitedDecls.empty());
|
2017-04-28 08:31:30 +08:00
|
|
|
if (!::IsStructurallyEquivalent(*this, T1, T2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return !Finish();
|
|
|
|
}
|
|
|
|
|
2018-08-08 23:04:27 +08:00
|
|
|
bool StructuralEquivalenceContext::CheckCommonEquivalence(Decl *D1, Decl *D2) {
|
|
|
|
// Check for equivalent described template.
|
|
|
|
TemplateDecl *Template1 = D1->getDescribedTemplate();
|
|
|
|
TemplateDecl *Template2 = D2->getDescribedTemplate();
|
|
|
|
if ((Template1 != nullptr) != (Template2 != nullptr))
|
|
|
|
return false;
|
|
|
|
if (Template1 && !IsStructurallyEquivalent(*this, Template1, Template2))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// FIXME: Move check for identifier names into this function.
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool StructuralEquivalenceContext::CheckKindSpecificEquivalence(
|
|
|
|
Decl *D1, Decl *D2) {
|
|
|
|
// FIXME: Switch on all declaration kinds. For now, we're just going to
|
|
|
|
// check the obvious ones.
|
|
|
|
if (auto *Record1 = dyn_cast<RecordDecl>(D1)) {
|
|
|
|
if (auto *Record2 = dyn_cast<RecordDecl>(D2)) {
|
|
|
|
// Check for equivalent structure names.
|
|
|
|
IdentifierInfo *Name1 = Record1->getIdentifier();
|
|
|
|
if (!Name1 && Record1->getTypedefNameForAnonDecl())
|
|
|
|
Name1 = Record1->getTypedefNameForAnonDecl()->getIdentifier();
|
|
|
|
IdentifierInfo *Name2 = Record2->getIdentifier();
|
|
|
|
if (!Name2 && Record2->getTypedefNameForAnonDecl())
|
|
|
|
Name2 = Record2->getTypedefNameForAnonDecl()->getIdentifier();
|
|
|
|
if (!::IsStructurallyEquivalent(Name1, Name2) ||
|
|
|
|
!::IsStructurallyEquivalent(*this, Record1, Record2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Record/non-record mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (auto *Enum1 = dyn_cast<EnumDecl>(D1)) {
|
|
|
|
if (auto *Enum2 = dyn_cast<EnumDecl>(D2)) {
|
|
|
|
// Check for equivalent enum names.
|
|
|
|
IdentifierInfo *Name1 = Enum1->getIdentifier();
|
|
|
|
if (!Name1 && Enum1->getTypedefNameForAnonDecl())
|
|
|
|
Name1 = Enum1->getTypedefNameForAnonDecl()->getIdentifier();
|
|
|
|
IdentifierInfo *Name2 = Enum2->getIdentifier();
|
|
|
|
if (!Name2 && Enum2->getTypedefNameForAnonDecl())
|
|
|
|
Name2 = Enum2->getTypedefNameForAnonDecl()->getIdentifier();
|
|
|
|
if (!::IsStructurallyEquivalent(Name1, Name2) ||
|
|
|
|
!::IsStructurallyEquivalent(*this, Enum1, Enum2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Enum/non-enum mismatch
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (const auto *Typedef1 = dyn_cast<TypedefNameDecl>(D1)) {
|
|
|
|
if (const auto *Typedef2 = dyn_cast<TypedefNameDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(Typedef1->getIdentifier(),
|
|
|
|
Typedef2->getIdentifier()) ||
|
|
|
|
!::IsStructurallyEquivalent(*this, Typedef1->getUnderlyingType(),
|
|
|
|
Typedef2->getUnderlyingType()))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Typedef/non-typedef mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (auto *ClassTemplate1 = dyn_cast<ClassTemplateDecl>(D1)) {
|
|
|
|
if (auto *ClassTemplate2 = dyn_cast<ClassTemplateDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, ClassTemplate1,
|
|
|
|
ClassTemplate2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Class template/non-class-template mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (auto *FunctionTemplate1 = dyn_cast<FunctionTemplateDecl>(D1)) {
|
|
|
|
if (auto *FunctionTemplate2 = dyn_cast<FunctionTemplateDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, FunctionTemplate1,
|
|
|
|
FunctionTemplate2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Class template/non-class-template mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
2019-07-11 05:25:49 +08:00
|
|
|
} else if (auto *ConceptDecl1 = dyn_cast<ConceptDecl>(D1)) {
|
|
|
|
if (auto *ConceptDecl2 = dyn_cast<ConceptDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, ConceptDecl1, ConceptDecl2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Concept/non-concept mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
2018-08-08 23:04:27 +08:00
|
|
|
} else if (auto *TTP1 = dyn_cast<TemplateTypeParmDecl>(D1)) {
|
|
|
|
if (auto *TTP2 = dyn_cast<TemplateTypeParmDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, TTP1, TTP2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Kind mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (auto *NTTP1 = dyn_cast<NonTypeTemplateParmDecl>(D1)) {
|
|
|
|
if (auto *NTTP2 = dyn_cast<NonTypeTemplateParmDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, NTTP1, NTTP2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Kind mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (auto *TTP1 = dyn_cast<TemplateTemplateParmDecl>(D1)) {
|
|
|
|
if (auto *TTP2 = dyn_cast<TemplateTemplateParmDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, TTP1, TTP2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Kind mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (auto *MD1 = dyn_cast<CXXMethodDecl>(D1)) {
|
|
|
|
if (auto *MD2 = dyn_cast<CXXMethodDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, MD1, MD2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Kind mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (FunctionDecl *FD1 = dyn_cast<FunctionDecl>(D1)) {
|
|
|
|
if (FunctionDecl *FD2 = dyn_cast<FunctionDecl>(D2)) {
|
2019-02-08 16:55:32 +08:00
|
|
|
if (FD1->isOverloadedOperator()) {
|
|
|
|
if (!FD2->isOverloadedOperator())
|
|
|
|
return false;
|
|
|
|
if (FD1->getOverloadedOperator() != FD2->getOverloadedOperator())
|
|
|
|
return false;
|
|
|
|
}
|
2018-08-08 23:04:27 +08:00
|
|
|
if (!::IsStructurallyEquivalent(FD1->getIdentifier(),
|
|
|
|
FD2->getIdentifier()))
|
|
|
|
return false;
|
|
|
|
if (!::IsStructurallyEquivalent(*this, FD1, FD2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Kind mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else if (FriendDecl *FrD1 = dyn_cast<FriendDecl>(D1)) {
|
|
|
|
if (FriendDecl *FrD2 = dyn_cast<FriendDecl>(D2)) {
|
|
|
|
if (!::IsStructurallyEquivalent(*this, FrD1, FrD2))
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
// Kind mismatch.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
bool StructuralEquivalenceContext::Finish() {
|
|
|
|
while (!DeclsToCheck.empty()) {
|
|
|
|
// Check the next declaration.
|
2019-09-02 19:01:09 +08:00
|
|
|
std::pair<Decl *, Decl *> P = DeclsToCheck.front();
|
|
|
|
DeclsToCheck.pop();
|
2017-04-28 08:31:30 +08:00
|
|
|
|
2019-09-02 19:01:09 +08:00
|
|
|
Decl *D1 = P.first;
|
|
|
|
Decl *D2 = P.second;
|
2017-04-28 08:31:30 +08:00
|
|
|
|
2018-08-08 23:04:27 +08:00
|
|
|
bool Equivalent =
|
|
|
|
CheckCommonEquivalence(D1, D2) && CheckKindSpecificEquivalence(D1, D2);
|
2017-04-28 08:31:30 +08:00
|
|
|
|
|
|
|
if (!Equivalent) {
|
|
|
|
// Note that these two declarations are not equivalent (and we already
|
|
|
|
// know about it).
|
2019-09-02 19:01:09 +08:00
|
|
|
NonEquivalentDecls.insert(P);
|
|
|
|
|
2017-04-28 08:31:30 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|