| Doc. no.: | N4510 | 
|---|---|
| Date: | 2015-05-05 | 
| Project: | Programming Language C++, Library Working Group | 
| Revises: | N4390 | 
| Reply-to: | Zhihao Yuan <zy at miator dot net> | 
In the original version (N3890) of this paper, we explored the possibility to make all STL containers usable in the recursive data structure definitions, such as
struct Entry
{
    std::list<Entry> messages;
    // ...
};Based on the discussion on the Issaquah meeting, we achieved the consensus to processed with the approach – “Containers of Incomplete Types”, but limit the scope to std::vector, std::list, and std::forward_list, as the first step.
The approach itself is well-known shortly after the C++98 standard being shipped[1], is one of the main features provided by Boost.Container[2] and libstdc++, and is receiving increasing interests in libc++. By limiting the scope, MSVC STL, libstdc++, and libc++ already support the proposed solution.
Conditionally require the targeted containers to be complete types.
Define “allocator completeness requirements” as a dependency of 1).
Require std::allocator<T> to unconditionally satisfy the “allocator completeness requirements” defined in 2).
This wording is relative to N4431.
New section 17.6.3.5.1 [allocator.requirements.completeness]:
17.6.3.5.1 Allocator completeness requirements [allocator.requirements.completeness]
If
Xis an allocator class for typeT,Xadditionally satisfies the allocator completeness requirements if, whether or notTis a complete type:
Xis a complete type, and- all the member types of
allocator_traits<X>(20.7.8) other thanvalue_typeare complete types.
New paragraph in 20.7.9 [default.allocator], before the synopsis, as the first paragraph:
All specializations of the default allocator satisfy the allocator completeness requirements (17.6.3.5.1).
New paragraph in 23.3.4.1 [forwardlist.overview], as paragraph 4, after the synopsis:
An incomplete type
Tmay be used when instantiatingforward_listif the allocator satisfies the allocator completeness requirements (17.6.3.5.1).Tshall be complete before any member of the resulting specialization offorward_listis referenced.
New paragraph in 23.3.5.1 [list.overview], as paragraph 3, after the synopsis:
An incomplete type
Tmay be used when instantiatinglistif the allocator satisfies the allocator completeness requirements (17.6.3.5.1).Tshall be complete before any member of the resulting specialization oflistis referenced.
New paragraph in 23.3.6.1 [vector.overview], as paragraph 3, after the synopsis:
An incomplete type
Tmay be used when instantiatingvectorif the allocator satisfies the allocator completeness requirements (17.6.3.5.1).Tshall be complete before any member of the resulting specialization ofvectoris referenced.
Let X be an allocator type for T.
“whether or not T is a complete type… X is a complete type” is unconditional regarding to the completeness of T, which could be a type which is not yet complete, or a type which cannot be incomplete, e.g. int, while X may be X<T>, X<T, Args...>, or even a non-template class IntAllocator. The completeness of X is required because a container may store an instance of the allocator.
“all the member types of allocator_traits<X>… other than value_type are complete types” allows a container to query minimal information from the allocator type to finish the class definition. For a non-template container, the class definition must not require a complete T if T could be incomplete; for a specialization the outcome of the rule is more “flexible”, since the separately-instantiated entities do not require a complete T when the specialization is implicitly instantiated.
Let C be a sequence container.
“An incomplete type T may be used…” revokes [res.on.function]/2.5. The rule is unconditional regarding to the completeness of T. If the rule is extended to cover associative/hash containers, the rule may be unconditional regarding to the completeness of K and/or V.
“… if the allocator satisfies the allocator completeness requirements” ensures that the specialization can be instantiated from a user’s point of view; extra template parameters or meta-programming within the class definition must not make the instantiation ill-formed. If the rule is extended to cover associative/hash containers, the rule may be conditional regarding to the completeness requirements on other template parameters, e.g. Compare and Hash, as well.
“T shall be complete before any member of the resulting specialization… is referenced.” avoids ill-formed programs caused by instantiating separately-instantiated entities before T is complete. It seldom happens for a typical recursive definition use case, but if a user try the following,
struct NT
{
    C<T> ct;
};, he/she has a chance to reference, for example, the default constructor of C<T> after the definition of NT, while this rule requires T to be completed before the default constructor is referenced.
[1] Austern, Matthew H. The Standard Librarian: Containers of Incomplete Types. http://www.drdobbs.com/the-standard-librarian-containers-of-inc/184403814
[2] Main features: Containers of Incomplete Types. “Boost.Container” http://www.boost.org/doc/libs/1_55_0/doc/html/container/main_features.html#container.main_features.containers_of_incomplete_types
Thanks to the library implementers who helped refine the idea, run my test code, and review the patches and the wording.
Special thanks go to Jonathan Wakely, who reviewed this paper many times, and Richard Smith, who wrote the new wording.