| //===----------------------------------------------------------------------===// |
| // |
| // Part of the LLVM Project, 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 |
| // |
| //===----------------------------------------------------------------------===// |
| |
| // <string> |
| |
| // Make sure the size we allocate and deallocate match. See https://github.com/llvm/llvm-project/pull/90292. |
| |
| #include <string> |
| #include <cassert> |
| #include <cstdint> |
| #include <type_traits> |
| |
| #include "test_macros.h" |
| |
| static int allocated_; |
| |
| template <class T, class Sz> |
| struct test_alloc { |
| typedef Sz size_type; |
| typedef typename std::make_signed<Sz>::type difference_type; |
| typedef T value_type; |
| typedef value_type* pointer; |
| typedef const value_type* const_pointer; |
| typedef typename std::add_lvalue_reference<value_type>::type reference; |
| typedef typename std::add_lvalue_reference<const value_type>::type const_reference; |
| |
| template <class U> |
| struct rebind { |
| typedef test_alloc<U, Sz> other; |
| }; |
| |
| TEST_CONSTEXPR test_alloc() TEST_NOEXCEPT {} |
| |
| template <class U> |
| TEST_CONSTEXPR test_alloc(const test_alloc<U, Sz>&) TEST_NOEXCEPT {} |
| |
| pointer allocate(size_type n, const void* = nullptr) { |
| allocated_ += n; |
| return std::allocator<value_type>().allocate(n); |
| } |
| |
| void deallocate(pointer p, size_type s) { |
| allocated_ -= s; |
| std::allocator<value_type>().deallocate(p, s); |
| } |
| |
| template <class U> |
| friend TEST_CONSTEXPR bool operator==(const test_alloc&, const test_alloc<U, Sz>&) TEST_NOEXCEPT { |
| return true; |
| } |
| |
| #if TEST_STD_VER < 20 |
| template <class U> |
| friend TEST_CONSTEXPR bool operator!=(const test_alloc&, const test_alloc<U, Sz>&) TEST_NOEXCEPT { |
| return false; |
| } |
| #endif |
| }; |
| |
| template <class Sz> |
| void test() { |
| for (int i = 1; i < 1000; ++i) { |
| using Str = std::basic_string<char, std::char_traits<char>, test_alloc<char, Sz> >; |
| { |
| Str s(i, 't'); |
| assert(allocated_ == 0 || allocated_ >= i); |
| } |
| } |
| assert(allocated_ == 0); |
| } |
| |
| int main(int, char**) { |
| test<uint32_t>(); |
| test<uint64_t>(); |
| test<size_t>(); |
| |
| return 0; |
| } |