blob: 5926ef06214e5d97eed04893b8658ebcce77450f [file] [log] [blame]
// 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 <array>
#include <vector>
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
TEST(Spancpy, FitsEntirely) {
std::vector<char> src(4, 'A');
std::vector<char> dst(4, 'B');
auto remain = fxcrt::spancpy(pdfium::span(dst), pdfium::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<char> src(2, 'A');
std::vector<char> dst(4, 'B');
// Also show that a const src argument is acceptable.
auto remain = fxcrt::spancpy(pdfium::span(dst).subspan<1u>(),
pdfium::span<const char>(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<char> src(2, 'A');
std::vector<char> dst(4, 'B');
auto remain = fxcrt::spancpy(pdfium::span(dst).subspan<1u>(),
pdfium::span(src).subspan<2u>());
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<char> src(2, 'A');
std::vector<char> dst(4, 'B');
auto remain = fxcrt::spancpy(pdfium::span(dst).subspan<4u>(),
pdfium::span(src).subspan<2u>());
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<char> src(4, 'A');
std::vector<char> dst(4, 'B');
EXPECT_TRUE(fxcrt::try_spancpy(pdfium::span(dst), pdfium::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<char> src(4, 'A');
std::vector<char> dst(3, 'B');
EXPECT_FALSE(fxcrt::try_spancpy(pdfium::span(dst), pdfium::span(src)));
EXPECT_EQ(dst[0], 'B');
EXPECT_EQ(dst[1], 'B');
EXPECT_EQ(dst[2], 'B');
}
TEST(Spanmove, FitsWithin) {
std::vector<char> src(2, 'A');
std::vector<char> dst(4, 'B');
// Also show that a const src argument is acceptable.
auto remain = fxcrt::spanmove(pdfium::span(dst).subspan<1u>(),
pdfium::span<const char>(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<char> src(4, 'A');
std::vector<char> dst(4, 'B');
EXPECT_TRUE(fxcrt::try_spanmove(pdfium::span(dst), pdfium::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<char> vec = {'A', 'B', 'C', 'D'};
EXPECT_TRUE(fxcrt::try_spanmove(pdfium::span(vec).first(3u),
pdfium::span(vec).last(3u)));
EXPECT_EQ(vec[0], 'B');
EXPECT_EQ(vec[1], 'C');
EXPECT_EQ(vec[2], 'D');
EXPECT_EQ(vec[3], 'D');
}
{
std::vector<char> vec = {'A', 'B', 'C', 'D'};
EXPECT_TRUE(fxcrt::try_spanmove(pdfium::span(vec).last(3u),
pdfium::span(vec).first(3u)));
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<char> src(4, 'A');
std::vector<char> dst(3, 'B');
EXPECT_FALSE(fxcrt::try_spanmove(pdfium::span(dst), pdfium::span(src)));
EXPECT_EQ(dst[0], 'B');
EXPECT_EQ(dst[1], 'B');
EXPECT_EQ(dst[2], 'B');
}
TEST(Span, AssignOverOnePastEnd) {
std::vector<char> src(2, 'A');
pdfium::span<char> span = pdfium::span(src);
span = span.subspan<2u>();
span = pdfium::span(src);
EXPECT_EQ(span.size(), 2u);
}
TEST(ReinterpretSpan, Empty) {
pdfium::span<uint8_t> empty;
pdfium::span<uint32_t> converted = fxcrt::reinterpret_span<uint32_t>(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<uint8_t> original = pdfium::span(aaaabbbb);
pdfium::span<uint32_t> converted =
fxcrt::reinterpret_span<uint32_t>(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<uint32_t>(pdfium::span(abcabc).subspan<1u, 4u>()),
"");
}
TEST(ReinterpretSpan, FlattenMultiDimension) {
struct Pt {
float x;
float y;
};
using Line = std::array<Pt, 4>;
using Plane = std::array<Line, 4>;
using Volume = std::array<Plane, 4>;
Volume box = {{{{{{}}}}}};
auto flat = fxcrt::reinterpret_span<Pt>(pdfium::span(box));
EXPECT_EQ(64u, flat.size());
float ctr = 0.0f;
for (auto& pt : flat) {
pt.x = ctr++;
pt.y = ctr++;
}
EXPECT_EQ(box[3][3][3].x, 126.0f);
EXPECT_EQ(box[3][3][3].y, 127.0f);
}
TEST(Spanpos, Empty) {
pdfium::span<const uint32_t> 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::span(kHaystack), kEmpty));
EXPECT_FALSE(fxcrt::spanpos(kEmpty, pdfium::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::span(kHaystack), pdfium::span(kStartMatch)),
testing::Optional(0u));
EXPECT_THAT(fxcrt::spanpos(pdfium::span(kHaystack), pdfium::span(kEndMatch)),
testing::Optional(4u));
EXPECT_FALSE(
fxcrt::spanpos(pdfium::span(kHaystack), pdfium::span(kNotFound)));
EXPECT_FALSE(fxcrt::spanpos(pdfium::span(kHaystack), pdfium::span(kTooLong)));
}