#include "src/__support/CPP/span.h"
#include "src/string/memccpy.h"
#include "test/UnitTest/Test.h"
#include <stddef.h>
class LlvmLibcMemccpyTest : public LIBC_NAMESPACE::testing::Test {
public:
void check_memccpy(LIBC_NAMESPACE::cpp::span<char> dst,
const LIBC_NAMESPACE::cpp::span<const char> src, int end,
size_t count,
const LIBC_NAMESPACE::cpp::span<const char> expected,
size_t expectedCopied, bool shouldReturnNull = false) {
ASSERT_GE(dst.size(), count);
void *result = LIBC_NAMESPACE::memccpy(dst.data(), src.data(), end, count);
if (shouldReturnNull) {
ASSERT_EQ(result, static_cast<void *>(nullptr));
} else {
ASSERT_EQ(result, static_cast<void *>(dst.data() + expectedCopied));
}
ASSERT_EQ(dst.size(), expected.size());
for (size_t i = 0; i < expected.size(); ++i)
ASSERT_EQ(expected[i], dst[i]);
}
};
TEST_F(LlvmLibcMemccpyTest, UntouchedUnrelatedEnd) {
char dst[] = {'a', 'b'};
const char src[] = {'x', '\0'};
const char expected[] = {'a', 'b'};
check_memccpy(dst, src, 'z', 0, expected, 0, true);
}
TEST_F(LlvmLibcMemccpyTest, UntouchedStartsWithEnd) {
char dst[] = {'a', 'b'};
const char src[] = {'x', '\0'};
const char expected[] = {'a', 'b'};
check_memccpy(dst, src, 'x', 0, expected, 0, true);
}
TEST_F(LlvmLibcMemccpyTest, CopyOneUnrelatedEnd) {
char dst[] = {'a', 'b'};
const char src[] = {'x', 'y'};
const char expected[] = {'x', 'b'};
check_memccpy(dst, src, 'z', 1, expected, 1, true);
}
TEST_F(LlvmLibcMemccpyTest, CopyOneStartsWithEnd) {
char dst[] = {'a', 'b'};
const char src[] = {'x', 'y'};
const char expected[] = {'x', 'b'};
check_memccpy(dst, src, 'x', 1, expected, 1);
}
TEST_F(LlvmLibcMemccpyTest, CopyTwoUnrelatedEnd) {
char dst[] = {'a', 'b'};
const char src[] = {'x', 'y'};
const char expected[] = {'x', 'y'};
check_memccpy(dst, src, 'z', 2, expected, 2, true);
}
TEST_F(LlvmLibcMemccpyTest, CopyTwoStartsWithEnd) {
char dst[] = {'a', 'b'};
const char src[] = {'x', 'y'};
const char expected[] = {'x', 'b'};
check_memccpy(dst, src, 'x', 2, expected, 1);
}