Parse cpp_namespace_alias

This commit is contained in:
Jonathan Müller 2017-02-22 11:04:34 +01:00
commit 45e9e5305b
9 changed files with 161 additions and 11 deletions

View file

@ -9,6 +9,14 @@
namespace cppast
{
enum class cpp_entity_kind;
/// \exclude
namespace detail
{
void check_entity_cast(const cpp_entity& e, cpp_entity_kind expected_kind);
} // namespace detail
/// A basic reference to some kind of [cppast::cpp_entity]().
template <typename T, typename Predicate>
class basic_cpp_entity_ref
@ -36,6 +44,9 @@ namespace cppast
type_safe::optional_ref<const T> get(const cpp_entity_index& idx) const noexcept
{
auto entity = idx.lookup(target_);
if (!entity)
return type_safe::nullopt;
detail::check_entity_cast(entity.value(), T::kind());
return static_cast<const T&>(entity.value());
}

View file

@ -88,12 +88,16 @@ namespace cppast
class cpp_namespace_alias final : public cpp_entity
{
public:
static cpp_entity_kind kind() noexcept;
/// \returns A newly created and registered namespace alias.
static std::unique_ptr<cpp_namespace_alias> build(const cpp_entity_index& idx,
cpp_entity_id id, std::string name,
cpp_namespace_ref target);
/// \returns The [cppast::cpp_namespace_ref]() to the aliased namespace.
/// \notes If the namespace aliases aliases another namespace alias,
/// the target entity will still be the namespace, not the alias.
const cpp_namespace_ref& target() const noexcept
{
return target_;

15
src/cpp_entity_ref.cpp Normal file
View file

@ -0,0 +1,15 @@
// Copyright (C) 2017 Jonathan Müller <jonathanmueller.dev@gmail.com>
// This file is subject to the license terms in the LICENSE file
// found in the top-level directory of this distribution.
#include <cppast/cpp_entity_ref.hpp>
#include <cppast/cpp_entity.hpp>
using namespace cppast;
void detail::check_entity_cast(const cpp_entity& e, cpp_entity_kind expected_kind)
{
DEBUG_ASSERT(e.kind() == expected_kind, detail::precondition_error_handler{},
"mismatched entity kind");
}

View file

@ -33,11 +33,16 @@ std::unique_ptr<cpp_namespace_alias> cpp_namespace_alias::build(const cpp_entity
return ptr;
}
cpp_entity_kind cpp_namespace_alias::do_get_entity_kind() const noexcept
cpp_entity_kind cpp_namespace_alias::kind() noexcept
{
return cpp_entity_kind::namespace_alias_t;
}
cpp_entity_kind cpp_namespace_alias::do_get_entity_kind() const noexcept
{
return kind();
}
cpp_entity_kind cpp_using_directive::do_get_entity_kind() const noexcept
{
return cpp_entity_kind::using_directive_t;

View file

@ -15,15 +15,20 @@ namespace cppast
{
// visits direct children of an entity
template <typename Func>
void visit_children(CXCursor parent, Func f)
void visit_children(CXCursor parent, Func f, bool recurse = false)
{
clang_visitChildren(parent,
[](CXCursor cur, CXCursor, CXClientData data) {
auto& actual_cb = *static_cast<Func*>(data);
actual_cb(cur);
return CXChildVisit_Continue;
},
&f);
auto continue_lambda = [](CXCursor cur, CXCursor, CXClientData data) {
auto& actual_cb = *static_cast<Func*>(data);
actual_cb(cur);
return CXChildVisit_Continue;
};
auto recurse_lambda = [](CXCursor cur, CXCursor, CXClientData data) {
auto& actual_cb = *static_cast<Func*>(data);
actual_cb(cur);
return CXChildVisit_Recurse;
};
clang_visitChildren(parent, recurse ? recurse_lambda : continue_lambda, &f);
}
// visits a translation unit

View file

@ -12,7 +12,8 @@ using namespace cppast;
namespace
{
cpp_namespace::builder make_builder(const detail::parse_context& context, const CXCursor& cur)
cpp_namespace::builder make_ns_builder(const detail::parse_context& context,
const CXCursor& cur)
{
detail::tokenizer tokenizer(context.tu, context.file, cur);
detail::token_stream stream(tokenizer, cur);
@ -38,7 +39,7 @@ std::unique_ptr<cpp_entity> detail::parse_cpp_namespace(const detail::parse_cont
{
DEBUG_ASSERT(cur.kind == CXCursor_Namespace, detail::assert_handler{});
auto builder = make_builder(context, cur);
auto builder = make_ns_builder(context, cur);
detail::visit_children(cur, [&](const CXCursor& cur) {
auto entity = parse_entity(context, cur);
if (entity)
@ -46,3 +47,49 @@ std::unique_ptr<cpp_entity> detail::parse_cpp_namespace(const detail::parse_cont
});
return builder.finish(*context.idx, get_entity_id(cur));
}
namespace
{
cpp_entity_id ns_alias_target_id(const CXCursor& cur)
{
cpp_entity_id result("");
detail::visit_children(cur,
[&](const CXCursor& child) {
auto referenced = clang_getCursorReferenced(child);
auto kind = clang_getCursorKind(referenced);
if (kind == CXCursor_Namespace)
result = detail::get_entity_id(referenced);
else if (kind == CXCursor_NamespaceAlias)
// get target of namespace alias instead
result = ns_alias_target_id(referenced);
else
DEBUG_UNREACHABLE(detail::parse_error_handler{}, cur,
"unexpected child of namespace alias");
},
true);
return result;
}
}
std::unique_ptr<cpp_entity> detail::parse_cpp_namespace_alias(const detail::parse_context& context,
const CXCursor& cur)
{
DEBUG_ASSERT(cur.kind == CXCursor_NamespaceAlias, detail::assert_handler{});
detail::tokenizer tokenizer(context.tu, context.file, cur);
detail::token_stream stream(tokenizer, cur);
// namespace <identifier> = <nested identifier>;
detail::skip(stream, "namespace");
auto name = stream.get().c_str();
detail::skip(stream, "=");
// <nested identifier>;
std::string target_name;
while (!detail::skip_if(stream, ";"))
target_name += stream.get().c_str();
auto target = cpp_namespace_ref(ns_alias_target_id(cur), std::move(target_name));
return cpp_namespace_alias::build(*context.idx, get_entity_id(cur), std::move(name),
std::move(target));
}

View file

@ -21,6 +21,8 @@ std::unique_ptr<cpp_entity> detail::parse_entity(const detail::parse_context& co
{
case CXCursor_Namespace:
return parse_cpp_namespace(context, cur);
case CXCursor_NamespaceAlias:
return parse_cpp_namespace_alias(context, cur);
default:
break;

View file

@ -29,6 +29,9 @@ namespace cppast
std::unique_ptr<cpp_entity> parse_cpp_namespace(const parse_context& context,
const CXCursor& cur);
std::unique_ptr<cpp_entity> parse_cpp_namespace_alias(const parse_context& context,
const CXCursor& cur);
std::unique_ptr<cpp_entity> parse_entity(const parse_context& context, const CXCursor& cur);
}
} // namespace cppast::detail

View file

@ -50,3 +50,61 @@ namespace c
});
REQUIRE(count == 4u);
}
TEST_CASE("cpp_namespace_alias")
{
auto code = R"(
namespace ns1 {}
namespace ns2 {}
namespace a = ns1;
namespace b = ns2;
namespace outer
{
namespace ns {}
namespace c = ns;
namespace d = ns1;
}
namespace e = outer::ns;
namespace f = outer::c;
)";
cpp_entity_index idx;
auto check_alias = [&](const cpp_namespace_alias& alias, const char* target_name,
const char* target_full_name) {
auto& target = alias.target();
REQUIRE(target.name() == target_name);
auto entity = target.get(idx);
REQUIRE(entity);
REQUIRE(full_name(entity.value()) == target_full_name);
};
auto file = parse(idx, "cpp_namespace_alias.cpp", code);
auto count = test_visit<cpp_namespace_alias>(*file, [&](const cpp_namespace_alias& alias) {
if (alias.name() == "a")
check_alias(alias, "ns1", "ns1");
else if (alias.name() == "b")
check_alias(alias, "ns2", "ns2");
else if (alias.name() == "c")
{
check_parent(alias, "outer", "outer::c");
check_alias(alias, "ns", "outer::ns");
}
else if (alias.name() == "d")
{
check_parent(alias, "outer", "outer::d");
check_alias(alias, "ns1", "ns1");
}
else if (alias.name() == "e")
check_alias(alias, "outer::ns", "outer::ns");
else if (alias.name() == "f")
check_alias(alias, "outer::c", "outer::ns");
else
REQUIRE(false);
});
REQUIRE(count == 6u);
}