//===----------------------------------------------------------------------===// // // 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 // //===----------------------------------------------------------------------===// // UNSUPPORTED: no-exceptions // (bug report: https://llvm.org/PR58392) // Check that vector constructors don't leak memory when an operation inside the constructor throws an exception #include #include #include "count_new.h" #include "test_iterators.h" template struct Allocator { using value_type = T; using is_always_equal = std::false_type; Allocator(bool should_throw = true) { if (should_throw) throw 0; } T* allocate(int n) { return std::allocator().allocate(n); } void deallocate(T* ptr, int n) { std::allocator().deallocate(ptr, n); } friend bool operator==(const Allocator&, const Allocator&) { return false; } }; struct ThrowingT { int* throw_after_n_ = nullptr; ThrowingT() { throw 0; } ThrowingT(int& throw_after_n) : throw_after_n_(&throw_after_n) { if (throw_after_n == 0) throw 0; --throw_after_n; } ThrowingT(const ThrowingT&) { if (throw_after_n_ == nullptr || *throw_after_n_ == 0) throw 1; --*throw_after_n_; } ThrowingT& operator=(const ThrowingT&) { if (throw_after_n_ == nullptr || *throw_after_n_ == 0) throw 1; --*throw_after_n_; return *this; } }; template struct Iterator { using iterator_category = IterCat; using difference_type = std::ptrdiff_t; using value_type = int; using reference = int&; using pointer = int*; int i_; Iterator(int i = 0) : i_(i) {} int& operator*() { if (i_ == 1) throw 1; return i_; } friend bool operator==(const Iterator& lhs, const Iterator& rhs) { return lhs.i_ == rhs.i_; } friend bool operator!=(const Iterator& lhs, const Iterator& rhs) { return lhs.i_ != rhs.i_; } Iterator& operator++() { ++i_; return *this; } Iterator operator++(int) { auto tmp = *this; ++i_; return tmp; } }; void check_new_delete_called() { assert(globalMemCounter.new_called == globalMemCounter.delete_called); assert(globalMemCounter.new_array_called == globalMemCounter.delete_array_called); assert(globalMemCounter.aligned_new_called == globalMemCounter.aligned_delete_called); assert(globalMemCounter.aligned_new_array_called == globalMemCounter.aligned_delete_array_called); } int main(int, char**) { using AllocVec = std::vector >; try { // vector() AllocVec vec; } catch (int) { } check_new_delete_called(); try { // Throw in vector(size_type) from type std::vector get_alloc(1); } catch (int) { } check_new_delete_called(); #if TEST_STD_VER >= 14 try { // Throw in vector(size_type, value_type) from type int throw_after = 1; ThrowingT v(throw_after); std::vector get_alloc(1, v); } catch (int) { } check_new_delete_called(); try { // Throw in vector(size_type, const allocator_type&) from allocator Allocator alloc(false); AllocVec get_alloc(0, alloc); } catch (int) { } check_new_delete_called(); try { // Throw in vector(size_type, const allocator_type&) from the type std::vector vec(1, std::allocator()); } catch (int) { } check_new_delete_called(); #endif // TEST_STD_VER >= 14 try { // Throw in vector(InputIterator, InputIterator) from input iterator std::vector vec((Iterator()), Iterator(2)); } catch (int) { } check_new_delete_called(); try { // Throw in vector(InputIterator, InputIterator) from forward iterator std::vector vec((Iterator()), Iterator(2)); } catch (int) { } check_new_delete_called(); try { // Throw in vector(InputIterator, InputIterator) from allocator int a[] = {1, 2}; AllocVec vec(cpp17_input_iterator(a), cpp17_input_iterator(a + 2)); } catch (int) { } check_new_delete_called(); try { // Throw in vector(InputIterator, InputIterator, const allocator_type&) from input iterator std::allocator alloc; std::vector vec(Iterator(), Iterator(2), alloc); } catch (int) { } check_new_delete_called(); try { // Throw in vector(InputIterator, InputIterator, const allocator_type&) from forward iterator std::allocator alloc; std::vector vec(Iterator(), Iterator(2), alloc); } catch (int) { } check_new_delete_called(); try { // Throw in vector(InputIterator, InputIterator, const allocator_type&) from allocator int a[] = {1, 2}; Allocator alloc(false); AllocVec vec(cpp17_input_iterator(a), cpp17_input_iterator(a + 2), alloc); } catch (int) { } check_new_delete_called(); try { // Throw in vector(InputIterator, InputIterator, const allocator_type&) from allocator int a[] = {1, 2}; Allocator alloc(false); AllocVec vec(forward_iterator(a), forward_iterator(a + 2), alloc); } catch (int) { } check_new_delete_called(); try { // Throw in vector(const vector&) from type std::vector vec; int throw_after = 0; vec.emplace_back(throw_after); auto vec2 = vec; } catch (int) { } check_new_delete_called(); try { // Throw in vector(const vector&, const allocator_type&) from type std::vector vec; int throw_after = 1; vec.emplace_back(throw_after); std::vector vec2(vec, std::allocator()); } catch (int) { } check_new_delete_called(); try { // Throw in vector(vector&&, const allocator_type&) from type std::vector > vec(Allocator(false)); int throw_after = 1; vec.emplace_back(throw_after); std::vector > vec2(std::move(vec), Allocator(false)); } catch (int) { } check_new_delete_called(); #if TEST_STD_VER >= 11 try { // Throw in vector(initializer_list) from type int throw_after = 1; std::vector vec({ThrowingT(throw_after)}); } catch (int) { } check_new_delete_called(); try { // Throw in vector(initializer_list, const allocator_type&) constructor from type int throw_after = 1; std::vector vec({ThrowingT(throw_after)}, std::allocator()); } catch (int) { } check_new_delete_called(); #endif // TEST_STD_VER >= 11 return 0; }