mirror of git://gcc.gnu.org/git/gcc.git
PR libstdc++/83982 fix exception-safety guarantee of std::vector::resize
Construct new elements before moving existing ones, so that if a default constructor throws, the existing elements are not left in a moved-from state. 2018-06-14 Daniel Trebbien <dtrebbien@gmail.com> Jonathan Wakely <jwakely@redhat.com> PR libstdc++/83982 * include/bits/vector.tcc (vector::_M_default_append(size_type)): Default-construct new elements before moving existing ones. * testsuite/23_containers/vector/capacity/resize/strong_guarantee.cc: New. Co-Authored-By: Jonathan Wakely <jwakely@redhat.com> From-SVN: r261585
This commit is contained in:
parent
c7a42ade9b
commit
4c1d999a7e
|
|
@ -1,3 +1,12 @@
|
||||||
|
2018-06-14 Daniel Trebbien <dtrebbien@gmail.com>
|
||||||
|
Jonathan Wakely <jwakely@redhat.com>
|
||||||
|
|
||||||
|
PR libstdc++/83982
|
||||||
|
* include/bits/vector.tcc (vector::_M_default_append(size_type)):
|
||||||
|
Default-construct new elements before moving existing ones.
|
||||||
|
* testsuite/23_containers/vector/capacity/resize/strong_guarantee.cc:
|
||||||
|
New.
|
||||||
|
|
||||||
2018-06-13 Jonathan Wakely <jwakely@redhat.com>
|
2018-06-13 Jonathan Wakely <jwakely@redhat.com>
|
||||||
|
|
||||||
PR libstdc++/86127
|
PR libstdc++/86127
|
||||||
|
|
|
||||||
|
|
@ -582,7 +582,7 @@ _GLIBCXX_BEGIN_NAMESPACE_CONTAINER
|
||||||
{
|
{
|
||||||
if (__n != 0)
|
if (__n != 0)
|
||||||
{
|
{
|
||||||
size_type __size = size();
|
const size_type __size = size();
|
||||||
size_type __navail = size_type(this->_M_impl._M_end_of_storage
|
size_type __navail = size_type(this->_M_impl._M_end_of_storage
|
||||||
- this->_M_impl._M_finish);
|
- this->_M_impl._M_finish);
|
||||||
|
|
||||||
|
|
@ -601,23 +601,22 @@ _GLIBCXX_BEGIN_NAMESPACE_CONTAINER
|
||||||
{
|
{
|
||||||
const size_type __len =
|
const size_type __len =
|
||||||
_M_check_len(__n, "vector::_M_default_append");
|
_M_check_len(__n, "vector::_M_default_append");
|
||||||
const size_type __old_size = __size;
|
|
||||||
pointer __new_start(this->_M_allocate(__len));
|
pointer __new_start(this->_M_allocate(__len));
|
||||||
pointer __new_finish(__new_start);
|
pointer __destroy_from = pointer();
|
||||||
__try
|
__try
|
||||||
{
|
{
|
||||||
__new_finish
|
std::__uninitialized_default_n_a(__new_start + __size,
|
||||||
= std::__uninitialized_move_if_noexcept_a
|
__n, _M_get_Tp_allocator());
|
||||||
(this->_M_impl._M_start, this->_M_impl._M_finish,
|
__destroy_from = __new_start + __size;
|
||||||
__new_start, _M_get_Tp_allocator());
|
std::__uninitialized_move_if_noexcept_a(
|
||||||
__new_finish =
|
this->_M_impl._M_start, this->_M_impl._M_finish,
|
||||||
std::__uninitialized_default_n_a(__new_finish, __n,
|
__new_start, _M_get_Tp_allocator());
|
||||||
_M_get_Tp_allocator());
|
|
||||||
}
|
}
|
||||||
__catch(...)
|
__catch(...)
|
||||||
{
|
{
|
||||||
std::_Destroy(__new_start, __new_finish,
|
if (__destroy_from)
|
||||||
_M_get_Tp_allocator());
|
std::_Destroy(__destroy_from, __destroy_from + __n,
|
||||||
|
_M_get_Tp_allocator());
|
||||||
_M_deallocate(__new_start, __len);
|
_M_deallocate(__new_start, __len);
|
||||||
__throw_exception_again;
|
__throw_exception_again;
|
||||||
}
|
}
|
||||||
|
|
@ -628,7 +627,7 @@ _GLIBCXX_BEGIN_NAMESPACE_CONTAINER
|
||||||
this->_M_impl._M_end_of_storage
|
this->_M_impl._M_end_of_storage
|
||||||
- this->_M_impl._M_start);
|
- this->_M_impl._M_start);
|
||||||
this->_M_impl._M_start = __new_start;
|
this->_M_impl._M_start = __new_start;
|
||||||
this->_M_impl._M_finish = __new_finish;
|
this->_M_impl._M_finish = __new_start + __size + __n;
|
||||||
this->_M_impl._M_end_of_storage = __new_start + __len;
|
this->_M_impl._M_end_of_storage = __new_start + __len;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,60 @@
|
||||||
|
// Copyright (C) 2018 Free Software Foundation, Inc.
|
||||||
|
//
|
||||||
|
// This file is part of the GNU ISO C++ Library. This library is free
|
||||||
|
// software; you can redistribute it and/or modify it under the
|
||||||
|
// terms of the GNU General Public License as published by the
|
||||||
|
// Free Software Foundation; either version 3, or (at your option)
|
||||||
|
// any later version.
|
||||||
|
|
||||||
|
// This library is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License along
|
||||||
|
// with this library; see the file COPYING3. If not see
|
||||||
|
// <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include <testsuite_hooks.h>
|
||||||
|
|
||||||
|
struct X
|
||||||
|
{
|
||||||
|
X() : data(1)
|
||||||
|
{
|
||||||
|
if (fail)
|
||||||
|
throw 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool fail;
|
||||||
|
|
||||||
|
std::vector<int> data;
|
||||||
|
};
|
||||||
|
|
||||||
|
bool X::fail = false;
|
||||||
|
|
||||||
|
void
|
||||||
|
test01()
|
||||||
|
{
|
||||||
|
std::vector<X> v(2);
|
||||||
|
X* const addr = &v[0];
|
||||||
|
bool caught = false;
|
||||||
|
try {
|
||||||
|
X::fail = true;
|
||||||
|
v.resize(v.capacity() + 1); // force reallocation
|
||||||
|
} catch (int) {
|
||||||
|
caught = true;
|
||||||
|
}
|
||||||
|
VERIFY( caught );
|
||||||
|
VERIFY( v.size() == 2 );
|
||||||
|
VERIFY( &v[0] == addr );
|
||||||
|
// PR libstdc++/83982
|
||||||
|
VERIFY( ! v[0].data.empty() );
|
||||||
|
VERIFY( ! v[1].data.empty() );
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
main()
|
||||||
|
{
|
||||||
|
test01();
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue