blob: d23aa0f0b3af1ef40555ac4e8292c13691698193 [file] [log] [blame]
//===-- Unittests for wmemmove --------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "hdr/types/size_t.h"
#include "hdr/types/wchar_t.h"
#include "src/wchar/wmemmove.h"
#include "test/UnitTest/Test.h"
TEST(LlvmLibcWMemmoveTest, MoveZeroByte) {
wchar_t buffer[] = {L'a', L'b', L'y', L'z'};
wchar_t *ret = LIBC_NAMESPACE::wmemmove(buffer, buffer + 2, 0);
EXPECT_EQ(ret, buffer);
const wchar_t expected[] = {L'a', L'b', L'y', L'z'};
EXPECT_TRUE(buffer[0] == expected[0]);
EXPECT_TRUE(buffer[1] == expected[1]);
EXPECT_TRUE(buffer[2] == expected[2]);
EXPECT_TRUE(buffer[3] == expected[3]);
}
TEST(LlvmLibcWMemmoveTest, DstAndSrcPointToSameAddress) {
wchar_t buffer[] = {L'a', L'b'};
wchar_t *ret = LIBC_NAMESPACE::wmemmove(buffer, buffer, 1);
EXPECT_EQ(ret, buffer);
const wchar_t expected[] = {L'a', L'b'};
EXPECT_TRUE(buffer[0] == expected[0]);
EXPECT_TRUE(buffer[1] == expected[1]);
}
TEST(LlvmLibcWMemmoveTest, DstStartsBeforeSrc) {
// Set boundary at beginning and end for not overstepping when
// copy forward or backward.
wchar_t buffer[] = {L'z', L'a', L'b', L'c', L'z'};
wchar_t *dst = buffer + 1;
wchar_t *ret = LIBC_NAMESPACE::wmemmove(dst, buffer + 2, 2);
EXPECT_EQ(ret, dst);
const wchar_t expected[] = {L'z', L'b', L'c', L'c', L'z'};
EXPECT_TRUE(buffer[0] == expected[0]);
EXPECT_TRUE(buffer[1] == expected[1]);
EXPECT_TRUE(buffer[2] == expected[2]);
EXPECT_TRUE(buffer[3] == expected[3]);
EXPECT_TRUE(buffer[4] == expected[4]);
}
TEST(LlvmLibcWMemmoveTest, DstStartsAfterSrc) {
wchar_t buffer[] = {L'z', L'a', L'b', L'c', L'z'};
wchar_t *dst = buffer + 2;
wchar_t *ret = LIBC_NAMESPACE::wmemmove(dst, buffer + 1, 2);
EXPECT_EQ(ret, dst);
const wchar_t expected[] = {L'z', L'a', L'a', L'b', L'z'};
EXPECT_TRUE(buffer[0] == expected[0]);
EXPECT_TRUE(buffer[1] == expected[1]);
EXPECT_TRUE(buffer[2] == expected[2]);
EXPECT_TRUE(buffer[3] == expected[3]);
EXPECT_TRUE(buffer[4] == expected[4]);
}
// e.g. `Dst` follow `src`.
// str: [abcdefghij]
// [__src_____]
// [_____Dst__]
TEST(LlvmLibcWMemmoveTest, SrcFollowDst) {
wchar_t buffer[] = {L'z', L'a', L'b', L'z'};
wchar_t *dst = buffer + 1;
wchar_t *ret = LIBC_NAMESPACE::wmemmove(dst, buffer + 2, 1);
EXPECT_EQ(ret, dst);
const char expected[] = {L'z', L'b', L'b', L'z'};
EXPECT_TRUE(buffer[0] == expected[0]);
EXPECT_TRUE(buffer[1] == expected[1]);
EXPECT_TRUE(buffer[2] == expected[2]);
EXPECT_TRUE(buffer[3] == expected[3]);
}
TEST(LlvmLibcWMemmoveTest, DstFollowSrc) {
wchar_t buffer[] = {L'z', L'a', L'b', L'z'};
wchar_t *dst = buffer + 2;
wchar_t *ret = LIBC_NAMESPACE::wmemmove(dst, buffer + 1, 1);
EXPECT_EQ(ret, dst);
const char expected[] = {L'z', L'a', L'a', L'z'};
EXPECT_TRUE(buffer[0] == expected[0]);
EXPECT_TRUE(buffer[1] == expected[1]);
EXPECT_TRUE(buffer[2] == expected[2]);
EXPECT_TRUE(buffer[3] == expected[3]);
}
#if defined(LIBC_ADD_NULL_CHECKS) && !defined(LIBC_HAS_SANITIZER)
TEST(LlvmLibcWMemmoveTest, NullptrCrash) {
wchar_t buffer[] = {L'a', L'b'};
// Passing in a nullptr should crash the program.
EXPECT_DEATH([&buffer] { LIBC_NAMESPACE::wmemmove(buffer, nullptr, 2); },
WITH_SIGNAL(-1));
EXPECT_DEATH([&buffer] { LIBC_NAMESPACE::wmemmove(nullptr, buffer, 2); },
WITH_SIGNAL(-1));
}
#endif // LIBC_HAS_ADDRESS_SANITIZER