//===-- 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) 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_ADD_NULL_CHECKS