1*71db0c75SAndroid Build Coastguard Worker //===-- Unittests for memcpy ----------------------------------------------===//
2*71db0c75SAndroid Build Coastguard Worker //
3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information.
5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*71db0c75SAndroid Build Coastguard Worker //
7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
8*71db0c75SAndroid Build Coastguard Worker
9*71db0c75SAndroid Build Coastguard Worker #include "memory_utils/memory_check_utils.h"
10*71db0c75SAndroid Build Coastguard Worker #include "src/__support/macros/config.h"
11*71db0c75SAndroid Build Coastguard Worker #include "src/__support/macros/properties/os.h" // LIBC_TARGET_OS_IS_LINUX
12*71db0c75SAndroid Build Coastguard Worker #include "src/string/memcpy.h"
13*71db0c75SAndroid Build Coastguard Worker #include "test/UnitTest/Test.h"
14*71db0c75SAndroid Build Coastguard Worker
15*71db0c75SAndroid Build Coastguard Worker #if !defined(LIBC_FULL_BUILD) && defined(LIBC_TARGET_OS_IS_LINUX)
16*71db0c75SAndroid Build Coastguard Worker #include "memory_utils/protected_pages.h"
17*71db0c75SAndroid Build Coastguard Worker #endif // !defined(LIBC_FULL_BUILD) && defined(LIBC_TARGET_OS_IS_LINUX)
18*71db0c75SAndroid Build Coastguard Worker
19*71db0c75SAndroid Build Coastguard Worker namespace LIBC_NAMESPACE_DECL {
20*71db0c75SAndroid Build Coastguard Worker
21*71db0c75SAndroid Build Coastguard Worker // Adapt CheckMemcpy signature to memcpy.
Adaptor(cpp::span<char> dst,cpp::span<char> src,size_t size)22*71db0c75SAndroid Build Coastguard Worker static inline void Adaptor(cpp::span<char> dst, cpp::span<char> src,
23*71db0c75SAndroid Build Coastguard Worker size_t size) {
24*71db0c75SAndroid Build Coastguard Worker LIBC_NAMESPACE::memcpy(dst.begin(), src.begin(), size);
25*71db0c75SAndroid Build Coastguard Worker }
26*71db0c75SAndroid Build Coastguard Worker
TEST(LlvmLibcMemcpyTest,SizeSweep)27*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcMemcpyTest, SizeSweep) {
28*71db0c75SAndroid Build Coastguard Worker static constexpr size_t kMaxSize = 400;
29*71db0c75SAndroid Build Coastguard Worker Buffer SrcBuffer(kMaxSize);
30*71db0c75SAndroid Build Coastguard Worker Buffer DstBuffer(kMaxSize);
31*71db0c75SAndroid Build Coastguard Worker Randomize(SrcBuffer.span());
32*71db0c75SAndroid Build Coastguard Worker for (size_t size = 0; size < kMaxSize; ++size) {
33*71db0c75SAndroid Build Coastguard Worker auto src = SrcBuffer.span().subspan(0, size);
34*71db0c75SAndroid Build Coastguard Worker auto dst = DstBuffer.span().subspan(0, size);
35*71db0c75SAndroid Build Coastguard Worker ASSERT_TRUE(CheckMemcpy<Adaptor>(dst, src, size));
36*71db0c75SAndroid Build Coastguard Worker }
37*71db0c75SAndroid Build Coastguard Worker }
38*71db0c75SAndroid Build Coastguard Worker
39*71db0c75SAndroid Build Coastguard Worker #if !defined(LIBC_FULL_BUILD) && defined(LIBC_TARGET_OS_IS_LINUX)
40*71db0c75SAndroid Build Coastguard Worker
TEST(LlvmLibcMemcpyTest,CheckAccess)41*71db0c75SAndroid Build Coastguard Worker TEST(LlvmLibcMemcpyTest, CheckAccess) {
42*71db0c75SAndroid Build Coastguard Worker static constexpr size_t MAX_SIZE = 1024;
43*71db0c75SAndroid Build Coastguard Worker LIBC_ASSERT(MAX_SIZE < GetPageSize());
44*71db0c75SAndroid Build Coastguard Worker ProtectedPages pages;
45*71db0c75SAndroid Build Coastguard Worker const Page write_buffer = pages.GetPageA().WithAccess(PROT_WRITE);
46*71db0c75SAndroid Build Coastguard Worker const Page read_buffer = [&]() {
47*71db0c75SAndroid Build Coastguard Worker // We fetch page B in write mode.
48*71db0c75SAndroid Build Coastguard Worker auto page = pages.GetPageB().WithAccess(PROT_WRITE);
49*71db0c75SAndroid Build Coastguard Worker // And fill it with random numbers.
50*71db0c75SAndroid Build Coastguard Worker for (size_t i = 0; i < page.page_size; ++i)
51*71db0c75SAndroid Build Coastguard Worker page.page_ptr[i] = rand();
52*71db0c75SAndroid Build Coastguard Worker // Then return it in read mode.
53*71db0c75SAndroid Build Coastguard Worker return page.WithAccess(PROT_READ);
54*71db0c75SAndroid Build Coastguard Worker }();
55*71db0c75SAndroid Build Coastguard Worker for (size_t size = 0; size < MAX_SIZE; ++size) {
56*71db0c75SAndroid Build Coastguard Worker // We cross-check the function with two sources and two destinations.
57*71db0c75SAndroid Build Coastguard Worker // - The first of them (bottom) is always page aligned and faults when
58*71db0c75SAndroid Build Coastguard Worker // accessing bytes before it.
59*71db0c75SAndroid Build Coastguard Worker // - The second one (top) is not necessarily aligned and faults when
60*71db0c75SAndroid Build Coastguard Worker // accessing bytes after it.
61*71db0c75SAndroid Build Coastguard Worker const uint8_t *sources[2] = {read_buffer.bottom(size),
62*71db0c75SAndroid Build Coastguard Worker read_buffer.top(size)};
63*71db0c75SAndroid Build Coastguard Worker uint8_t *destinations[2] = {write_buffer.bottom(size),
64*71db0c75SAndroid Build Coastguard Worker write_buffer.top(size)};
65*71db0c75SAndroid Build Coastguard Worker for (const uint8_t *src : sources) {
66*71db0c75SAndroid Build Coastguard Worker for (uint8_t *dst : destinations) {
67*71db0c75SAndroid Build Coastguard Worker LIBC_NAMESPACE::memcpy(dst, src, size);
68*71db0c75SAndroid Build Coastguard Worker }
69*71db0c75SAndroid Build Coastguard Worker }
70*71db0c75SAndroid Build Coastguard Worker }
71*71db0c75SAndroid Build Coastguard Worker }
72*71db0c75SAndroid Build Coastguard Worker
73*71db0c75SAndroid Build Coastguard Worker #endif // !defined(LIBC_FULL_BUILD) && defined(LIBC_TARGET_OS_IS_LINUX)
74*71db0c75SAndroid Build Coastguard Worker
75*71db0c75SAndroid Build Coastguard Worker } // namespace LIBC_NAMESPACE_DECL
76