Files
clice/unittests/Compiler/Resolver.cpp
2024-12-08 15:21:27 +08:00

488 lines
9.4 KiB
C++

#include "../Test.h"
#include <Compiler/Compiler.h>
#include <Compiler/Resolver.h>
namespace clice {
namespace {
struct TemplateResolverTester : public clang::RecursiveASTVisitor<TemplateResolverTester> {
TemplateResolverTester(llvm::StringRef code) {
compileArgs = {
"clang++",
"-std=c++20",
"main.cpp",
"-resource-dir",
"/home/ykiko/C++/clice2/build/lib/clang/20",
};
CompliationParams params;
params.srcPath = "main.cpp";
params.content = code;
params.args = compileArgs;
auto info = compile(params);
if(!info) {
llvm::errs() << info.takeError() << "\n";
return;
}
compiler = std::move(*info);
test();
}
bool VisitTypeAliasDecl(clang::TypeAliasDecl* decl) {
if(decl->getName() == "input") {
input = decl->getUnderlyingType();
}
if(decl->getName() == "expect") {
expect = decl->getUnderlyingType();
}
return true;
}
void test(std::source_location location = std::source_location::current()) {
TraverseDecl(compiler.tu());
EXPECT_FALSE(input.isNull());
EXPECT_FALSE(expect.isNull());
auto& resolver = compiler.resolver();
clang::QualType result = resolver.resolve(input);
EXPECT_EQ(result.getCanonicalType(), expect.getCanonicalType());
if(result.getCanonicalType() != expect.getCanonicalType()) {
result.dump();
expect.dump();
}
/// Test whether cache works.
// clang::QualType result2 = resolver.resolve(input);
// result.dump();
// result2.dump();
// EXPECT_EQ(result, result2);
}
clang::QualType input;
clang::QualType expect;
std::vector<const char*> compileArgs;
ASTInfo compiler;
};
TEST(TemplateResolver, TypeParameterType) {
TemplateResolverTester tester(R"cpp(
template <typename T>
struct A {
using type = T;
};
template <typename X>
struct test {
using input = typename A<X>::type;
using expect = X;
};
)cpp");
}
TEST(TemplateResolver, SingleLevel) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T>
struct A {
using type = type_list<T>;
};
template <typename X>
struct test {
using input = typename A<X>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, SingleLevelNotDependent) {
TemplateResolverTester tester(R"cpp(
template <typename T>
struct A {
using type = int;
};
template <typename X>
struct test {
using input = typename A<X>::type;
using expect = int;
};
)cpp");
}
TEST(TemplateResolver, MultiLevel) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using type = type_list<T1>;
};
template <typename T2>
struct B {
using type = typename A<T2>::type;
};
template <typename T3>
struct C {
using type = typename B<T3>::type;
};
template <typename X>
struct test {
using input = typename C<X>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, MultiLevelNotDependent) {
TemplateResolverTester tester(R"cpp(
template <typename T1>
struct A {
using type = int;
};
template <typename T2>
struct B {
using type = typename A<T2>::type;
};
template <typename T3>
struct C {
using type = typename B<T3>::type;
};
template <typename X>
struct test {
using input = typename C<X>::type;
using expect = int;
};
)cpp");
}
TEST(TemplateResolver, ArgumentDependent) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using type = T1;
};
template <typename T2>
struct B {
using type = type_list<T2>;
};
template <typename X>
struct test {
using input = typename B<typename A<X>::type>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, AliasArgument) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using type = T1;
};
template <typename T2>
struct B {
using base = A<T2>;
using type = type_list<typename base::type>;
};
template <typename X>
struct test {
using input = typename B<X>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, AliasDependent) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using type = type_list<T1>;
};
template <typename T2>
struct B {
using base = A<T2>;
using type = typename base::type;
};
template <typename X>
struct test {
using input = typename B<X>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, AliasTemplate) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1, typename U1>
struct A {
using type = type_list<T1, U1>;
};
template <typename T2>
struct B {
template <typename U2>
using type = typename A<T2, U2>::type;
};
template <typename X, typename Y>
struct test {
using input = typename B<X>::template type<Y>;
using expect = type_list<X, Y>;
};
)cpp");
}
TEST(TemplateResolver, BaseDependent) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using type = type_list<T1>;
};
template <typename U2>
struct B : A<U2> {};
template <typename X>
struct test {
using input = typename B<X>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, MultiNested) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using self = A<T1>;
using type = type_list<T1>;
};
template <typename X>
struct test {
using input = typename A<X>::self::self::self::self::self::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, OuterDependentMemberClass) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
template <typename T2>
struct B {
template <typename T3>
struct C {
using type = type_list<T1, T2, T3>;
};
};
};
template <typename X, typename Y, typename Z>
struct test {
using input = typename A<X>::template B<Y>::template C<Z>::type;
using expect = type_list<X, Y, Z>;
};
)cpp");
}
TEST(TemplateResolver, InnerDependentMemberClass) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T>
struct test {
template <int N, typename U>
struct B {
using type = type_list<U, T>;
};
using input = typename B<1, T>::type;
using expect = type_list<T, T>;
};
)cpp");
}
TEST(TemplateResolver, InnerDependentPartialMemberClass) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T, typename U>
struct test {};
template <typename T>
struct test<T, T> {
template <int N, typename U>
struct A {
using type = type_list<U, T>;
};
using input = typename A<1, T>::type;
using expect = type_list<T, T>;
};
)cpp");
}
TEST(TemplateResolver, PartialSpecialization) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {};
template <typename U2>
struct B {};
template <typename U2, template <typename...> typename HKT>
struct B<HKT<U2>> {
using type = type_list<U2>;
};
template <typename X>
struct test {
using input = typename B<A<X>>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, PartialDefaultArgument) {
TemplateResolverTester tester(R"cpp(
template <typename T, typename U = T>
struct X {};
template <typename T>
struct X<T, T> {
using type = T;
};
template <typename T>
struct test {
using input = typename X<T>::type;
using expect = T;
};
)cpp");
}
TEST(TemplateResolver, DefaultArgument) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename T1>
struct A {
using type = type_list<T1>;
};
template <typename U1, typename U2 = A<U1>>
struct B {
using type = typename U2::type;
};
template <typename X>
struct test {
using input = typename B<X>::type;
using expect = type_list<X>;
};
)cpp");
}
TEST(TemplateResolver, PackExpansion) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename U, typename... Us>
struct X {
using type = type_list<Us...>;
};
template <typename... Ts>
struct test {
using input = typename X<int, Ts...>::type;
using expect = type_list<Ts...>;
};
)cpp");
}
TEST(TemplateResolver, BasePackExpansion) {
TemplateResolverTester tester(R"cpp(
template <typename... Ts>
struct type_list {};
template <typename U, typename... Us>
struct X {
using type = type_list<Us...>;
};
template <typename... Us>
struct Y : X<int, Us...> {};
template <typename... Ts>
struct test {
using input = typename Y<Ts...>::type;
using expect = type_list<Ts...>;
};
)cpp");
}
TEST(TemplateResolver, Standard) {
TemplateResolverTester tester(R"cpp(
#include <vector>
template <typename T>
struct test {
using input = typename std::vector<T>::reference;
using expect = T&;
};
)cpp");
}
} // namespace
} // namespace clice