Index: include/scoped_allocator =================================================================== --- include/scoped_allocator +++ include/scoped_allocator @@ -500,6 +500,56 @@ void construct(_Tp* __p, _Args&& ...__args) {__construct(__uses_alloc_ctor<_Tp, inner_allocator_type, _Args...>(), __p, _VSTD::forward<_Args>(__args)...);} + + template + void construct(pair<_T1, _T2>* __p, piecewise_construct_t, + tuple<_Args1...> __x, tuple<_Args2...> __y) + { + typedef __outermost _OM; + allocator_traits::construct( + _OM()(outer_allocator()), __p, piecewise_construct + , __transform_tuple( + typename __uses_alloc_ctor< + _T1, inner_allocator_type, _Args1... + >::type() + , _VSTD::move(__x) + , typename __make_tuple_indices::type{} + ) + , __transform_tuple( + typename __uses_alloc_ctor< + _T2, inner_allocator_type, _Args2... + >::type() + , _VSTD::move(__y) + , typename __make_tuple_indices::type{} + ) + ); + } + + template + void construct(pair<_T1, _T2>* __p) + { construct(__p, piecewise_construct, tuple<>{}, tuple<>{}); } + + template + void construct(pair<_T1, _T2>* __p, _Up&& __x, _Vp&& __y) { + construct(__p, piecewise_construct, + _VSTD::forward_as_tuple(_VSTD::forward<_Up>(__x)), + _VSTD::forward_as_tuple(_VSTD::forward<_Vp>(__y))); + } + + template + void construct(pair<_T1, _T2>* __p, const pair<_Up, _Vp>& __x) { + construct(__p, piecewise_construct, + _VSTD::forward_as_tuple(__x.first), + _VSTD::forward_as_tuple(__x.second)); + } + + template + void construct(pair<_T1, _T2>* __p, pair<_Up, _Vp>&& __x) { + construct(__p, piecewise_construct, + _VSTD::forward_as_tuple(_VSTD::forward<_Up>(__x.first)), + _VSTD::forward_as_tuple(_VSTD::forward<_Vp>(__x.second))); + } + template _LIBCPP_INLINE_VISIBILITY void destroy(_Tp* __p) @@ -515,6 +565,7 @@ private: + template ::value @@ -566,6 +617,36 @@ ); } + template + _LIBCPP_INLINE_VISIBILITY + tuple<_Args&&...> + __transform_tuple(integral_constant, tuple<_Args...>&& __t, + __tuple_indices<_Idx...>) + { + return _VSTD::forward_as_tuple(_VSTD::get<_Idx>(_VSTD::move(__t))...); + } + + template + _LIBCPP_INLINE_VISIBILITY + tuple + __transform_tuple(integral_constant, tuple<_Args...> && __t, + __tuple_indices<_Idx...>) + { + using _Tup = tuple; + return _Tup(allocator_arg, inner_allocator(), + _VSTD::get<_Idx>(_VSTD::move(__t))...); + } + + template + _LIBCPP_INLINE_VISIBILITY + tuple<_Args&&..., inner_allocator_type&> + __transform_tuple(integral_constant, tuple<_Args...> && __t, + __tuple_indices<_Idx...>) + { + using _Tup = tuple<_Args&&..., inner_allocator_type&>; + return _Tup(_VSTD::get<_Idx>(_VSTD::move(__t))..., inner_allocator()); + } + template friend class __scoped_allocator_storage; }; Index: test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair.pass.cpp =================================================================== --- /dev/null +++ test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair.pass.cpp @@ -0,0 +1,139 @@ +//===----------------------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03 + +// + +// template +// class scoped_allocator_adaptor + +// template +// void scoped_allocator_adaptor::construct(pair*) + +#include +#include +#include +#include +#include +#include +#include "uses_alloc_types.hpp" +#include "controlled_allocators.hpp" + + +void test_no_inner_alloc() +{ + using VoidAlloc = CountingAllocator; + AllocController P; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + A.construct(ptr); + assert(checkConstruct<>(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct<>(ptr->second, UA_AllocLast, CA)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + + } + P.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + A.construct(ptr); + assert(checkConstruct<>(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct<>(ptr->second, UA_None)); + assert((P.checkConstruct&&, + std::tuple<>&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} + +void test_with_inner_alloc() +{ + using VoidAlloc1 = CountingAllocator; + using VoidAlloc2 = CountingAllocator; + + AllocController POuter; + AllocController PInner; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + A.construct(ptr); + assert(checkConstruct<>(ptr->first, UA_AllocArg, I)); + assert(checkConstruct<>(ptr->second, UA_AllocLast)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } + PInner.reset(); + POuter.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + A.construct(ptr); + assert(checkConstruct<>(ptr->first, UA_AllocArg, I)); + assert(checkConstruct<>(ptr->second, UA_None)); + assert((POuter.checkConstruct&&, + std::tuple<>&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} +int main() { + test_no_inner_alloc(); + test_with_inner_alloc(); +} Index: test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_const_lvalue_pair.pass.cpp =================================================================== --- /dev/null +++ test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_const_lvalue_pair.pass.cpp @@ -0,0 +1,155 @@ +//===----------------------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03 + +// + +// template +// class scoped_allocator_adaptor + +// template +// void scoped_allocator_adaptor::construct(pair*, pairconst&) + +#include +#include +#include +#include +#include +#include +#include "uses_alloc_types.hpp" +#include "controlled_allocators.hpp" + + +void test_no_inner_alloc() +{ + using VoidAlloc = CountingAllocator; + AllocController P; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + const PairIn in(x, std::move(y)); + A.construct(ptr, in); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_AllocLast, CA)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + + } + P.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + const PairIn in(x, y); + A.construct(ptr, in); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_None)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} + +void test_with_inner_alloc() +{ + using VoidAlloc1 = CountingAllocator; + using VoidAlloc2 = CountingAllocator; + + AllocController POuter; + AllocController PInner; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + const PairIn in(x, std::move(y)); + A.construct(ptr, in); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_AllocLast)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } + PInner.reset(); + POuter.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + const PairIn in(x, y); + A.construct(ptr, in); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_None)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} +int main() { + test_no_inner_alloc(); + test_with_inner_alloc(); +} Index: test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_piecewise.pass.cpp =================================================================== --- /dev/null +++ test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_piecewise.pass.cpp @@ -0,0 +1,156 @@ +//===----------------------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03 + +// + +// template +// class scoped_allocator_adaptor + +// template +// void scoped_allocator_adaptor::construct(pair*, +// piecewise_construct_t, tuple, tuple) + +#include +#include +#include +#include +#include +#include +#include "uses_alloc_types.hpp" +#include "controlled_allocators.hpp" + + +void test_no_inner_alloc() +{ + using VoidAlloc = CountingAllocator; + AllocController P; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + A.construct(ptr, std::piecewise_construct, + std::forward_as_tuple(x), + std::forward_as_tuple(std::move(y))); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_AllocLast, CA)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + + } + P.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + A.construct(ptr, std::piecewise_construct, + std::forward_as_tuple(std::move(x)), + std::forward_as_tuple(y)); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_None)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} + +void test_with_inner_alloc() +{ + using VoidAlloc1 = CountingAllocator; + using VoidAlloc2 = CountingAllocator; + + AllocController POuter; + AllocController PInner; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + int x = 42; + int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + A.construct(ptr, std::piecewise_construct, + std::forward_as_tuple(x), + std::forward_as_tuple(std::move(y))); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_AllocLast)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } + PInner.reset(); + POuter.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + int x = 42; + const int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + A.construct(ptr, std::piecewise_construct, + std::forward_as_tuple(std::move(x)), + std::forward_as_tuple(std::move(y))); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_None)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} +int main() { + test_no_inner_alloc(); + test_with_inner_alloc(); +} Index: test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_rvalue.pass.cpp =================================================================== --- /dev/null +++ test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_rvalue.pass.cpp @@ -0,0 +1,155 @@ +//===----------------------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03 + +// + +// template +// class scoped_allocator_adaptor + +// template +// void scoped_allocator_adaptor::construct(pair*, pair&&) + +#include +#include +#include +#include +#include +#include +#include "uses_alloc_types.hpp" +#include "controlled_allocators.hpp" + + +void test_no_inner_alloc() +{ + using VoidAlloc = CountingAllocator; + AllocController P; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + PairIn in(x, std::move(y)); + A.construct(ptr, std::move(in)); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_AllocLast, CA)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + + } + P.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + PairIn in(x, y); + A.construct(ptr, std::move(in)); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_None)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} + +void test_with_inner_alloc() +{ + using VoidAlloc1 = CountingAllocator; + using VoidAlloc2 = CountingAllocator; + + AllocController POuter; + AllocController PInner; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + PairIn in(x, std::move(y)); + A.construct(ptr, std::move(in)); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_AllocLast)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } + PInner.reset(); + POuter.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + using PairIn = std::pair; + int x = 42; + int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + PairIn in(x, y); + A.construct(ptr, std::move(in)); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_None)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} +int main() { + test_no_inner_alloc(); + test_with_inner_alloc(); +} Index: test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_values.pass.cpp =================================================================== --- /dev/null +++ test/std/utilities/allocator.adaptor/allocator.adaptor.members/construct_pair_values.pass.cpp @@ -0,0 +1,147 @@ +//===----------------------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is dual licensed under the MIT and the University of Illinois Open +// Source Licenses. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +// UNSUPPORTED: c++98, c++03 + +// + +// template +// class scoped_allocator_adaptor + +// template +// void scoped_allocator_adaptor::construct(pair*, Tp&&, Up&&) + +#include +#include +#include +#include +#include +#include +#include "uses_alloc_types.hpp" +#include "controlled_allocators.hpp" + + +void test_no_inner_alloc() +{ + using VoidAlloc = CountingAllocator; + AllocController P; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + A.construct(ptr, x, std::move(y)); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_AllocLast, CA)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + + } + P.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + int x = 42; + const int y = 101; + using Alloc = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + static_assert(std::uses_allocator >::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Alloc CA(P); + SA A(CA); + A.construct(ptr, std::move(x), y); + assert(checkConstruct(ptr->first, UA_AllocArg, CA)); + assert(checkConstruct(ptr->second, UA_None)); + assert((P.checkConstruct&&, + std::tuple&& + >(CA, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} + +void test_with_inner_alloc() +{ + using VoidAlloc1 = CountingAllocator; + using VoidAlloc2 = CountingAllocator; + + AllocController POuter; + AllocController PInner; + { + using T = UsesAllocatorV1; + using U = UsesAllocatorV2; + using Pair = std::pair; + int x = 42; + int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + A.construct(ptr, x, std::move(y)); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_AllocLast)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } + PInner.reset(); + POuter.reset(); + { + using T = UsesAllocatorV3; + using U = NotUsesAllocator; + using Pair = std::pair; + int x = 42; + const int y = 101; + using Outer = CountingAllocator; + using Inner = CountingAllocator; + using SA = std::scoped_allocator_adaptor; + using SAInner = std::scoped_allocator_adaptor; + static_assert(!std::uses_allocator::value, ""); + static_assert(std::uses_allocator::value, ""); + Pair * ptr = (Pair*)std::malloc(sizeof(Pair)); + Outer O(POuter); + Inner I(PInner); + SA A(O, I); + A.construct(ptr, std::move(x), std::move(y)); + assert(checkConstruct(ptr->first, UA_AllocArg, I)); + assert(checkConstruct(ptr->second, UA_None)); + assert((POuter.checkConstruct&&, + std::tuple&& + >(O, ptr))); + A.destroy(ptr); + std::free(ptr); + } +} +int main() { + test_no_inner_alloc(); + test_with_inner_alloc(); +}