From e919a61df1ff764b1c92d9388dc4f8c18abf18d3 Mon Sep 17 00:00:00 2001 From: alandefreitas Date: Wed, 30 Oct 2024 15:38:27 -0300 Subject: [PATCH] refactor: unify dom lazy array classes The DomCorpus uses multiple classes that represented lazy arrays. This commit simplifies the dom by unifying all of these array implementations. --- include/mrdocs/Metadata/Source.hpp | 27 ++ src/lib/Dom/LazyArray.hpp | 170 +++++++++ src/lib/Metadata/DomCorpus.cpp | 579 +++++++++++------------------ src/lib/Metadata/Source.cpp | 53 +++ 4 files changed, 463 insertions(+), 366 deletions(-) create mode 100644 src/lib/Dom/LazyArray.hpp diff --git a/include/mrdocs/Metadata/Source.hpp b/include/mrdocs/Metadata/Source.hpp index 307a9838d..5adb6acc2 100644 --- a/include/mrdocs/Metadata/Source.hpp +++ b/include/mrdocs/Metadata/Source.hpp @@ -36,6 +36,15 @@ MRDOCS_DECL std::string_view toString(FileKind kind); +namespace dom { + template <> + struct ToValue + { + std::string_view + operator()(FileKind id) const; + }; +} + struct MRDOCS_DECL Location { @@ -76,6 +85,15 @@ struct MRDOCS_DECL } }; +namespace dom { + template <> + struct ToValue + { + dom::Object + operator()(Location const& loc) const; + }; +} + struct LocationEmptyPredicate { constexpr bool operator()( @@ -113,6 +131,15 @@ struct MRDOCS_DECL SourceInfo() = default; }; +namespace dom { + template <> + struct ToValue + { + dom::Object + operator()(SourceInfo const& info) const; + }; +} + } // mrdocs } // clang diff --git a/src/lib/Dom/LazyArray.hpp b/src/lib/Dom/LazyArray.hpp new file mode 100644 index 000000000..f7ca58b29 --- /dev/null +++ b/src/lib/Dom/LazyArray.hpp @@ -0,0 +1,170 @@ +// +// Licensed 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 +// +// Copyright (c) 2024 Alan de Freitas (alandefreitas@gmail.com) +// +// Official repository: https://github.com/cppalliance/mrdocs +// + +#ifndef MRDOCS_LIB_DOM_LAZY_ARRAY_HPP +#define MRDOCS_LIB_DOM_LAZY_ARRAY_HPP + +#include "mrdocs/Dom.hpp" +#include "mrdocs/Platform.hpp" +#include "mrdocs/Support/Error.hpp" +#include +#include + +namespace clang { +namespace mrdocs { +namespace dom { + +namespace detail { + struct noop { + template + auto operator()(T&& t) const + { + return t; + } + }; + + struct no_size_tag {}; +} + +/** Lazy array implementation + + This array type is used to define a dom::Array + whose members are evaluated on demand + as they are accessed. + + Each member can goes through a transform + function before being returned as a Value so + that all types can be converted to dom::Value. + + The underlying representation of the array is + a range from where the elements are extracted. + Elements in this range should be convertible + to dom::Value. + + This class is typically useful for + implementing arrays that are expensive + and have recursive dependencies, as these + recursive dependencies can also be deferred. + + Unlike a LazyObjectImpl, which contains an + overlay object, this implementation is + read-only. The `set` and `emplace_back` + methods are not implemented. + +*/ +template +requires + std::invocable> && + std::constructible_from>> +class LazyArrayImpl : public ArrayImpl +{ + using const_iterator_t = decltype(std::ranges::cbegin(std::declval())); + using const_sentinel_t = decltype(std::ranges::cend(std::declval())); + using size_type = std::conditional_t< + std::ranges::sized_range, + std::ranges::range_size_t, + detail::no_size_tag>; + + const_iterator_t begin_; + const_sentinel_t end_; + [[no_unique_address]] size_type size_; + [[no_unique_address]] F transform_; + +public: + explicit + LazyArrayImpl(R const& arr) + : begin_(std::ranges::begin(arr)) + , end_(std::ranges::end(arr)) + { + if constexpr (std::ranges::sized_range) + { + size_ = std::ranges::size(arr); + } + } + + explicit + LazyArrayImpl(R const& arr, F transform) + : begin_(std::ranges::begin(arr)) + , end_(std::ranges::end(arr)) + , transform_(std::move(transform)) + { + if constexpr (std::ranges::sized_range) + { + size_ = std::ranges::size(arr); + } + } + + ~LazyArrayImpl() override = default; + + /// @copydoc ObjectImpl::type_key + char const* + type_key() const noexcept override + { + return "LazyArray"; + } + + std::size_t size() const noexcept override + { + if constexpr (std::ranges::sized_range) + { + return size_; + } + else + { + return std::ranges::distance(begin_, end_); + } + } + + dom::Value get(std::size_t i) const override + { + if (i >= size()) + { + return {}; + } + auto it = begin_; + std::ranges::advance(it, i); + if constexpr (std::is_same_v) + { + return Value(*it); + } + else + { + return Value(transform_(*it)); + } + } +}; + +/** Return a new dom::Array based on a lazy array implementation. +*/ +template +requires std::constructible_from> +Array +LazyArray(T const& arr) +{ + return newArray>(arr); +} + +/** Return a new dom::Array based on a transformed lazy array implementation. +*/ +template +requires + std::invocable> && + std::constructible_from>> +Array +LazyArray(T const& arr, F transform) +{ + return newArray>(arr, std::move(transform)); +} + +} // dom +} // mrdocs +} // clang + +#endif diff --git a/src/lib/Metadata/DomCorpus.cpp b/src/lib/Metadata/DomCorpus.cpp index 34c25deef..c7d50007b 100644 --- a/src/lib/Metadata/DomCorpus.cpp +++ b/src/lib/Metadata/DomCorpus.cpp @@ -5,6 +5,7 @@ // // Copyright (c) 2023 Vinnie Falco (vinnie.falco@gmail.com) // Copyright (c) 2023 Krystian Stasiowski (sdkrystian@gmail.com) +// Copyright (c) 2024 Alan de Freitas (alandefreitas@gmail.com) // // Official repository: https://github.com/cppalliance/mrdocs // @@ -12,6 +13,7 @@ #include "lib/Support/Radix.hpp" #include "lib/Support/LegibleNames.hpp" #include "lib/Dom/LazyObject.hpp" +#include "lib/Dom/LazyArray.hpp" #include #include #include @@ -29,241 +31,55 @@ namespace { // //------------------------------------------------ -dom::Value -domCreate( - std::unique_ptr const& jd, +/* Get overloads for a scope + + This function takes a type derived from ScopeInfo, + such as Namespace or Record, and returns a vector + of overloads in this scope using the DomCorpus + to resolve the SymbolIDs in this scope. + + If the symbol in the scope has an ID, the ID + is included in the result. + Otherwise, the symbol is an OverloadSet, and + the OverloadSet is included. + + Unfortunately, this information is not readily + available in the Corpus, so we can't have lazy + references to them. + Instead, we need to traverse the overloads and + generate the data whenever the information + is requested. + */ +template ScopeInfoTy> +dom::Array +generateScopeOverloadsArray( + ScopeInfoTy const& I, DomCorpus const& domCorpus) { - if(!jd) - return nullptr; - return domCorpus.getJavadoc(*jd); -} - -/* A Lazy DOM Array type that replaces symbol IDs with their - corresponding DOM objects. -*/ -class DomSymbolArray : public dom::ArrayImpl -{ - std::span list_; - DomCorpus const& domCorpus_; - -public: - DomSymbolArray( - std::span list, - DomCorpus const& domCorpus) noexcept - : list_(list) - , domCorpus_(domCorpus) - { - } - - std::size_t size() const noexcept override + dom::Array res; + domCorpus->traverseOverloads(I, + [&](const auto& C) { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - if (i < list_.size()) + if constexpr(requires { C.id; }) { - return domCorpus_.get(list_[i]); + res.push_back(domCorpus.get(C.id)); } - return dom::Value{}; - } -}; - -//------------------------------------------------ - -dom::Object -domCreate( - OverloadSet const& overloads, - DomCorpus const& domCorpus) -{ - return dom::Object({ - { "kind", "overload"}, - { "name", overloads.Name }, - { "members", dom::newArray( - overloads.Members, domCorpus) }, - { "namespace", dom::newArray( - overloads.Namespace, domCorpus) }, - { "parent", domCorpus.get(overloads.Parent) } - }); -} - -class DomOverloadsArray : public dom::ArrayImpl -{ - std::vector> overloads_; - - DomCorpus const& domCorpus_; - -public: - template ScopeInfoTy> - DomOverloadsArray( - const ScopeInfoTy& I, - DomCorpus const& domCorpus) noexcept - : domCorpus_(domCorpus) - { - overloads_.reserve(I.Lookups.size()); - domCorpus_->traverseOverloads(I, - [&](const auto& C) + else { - if constexpr(requires { C.id; }) - overloads_.emplace_back(C.id); - else - overloads_.emplace_back(C); - }); - } - - std::size_t size() const noexcept override - { - return overloads_.size(); - } - - dom::Value get(std::size_t index) const override - { - MRDOCS_ASSERT(index < size()); - const auto& member = overloads_[index]; - if(auto* id = std::get_if(&member)) - return domCorpus_.get(*id); - return domCorpus_.getOverloads( - std::get(member)); - } -}; - -//------------------------------------------------ -// -// Location -// SourceInfo -// -//------------------------------------------------ - -dom::Object -domCreate(Location const& loc) -{ - return dom::Object({ - { "path", loc.Path }, - { "file", loc.Filename }, - { "line", loc.LineNumber }, - { "kind", toString(loc.Kind) }, - { "documented", loc.Documented } - }); -} - -class DomLocationArray : public dom::ArrayImpl -{ - std::vector const& list_; - -public: - explicit - DomLocationArray( - std::vector const& list) noexcept - : list_(list) - { - } - - std::size_t size() const noexcept override - { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - MRDOCS_ASSERT(i < list_.size()); - return domCreate(list_[i]); - } -}; - -dom::Object -domCreate(SourceInfo const& I) -{ - dom::Object::storage_type entries; - if(I.DefLoc) - entries.emplace_back("def", domCreate(*I.DefLoc)); - if(! I.Loc.empty()) - entries.emplace_back("decl", dom::newArray(I.Loc)); - return dom::Object(std::move(entries)); + res.push_back(domCorpus.getOverloads(C)); + } + }); + return res; } //------------------------------------------------ // -// TypeInfo +// domCreate // //------------------------------------------------ static dom::Value domCreate( std::unique_ptr const&, DomCorpus const&); - -class DomTypeInfoArray : public dom::ArrayImpl -{ - std::vector> const& list_; - DomCorpus const& domCorpus_; - -public: - DomTypeInfoArray( - std::vector> const& list, - DomCorpus const& domCorpus) noexcept - : list_(list) - , domCorpus_(domCorpus) - { - } - - std::size_t size() const noexcept override - { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - MRDOCS_ASSERT(i < list_.size()); - return domCreate(list_[i], domCorpus_); - } -}; - -//------------------------------------------------ -// -// Param -// -//------------------------------------------------ - -/** An array of function parameters -*/ -class DomParamArray : public dom::ArrayImpl -{ - std::vector const& list_; - DomCorpus const& domCorpus_; - -public: - DomParamArray( - std::vector const& list, - DomCorpus const& domCorpus) noexcept - : list_(list) - , domCorpus_(domCorpus) - { - } - - std::size_t size() const noexcept override - { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - MRDOCS_ASSERT(i < list_.size()); - auto const& I = list_[i]; - return dom::Object({ - { "name", dom::stringOrNull(I.Name) }, - { "type", domCreate(I.Type, domCorpus_) }, - { "default", dom::stringOrNull(I.Default) } - }); - } -}; - -//------------------------------------------------ -// -// TemplateInfo -// -//------------------------------------------------ - static dom::Value domCreate( std::unique_ptr const&, DomCorpus const&); static dom::Value domCreate( @@ -273,65 +89,6 @@ static dom::Value domCreate( static dom::Value domCreate( std::unique_ptr const& I, DomCorpus const&); -//------------------------------------------------ - -/** An array of template arguments -*/ -class DomTArgArray : public dom::ArrayImpl -{ - std::vector> const& list_; - DomCorpus const& domCorpus_; - -public: - DomTArgArray( - std::vector> const& list, - DomCorpus const& domCorpus) noexcept - : list_(list) - , domCorpus_(domCorpus) - { - } - - std::size_t size() const noexcept override - { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - MRDOCS_ASSERT(i < list_.size()); - return domCreate(list_[i], domCorpus_); - } -}; - -/** An array of template parameters -*/ -class DomTParamArray : public dom::ArrayImpl -{ - std::vector> const& list_; - DomCorpus const& domCorpus_; - -public: - DomTParamArray( - std::vector> const& list, - DomCorpus const& domCorpus) noexcept - : list_(list) - , domCorpus_(domCorpus) - { - } - - std::size_t size() const noexcept override - { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - return domCreate(list_[i], domCorpus_); - } -}; - -//------------------------------------------------ - static dom::Value domCreate( @@ -402,8 +159,10 @@ domCreate( if constexpr(T::isTemplate()) { entries.emplace_back("params", - dom::newArray( - t.Params, domCorpus)); + dom::LazyArray(t.Params, [&](std::unique_ptr const& I) + { + return domCreate(I, domCorpus); + })); } }); return dom::Object(std::move(entries)); @@ -420,8 +179,14 @@ domCreate( return dom::Object({ { "kind", toString(I->specializationKind()) }, { "primary", domCorpus.get(I->Primary) }, - { "params", dom::newArray( I->Params, domCorpus) }, - { "args", dom::newArray(I->Args, domCorpus) }, + { "params", dom::LazyArray(I->Params, [&](std::unique_ptr const& I) + { + return domCreate(I, domCorpus); + }) }, + { "args", dom::LazyArray(I->Args, [&](std::unique_ptr const& I) + { + return domCreate(I, domCorpus); + }) }, { "requires", dom::stringOrNull(I->Requires.Written) } }); } @@ -446,8 +211,10 @@ domCreate( if constexpr(requires { t.TemplateArgs; }) entries.emplace_back("args", - dom::newArray(t.TemplateArgs, domCorpus)); - + dom::LazyArray(t.TemplateArgs, [&](std::unique_ptr const& I) + { + return domCreate(I, domCorpus); + })); entries.emplace_back("prefix", domCreate(t.Prefix, domCorpus)); }); return dom::Object(std::move(entries)); @@ -505,13 +272,15 @@ domCreate( entries.emplace_back("bounds-expr", t.Bounds.Written); } - if constexpr(T::isFunction()) { entries.emplace_back("return-type", domCreate(t.ReturnType, domCorpus)); entries.emplace_back("param-types", - dom::newArray(t.ParamTypes, domCorpus)); + dom::LazyArray(t.ParamTypes, [&](std::unique_ptr const& I) + { + return domCreate(I, domCorpus); + })); entries.emplace_back("exception-spec", toString(t.ExceptionSpec)); entries.emplace_back("ref-qualifier", @@ -522,42 +291,6 @@ domCreate( return dom::Object(std::move(entries)); } -//------------------------------------------------ -// -// BaseInfo -// -//------------------------------------------------ - -class DomBaseArray : public dom::ArrayImpl -{ - std::vector const& list_; - DomCorpus const& domCorpus_; - -public: - DomBaseArray( - std::vector const& list, - DomCorpus const& domCorpus) noexcept - : list_(list) - , domCorpus_(domCorpus) - { - } - - std::size_t size() const noexcept override - { - return list_.size(); - } - - dom::Value get(std::size_t i) const override - { - auto const& I = list_[i]; - return dom::Object({ - { "access", toString(I.Access) }, - { "isVirtual", I.IsVirtual }, - { "type", domCreate(I.Type, domCorpus_) } - }); - } -}; - //------------------------------------------------ // // Interface @@ -567,7 +300,6 @@ class DomBaseArray : public dom::ArrayImpl class DomTranche : public dom::DefaultObjectImpl { std::shared_ptr tranche_; - DomCorpus const& domCorpus_; static dom::Value @@ -575,7 +307,10 @@ class DomTranche : public dom::DefaultObjectImpl std::span list, DomCorpus const& domCorpus) { - return dom::newArray(list, domCorpus); + return dom::LazyArray(list, [&](SymbolID const& id) + { + return domCorpus.get(id); + }); } static @@ -584,7 +319,7 @@ class DomTranche : public dom::DefaultObjectImpl const ScopeInfo& scope, DomCorpus const& domCorpus) { - return dom::newArray(scope, domCorpus); + return generateScopeOverloadsArray(scope, domCorpus); } public: @@ -601,7 +336,6 @@ class DomTranche : public dom::DefaultObjectImpl { "staticoverloads", init(tranche->StaticOverloads, domCorpus) }, }) , tranche_(tranche) - , domCorpus_(domCorpus) { } }; @@ -773,6 +507,48 @@ namespace dom { { } + /* Resolve types that use SymbolIDs to the corresponding Info object + */ + template + Value + resolve(U const& I) const + { + if constexpr (std::same_as) + { + return domCorpus_->get(I); + } + else if constexpr (std::same_as) + { + return domCorpus_->getJavadoc(I); + } + else if constexpr ( + std::ranges::range && + std::same_as, SymbolID>) + { + return dom::LazyArray(I, [this](SymbolID const& id) + { + return domCorpus_->get(id); + }); + } + else if constexpr ( + std::ranges::range && + std::same_as, Param>) + { + return dom::LazyArray(I, [&](Param const& p) + { + return dom::Object({ + { "name", dom::stringOrNull(p.Name) }, + { "type", domCreate(p.Type, *domCorpus_) }, + { "default", dom::stringOrNull(p.Default) } + }); + }); + } + else + { + MRDOCS_UNREACHABLE(); + } + } + template void map(IO &io, T const& I) const @@ -786,51 +562,78 @@ namespace dom { io.map("kind", I.Kind); io.map("access", I.Access); io.map("implicit", I.Implicit); - io.defer("namespace", [&]{ return dom::newArray(I.Namespace, *domCorpus_); }); - io.defer("doc", [&]{ return domCreate(I.javadoc, *domCorpus_); }); + io.defer("namespace", [&]{ + return resolve(I.Namespace); + }); if (!I.Namespace.empty()) { - io.defer("parent", [&]{ return domCorpus_->get(I.Namespace.front()); }); + io.defer("parent",[&]{ + return resolve(I.Namespace.front()); + }); + } + if (I.javadoc) + { + io.defer("doc",[&]{ + return resolve(*I.javadoc); + }); } if constexpr(std::derived_from) { - io.defer("members", [&]{ return dom::newArray(I.Members, *domCorpus_); }); - io.defer("overloads", [&]{ return dom::newArray(I, *domCorpus_); }); + io.defer("members", [&]{ + return resolve(I.Members); + }); + io.defer("overloads",[&]{ + return generateScopeOverloadsArray(I, *domCorpus_); + }); } if constexpr(std::derived_from) { - io.defer("loc", [&]{ return domCreate(I); }); + io.map("loc", static_cast(I)); } if constexpr(T::isNamespace()) { - io.defer("interface", [&]{ return dom::newObject( - std::make_shared( - makeTranche(I, **domCorpus_)), - *domCorpus_); }); - io.defer("usingDirectives", [&]{ return dom::newArray( - I.UsingDirectives, *domCorpus_); }); + io.defer("interface", [&]{ + return dom::newObject( + std::make_shared( + makeTranche(I, **domCorpus_)), + *domCorpus_); + }); + io.defer("usingDirectives", [&]{ + return resolve(I.UsingDirectives); + }); } if constexpr (T::isRecord()) { io.map("tag", I.KeyKind); - io.defer("defaultAccess", [&]{ return getDefaultAccess(I); }); + io.defer("defaultAccess", [&]{ + return getDefaultAccess(I); + }); io.map("isTypedef", I.IsTypeDef); - io.defer("bases", [&]{ return dom::newArray(I.Bases, *domCorpus_); }); + io.map("bases", dom::LazyArray(I.Bases, [&](BaseInfo const& I) + { + return dom::Object({ + { "access", toString(I.Access) }, + { "isVirtual", I.IsVirtual }, + { "type", domCreate(I.Type, *domCorpus_) } + }); + })); io.defer("interface", [&]{ auto sp = std::make_shared(makeInterface(I, domCorpus_->getCorpus())); return dom::Object({ { "public", dom::newObject(sp->Public, *domCorpus_) }, { "protected", dom::newObject(sp->Protected, *domCorpus_) }, - { "private", dom::newObject(sp->Private, *domCorpus_) }, - // { "overloads", dom::newArray(sp->Overloads, *domCorpus_) }, - // { "static-overloads", dom::newArray(sp->StaticOverloads, *domCorpus_) } + { "private", dom::newObject(sp->Private, *domCorpus_) } }); }); - io.defer("template", [&]{ return domCreate(I.Template, *domCorpus_); }); + io.defer("template", [&]{ + return domCreate(I.Template, *domCorpus_); + }); } if constexpr (T::isEnum()) { - io.defer("type", [&]{ return domCreate(I.UnderlyingType, *domCorpus_); }); + io.defer("type", [&]{ + return domCreate(I.UnderlyingType, *domCorpus_); + }); io.map("isScoped", I.Scoped); } if constexpr (T::isFunction()) @@ -864,7 +667,7 @@ namespace dom { io.map("refQualifier", I.RefQualifier); } io.map("class", I.Class); - io.defer("params", [&]{ return dom::newArray(I.Params, *domCorpus_); }); + io.defer("params", [&]{ return resolve(I.Params); }); io.defer("return", [&]{ return domCreate(I.ReturnType, *domCorpus_); }); io.defer("template", [&]{ return domCreate(I.Template, *domCorpus_); }); io.map("overloadedOperator", I.OverloadedOperator); @@ -877,14 +680,22 @@ namespace dom { } if constexpr (T::isTypedef()) { - io.defer("type", [&]{ return domCreate(I.Type, *domCorpus_); }); - io.defer("template", [&]{ return domCreate(I.Template, *domCorpus_); }); + io.defer("type", [&]{ + return domCreate(I.Type, *domCorpus_); + }); + io.defer("template", [&]{ + return domCreate(I.Template, *domCorpus_); + }); io.map("isUsing", I.IsUsing); } if constexpr (T::isVariable()) { - io.defer("type", [&]{ return domCreate(I.Type, *domCorpus_); }); - io.defer("template", [&]{ return domCreate(I.Template, *domCorpus_); }); + io.defer("type", [&]{ + return domCreate(I.Type, *domCorpus_); + }); + io.defer("template", [&]{ + return domCreate(I.Template, *domCorpus_); + }); if (I.Constexpr != ConstexprKind::None) { io.map("constexprKind", I.Constexpr); @@ -902,7 +713,9 @@ namespace dom { } if constexpr (T::isField()) { - io.defer("type", [&]{ return domCreate(I.Type, *domCorpus_); }); + io.defer("type", [&]{ + return domCreate(I.Type, *domCorpus_); + }); if (!I.Default.Written.empty()) { io.map("default", I.Default.Written); @@ -924,25 +737,39 @@ namespace dom { { if (I.FriendSymbol) { - io.defer("name", [&]{ return domCorpus_->get(I.FriendSymbol).get("name"); }); - io.defer("symbol", [&]{ return domCorpus_->get(I.FriendSymbol); }); + io.defer("name", [&]{ + return domCorpus_->get(I.FriendSymbol).get("name"); + }); + io.defer("symbol", [&]{ + return domCorpus_->get(I.FriendSymbol); + }); } else if (I.FriendType) { - io.defer("name", [&]{ return domCreate(I.FriendType, *domCorpus_).get("name"); }); - io.defer("type", [&]{ return domCreate(I.FriendType, *domCorpus_); }); + io.defer("name", [&]{ + return domCreate(I.FriendType, *domCorpus_).get("name"); + }); + io.defer("type", [&]{ + return domCreate(I.FriendType, *domCorpus_); + }); } } if constexpr (T::isAlias()) { MRDOCS_ASSERT(I.AliasedSymbol); - io.defer("aliasedSymbol", [&]{ return domCreate(I.AliasedSymbol, *domCorpus_); }); + io.defer("aliasedSymbol", [&]{ + return domCreate(I.AliasedSymbol, *domCorpus_); + }); } if constexpr (T::isUsing()) { io.map("class", I.Class); - io.defer("shadows", [&]{ return dom::newArray(I.UsingSymbols, *domCorpus_); }); - io.defer("qualifier", [&]{ return domCreate(I.Qualifier, *domCorpus_); }); + io.defer("shadows", [&]{ + return resolve(I.UsingSymbols); + }); + io.defer("qualifier", [&]{ + return domCreate(I.Qualifier, *domCorpus_); + }); } if constexpr (T::isEnumerator()) { @@ -953,14 +780,24 @@ namespace dom { } if constexpr (T::isGuide()) { - io.defer("params", [&]{ return dom::newArray(I.Params, *domCorpus_); }); - io.defer("deduced", [&]{ return domCreate(I.Deduced, *domCorpus_); }); - io.defer("template", [&]{ return domCreate(I.Template, *domCorpus_); }); - io.defer("explicitSpec", [&]{ return toString(I.Explicit); }); + io.defer("params", [&]{ + return resolve(I.Params); + }); + io.defer("deduced", [&]{ + return domCreate(I.Deduced, *domCorpus_); + }); + io.defer("template", [&]{ + return domCreate(I.Template, *domCorpus_); + }); + io.defer("explicitSpec", [&]{ + return toString(I.Explicit); + }); } if constexpr (T::isConcept()) { - io.defer("template", [&]{ return domCreate(I.Template, *domCorpus_); }); + io.defer("template", [&]{ + return domCreate(I.Template, *domCorpus_); + }); if (!I.Constraint.Written.empty()) { io.map("constraint", I.Constraint.Written); @@ -1002,9 +839,19 @@ getJavadoc( dom::Object DomCorpus:: getOverloads( - OverloadSet const& os) const + OverloadSet const& overloads) const { - return domCreate(os, *this); + auto resolveFn = [this](SymbolID const& id) + { + return this->get(id); + }; + return dom::Object({ + { "kind", "overload"}, + { "name", overloads.Name }, + { "members", dom::LazyArray(overloads.Members, resolveFn) }, + { "namespace", dom::LazyArray(overloads.Namespace, resolveFn) }, + { "parent", this->get(overloads.Parent) } + }); } } // mrdocs diff --git a/src/lib/Metadata/Source.cpp b/src/lib/Metadata/Source.cpp index 82691fff8..6687af6b0 100644 --- a/src/lib/Metadata/Source.cpp +++ b/src/lib/Metadata/Source.cpp @@ -10,6 +10,8 @@ // #include +#include "lib/Dom/LazyObject.hpp" +#include "lib/Dom/LazyArray.hpp" namespace clang { namespace mrdocs { @@ -30,5 +32,56 @@ toString(FileKind kind) }; } +namespace dom { + std::string_view + ToValue::operator()(FileKind kind) const + { + return toString(kind); + } + + template<> + struct MappingTraits + { + template + void map(IO &io, Location const& loc) const + { + io.map("path", loc.Path); + io.map("file", loc.Filename); + io.map("line", loc.LineNumber); + io.map("kind", loc.Kind); + io.map("documented", loc.Documented); + } + }; + + dom::Object + ToValue::operator()(Location const& loc) const + { + return dom::newObject>(loc); + } + + template<> + struct MappingTraits + { + template + void map(IO &io, SourceInfo const& I) const + { + if (I.DefLoc) + { + io.map("def", *I.DefLoc); + } + if (!I.Loc.empty()) + { + io.map("decl", dom::newArray>>(I.Loc)); + } + } + }; + + dom::Object + ToValue::operator()(SourceInfo const& loc) const + { + return dom::newObject>(loc); + } +} + } // mrdocs } // clang