// Copyright (C) 2010-2020 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
// .
//
#include
#include
#include
#ifndef _GLIBCXX_DEBUG
# include
# include
# include
#endif
#include
namespace __gnu_test
{
template
struct CopyableValueType
{
typedef _Tp value_type;
};
template
struct CopyableValueType >
{
typedef std::pair<_Tp1, _Tp2> value_type;
};
template
struct generate_unique
{
typedef _Tp value_type;
value_type build()
{
static value_type _S_;
++_S_;
return _S_;
}
};
template<>
struct generate_unique
{
typedef bool value_type;
value_type build()
{
static value_type _S_;
_S_ = !_S_;
return _S_;
}
};
template
struct generate_unique >
{
typedef _Tp1 first_type;
typedef _Tp2 second_type;
typedef std::pair<_Tp1, _Tp2> pair_type;
pair_type build()
{
static first_type _S_1;
static second_type _S_2;
++_S_1;
++_S_2;
return pair_type(_S_1, _S_2);
}
};
// Check that invalid range of pointers is detected
template
void
check_assign1()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::vector vector_type;
generate_unique gu;
vector_type v;
for (int i = 0; i != 5; ++i)
v.push_back(gu.build());
VERIFY(v.size() == 5);
const val_type* first = &v.front() + 1;
const val_type* last = first + 2;
cont_type c1;
c1.assign(first, last);
VERIFY(c1.size() == 2);
cont_type c2;
c2.assign(last, first); // Expected failure
}
// Check that invalid range of debug random iterators is detected
template
void
check_assign2()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::vector vector_type;
generate_unique gu;
vector_type v;
for (int i = 0; i != 5; ++i)
v.push_back(gu.build());
VERIFY(v.size() == 5);
typename vector_type::iterator first = v.begin() + 1;
typename vector_type::iterator last = first + 2;
cont_type c1;
c1.assign(first, last);
VERIFY(c1.size() == 2);
cont_type c2;
c2.assign(last, first); // Expected failure
}
// Check that invalid range of debug not random iterators is detected
template
void
check_assign3()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::list list_type;
generate_unique gu;
list_type l;
for (int i = 0; i != 5; ++i)
l.push_back(gu.build());
VERIFY(l.size() == 5);
typename list_type::iterator first = l.begin(); ++first;
typename list_type::iterator last = first; ++last; ++last;
cont_type c1;
c1.assign(first, last);
VERIFY(c1.size() == 2);
cont_type c2;
c2.assign(last, first); // Expected failure
}
// Check that invalid range of pointers is detected
template
void
check_construct1()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::vector vector_type;
generate_unique gu;
vector_type v;
for (int i = 0; i != 5; ++i)
v.push_back(gu.build());
VERIFY(v.size() == 5);
val_type *first = &v.front() + 1;
val_type *last = first + 2;
cont_type c(last, first); // Expected failure
}
// Check that invalid range of debug random iterators is detected
template
void
check_construct2()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::vector vector_type;
generate_unique gu;
vector_type v;
for (int i = 0; i != 5; ++i)
v.push_back(gu.build());
VERIFY(v.size() == 5);
typename vector_type::iterator first = v.begin() + 1;
typename vector_type::iterator last = first + 2;
cont_type c(last, first); // Expected failure
}
// Check that invalid range of debug not random iterators is detected
template
void
check_construct3()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::list list_type;
generate_unique gu;
list_type l;
for (int i = 0; i != 5; ++i)
l.push_back(gu.build());
VERIFY(l.size() == 5);
typename list_type::iterator first = l.begin(); ++first;
typename list_type::iterator last = first; ++last; ++last;
cont_type c(last, first); // Expected failure
}
template
struct InsertRangeHelper
{
template
static void
Insert(_Cont& cont, _It first, _It last)
{ cont.insert(first, last); }
};
template
struct InsertRangeHelperAux
{
template
static void
Insert(_Cont& cont, _It first, _It last)
{ cont.insert(cont.begin(), first, last); }
};
template
struct InsertRangeHelper >
: InsertRangeHelperAux >
{ };
template
struct InsertRangeHelper >
: InsertRangeHelperAux >
{ };
template
struct InsertRangeHelper >
: InsertRangeHelperAux >
{ };
#ifndef _GLIBCXX_DEBUG
template
struct InsertRangeHelper<__gnu_debug::vector<_Tp1, _Tp2> >
: InsertRangeHelperAux<__gnu_debug::vector<_Tp1, _Tp2> >
{ };
template
struct InsertRangeHelper<__gnu_debug::deque<_Tp1, _Tp2> >
: InsertRangeHelperAux<__gnu_debug::deque<_Tp1, _Tp2> >
{ };
template
struct InsertRangeHelper<__gnu_debug::list<_Tp1, _Tp2> >
: InsertRangeHelperAux<__gnu_debug::list<_Tp1, _Tp2> >
{ };
#endif
template
void
check_insert1()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::vector vector_type;
generate_unique gu;
vector_type v;
for (int i = 0; i != 5; ++i)
v.push_back(gu.build());
VERIFY(v.size() == 5);
const val_type* first = &v.front() + 1;
const val_type* last = first + 2;
cont_type c1;
InsertRangeHelper::Insert(c1, first, last);
VERIFY(c1.size() == 2);
cont_type c2;
InsertRangeHelper::Insert(c2, last, first); // Expected failure
}
template
void
check_insert2()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::vector vector_type;
generate_unique gu;
vector_type v;
for (int i = 0; i != 5; ++i)
v.push_back(gu.build());
VERIFY(v.size() == 5);
typename vector_type::iterator first = v.begin() + 1;
typename vector_type::iterator last = first + 2;
cont_type c1;
InsertRangeHelper::Insert(c1, first, last);
VERIFY(c1.size() == 2);
cont_type c2;
InsertRangeHelper::Insert(c2, last, first); // Expected failure
}
template
void
check_insert3()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::list list_type;
generate_unique gu;
list_type l;
for (int i = 0; i != 5; ++i)
l.push_back(gu.build());
VERIFY(l.size() == 5);
typename list_type::iterator first = l.begin(); ++first;
typename list_type::iterator last = first; ++last; ++last;
cont_type c1;
InsertRangeHelper::Insert(c1, first, last);
VERIFY(c1.size() == 2);
cont_type c2;
InsertRangeHelper::Insert(c2, last, first); // Expected failure
}
template
void
check_insert4()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
typedef std::list list_type;
generate_unique gu;
list_type l;
for (int i = 0; i != 5; ++i)
l.push_back(gu.build());
VERIFY(l.size() == 5);
typename list_type::iterator first = l.begin(); ++first;
typename list_type::iterator last = first; ++last; ++last;
cont_type c1;
InsertRangeHelper::Insert(c1, l.begin(), l.end());
VERIFY(c1.size() == 5);
c1.insert(c1.begin(), c1.begin(), c1.end()); // Expected failure.
}
template
void use_invalid_iterator()
{
typedef _Tp cont_type;
typedef typename cont_type::value_type cont_val_type;
typedef typename CopyableValueType::value_type val_type;
generate_unique gu;
cont_type c;
for (size_t i = 0; i != 5; ++i)
c.insert(gu.build());
typename cont_type::iterator it = c.begin();
cont_val_type val = *it;
c.clear();
VERIFY( *it == val );
}
}