#include "asl/span.hpp" #include "asl/testing/testing.hpp" #include "asl/tests/test_types.hpp" static_assert(asl::trivially_destructible>); static_assert(asl::trivially_destructible>); static_assert(asl::trivially_copyable>); static_assert(asl::trivially_copyable>); static_assert(asl::size_of> == asl::size_of * 2); static_assert(asl::size_of> == asl::size_of); ASL_TEST(empty_dynamic) { asl::span s; ASL_TEST_EXPECT(s.size() == 0); ASL_TEST_EXPECT(s.size_bytes() == 0); ASL_TEST_EXPECT(s.is_empty()); } ASL_TEST(empty_static) { asl::span s; ASL_TEST_EXPECT(s.size() == 0); ASL_TEST_EXPECT(s.size_bytes() == 0); ASL_TEST_EXPECT(s.is_empty()); } ASL_TEST(from_array_dynamic) { int array[] = {1, 2, 3}; asl::span span = array; ASL_TEST_ASSERT(span.size() == 3); ASL_TEST_EXPECT(span[0] == 1); ASL_TEST_EXPECT(span[1] == 2); ASL_TEST_EXPECT(span[2] == 3); } static_assert(asl::default_constructible>); static_assert(asl::default_constructible>); static_assert(!asl::default_constructible>); static_assert(asl::constructible_from, int32_t(&)[8]>); static_assert(!asl::constructible_from, const int32_t(&)[8]>); static_assert(asl::constructible_from, int32_t(&)[8]>); static_assert(asl::constructible_from, const int32_t(&)[8]>); static_assert(asl::constructible_from, int32_t(&)[8]>); static_assert(!asl::constructible_from, int32_t(&)[10]>); ASL_TEST(from_array_static) { int array[] = {1, 2, 3}; asl::span span = array; ASL_TEST_ASSERT(span.size() == 3); ASL_TEST_EXPECT(span[0] == 1); ASL_TEST_EXPECT(span[1] == 2); ASL_TEST_EXPECT(span[2] == 3); } static_assert(asl::constructible_from, asl::span>); static_assert(!asl::constructible_from, asl::span>); static_assert(!asl::constructible_from, asl::span>); static_assert(!asl::constructible_from, asl::span>); static_assert(asl::constructible_from, asl::span>); static_assert(asl::constructible_from, asl::span>); static_assert(asl::constructible_from, asl::span>); static_assert(asl::constructible_from, asl::span>); static_assert(!asl::constructible_from, asl::span>); ASL_TEST(conversion) { int array[] = {1, 2, 3}; asl::span span1 = array; asl::span span2{span1}; ASL_TEST_ASSERT(span2.size() == 3); ASL_TEST_EXPECT(span2[0] == 1); ASL_TEST_EXPECT(span2[1] == 2); ASL_TEST_EXPECT(span2[2] == 3); asl::span span3 = span2; ASL_TEST_ASSERT(span3.size() == 3); ASL_TEST_EXPECT(span3[0] == 1); ASL_TEST_EXPECT(span3[1] == 2); ASL_TEST_EXPECT(span3[2] == 3); asl::span span4{span2}; ASL_TEST_ASSERT(span4.size() == 3); ASL_TEST_EXPECT(span4[0] == 1); ASL_TEST_EXPECT(span4[1] == 2); ASL_TEST_EXPECT(span4[2] == 3); } template concept IsValidSubspan = requires (Span s) { s.template subspan(); }; static_assert(asl::same_as, decltype(asl::declval>().subspan<0>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<1>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<2>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<3>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<4>())>); static_assert(!IsValidSubspan, 5>); static_assert(asl::same_as, decltype(asl::declval>().subspan<0>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<0, 4>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<0, 4>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<1, 2>())>); static_assert(asl::same_as, decltype(asl::declval>().subspan<2, 2>())>); static_assert(!IsValidSubspan, 2, 3>); ASL_TEST(subspan_static_from_static) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.subspan<0>(); ASL_TEST_ASSERT(s1.size() == 4); ASL_TEST_EXPECT(s1[0] == 1); ASL_TEST_EXPECT(s1[1] == 2); ASL_TEST_EXPECT(s1[2] == 3); ASL_TEST_EXPECT(s1[3] == 4); auto s2 = span.subspan<2>(); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 3); ASL_TEST_EXPECT(s2[1] == 4); auto s3 = span.subspan<4>(); ASL_TEST_ASSERT(s3.size() == 0); auto s4 = span.subspan<1, 2>(); ASL_TEST_ASSERT(s4.size() == 2); ASL_TEST_EXPECT(s4[0] == 2); ASL_TEST_EXPECT(s4[1] == 3); } ASL_TEST(subspan_static_from_dynamic) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.subspan<0>(); ASL_TEST_ASSERT(s1.size() == 4); ASL_TEST_EXPECT(s1[0] == 1); ASL_TEST_EXPECT(s1[1] == 2); ASL_TEST_EXPECT(s1[2] == 3); ASL_TEST_EXPECT(s1[3] == 4); auto s2 = span.subspan<2>(); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 3); ASL_TEST_EXPECT(s2[1] == 4); auto s3 = span.subspan<4>(); ASL_TEST_ASSERT(s3.size() == 0); auto s4 = span.subspan<1, 2>(); ASL_TEST_ASSERT(s4.size() == 2); ASL_TEST_EXPECT(s4[0] == 2); ASL_TEST_EXPECT(s4[1] == 3); } ASL_TEST(subspan_dynamic) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.subspan(0); ASL_TEST_ASSERT(s1.size() == 4); ASL_TEST_EXPECT(s1[0] == 1); ASL_TEST_EXPECT(s1[1] == 2); ASL_TEST_EXPECT(s1[2] == 3); ASL_TEST_EXPECT(s1[3] == 4); auto s2 = span.subspan(2); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 3); ASL_TEST_EXPECT(s2[1] == 4); auto s3 = span.subspan(4); ASL_TEST_ASSERT(s3.size() == 0); auto s4 = span.subspan(1, 2); ASL_TEST_ASSERT(s4.size() == 2); ASL_TEST_EXPECT(s4[0] == 2); ASL_TEST_EXPECT(s4[1] == 3); } template concept IsValidFirst = requires (Span s) { s.template first(); }; static_assert(asl::same_as, decltype(asl::declval>().first<0>())>); static_assert(asl::same_as, decltype(asl::declval>().first<1>())>); static_assert(asl::same_as, decltype(asl::declval>().first<2>())>); static_assert(asl::same_as, decltype(asl::declval>().first<3>())>); static_assert(asl::same_as, decltype(asl::declval>().first<0>())>); static_assert(asl::same_as, decltype(asl::declval>().first<1>())>); static_assert(asl::same_as, decltype(asl::declval>().first<2>())>); static_assert(asl::same_as, decltype(asl::declval>().first<3>())>); static_assert(asl::same_as, decltype(asl::declval>().first<4>())>); static_assert(!IsValidFirst, 5>); static_assert(!IsValidFirst, asl::dynamic_size>); ASL_TEST(first_static_from_static) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.first<0>(); ASL_TEST_ASSERT(s1.size() == 0); auto s2 = span.first<2>(); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 1); ASL_TEST_EXPECT(s2[1] == 2); auto s3 = span.first<4>(); ASL_TEST_ASSERT(s3.size() == 4); ASL_TEST_EXPECT(s3[0] == 1); ASL_TEST_EXPECT(s3[1] == 2); ASL_TEST_EXPECT(s3[2] == 3); ASL_TEST_EXPECT(s3[3] == 4); } ASL_TEST(first_static_from_dynamic) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.first<0>(); ASL_TEST_ASSERT(s1.size() == 0); auto s2 = span.first<2>(); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 1); ASL_TEST_EXPECT(s2[1] == 2); auto s3 = span.first<4>(); ASL_TEST_ASSERT(s3.size() == 4); ASL_TEST_EXPECT(s3[0] == 1); ASL_TEST_EXPECT(s3[1] == 2); ASL_TEST_EXPECT(s3[2] == 3); ASL_TEST_EXPECT(s3[3] == 4); } ASL_TEST(first_dynamic) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.first(0); ASL_TEST_ASSERT(s1.size() == 0); auto s2 = span.first(2); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 1); ASL_TEST_EXPECT(s2[1] == 2); auto s3 = span.first(4); ASL_TEST_ASSERT(s3.size() == 4); ASL_TEST_EXPECT(s3[0] == 1); ASL_TEST_EXPECT(s3[1] == 2); ASL_TEST_EXPECT(s3[2] == 3); ASL_TEST_EXPECT(s3[3] == 4); } template concept IsValidLast = requires (Span s) { s.template last(); }; static_assert(asl::same_as, decltype(asl::declval>().last<0>())>); static_assert(asl::same_as, decltype(asl::declval>().last<1>())>); static_assert(asl::same_as, decltype(asl::declval>().last<2>())>); static_assert(asl::same_as, decltype(asl::declval>().last<3>())>); static_assert(asl::same_as, decltype(asl::declval>().last<0>())>); static_assert(asl::same_as, decltype(asl::declval>().last<1>())>); static_assert(asl::same_as, decltype(asl::declval>().last<2>())>); static_assert(asl::same_as, decltype(asl::declval>().last<3>())>); static_assert(asl::same_as, decltype(asl::declval>().last<4>())>); static_assert(!IsValidLast, 5>); static_assert(!IsValidLast, asl::dynamic_size>); ASL_TEST(last_static_from_static) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.last<0>(); ASL_TEST_ASSERT(s1.size() == 0); auto s2 = span.last<2>(); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 3); ASL_TEST_EXPECT(s2[1] == 4); auto s3 = span.last<4>(); ASL_TEST_ASSERT(s3.size() == 4); ASL_TEST_EXPECT(s3[0] == 1); ASL_TEST_EXPECT(s3[1] == 2); ASL_TEST_EXPECT(s3[2] == 3); ASL_TEST_EXPECT(s3[3] == 4); } ASL_TEST(last_static_from_dynamic) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.last<0>(); ASL_TEST_ASSERT(s1.size() == 0); auto s2 = span.last<2>(); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 3); ASL_TEST_EXPECT(s2[1] == 4); auto s3 = span.last<4>(); ASL_TEST_ASSERT(s3.size() == 4); ASL_TEST_EXPECT(s3[0] == 1); ASL_TEST_EXPECT(s3[1] == 2); ASL_TEST_EXPECT(s3[2] == 3); ASL_TEST_EXPECT(s3[3] == 4); } ASL_TEST(last_dynamic) { int array[] = {1, 2, 3, 4}; asl::span span{array}; auto s1 = span.last(0); ASL_TEST_ASSERT(s1.size() == 0); auto s2 = span.last(2); ASL_TEST_ASSERT(s2.size() == 2); ASL_TEST_EXPECT(s2[0] == 3); ASL_TEST_EXPECT(s2[1] == 4); auto s3 = span.last(4); ASL_TEST_ASSERT(s3.size() == 4); ASL_TEST_EXPECT(s3[0] == 1); ASL_TEST_EXPECT(s3[1] == 2); ASL_TEST_EXPECT(s3[2] == 3); ASL_TEST_EXPECT(s3[3] == 4); } template concept HasAsMutableBytes = requires(asl::span s) { asl::as_mutable_bytes(s); }; static_assert(HasAsMutableBytes); static_assert(!HasAsMutableBytes); static_assert(!HasAsMutableBytes); static_assert(HasAsMutableBytes); ASL_TEST(as_bytes) { uint32_t data[] = {0x01020304, 0x05060708}; asl::span s1(data); asl::span s2 = asl::as_bytes(s1); ASL_TEST_ASSERT(s2.size() == 8); ASL_TEST_ASSERT(static_cast(s2[0]) == 0x04); ASL_TEST_ASSERT(static_cast(s2[1]) == 0x03); ASL_TEST_ASSERT(static_cast(s2[2]) == 0x02); ASL_TEST_ASSERT(static_cast(s2[3]) == 0x01); ASL_TEST_ASSERT(static_cast(s2[4]) == 0x08); ASL_TEST_ASSERT(static_cast(s2[5]) == 0x07); ASL_TEST_ASSERT(static_cast(s2[6]) == 0x06); ASL_TEST_ASSERT(static_cast(s2[7]) == 0x05); } ASL_TEST(as_mutable_bytes) { uint32_t data[] = {0x01020304, 0x05060708}; asl::span s1(data); asl::span s2 = asl::as_mutable_bytes(s1); ASL_TEST_ASSERT(s2.size() == 8); ASL_TEST_ASSERT(static_cast(s2[0]) == 0x04); ASL_TEST_ASSERT(static_cast(s2[1]) == 0x03); ASL_TEST_ASSERT(static_cast(s2[2]) == 0x02); ASL_TEST_ASSERT(static_cast(s2[3]) == 0x01); ASL_TEST_ASSERT(static_cast(s2[4]) == 0x08); ASL_TEST_ASSERT(static_cast(s2[5]) == 0x07); ASL_TEST_ASSERT(static_cast(s2[6]) == 0x06); ASL_TEST_ASSERT(static_cast(s2[7]) == 0x05); }