// Copyright 2021 The PDFium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "core/fxcrt/span_util.h" #include #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" TEST(Spancpy, FitsEntirely) { std::vector src(4, 'A'); std::vector dst(4, 'B'); auto remain = fxcrt::spancpy(pdfium::make_span(dst), pdfium::make_span(src)); EXPECT_EQ(dst[0], 'A'); EXPECT_EQ(dst[1], 'A'); EXPECT_EQ(dst[2], 'A'); EXPECT_EQ(dst[3], 'A'); EXPECT_TRUE(remain.empty()); } TEST(Spancpy, FitsWithin) { std::vector src(2, 'A'); std::vector dst(4, 'B'); // Also show that a const src argument is acceptable. auto remain = fxcrt::spancpy(pdfium::make_span(dst).subspan(1), pdfium::span(src)); EXPECT_EQ(dst[0], 'B'); EXPECT_EQ(dst[1], 'A'); EXPECT_EQ(dst[2], 'A'); EXPECT_EQ(dst[3], 'B'); EXPECT_EQ(remain.size(), 1u); EXPECT_EQ(remain.data(), &dst[3]); } TEST(Spancpy, EmptyCopyWithin) { std::vector src(2, 'A'); std::vector dst(4, 'B'); auto remain = fxcrt::spancpy(pdfium::make_span(dst).subspan(1), pdfium::make_span(src).subspan(2)); EXPECT_EQ(dst[0], 'B'); EXPECT_EQ(dst[1], 'B'); EXPECT_EQ(dst[2], 'B'); EXPECT_EQ(dst[3], 'B'); EXPECT_EQ(remain.size(), 3u); EXPECT_EQ(remain.data(), &dst[1]); } TEST(Spancpy, EmptyCopyToEmpty) { std::vector src(2, 'A'); std::vector dst(4, 'B'); auto remain = fxcrt::spancpy(pdfium::make_span(dst).subspan(4), pdfium::make_span(src).subspan(2)); EXPECT_EQ(dst[0], 'B'); EXPECT_EQ(dst[1], 'B'); EXPECT_EQ(dst[2], 'B'); EXPECT_EQ(dst[3], 'B'); EXPECT_TRUE(remain.empty()); } TEST(Spancpy, TryFitsEntirely) { std::vector src(4, 'A'); std::vector dst(4, 'B'); EXPECT_TRUE( fxcrt::try_spancpy(pdfium::make_span(dst), pdfium::make_span(src))); EXPECT_EQ(dst[0], 'A'); EXPECT_EQ(dst[1], 'A'); EXPECT_EQ(dst[2], 'A'); EXPECT_EQ(dst[3], 'A'); } TEST(Spancpy, TryDoesNotFit) { std::vector src(4, 'A'); std::vector dst(3, 'B'); EXPECT_FALSE( fxcrt::try_spancpy(pdfium::make_span(dst), pdfium::make_span(src))); EXPECT_EQ(dst[0], 'B'); EXPECT_EQ(dst[1], 'B'); EXPECT_EQ(dst[2], 'B'); } TEST(Spanmove, FitsWithin) { std::vector src(2, 'A'); std::vector dst(4, 'B'); // Also show that a const src argument is acceptable. auto remain = fxcrt::spanmove(pdfium::make_span(dst).subspan(1), pdfium::span(src)); EXPECT_EQ(dst[0], 'B'); EXPECT_EQ(dst[1], 'A'); EXPECT_EQ(dst[2], 'A'); EXPECT_EQ(dst[3], 'B'); EXPECT_EQ(remain.size(), 1u); EXPECT_EQ(remain.data(), &dst[3]); } TEST(Spanmove, TryFitsEntirely) { std::vector src(4, 'A'); std::vector dst(4, 'B'); EXPECT_TRUE( fxcrt::try_spanmove(pdfium::make_span(dst), pdfium::make_span(src))); EXPECT_EQ(dst[0], 'A'); EXPECT_EQ(dst[1], 'A'); EXPECT_EQ(dst[2], 'A'); EXPECT_EQ(dst[3], 'A'); } TEST(Spanmove, TrySelfIntersect) { { std::vector vec = {'A', 'B', 'C', 'D'}; EXPECT_TRUE(fxcrt::try_spanmove(pdfium::make_span(vec).first(3), pdfium::make_span(vec).last(3))); EXPECT_EQ(vec[0], 'B'); EXPECT_EQ(vec[1], 'C'); EXPECT_EQ(vec[2], 'D'); EXPECT_EQ(vec[3], 'D'); } { std::vector vec = {'A', 'B', 'C', 'D'}; EXPECT_TRUE(fxcrt::try_spanmove(pdfium::make_span(vec).last(3), pdfium::make_span(vec).first(3))); EXPECT_EQ(vec[0], 'A'); EXPECT_EQ(vec[1], 'A'); EXPECT_EQ(vec[2], 'B'); EXPECT_EQ(vec[3], 'C'); } } TEST(Spanmove, TryDoesNotFit) { std::vector src(4, 'A'); std::vector dst(3, 'B'); EXPECT_FALSE( fxcrt::try_spanmove(pdfium::make_span(dst), pdfium::make_span(src))); EXPECT_EQ(dst[0], 'B'); EXPECT_EQ(dst[1], 'B'); EXPECT_EQ(dst[2], 'B'); } TEST(SpanEquals, Empty) { std::vector vec = {1, 2}; std::vector vec2 = {3, 4}; pdfium::span empty; pdfium::span some = pdfium::make_span(vec); pdfium::span some2 = pdfium::make_span(vec2); EXPECT_FALSE(fxcrt::span_equals(empty, some)); EXPECT_FALSE(fxcrt::span_equals(some, empty)); EXPECT_TRUE(fxcrt::span_equals(empty, empty)); EXPECT_TRUE(fxcrt::span_equals(empty, some.first(0))); EXPECT_TRUE(fxcrt::span_equals(some.first(0), empty)); EXPECT_TRUE(fxcrt::span_equals(some2.first(0), some.first(0))); EXPECT_TRUE(fxcrt::span_equals(some.first(0), some2.first(0))); } TEST(SpanEquals, NonEmpty) { std::vector vec = {1, 2, 3}; std::vector vec2 = {1, 2, 4}; pdfium::span some = pdfium::make_span(vec); pdfium::span some2 = pdfium::make_span(vec2); EXPECT_FALSE(fxcrt::span_equals(some, some2)); EXPECT_FALSE(fxcrt::span_equals(some.first(2), some2)); EXPECT_FALSE(fxcrt::span_equals(some, some2.first(2))); EXPECT_TRUE(fxcrt::span_equals(some.first(2), some2.first(2))); } TEST(Span, AssignOverOnePastEnd) { std::vector src(2, 'A'); pdfium::span span = pdfium::make_span(src); span = span.subspan(2); span = pdfium::make_span(src); EXPECT_EQ(span.size(), 2u); } TEST(ReinterpretSpan, Empty) { pdfium::span empty; pdfium::span converted = fxcrt::reinterpret_span(empty); EXPECT_EQ(converted.data(), nullptr); EXPECT_EQ(converted.size(), 0u); } TEST(ReinterpretSpan, LegalConversions) { uint8_t aaaabbbb[8] = {0x61, 0x61, 0x61, 0x61, 0x62, 0x62, 0x62, 0x62}; pdfium::span original = pdfium::make_span(aaaabbbb); pdfium::span converted = fxcrt::reinterpret_span(original); ASSERT_NE(converted.data(), nullptr); ASSERT_EQ(converted.size(), 2u); EXPECT_EQ(converted[0], 0x61616161u); EXPECT_EQ(converted[1], 0x62626262u); } TEST(ReinterpretSpan, BadAlignment) { uint8_t abcabc[6] = {0x61, 0x62, 0x63, 0x61, 0x62, 0x63}; EXPECT_DEATH(fxcrt::reinterpret_span( pdfium::make_span(abcabc).subspan(1, 4)), ""); } TEST(Spanpos, Empty) { pdfium::span kEmpty; const uint32_t kHaystack[] = {0, 1, 2, 3, 4, 5}; const uint32_t kNeedle[] = {1, 2}; EXPECT_FALSE(fxcrt::spanpos(kEmpty, kEmpty)); EXPECT_FALSE(fxcrt::spanpos(pdfium::make_span(kHaystack), kEmpty)); EXPECT_FALSE(fxcrt::spanpos(kEmpty, pdfium::make_span(kNeedle))); } TEST(Spanpos, NotEmpty) { const uint32_t kHaystack[] = {0, 1, 2, 3, 4, 5}; const uint32_t kStartMatch[] = {0, 1}; const uint32_t kEndMatch[] = {4, 5}; const uint32_t kNotFound[] = {256, 512}; // test byte-shifted {1,2}. const uint32_t kTooLong[] = {0, 1, 2, 3, 4, 5, 6}; EXPECT_THAT(fxcrt::spanpos(pdfium::make_span(kHaystack), pdfium::make_span(kStartMatch)), testing::Optional(0u)); EXPECT_THAT(fxcrt::spanpos(pdfium::make_span(kHaystack), pdfium::make_span(kEndMatch)), testing::Optional(4u)); EXPECT_FALSE(fxcrt::spanpos(pdfium::make_span(kHaystack), pdfium::make_span(kNotFound))); EXPECT_FALSE(fxcrt::spanpos(pdfium::make_span(kHaystack), pdfium::make_span(kTooLong))); }