xref: /llvm-project/libc/test/src/sys/mman/linux/mprotect_test.cpp (revision 3eb1e6d8e930f5aff17b8d6bcc160f5bbf8cabc7)
1 //===-- Unittests for mprotect --------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "src/errno/libc_errno.h"
10 #include "src/sys/mman/mmap.h"
11 #include "src/sys/mman/mprotect.h"
12 #include "src/sys/mman/munmap.h"
13 #include "test/UnitTest/ErrnoSetterMatcher.h"
14 #include "test/UnitTest/Test.h"
15 
16 #include <signal.h>
17 #include <sys/mman.h>
18 
19 using LIBC_NAMESPACE::testing::ErrnoSetterMatcher::Fails;
20 using LIBC_NAMESPACE::testing::ErrnoSetterMatcher::Succeeds;
21 
TEST(LlvmLibcMProtectTest,NoError)22 TEST(LlvmLibcMProtectTest, NoError) {
23   size_t alloc_size = 128;
24   LIBC_NAMESPACE::libc_errno = 0;
25   void *addr = LIBC_NAMESPACE::mmap(nullptr, alloc_size, PROT_READ,
26                                     MAP_ANONYMOUS | MAP_PRIVATE, -1, 0);
27   ASSERT_ERRNO_SUCCESS();
28   EXPECT_NE(addr, MAP_FAILED);
29 
30   int *array = reinterpret_cast<int *>(addr);
31   // Reading from the memory should not crash the test.
32   // Since we used the MAP_ANONYMOUS flag, the contents of the newly
33   // allocated memory should be initialized to zero.
34   EXPECT_EQ(array[0], 0);
35 
36   // By setting the memory protection to read and write, we should be able to
37   // modify that memory.
38   EXPECT_THAT(
39       LIBC_NAMESPACE::mprotect(addr, alloc_size, PROT_READ | PROT_WRITE),
40       Succeeds());
41   array[0] = 1;
42   EXPECT_EQ(array[0], 1);
43 
44   EXPECT_THAT(LIBC_NAMESPACE::munmap(addr, alloc_size), Succeeds());
45 }
46 
47 // This test is disabled currently due to flakeyness. It will be re-enabled once
48 // it is less flakey.
49 /*
50 TEST(LlvmLibcMProtectTest, Error_InvalidWrite) {
51   // attempting to write to a read-only protected part of memory should cause a
52   // segfault.
53   EXPECT_DEATH(
54       [] {
55         size_t alloc_size = 128;
56         void *addr =
57             LIBC_NAMESPACE::mmap(nullptr, alloc_size, PROT_READ | PROT_WRITE,
58                               MAP_ANONYMOUS | MAP_PRIVATE, -1, 0);
59         LIBC_NAMESPACE::mprotect(addr, alloc_size, PROT_READ);
60 
61         (reinterpret_cast<char *>(addr))[0] = 'A';
62       },
63       WITH_SIGNAL(SIGSEGV));
64   // Reading from a write only segment may succeed on some platforms, so there's
65   // no test to check that.
66 }
67 */
68