2018-10-17 07:06:42 +08:00
|
|
|
//===-- clang-doc/SerializeTest.cpp ---------------------------------------===//
|
|
|
|
//
|
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
|
2018-10-17 07:06:42 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "Serialize.h"
|
|
|
|
#include "ClangDocTest.h"
|
|
|
|
#include "Representation.h"
|
|
|
|
#include "clang/AST/Comment.h"
|
|
|
|
#include "clang/AST/RecursiveASTVisitor.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace doc {
|
|
|
|
|
|
|
|
class ClangDocSerializeTestVisitor
|
|
|
|
: public RecursiveASTVisitor<ClangDocSerializeTestVisitor> {
|
|
|
|
|
|
|
|
EmittedInfoList &EmittedInfos;
|
|
|
|
bool Public;
|
|
|
|
|
|
|
|
comments::FullComment *getComment(const NamedDecl *D) const {
|
|
|
|
if (RawComment *Comment =
|
|
|
|
D->getASTContext().getRawCommentForDeclNoCache(D)) {
|
|
|
|
Comment->setAttached();
|
|
|
|
return Comment->parse(D->getASTContext(), nullptr, D);
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
ClangDocSerializeTestVisitor(EmittedInfoList &EmittedInfos, bool Public)
|
|
|
|
: EmittedInfos(EmittedInfos), Public(Public) {}
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
template <typename T> bool mapDecl(const T *D) {
|
2018-10-17 07:06:42 +08:00
|
|
|
auto I = serialize::emitInfo(D, getComment(D), /*Line=*/0,
|
2019-08-10 01:49:41 +08:00
|
|
|
/*File=*/"test.cpp", true, Public);
|
2019-07-03 03:59:56 +08:00
|
|
|
if (I.first)
|
|
|
|
EmittedInfos.emplace_back(std::move(I.first));
|
|
|
|
if (I.second)
|
|
|
|
EmittedInfos.emplace_back(std::move(I.second));
|
2018-10-17 07:06:42 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
bool VisitNamespaceDecl(const NamespaceDecl *D) { return mapDecl(D); }
|
|
|
|
|
2018-10-17 07:06:42 +08:00
|
|
|
bool VisitFunctionDecl(const FunctionDecl *D) {
|
|
|
|
// Don't visit CXXMethodDecls twice
|
|
|
|
if (dyn_cast<CXXMethodDecl>(D))
|
|
|
|
return true;
|
2019-07-03 03:59:56 +08:00
|
|
|
return mapDecl(D);
|
2018-10-17 07:06:42 +08:00
|
|
|
}
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
bool VisitCXXMethodDecl(const CXXMethodDecl *D) { return mapDecl(D); }
|
2018-10-17 07:06:42 +08:00
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
bool VisitRecordDecl(const RecordDecl *D) { return mapDecl(D); }
|
2018-10-17 07:06:42 +08:00
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
bool VisitEnumDecl(const EnumDecl *D) { return mapDecl(D); }
|
2018-10-17 07:06:42 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void ExtractInfosFromCode(StringRef Code, size_t NumExpectedInfos, bool Public,
|
|
|
|
EmittedInfoList &EmittedInfos) {
|
|
|
|
auto ASTUnit = clang::tooling::buildASTFromCode(Code);
|
|
|
|
auto TU = ASTUnit->getASTContext().getTranslationUnitDecl();
|
|
|
|
ClangDocSerializeTestVisitor Visitor(EmittedInfos, Public);
|
|
|
|
Visitor.TraverseTranslationUnitDecl(TU);
|
|
|
|
ASSERT_EQ(NumExpectedInfos, EmittedInfos.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ExtractInfosFromCodeWithArgs(StringRef Code, size_t NumExpectedInfos,
|
|
|
|
bool Public, EmittedInfoList &EmittedInfos,
|
|
|
|
std::vector<std::string> &Args) {
|
|
|
|
auto ASTUnit = clang::tooling::buildASTFromCodeWithArgs(Code, Args);
|
|
|
|
auto TU = ASTUnit->getASTContext().getTranslationUnitDecl();
|
|
|
|
ClangDocSerializeTestVisitor Visitor(EmittedInfos, Public);
|
|
|
|
Visitor.TraverseTranslationUnitDecl(TU);
|
|
|
|
ASSERT_EQ(NumExpectedInfos, EmittedInfos.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test serialization of namespace declarations.
|
|
|
|
TEST(SerializeTest, emitNamespaceInfo) {
|
|
|
|
EmittedInfoList Infos;
|
2019-07-03 03:59:56 +08:00
|
|
|
ExtractInfosFromCode("namespace A { namespace B { void f() {} } }", 5,
|
2018-10-17 07:06:42 +08:00
|
|
|
/*Public=*/false, Infos);
|
|
|
|
|
|
|
|
NamespaceInfo *A = InfoAsNamespace(Infos[0].get());
|
|
|
|
NamespaceInfo ExpectedA(EmptySID, "A");
|
|
|
|
CheckNamespaceInfo(&ExpectedA, A);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
NamespaceInfo *B = InfoAsNamespace(Infos[2].get());
|
2019-08-15 04:19:33 +08:00
|
|
|
NamespaceInfo ExpectedB(EmptySID, /*Name=*/"B", /*Path=*/"A");
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedB.Namespace.emplace_back(EmptySID, "A", InfoType::IT_namespace);
|
|
|
|
CheckNamespaceInfo(&ExpectedB, B);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
NamespaceInfo *BWithFunction = InfoAsNamespace(Infos[4].get());
|
2018-10-17 07:06:42 +08:00
|
|
|
NamespaceInfo ExpectedBWithFunction(EmptySID);
|
|
|
|
FunctionInfo F;
|
|
|
|
F.Name = "f";
|
|
|
|
F.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
F.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
F.Namespace.emplace_back(EmptySID, "B", InfoType::IT_namespace);
|
|
|
|
F.Namespace.emplace_back(EmptySID, "A", InfoType::IT_namespace);
|
2019-08-16 07:04:27 +08:00
|
|
|
F.Access = AccessSpecifier::AS_none;
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedBWithFunction.ChildFunctions.emplace_back(std::move(F));
|
|
|
|
CheckNamespaceInfo(&ExpectedBWithFunction, BWithFunction);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitAnonymousNamespaceInfo) {
|
|
|
|
EmittedInfoList Infos;
|
2019-07-03 03:59:56 +08:00
|
|
|
ExtractInfosFromCode("namespace { }", 2, /*Public=*/false, Infos);
|
2018-10-17 07:06:42 +08:00
|
|
|
|
|
|
|
NamespaceInfo *A = InfoAsNamespace(Infos[0].get());
|
|
|
|
NamespaceInfo ExpectedA(EmptySID);
|
2019-06-29 03:07:56 +08:00
|
|
|
ExpectedA.Name = "@nonymous_namespace";
|
2018-10-17 07:06:42 +08:00
|
|
|
CheckNamespaceInfo(&ExpectedA, A);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test serialization of record declarations.
|
|
|
|
TEST(SerializeTest, emitRecordInfo) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode(R"raw(class E {
|
|
|
|
public:
|
|
|
|
E() {}
|
|
|
|
protected:
|
|
|
|
void ProtectedMethod();
|
2019-06-25 03:31:02 +08:00
|
|
|
};
|
|
|
|
template <typename T>
|
|
|
|
struct F {
|
|
|
|
void TemplateMethod();
|
|
|
|
};
|
|
|
|
template <>
|
|
|
|
void F<int>::TemplateMethod();
|
|
|
|
typedef struct {} G;)raw",
|
2019-07-03 03:59:56 +08:00
|
|
|
10, /*Public=*/false, Infos);
|
2018-10-17 07:06:42 +08:00
|
|
|
|
|
|
|
RecordInfo *E = InfoAsRecord(Infos[0].get());
|
|
|
|
RecordInfo ExpectedE(EmptySID, "E");
|
|
|
|
ExpectedE.TagType = TagTypeKind::TTK_Class;
|
|
|
|
ExpectedE.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
CheckRecordInfo(&ExpectedE, E);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *RecordWithEConstructor = InfoAsRecord(Infos[2].get());
|
2018-10-17 07:06:42 +08:00
|
|
|
RecordInfo ExpectedRecordWithEConstructor(EmptySID);
|
|
|
|
FunctionInfo EConstructor;
|
|
|
|
EConstructor.Name = "E";
|
|
|
|
EConstructor.Parent = Reference(EmptySID, "E", InfoType::IT_record);
|
|
|
|
EConstructor.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
EConstructor.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
EConstructor.Namespace.emplace_back(EmptySID, "E", InfoType::IT_record);
|
|
|
|
EConstructor.Access = AccessSpecifier::AS_public;
|
|
|
|
EConstructor.IsMethod = true;
|
|
|
|
ExpectedRecordWithEConstructor.ChildFunctions.emplace_back(
|
|
|
|
std::move(EConstructor));
|
|
|
|
CheckRecordInfo(&ExpectedRecordWithEConstructor, RecordWithEConstructor);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *RecordWithMethod = InfoAsRecord(Infos[3].get());
|
2018-10-17 07:06:42 +08:00
|
|
|
RecordInfo ExpectedRecordWithMethod(EmptySID);
|
|
|
|
FunctionInfo Method;
|
|
|
|
Method.Name = "ProtectedMethod";
|
|
|
|
Method.Parent = Reference(EmptySID, "E", InfoType::IT_record);
|
|
|
|
Method.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
Method.Loc.emplace_back(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
Method.Namespace.emplace_back(EmptySID, "E", InfoType::IT_record);
|
|
|
|
Method.Access = AccessSpecifier::AS_protected;
|
|
|
|
Method.IsMethod = true;
|
|
|
|
ExpectedRecordWithMethod.ChildFunctions.emplace_back(std::move(Method));
|
|
|
|
CheckRecordInfo(&ExpectedRecordWithMethod, RecordWithMethod);
|
2019-06-25 03:31:02 +08:00
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *F = InfoAsRecord(Infos[4].get());
|
2019-06-25 03:31:02 +08:00
|
|
|
RecordInfo ExpectedF(EmptySID, "F");
|
|
|
|
ExpectedF.TagType = TagTypeKind::TTK_Struct;
|
|
|
|
ExpectedF.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
CheckRecordInfo(&ExpectedF, F);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *RecordWithTemplateMethod = InfoAsRecord(Infos[6].get());
|
2019-06-25 03:31:02 +08:00
|
|
|
RecordInfo ExpectedRecordWithTemplateMethod(EmptySID);
|
|
|
|
FunctionInfo TemplateMethod;
|
|
|
|
TemplateMethod.Name = "TemplateMethod";
|
|
|
|
TemplateMethod.Parent = Reference(EmptySID, "F", InfoType::IT_record);
|
|
|
|
TemplateMethod.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
TemplateMethod.Loc.emplace_back(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
TemplateMethod.Namespace.emplace_back(EmptySID, "F", InfoType::IT_record);
|
|
|
|
TemplateMethod.Access = AccessSpecifier::AS_public;
|
|
|
|
TemplateMethod.IsMethod = true;
|
|
|
|
ExpectedRecordWithTemplateMethod.ChildFunctions.emplace_back(
|
|
|
|
std::move(TemplateMethod));
|
|
|
|
CheckRecordInfo(&ExpectedRecordWithTemplateMethod, RecordWithTemplateMethod);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *TemplatedRecord = InfoAsRecord(Infos[7].get());
|
2019-06-25 03:31:02 +08:00
|
|
|
RecordInfo ExpectedTemplatedRecord(EmptySID);
|
|
|
|
FunctionInfo SpecializedTemplateMethod;
|
|
|
|
SpecializedTemplateMethod.Name = "TemplateMethod";
|
|
|
|
SpecializedTemplateMethod.Parent =
|
|
|
|
Reference(EmptySID, "F", InfoType::IT_record);
|
|
|
|
SpecializedTemplateMethod.ReturnType =
|
|
|
|
TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
SpecializedTemplateMethod.Loc.emplace_back(0,
|
|
|
|
llvm::SmallString<16>{"test.cpp"});
|
|
|
|
SpecializedTemplateMethod.Namespace.emplace_back(EmptySID, "F",
|
|
|
|
InfoType::IT_record);
|
|
|
|
SpecializedTemplateMethod.Access = AccessSpecifier::AS_public;
|
|
|
|
SpecializedTemplateMethod.IsMethod = true;
|
|
|
|
ExpectedTemplatedRecord.ChildFunctions.emplace_back(
|
|
|
|
std::move(SpecializedTemplateMethod));
|
|
|
|
CheckRecordInfo(&ExpectedTemplatedRecord, TemplatedRecord);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *G = InfoAsRecord(Infos[8].get());
|
2019-06-25 03:31:02 +08:00
|
|
|
RecordInfo ExpectedG(EmptySID, "G");
|
|
|
|
ExpectedG.TagType = TagTypeKind::TTK_Struct;
|
|
|
|
ExpectedG.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExpectedG.IsTypeDef = true;
|
|
|
|
CheckRecordInfo(&ExpectedG, G);
|
2018-10-17 07:06:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Test serialization of enum declarations.
|
|
|
|
TEST(SerializeTest, emitEnumInfo) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode("enum E { X, Y }; enum class G { A, B };", 2,
|
|
|
|
/*Public=*/false, Infos);
|
|
|
|
|
|
|
|
NamespaceInfo *NamespaceWithEnum = InfoAsNamespace(Infos[0].get());
|
|
|
|
NamespaceInfo ExpectedNamespaceWithEnum(EmptySID);
|
|
|
|
EnumInfo E;
|
|
|
|
E.Name = "E";
|
|
|
|
E.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
E.Members.emplace_back("X");
|
|
|
|
E.Members.emplace_back("Y");
|
|
|
|
ExpectedNamespaceWithEnum.ChildEnums.emplace_back(std::move(E));
|
|
|
|
CheckNamespaceInfo(&ExpectedNamespaceWithEnum, NamespaceWithEnum);
|
|
|
|
|
|
|
|
NamespaceInfo *NamespaceWithScopedEnum = InfoAsNamespace(Infos[1].get());
|
|
|
|
NamespaceInfo ExpectedNamespaceWithScopedEnum(EmptySID);
|
|
|
|
EnumInfo G;
|
|
|
|
G.Name = "G";
|
|
|
|
G.Scoped = true;
|
|
|
|
G.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
G.Members.emplace_back("A");
|
|
|
|
G.Members.emplace_back("B");
|
|
|
|
ExpectedNamespaceWithScopedEnum.ChildEnums.emplace_back(std::move(G));
|
|
|
|
CheckNamespaceInfo(&ExpectedNamespaceWithScopedEnum, NamespaceWithScopedEnum);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitUndefinedRecordInfo) {
|
|
|
|
EmittedInfoList Infos;
|
2019-07-03 03:59:56 +08:00
|
|
|
ExtractInfosFromCode("class E;", 2, /*Public=*/false, Infos);
|
2018-10-17 07:06:42 +08:00
|
|
|
|
|
|
|
RecordInfo *E = InfoAsRecord(Infos[0].get());
|
|
|
|
RecordInfo ExpectedE(EmptySID, "E");
|
|
|
|
ExpectedE.TagType = TagTypeKind::TTK_Class;
|
|
|
|
ExpectedE.Loc.emplace_back(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
CheckRecordInfo(&ExpectedE, E);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitRecordMemberInfo) {
|
|
|
|
EmittedInfoList Infos;
|
2019-07-03 03:59:56 +08:00
|
|
|
ExtractInfosFromCode("struct E { int I; };", 2, /*Public=*/false, Infos);
|
2018-10-17 07:06:42 +08:00
|
|
|
|
|
|
|
RecordInfo *E = InfoAsRecord(Infos[0].get());
|
|
|
|
RecordInfo ExpectedE(EmptySID, "E");
|
|
|
|
ExpectedE.TagType = TagTypeKind::TTK_Struct;
|
|
|
|
ExpectedE.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExpectedE.Members.emplace_back("int", "I", AccessSpecifier::AS_public);
|
|
|
|
CheckRecordInfo(&ExpectedE, E);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitInternalRecordInfo) {
|
|
|
|
EmittedInfoList Infos;
|
2019-07-03 03:59:56 +08:00
|
|
|
ExtractInfosFromCode("class E { class G {}; };", 4, /*Public=*/false, Infos);
|
2018-10-17 07:06:42 +08:00
|
|
|
|
|
|
|
RecordInfo *E = InfoAsRecord(Infos[0].get());
|
|
|
|
RecordInfo ExpectedE(EmptySID, "E");
|
|
|
|
ExpectedE.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExpectedE.TagType = TagTypeKind::TTK_Class;
|
|
|
|
CheckRecordInfo(&ExpectedE, E);
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
RecordInfo *G = InfoAsRecord(Infos[2].get());
|
2019-08-15 04:19:33 +08:00
|
|
|
RecordInfo ExpectedG(EmptySID, /*Name=*/"G", /*Path=*/"E");
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedG.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExpectedG.TagType = TagTypeKind::TTK_Class;
|
|
|
|
ExpectedG.Namespace.emplace_back(EmptySID, "E", InfoType::IT_record);
|
|
|
|
CheckRecordInfo(&ExpectedG, G);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitPublicAnonymousNamespaceInfo) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode("namespace { class A; }", 0, /*Public=*/true, Infos);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitPublicFunctionInternalInfo) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode("int F() { class G {}; return 0; };", 1, /*Public=*/true,
|
|
|
|
Infos);
|
|
|
|
|
|
|
|
NamespaceInfo *BWithFunction = InfoAsNamespace(Infos[0].get());
|
|
|
|
NamespaceInfo ExpectedBWithFunction(EmptySID);
|
|
|
|
FunctionInfo F;
|
|
|
|
F.Name = "F";
|
|
|
|
F.ReturnType = TypeInfo(EmptySID, "int", InfoType::IT_default);
|
|
|
|
F.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
2019-08-16 07:04:27 +08:00
|
|
|
F.Access = AccessSpecifier::AS_none;
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedBWithFunction.ChildFunctions.emplace_back(std::move(F));
|
|
|
|
CheckNamespaceInfo(&ExpectedBWithFunction, BWithFunction);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitInlinedFunctionInfo) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode("inline void F(int I) { };", 1, /*Public=*/true, Infos);
|
|
|
|
|
|
|
|
NamespaceInfo *BWithFunction = InfoAsNamespace(Infos[0].get());
|
|
|
|
NamespaceInfo ExpectedBWithFunction(EmptySID);
|
|
|
|
FunctionInfo F;
|
|
|
|
F.Name = "F";
|
|
|
|
F.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
F.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
F.Params.emplace_back("int", "I");
|
2019-08-16 07:04:27 +08:00
|
|
|
F.Access = AccessSpecifier::AS_none;
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedBWithFunction.ChildFunctions.emplace_back(std::move(F));
|
|
|
|
CheckNamespaceInfo(&ExpectedBWithFunction, BWithFunction);
|
|
|
|
}
|
|
|
|
|
2019-08-16 08:10:49 +08:00
|
|
|
TEST(SerializeTest, emitInheritedRecordInfo) {
|
2018-10-17 07:06:42 +08:00
|
|
|
EmittedInfoList Infos;
|
2019-08-16 08:10:49 +08:00
|
|
|
ExtractInfosFromCode(R"raw(class F { protected: void set(int N); };
|
|
|
|
class G { public: int get() { return 1; } protected: int I; };
|
2019-06-25 03:31:02 +08:00
|
|
|
class E : public F, virtual private G {};
|
2019-08-16 08:10:49 +08:00
|
|
|
class H : private E {};
|
2019-06-25 03:31:02 +08:00
|
|
|
template <typename T>
|
2019-08-16 08:10:49 +08:00
|
|
|
class I {} ;
|
|
|
|
class J : public I<int> {} ;)raw",
|
|
|
|
14, /*Public=*/false, Infos);
|
2018-10-17 07:06:42 +08:00
|
|
|
|
|
|
|
RecordInfo *F = InfoAsRecord(Infos[0].get());
|
|
|
|
RecordInfo ExpectedF(EmptySID, "F");
|
|
|
|
ExpectedF.TagType = TagTypeKind::TTK_Class;
|
|
|
|
ExpectedF.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
CheckRecordInfo(&ExpectedF, F);
|
|
|
|
|
2019-08-16 08:10:49 +08:00
|
|
|
RecordInfo *G = InfoAsRecord(Infos[3].get());
|
2018-10-17 07:06:42 +08:00
|
|
|
RecordInfo ExpectedG(EmptySID, "G");
|
|
|
|
ExpectedG.TagType = TagTypeKind::TTK_Class;
|
|
|
|
ExpectedG.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
2019-08-16 08:10:49 +08:00
|
|
|
ExpectedG.Members.emplace_back("int", "I", AccessSpecifier::AS_protected);
|
2018-10-17 07:06:42 +08:00
|
|
|
CheckRecordInfo(&ExpectedG, G);
|
|
|
|
|
2019-08-16 08:10:49 +08:00
|
|
|
RecordInfo *E = InfoAsRecord(Infos[6].get());
|
2018-10-17 07:06:42 +08:00
|
|
|
RecordInfo ExpectedE(EmptySID, "E");
|
|
|
|
ExpectedE.Parents.emplace_back(EmptySID, "F", InfoType::IT_record);
|
|
|
|
ExpectedE.VirtualParents.emplace_back(EmptySID, "G", InfoType::IT_record);
|
2019-08-16 08:10:49 +08:00
|
|
|
ExpectedE.Bases.emplace_back(EmptySID, "F", "", false,
|
|
|
|
AccessSpecifier::AS_public, true);
|
|
|
|
FunctionInfo FunctionSet;
|
|
|
|
FunctionSet.Name = "set";
|
|
|
|
FunctionSet.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
FunctionSet.Loc.emplace_back();
|
|
|
|
FunctionSet.Params.emplace_back("int", "N");
|
|
|
|
FunctionSet.Namespace.emplace_back(EmptySID, "F", InfoType::IT_record);
|
|
|
|
FunctionSet.Access = AccessSpecifier::AS_protected;
|
|
|
|
FunctionSet.IsMethod = true;
|
|
|
|
ExpectedE.Bases.back().ChildFunctions.emplace_back(std::move(FunctionSet));
|
|
|
|
ExpectedE.Bases.emplace_back(EmptySID, "G", "", true,
|
|
|
|
AccessSpecifier::AS_private, true);
|
|
|
|
FunctionInfo FunctionGet;
|
|
|
|
FunctionGet.Name = "get";
|
|
|
|
FunctionGet.ReturnType = TypeInfo(EmptySID, "int", InfoType::IT_default);
|
|
|
|
FunctionGet.DefLoc = Location();
|
|
|
|
FunctionGet.Namespace.emplace_back(EmptySID, "G", InfoType::IT_record);
|
|
|
|
FunctionGet.Access = AccessSpecifier::AS_private;
|
|
|
|
FunctionGet.IsMethod = true;
|
|
|
|
ExpectedE.Bases.back().ChildFunctions.emplace_back(std::move(FunctionGet));
|
|
|
|
ExpectedE.Bases.back().Members.emplace_back("int", "I",
|
|
|
|
AccessSpecifier::AS_private);
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedE.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExpectedE.TagType = TagTypeKind::TTK_Class;
|
|
|
|
CheckRecordInfo(&ExpectedE, E);
|
2019-06-25 03:31:02 +08:00
|
|
|
|
2019-08-16 08:10:49 +08:00
|
|
|
RecordInfo *H = InfoAsRecord(Infos[8].get());
|
2019-06-25 03:31:02 +08:00
|
|
|
RecordInfo ExpectedH(EmptySID, "H");
|
|
|
|
ExpectedH.TagType = TagTypeKind::TTK_Class;
|
|
|
|
ExpectedH.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
2019-08-16 08:10:49 +08:00
|
|
|
ExpectedH.Parents.emplace_back(EmptySID, "E", InfoType::IT_record);
|
|
|
|
ExpectedH.VirtualParents.emplace_back(EmptySID, "G", InfoType::IT_record);
|
|
|
|
ExpectedH.Bases.emplace_back(EmptySID, "E", "", false,
|
|
|
|
AccessSpecifier::AS_private, true);
|
|
|
|
ExpectedH.Bases.emplace_back(EmptySID, "F", "", false,
|
|
|
|
AccessSpecifier::AS_private, false);
|
|
|
|
FunctionInfo FunctionSetNew;
|
|
|
|
FunctionSetNew.Name = "set";
|
|
|
|
FunctionSetNew.ReturnType = TypeInfo(EmptySID, "void", InfoType::IT_default);
|
|
|
|
FunctionSetNew.Loc.emplace_back();
|
|
|
|
FunctionSetNew.Params.emplace_back("int", "N");
|
|
|
|
FunctionSetNew.Namespace.emplace_back(EmptySID, "F", InfoType::IT_record);
|
|
|
|
FunctionSetNew.Access = AccessSpecifier::AS_private;
|
|
|
|
FunctionSetNew.IsMethod = true;
|
|
|
|
ExpectedH.Bases.back().ChildFunctions.emplace_back(std::move(FunctionSetNew));
|
|
|
|
ExpectedH.Bases.emplace_back(EmptySID, "G", "", true,
|
|
|
|
AccessSpecifier::AS_private, false);
|
|
|
|
FunctionInfo FunctionGetNew;
|
|
|
|
FunctionGetNew.Name = "get";
|
|
|
|
FunctionGetNew.ReturnType = TypeInfo(EmptySID, "int", InfoType::IT_default);
|
|
|
|
FunctionGetNew.DefLoc = Location();
|
|
|
|
FunctionGetNew.Namespace.emplace_back(EmptySID, "G", InfoType::IT_record);
|
|
|
|
FunctionGetNew.Access = AccessSpecifier::AS_private;
|
|
|
|
FunctionGetNew.IsMethod = true;
|
|
|
|
ExpectedH.Bases.back().ChildFunctions.emplace_back(std::move(FunctionGetNew));
|
|
|
|
ExpectedH.Bases.back().Members.emplace_back("int", "I",
|
|
|
|
AccessSpecifier::AS_private);
|
2019-06-25 03:31:02 +08:00
|
|
|
CheckRecordInfo(&ExpectedH, H);
|
|
|
|
|
2019-08-16 08:10:49 +08:00
|
|
|
RecordInfo *I = InfoAsRecord(Infos[10].get());
|
2019-06-25 03:31:02 +08:00
|
|
|
RecordInfo ExpectedI(EmptySID, "I");
|
|
|
|
ExpectedI.TagType = TagTypeKind::TTK_Class;
|
2019-08-16 08:10:49 +08:00
|
|
|
ExpectedI.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
2019-06-25 03:31:02 +08:00
|
|
|
CheckRecordInfo(&ExpectedI, I);
|
2019-08-16 08:10:49 +08:00
|
|
|
|
|
|
|
RecordInfo *J = InfoAsRecord(Infos[12].get());
|
|
|
|
RecordInfo ExpectedJ(EmptySID, "J");
|
|
|
|
ExpectedJ.Parents.emplace_back(EmptySID, "I<int>", InfoType::IT_record);
|
|
|
|
ExpectedJ.Bases.emplace_back(EmptySID, "I<int>", "", false,
|
|
|
|
AccessSpecifier::AS_public, true);
|
|
|
|
ExpectedJ.DefLoc = Location(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExpectedJ.TagType = TagTypeKind::TTK_Class;
|
|
|
|
CheckRecordInfo(&ExpectedJ, J);
|
2018-10-17 07:06:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(SerializeTest, emitModulePublicLFunctions) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
std::vector<std::string> Args;
|
|
|
|
Args.push_back("-fmodules-ts");
|
|
|
|
ExtractInfosFromCodeWithArgs(R"raw(export module M;
|
|
|
|
int moduleFunction(int x);
|
|
|
|
static int staticModuleFunction(int x);
|
|
|
|
export double exportedModuleFunction(double y);)raw",
|
|
|
|
2, /*Public=*/true, Infos, Args);
|
|
|
|
|
|
|
|
NamespaceInfo *BWithFunction = InfoAsNamespace(Infos[0].get());
|
|
|
|
NamespaceInfo ExpectedBWithFunction(EmptySID);
|
|
|
|
FunctionInfo F;
|
|
|
|
F.Name = "moduleFunction";
|
|
|
|
F.ReturnType = TypeInfo(EmptySID, "int", InfoType::IT_default);
|
|
|
|
F.Loc.emplace_back(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
F.Params.emplace_back("int", "x");
|
2019-08-16 07:04:27 +08:00
|
|
|
F.Access = AccessSpecifier::AS_none;
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedBWithFunction.ChildFunctions.emplace_back(std::move(F));
|
|
|
|
CheckNamespaceInfo(&ExpectedBWithFunction, BWithFunction);
|
|
|
|
|
|
|
|
NamespaceInfo *BWithExportedFunction = InfoAsNamespace(Infos[1].get());
|
|
|
|
NamespaceInfo ExpectedBWithExportedFunction(EmptySID);
|
|
|
|
FunctionInfo ExportedF;
|
|
|
|
ExportedF.Name = "exportedModuleFunction";
|
|
|
|
ExportedF.ReturnType = TypeInfo(EmptySID, "double", InfoType::IT_default);
|
|
|
|
ExportedF.Loc.emplace_back(0, llvm::SmallString<16>{"test.cpp"});
|
|
|
|
ExportedF.Params.emplace_back("double", "y");
|
2019-08-16 07:04:27 +08:00
|
|
|
ExportedF.Access = AccessSpecifier::AS_none;
|
2018-10-17 07:06:42 +08:00
|
|
|
ExpectedBWithExportedFunction.ChildFunctions.emplace_back(
|
|
|
|
std::move(ExportedF));
|
|
|
|
CheckNamespaceInfo(&ExpectedBWithExportedFunction, BWithExportedFunction);
|
|
|
|
}
|
|
|
|
|
2019-07-03 03:59:56 +08:00
|
|
|
// Test serialization of child records in namespaces and other records
|
|
|
|
TEST(SerializeTest, emitChildRecords) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode("class A { class B {}; }; namespace { class C {}; } ", 8,
|
|
|
|
/*Public=*/false, Infos);
|
|
|
|
|
|
|
|
NamespaceInfo *ParentA = InfoAsNamespace(Infos[1].get());
|
|
|
|
NamespaceInfo ExpectedParentA(EmptySID);
|
|
|
|
ExpectedParentA.ChildRecords.emplace_back(EmptySID, "A", InfoType::IT_record);
|
|
|
|
CheckNamespaceInfo(&ExpectedParentA, ParentA);
|
|
|
|
|
|
|
|
RecordInfo *ParentB = InfoAsRecord(Infos[3].get());
|
|
|
|
RecordInfo ExpectedParentB(EmptySID);
|
2019-08-13 02:42:46 +08:00
|
|
|
ExpectedParentB.ChildRecords.emplace_back(EmptySID, "B", InfoType::IT_record,
|
|
|
|
"A");
|
2019-07-03 03:59:56 +08:00
|
|
|
CheckRecordInfo(&ExpectedParentB, ParentB);
|
|
|
|
|
|
|
|
NamespaceInfo *ParentC = InfoAsNamespace(Infos[7].get());
|
|
|
|
NamespaceInfo ExpectedParentC(EmptySID);
|
2019-08-13 02:42:46 +08:00
|
|
|
ExpectedParentC.ChildRecords.emplace_back(EmptySID, "C", InfoType::IT_record,
|
|
|
|
"@nonymous_namespace");
|
2019-07-03 03:59:56 +08:00
|
|
|
CheckNamespaceInfo(&ExpectedParentC, ParentC);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test serialization of child namespaces
|
|
|
|
TEST(SerializeTest, emitChildNamespaces) {
|
|
|
|
EmittedInfoList Infos;
|
|
|
|
ExtractInfosFromCode("namespace A { namespace B { } }", 4, /*Public=*/false,
|
|
|
|
Infos);
|
|
|
|
|
|
|
|
NamespaceInfo *ParentA = InfoAsNamespace(Infos[1].get());
|
|
|
|
NamespaceInfo ExpectedParentA(EmptySID);
|
|
|
|
ExpectedParentA.ChildNamespaces.emplace_back(EmptySID, "A",
|
|
|
|
InfoType::IT_namespace);
|
|
|
|
CheckNamespaceInfo(&ExpectedParentA, ParentA);
|
|
|
|
|
|
|
|
NamespaceInfo *ParentB = InfoAsNamespace(Infos[3].get());
|
|
|
|
NamespaceInfo ExpectedParentB(EmptySID);
|
|
|
|
ExpectedParentB.ChildNamespaces.emplace_back(EmptySID, "B",
|
2019-08-13 02:42:46 +08:00
|
|
|
InfoType::IT_namespace, "A");
|
2019-07-03 03:59:56 +08:00
|
|
|
CheckNamespaceInfo(&ExpectedParentB, ParentB);
|
|
|
|
}
|
|
|
|
|
2018-10-17 07:06:42 +08:00
|
|
|
} // namespace doc
|
|
|
|
} // end namespace clang
|