/* Self tests for array_view for GDB, the GNU debugger.
Copyright (C) 2017-2023 Free Software Foundation, Inc.
This file is part of GDB.
This program 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 of the License, or
(at your option) any later version.
This program 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 program. If not, see . */
#include "defs.h"
#include "gdbsupport/selftest.h"
#include "gdbsupport/array-view.h"
#include
#include
namespace selftests {
namespace array_view_tests {
/* Triviality checks. */
#define CHECK_TRAIT(TRAIT) \
static_assert (std::TRAIT>::value, "")
#if HAVE_IS_TRIVIALLY_COPYABLE
CHECK_TRAIT (is_trivially_copyable);
CHECK_TRAIT (is_trivially_move_assignable);
CHECK_TRAIT (is_trivially_move_constructible);
CHECK_TRAIT (is_trivially_destructible);
#endif
#undef CHECK_TRAIT
/* Wrapper around std::is_convertible to make the code using it a bit
shorter. (With C++14 we'd use a variable template instead.) */
template
static constexpr bool
is_convertible ()
{
return std::is_convertible::value;
}
/* Check for implicit conversion to immutable and mutable views. */
static constexpr bool
check_convertible ()
{
using T = gdb_byte;
using gdb::array_view;
return (true
/* immutable array_view */
&& is_convertible> ()
&& is_convertible> ()
&& is_convertible> ()
&& is_convertible> ()
/* mutable array_view */
&& is_convertible> ()
&& !is_convertible> ()
&& is_convertible> ()
&& !is_convertible> ()
/* While float is implicitly convertible to gdb_byte, we
don't want implicit float->array_view
conversion. */
&& !is_convertible> ()
&& !is_convertible> ());
}
static_assert (check_convertible (), "");
namespace no_slicing
{
struct A { int i; };
struct B : A { int j; };
struct C : A { int l; };
/* Check that there's no array->view conversion for arrays of derived types or
subclasses. */
static constexpr bool
check ()
{
using gdb::array_view;
return (true
/* array->view */
&& is_convertible > ()
&& !is_convertible > ()
&& !is_convertible > ()
&& !is_convertible > ()
&& is_convertible > ()
&& !is_convertible > ()
/* elem->view */
&& is_convertible > ()
&& !is_convertible > ()
&& !is_convertible > ()
&& !is_convertible > ()
&& is_convertible > ()
&& !is_convertible > ());
}
/* Check that there's no container->view conversion for containers of derived
types or subclasses. */
template class Container>
static constexpr bool
check_ctor_from_container ()
{
using gdb::array_view;
return ( is_convertible , array_view> ()
&& !is_convertible , array_view> ()
&& !is_convertible , array_view> ()
&& !is_convertible , array_view> ()
&& is_convertible , array_view> ()
&& !is_convertible , array_view> ());
}
} /* namespace no_slicing */
/* std::array with only one template argument, so we can pass it to
check_ctor_from_container. */
template using StdArray1 = std::array;
static_assert (no_slicing::check (), "");
static_assert (no_slicing::check_ctor_from_container (), "");
static_assert (no_slicing::check_ctor_from_container (), "");
static_assert (no_slicing::check_ctor_from_container (), "");
/* Check that array_view implicitly converts from std::vector. */
static constexpr bool
check_convertible_from_std_vector ()
{
using gdb::array_view;
using T = gdb_byte;
/* Note there's no such thing as std::vector. */
return (true
&& is_convertible , array_view> ()
&& is_convertible , array_view> ());
}
static_assert (check_convertible_from_std_vector (), "");
/* Check that array_view implicitly converts from std::array. */
static constexpr bool
check_convertible_from_std_array ()
{
using gdb::array_view;
using T = gdb_byte;
/* Note: a non-const T view can't refer to a const T array. */
return (true
&& is_convertible , array_view> ()
&& is_convertible , array_view> ()
&& !is_convertible , array_view> ()
&& is_convertible , array_view> ());
}
static_assert (check_convertible_from_std_array (), "");
/* Check that VIEW views C (a container like std::vector/std::array)
correctly. */
template
static bool
check_container_view (const View &view, const Container &c)
{
if (view.empty ())
return false;
if (view.size () != c.size ())
return false;
if (view.data () != c.data ())
return false;
for (size_t i = 0; i < c.size (); i++)
{
if (&view[i] != &c[i])
return false;
if (view[i] != c[i])
return false;
}
return true;
}
/* Check that VIEW views E (an object of the type of a view element)
correctly. */
template
static bool
check_elem_view (const View &view, const Elem &e)
{
if (view.empty ())
return false;
if (view.size () != 1)
return false;
if (view.data () != &e)
return false;
if (&view[0] != &e)
return false;
if (view[0] != e)
return false;
return true;
}
/* Check for operator[]. The first overload is taken iff
'view()[0] = T()' is a valid expression. */
template ()[0]
= std::declval ())>
static bool
check_op_subscript (const View &view)
{
return true;
}
/* This overload is taken iff 'view()[0] = T()' is not a valid
expression. */
static bool
check_op_subscript (...)
{
return false;
}
/* Check construction with pointer + size. This is a template in
order to test both gdb_byte and const gdb_byte. */
template
static void
check_ptr_size_ctor ()
{
T data[] = {0x11, 0x22, 0x33, 0x44};
gdb::array_view view (data + 1, 2);
SELF_CHECK (!view.empty ());
SELF_CHECK (view.size () == 2);
SELF_CHECK (view.data () == &data[1]);
SELF_CHECK (view[0] == data[1]);
SELF_CHECK (view[1] == data[2]);
gdb::array_view cview (data + 1, 2);
SELF_CHECK (!cview.empty ());
SELF_CHECK (cview.size () == 2);
SELF_CHECK (cview.data () == &data[1]);
SELF_CHECK (cview[0] == data[1]);
SELF_CHECK (cview[1] == data[2]);
}
/* Asserts std::is_constructible. */
template
static constexpr bool
require_not_constructible ()
{
static_assert (!std::is_constructible::value, "");
/* constexpr functions can't return void in C++11 (N3444). */
return true;
};
/* Check the array_view(PTR, SIZE) ctor, when T is a pointer. */
static void
check_ptr_size_ctor2 ()
{
struct A {};
A an_a;
A *array[] = { &an_a };
const A * const carray[] = { &an_a };
gdb::array_view v1 = {array, ARRAY_SIZE (array)};
gdb::array_view v2 = {array, (char) ARRAY_SIZE (array)};
gdb::array_view v3 = {array, ARRAY_SIZE (array)};
gdb::array_view cv1 = {carray, ARRAY_SIZE (carray)};
require_not_constructible, decltype (carray), size_t> ();
SELF_CHECK (v1[0] == array[0]);
SELF_CHECK (v2[0] == array[0]);
SELF_CHECK (v3[0] == array[0]);
SELF_CHECK (!v1.empty ());
SELF_CHECK (v1.size () == 1);
SELF_CHECK (v1.data () == &array[0]);
SELF_CHECK (cv1[0] == carray[0]);
SELF_CHECK (!cv1.empty ());
SELF_CHECK (cv1.size () == 1);
SELF_CHECK (cv1.data () == &carray[0]);
}
/* Check construction with a pair of pointers. This is a template in
order to test both gdb_byte and const gdb_byte. */
template
static void
check_ptr_ptr_ctor ()
{
T data[] = {0x11, 0x22, 0x33, 0x44};
gdb::array_view view (data + 1, data + 3);
SELF_CHECK (!view.empty ());
SELF_CHECK (view.size () == 2);
SELF_CHECK (view.data () == &data[1]);
SELF_CHECK (view[0] == data[1]);
SELF_CHECK (view[1] == data[2]);
gdb_byte array[] = {0x11, 0x22, 0x33, 0x44};
const gdb_byte *p1 = array;
gdb_byte *p2 = array + ARRAY_SIZE (array);
gdb::array_view view2 (p1, p2);
}
/* Check construction with a pair of pointers of mixed constness. */
static void
check_ptr_ptr_mixed_cv ()
{
gdb_byte array[] = {0x11, 0x22, 0x33, 0x44};
const gdb_byte *cp = array;
gdb_byte *p = array;
gdb::array_view view1 (cp, p);
gdb::array_view view2 (p, cp);
SELF_CHECK (view1.empty ());
SELF_CHECK (view2.empty ());
}
/* Check range-for support (i.e., begin()/end()). This is a template
in order to test both gdb_byte and const gdb_byte. */
template
static void
check_range_for ()
{
T data[] = {1, 2, 3, 4};
gdb::array_view view (data);
typename std::decay::type sum = 0;
for (auto &elem : view)
sum += elem;
SELF_CHECK (sum == 1 + 2 + 3 + 4);
}
/* Entry point. */
static void
run_tests ()
{
/* Empty views. */
{
constexpr gdb::array_view view1;
constexpr gdb::array_view view2;
static_assert (view1.empty (), "");
static_assert (view1.data () == nullptr, "");
static_assert (view1.size () == 0, "");
static_assert (view2.empty (), "");
static_assert (view2.size () == 0, "");
static_assert (view2.data () == nullptr, "");
}
std::vector vec = {0x11, 0x22, 0x33, 0x44 };
std::array array = {{0x11, 0x22, 0x33, 0x44}};
/* Various tests of views over std::vector. */
{
gdb::array_view view = vec;
SELF_CHECK (check_container_view (view, vec));
gdb::array_view cview = vec;
SELF_CHECK (check_container_view (cview, vec));
}
/* Likewise, over std::array. */
{
gdb::array_view view = array;
SELF_CHECK (check_container_view (view, array));
gdb::array_view cview = array;
SELF_CHECK (check_container_view (cview, array));
}
/* op=(std::vector/std::array/elem) */
{
gdb::array_view view;
view = vec;
SELF_CHECK (check_container_view (view, vec));
view = std::move (vec);
SELF_CHECK (check_container_view (view, vec));
view = array;
SELF_CHECK (check_container_view (view, array));
view = std::move (array);
SELF_CHECK (check_container_view (view, array));
gdb_byte elem = 0;
view = elem;
SELF_CHECK (check_elem_view (view, elem));
view = std::move (elem);
SELF_CHECK (check_elem_view (view, elem));
}
/* Test copy/move ctor and mutable->immutable conversion. */
{
gdb_byte data[] = {0x11, 0x22, 0x33, 0x44};
gdb::array_view view1 = data;
gdb::array_view view2 = view1;
gdb::array_view view3 = std::move (view1);
gdb::array_view cview1 = data;
gdb::array_view cview2 = cview1;
gdb::array_view cview3 = std::move (cview1);
SELF_CHECK (view1[0] == data[0]);
SELF_CHECK (view2[0] == data[0]);
SELF_CHECK (view3[0] == data[0]);
SELF_CHECK (cview1[0] == data[0]);
SELF_CHECK (cview2[0] == data[0]);
SELF_CHECK (cview3[0] == data[0]);
}
/* Same, but op=(view). */
{
gdb_byte data[] = {0x55, 0x66, 0x77, 0x88};
gdb::array_view view1;
gdb::array_view view2;
gdb::array_view view3;
gdb::array_view cview1;
gdb::array_view cview2;
gdb::array_view cview3;
view1 = data;
view2 = view1;
view3 = std::move (view1);
cview1 = data;
cview2 = cview1;
cview3 = std::move (cview1);
SELF_CHECK (view1[0] == data[0]);
SELF_CHECK (view2[0] == data[0]);
SELF_CHECK (view3[0] == data[0]);
SELF_CHECK (cview1[0] == data[0]);
SELF_CHECK (cview2[0] == data[0]);
SELF_CHECK (cview3[0] == data[0]);
}
/* op[] */
{
std::vector vec2 = {0x11, 0x22};
gdb::array_view view = vec2;
gdb::array_view cview = vec2;
/* Check that op[] on a non-const view of non-const T returns a
mutable reference. */
view[0] = 0x33;
SELF_CHECK (vec2[0] == 0x33);
/* OTOH, check that assigning through op[] on a view of const T
wouldn't compile. */
SELF_CHECK (!check_op_subscript (cview));
/* For completeness. */
SELF_CHECK (check_op_subscript (view));
}
check_ptr_size_ctor ();
check_ptr_size_ctor ();
check_ptr_size_ctor2 ();
check_ptr_ptr_ctor ();
check_ptr_ptr_ctor ();
check_ptr_ptr_mixed_cv ();
check_range_for ();
check_range_for ();
/* Check that the right ctor overloads are taken when the element is
a container. */
{
using Vec = std::vector;
Vec vecs[3];
gdb::array_view view_array = vecs;
SELF_CHECK (view_array.size () == 3);
Vec elem;
gdb::array_view view_elem = elem;
SELF_CHECK (view_elem.size () == 1);
}
/* gdb::make_array_view, int length. */
{
gdb_byte data[] = {0x55, 0x66, 0x77, 0x88};
int len = sizeof (data) / sizeof (data[0]);
auto view = gdb::make_array_view (data, len);
SELF_CHECK (view.data () == data);
SELF_CHECK (view.size () == len);
for (size_t i = 0; i < len; i++)
SELF_CHECK (view[i] == data[i]);
}
/* Test slicing. */
{
gdb_byte data[] = {0x55, 0x66, 0x77, 0x88, 0x99};
gdb::array_view view = data;
{
auto slc = view.slice (1, 3);
SELF_CHECK (slc.data () == data + 1);
SELF_CHECK (slc.size () == 3);
SELF_CHECK (slc[0] == data[1]);
SELF_CHECK (slc[0] == view[1]);
}
{
auto slc = view.slice (2);
SELF_CHECK (slc.data () == data + 2);
SELF_CHECK (slc.size () == 3);
SELF_CHECK (slc[0] == view[2]);
SELF_CHECK (slc[0] == data[2]);
}
}
}
template
void
run_copy_test ()
{
/* Test non-overlapping copy. */
{
const std::vector src_v = {1, 2, 3, 4};
std::vector dest_v (4, -1);
SELF_CHECK (dest_v != src_v);
copy (gdb::array_view (src_v), gdb::array_view (dest_v));
SELF_CHECK (dest_v == src_v);
}
/* Test overlapping copy, where the source is before the destination. */
{
std::vector vec = {1, 2, 3, 4, 5, 6, 7, 8};
gdb::array_view v = vec;
copy (v.slice (1, 4),
v.slice (2, 4));
std::vector expected = {1, 2, 2, 3, 4, 5, 7, 8};
SELF_CHECK (vec == expected);
}
/* Test overlapping copy, where the source is after the destination. */
{
std::vector vec = {1, 2, 3, 4, 5, 6, 7, 8};
gdb::array_view v = vec;
copy (v.slice (2, 4),
v.slice (1, 4));
std::vector expected = {1, 3, 4, 5, 6, 6, 7, 8};
SELF_CHECK (vec == expected);
}
/* Test overlapping copy, where the source is the same as the destination. */
{
std::vector vec = {1, 2, 3, 4, 5, 6, 7, 8};
gdb::array_view v = vec;
copy (v.slice (2, 4),
v.slice (2, 4));
std::vector expected = {1, 2, 3, 4, 5, 6, 7, 8};
SELF_CHECK (vec == expected);
}
}
/* Class with a non-trivial copy assignment operator, used to test the
array_view copy function. */
struct foo
{
/* Can be implicitly constructed from an int, such that we can use the same
templated test function to test against array_view and
array_view. */
foo (int n)
: n (n)
{}
/* Needed to avoid -Wdeprecated-copy-with-user-provided-copy error with
Clang. */
foo (const foo &other) = default;
void operator= (const foo &other)
{
this->n = other.n;
this->n_assign_op_called++;
}
bool operator==(const foo &other) const
{
return this->n == other.n;
}
int n;
/* Number of times the assignment operator has been called. */
static int n_assign_op_called;
};
int foo::n_assign_op_called = 0;
/* Test the array_view copy free function. */
static void
run_copy_tests ()
{
/* Test with a trivial type. */
run_copy_test ();
/* Test with a non-trivial type. */
foo::n_assign_op_called = 0;
run_copy_test ();
/* Make sure that for the non-trivial type foo, the assignment operator was
called an amount of times that makes sense. */
SELF_CHECK (foo::n_assign_op_called == 12);
}
} /* namespace array_view_tests */
} /* namespace selftests */
void _initialize_array_view_selftests ();
void
_initialize_array_view_selftests ()
{
selftests::register_test ("array_view",
selftests::array_view_tests::run_tests);
selftests::register_test ("array_view-copy",
selftests::array_view_tests::run_copy_tests);
}