Skip to content

Commit

Permalink
TopoSorter: Only allow certain params to be incomplete
Browse files Browse the repository at this point in the history
For the containers which are allowed to be declared with incomplete
types, it is only the contained types which are allowed to be
incomplete. Other template parameters (e.g. allocators) must always be
defined before use.
  • Loading branch information
ajor committed Oct 25, 2023
1 parent fcb50df commit 1c63505
Show file tree
Hide file tree
Showing 2 changed files with 18 additions and 8 deletions.
16 changes: 10 additions & 6 deletions oi/type_graph/TopoSorter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -85,30 +85,34 @@ namespace {
* Other containers are not required to do this, but might still have this
* behaviour.
*/
bool containerAllowsIncompleteParams(const Container& c) {
bool containerAllowsIncompleteParam(const Container& c, size_t i) {
switch (c.containerInfo_.ctype) {
case SEQ_TYPE:
case LIST_TYPE:
case UNIQ_PTR_TYPE:
case SHRD_PTR_TYPE:
// Also std::forward_list, if we ever support that
// Would be good to have this as an option in the TOML files
return true;
if (i == 0)
return true;
return false;
default:
return false;
}
}
} // namespace

void TopoSorter::visit(Container& c) {
if (!containerAllowsIncompleteParams(c)) {
for (const auto& param : c.templateParams) {
for (size_t i = 0; i < c.templateParams.size(); i++) {
if (!containerAllowsIncompleteParam(c, i)) {
const auto& param = c.templateParams[i];
accept(param.type());
}
}
sortedTypes_.push_back(c);
if (containerAllowsIncompleteParams(c)) {
for (const auto& param : c.templateParams) {
for (size_t i = 0; i < c.templateParams.size(); i++) {
if (containerAllowsIncompleteParam(c, i)) {
const auto& param = c.templateParams[i];
acceptAfter(param.type());
}
}
Expand Down
10 changes: 8 additions & 2 deletions test/test_topo_sorter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -178,24 +178,30 @@ std::map
}

TEST(TopoSorterTest, ContainersVector) {
// std::vector allows forward declared template parameters
// std::vector allows a forward declared type
auto myparam = Class{1, Class::Kind::Struct, "MyParam", 13};
auto myalloc = Class{1, Class::Kind::Struct, "MyAlloc", 0};
auto mycontainer = getVector();
mycontainer.templateParams.push_back((myparam));
mycontainer.templateParams.push_back((myalloc));

test({mycontainer}, R"(
MyAlloc
std::vector
MyParam
)");
}

TEST(TopoSorterTest, ContainersList) {
// std::list allows forward declared template parameters
// std::list allows a forward declared type
auto myparam = Class{1, Class::Kind::Struct, "MyParam", 13};
auto myalloc = Class{1, Class::Kind::Struct, "MyAlloc", 0};
auto mycontainer = getList();
mycontainer.templateParams.push_back((myparam));
mycontainer.templateParams.push_back((myalloc));

test({mycontainer}, R"(
MyAlloc
std::list
MyParam
)");
Expand Down

0 comments on commit 1c63505

Please sign in to comment.