#include "clang-tidy/ClangTidyCheck.h"
#include "clang-tidy/ClangTidyModuleRegistry.h"
#include "llvm/ADT/ArrayRef.h"
#include "header_exportable_declarations.hpp"
#include <iostream>
#include <iterator>
#include <ranges>
#include <algorithm>
template <>
struct clang::tidy::OptionEnumMapping<libcpp::header_exportable_declarations::FileType> {
static llvm::ArrayRef<std::pair<libcpp::header_exportable_declarations::FileType, llvm::StringRef>> getEnumMapping() {
static constexpr std::pair<libcpp::header_exportable_declarations::FileType, llvm::StringRef> Mapping[] = {
{libcpp::header_exportable_declarations::FileType::Header, "Header"},
{libcpp::header_exportable_declarations::FileType::ModulePartition, "ModulePartition"},
{libcpp::header_exportable_declarations::FileType::Module, "Module"},
{libcpp::header_exportable_declarations::FileType::CHeader, "CHeader"},
{libcpp::header_exportable_declarations::FileType::CompatModulePartition, "CompatModulePartition"},
{libcpp::header_exportable_declarations::FileType::CompatModule, "CompatModule"}};
return ArrayRef(Mapping);
}
};
namespace libcpp {
header_exportable_declarations::header_exportable_declarations(
llvm::StringRef name, clang::tidy::ClangTidyContext* context)
: clang::tidy::ClangTidyCheck(name, context),
filename_(Options.get("Filename", "")),
file_type_(Options.get("FileType", header_exportable_declarations::FileType::Unknown)),
extra_header_(Options.get("ExtraHeader", "")) {
switch (file_type_) {
case header_exportable_declarations::FileType::CHeader:
case header_exportable_declarations::FileType::Header:
if (filename_.empty())
llvm::errs() << "No filename is provided.\n";
if (extra_header_.empty())
extra_header_ = "$^";
break;
case header_exportable_declarations::FileType::ModulePartition:
case header_exportable_declarations::FileType::CompatModulePartition:
if (filename_.empty())
llvm::errs() << "No filename is provided.\n";
[[fallthrough]];
case header_exportable_declarations::FileType::Module:
case header_exportable_declarations::FileType::CompatModule:
if (!extra_header_.empty())
llvm::errs() << "Extra headers are not allowed for modules.\n";
if (Options.get("SkipDeclarations"))
llvm::errs() << "Modules may not skip declarations.\n";
if (Options.get("ExtraDeclarations"))
llvm::errs() << "Modules may not have extra declarations.\n";
break;
case header_exportable_declarations::FileType::Unknown:
llvm::errs() << "No file type is provided.\n";
break;
}
std::optional<llvm::StringRef> list = Options.get("SkipDeclarations");
if (list)
for (auto decl : std::views::split(*list, ' ')) {
std::string s;
std::ranges::copy(decl, std::back_inserter(s));
skip_decls_.emplace(std::move(s));
}
decls_ = skip_decls_;
list = Options.get("ExtraDeclarations");
if (list)
for (auto decl : std::views::split(*list, ' '))
std::cout << "using ::" << std::string_view{decl.data(), decl.size()} << ";\n";
}
header_exportable_declarations::~header_exportable_declarations() {
for (const auto& name : global_decls_)
if (!skip_decls_.contains("std::" + name) && decls_.contains("std::" + name))
std::cout << "using ::" << name << ";\n";
}
void header_exportable_declarations::registerMatchers(clang::ast_matchers::MatchFinder* finder) {
using namespace clang::ast_matchers;
switch (file_type_) {
case FileType::Header:
finder->addMatcher(
namedDecl(
anyOf(isExpansionInFileMatching(("v1/__" + filename_ + "/").str()),
isExpansionInFileMatching(extra_header_),
isExpansionInFileMatching(("v1/__fwd/" + filename_ + "\\.h$").str()),
isExpansionInFileMatching(("v1/" + filename_ + "$").str())),
unless(hasAncestor(friendDecl())))
.bind("header_exportable_declarations"),
this);
break;
case FileType::CHeader:
finder->addMatcher(namedDecl().bind("cheader_exportable_declarations"), this);
[[fallthrough]];
case FileType::ModulePartition:
case FileType::CompatModulePartition:
finder->addMatcher(namedDecl(isExpansionInFileMatching(filename_)).bind("header_exportable_declarations"), this);
break;
case FileType::Module:
case FileType::CompatModule:
finder->addMatcher(namedDecl().bind("header_exportable_declarations"), this);
break;
case header_exportable_declarations::FileType::Unknown:
llvm::errs() << "This should be unreachable.\n";
break;
}
}
static std::string get_qualified_name(const clang::NamedDecl& decl) {
std::string result = decl.getNameAsString();
if (result.size() >= 2 && result[0] == '_')
if (result[1] == '_' || std::isupper(result[1]))
if (result != "_Exit")
return "";
for (auto* context = llvm::dyn_cast_or_null<clang::NamespaceDecl>(decl.getDeclContext());
context;
context = llvm::dyn_cast_or_null<clang::NamespaceDecl>(context->getDeclContext())) {
std::string ns = std::string(context->getName());
if (ns.starts_with("__")) {
if (!context->isInline())
return "";
} else
result = ns + "::" + result;
}
return result;
}
static bool is_viable_declaration(const clang::NamedDecl* decl) {
if (decl->getFriendObjectKind() != clang::Decl::FOK_None)
return false;
if (clang::CXXMethodDecl::classof(decl))
return false;
if (clang::CXXDeductionGuideDecl::classof(decl))
return false;
if (clang::FunctionDecl::classof(decl))
return true;
if (clang::CXXConstructorDecl::classof(decl))
return false;
if (const auto* r = llvm::dyn_cast_or_null<clang::RecordDecl>(decl))
return !r->isLambda() && !r->isImplicit();
return llvm::isa<clang::EnumDecl, clang::VarDecl, clang::ConceptDecl, clang::TypedefNameDecl, clang::UsingDecl>(decl);
}
static bool is_global_name_exported_by_std_module(std::string_view name) {
static const std::set<std::string_view> valid{
"operator delete", "operator delete[]", "operator new", "operator new[]"};
return valid.contains(name);
}
static bool is_valid_declaration_context(
const clang::NamedDecl& decl, std::string_view name, header_exportable_declarations::FileType file_type) {
const clang::DeclContext& context = *decl.getDeclContext();
if (context.isNamespace())
return true;
if (context.isFunctionOrMethod() || context.isRecord())
return false;
if (is_global_name_exported_by_std_module(name))
return true;
return file_type != header_exportable_declarations::FileType::Header;
}
static bool is_module(header_exportable_declarations::FileType file_type) {
switch (file_type) {
case header_exportable_declarations::FileType::Module:
case header_exportable_declarations::FileType::ModulePartition:
case header_exportable_declarations::FileType::CompatModule:
case header_exportable_declarations::FileType::CompatModulePartition:
return true;
case header_exportable_declarations::FileType::Header:
case header_exportable_declarations::FileType::CHeader:
return false;
case header_exportable_declarations::FileType::Unknown:
llvm::errs() << "This should be unreachable.\n";
return false;
}
}
void header_exportable_declarations::check(const clang::ast_matchers::MatchFinder::MatchResult& result) {
if (const auto* decl = result.Nodes.getNodeAs<clang::NamedDecl>("header_exportable_declarations"); decl != nullptr) {
if (!is_viable_declaration(decl))
return;
std::string name = get_qualified_name(*decl);
if (name.empty())
return;
if (is_module(file_type_))
if (decl->getModuleOwnershipKind() != clang::Decl::ModuleOwnershipKind::VisibleWhenImported)
return;
if (!is_valid_declaration_context(*decl, name, file_type_))
return;
if (decls_.contains(name)) {
if (file_type_ == FileType::ModulePartition || file_type_ == FileType::CompatModulePartition)
llvm::errs() << "Duplicated export of '" << name << "'.\n";
else
return;
}
std::cout << "using ::" << std::string{name} << ";\n";
decls_.insert(name);
} else if (const auto* decl = result.Nodes.getNodeAs<clang::NamedDecl>("cheader_exportable_declarations");
decl != nullptr) {
if (decl->getDeclContext()->isNamespace())
return;
if (!is_viable_declaration(decl))
return;
std::string name = get_qualified_name(*decl);
if (name.empty())
return;
if (global_decls_.contains(name))
return;
global_decls_.insert(name);
}
}
}