xref: /aosp_15_r20/system/unwinding/libunwindstack/tests/MemoryMteTest.cpp (revision eb293b8f56ee8303637c5595cfcdeef8039e85c6)
1*eb293b8fSAndroid Build Coastguard Worker /*
2*eb293b8fSAndroid Build Coastguard Worker  * Copyright (C) 2020 The Android Open Source Project
3*eb293b8fSAndroid Build Coastguard Worker  *
4*eb293b8fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*eb293b8fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*eb293b8fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*eb293b8fSAndroid Build Coastguard Worker  *
8*eb293b8fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*eb293b8fSAndroid Build Coastguard Worker  *
10*eb293b8fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*eb293b8fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*eb293b8fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*eb293b8fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*eb293b8fSAndroid Build Coastguard Worker  * limitations under the License.
15*eb293b8fSAndroid Build Coastguard Worker  */
16*eb293b8fSAndroid Build Coastguard Worker 
17*eb293b8fSAndroid Build Coastguard Worker #include <sys/mman.h>
18*eb293b8fSAndroid Build Coastguard Worker #include <sys/types.h>
19*eb293b8fSAndroid Build Coastguard Worker 
20*eb293b8fSAndroid Build Coastguard Worker #include <gtest/gtest.h>
21*eb293b8fSAndroid Build Coastguard Worker 
22*eb293b8fSAndroid Build Coastguard Worker #ifdef __ANDROID__
23*eb293b8fSAndroid Build Coastguard Worker #include <bionic/mte.h>
24*eb293b8fSAndroid Build Coastguard Worker #endif
25*eb293b8fSAndroid Build Coastguard Worker 
26*eb293b8fSAndroid Build Coastguard Worker #include "ForkTest.h"
27*eb293b8fSAndroid Build Coastguard Worker #include "MemoryLocal.h"
28*eb293b8fSAndroid Build Coastguard Worker #include "MemoryRemote.h"
29*eb293b8fSAndroid Build Coastguard Worker #include "PidUtils.h"
30*eb293b8fSAndroid Build Coastguard Worker #include "TestUtils.h"
31*eb293b8fSAndroid Build Coastguard Worker 
32*eb293b8fSAndroid Build Coastguard Worker namespace unwindstack {
33*eb293b8fSAndroid Build Coastguard Worker 
34*eb293b8fSAndroid Build Coastguard Worker using MemoryMteTest = ForkTest;
35*eb293b8fSAndroid Build Coastguard Worker 
CreateTagMapping()36*eb293b8fSAndroid Build Coastguard Worker static uintptr_t CreateTagMapping() {
37*eb293b8fSAndroid Build Coastguard Worker #if defined(__aarch64__)
38*eb293b8fSAndroid Build Coastguard Worker   uintptr_t mapping =
39*eb293b8fSAndroid Build Coastguard Worker       reinterpret_cast<uintptr_t>(mmap(nullptr, getpagesize(), PROT_READ | PROT_WRITE | PROT_MTE,
40*eb293b8fSAndroid Build Coastguard Worker                                        MAP_PRIVATE | MAP_ANONYMOUS, -1, 0));
41*eb293b8fSAndroid Build Coastguard Worker   if (reinterpret_cast<void*>(mapping) == MAP_FAILED) {
42*eb293b8fSAndroid Build Coastguard Worker     return 0;
43*eb293b8fSAndroid Build Coastguard Worker   }
44*eb293b8fSAndroid Build Coastguard Worker   __asm__ __volatile__(".arch_extension mte; stg %0, [%0]"
45*eb293b8fSAndroid Build Coastguard Worker                        :
46*eb293b8fSAndroid Build Coastguard Worker                        : "r"(mapping + (1ULL << 56))
47*eb293b8fSAndroid Build Coastguard Worker                        : "memory");
48*eb293b8fSAndroid Build Coastguard Worker   return mapping;
49*eb293b8fSAndroid Build Coastguard Worker #else
50*eb293b8fSAndroid Build Coastguard Worker   return 0;
51*eb293b8fSAndroid Build Coastguard Worker #endif
52*eb293b8fSAndroid Build Coastguard Worker }
53*eb293b8fSAndroid Build Coastguard Worker 
TEST_F(MemoryMteTest,remote_read_tag)54*eb293b8fSAndroid Build Coastguard Worker TEST_F(MemoryMteTest, remote_read_tag) {
55*eb293b8fSAndroid Build Coastguard Worker #if !defined(__aarch64__)
56*eb293b8fSAndroid Build Coastguard Worker   GTEST_SKIP() << "Requires aarch64";
57*eb293b8fSAndroid Build Coastguard Worker #else
58*eb293b8fSAndroid Build Coastguard Worker   if (!mte_supported()) {
59*eb293b8fSAndroid Build Coastguard Worker     GTEST_SKIP() << "Requires MTE";
60*eb293b8fSAndroid Build Coastguard Worker   }
61*eb293b8fSAndroid Build Coastguard Worker #endif
62*eb293b8fSAndroid Build Coastguard Worker 
63*eb293b8fSAndroid Build Coastguard Worker   uintptr_t mapping = CreateTagMapping();
64*eb293b8fSAndroid Build Coastguard Worker   ASSERT_NE(0U, mapping);
65*eb293b8fSAndroid Build Coastguard Worker 
66*eb293b8fSAndroid Build Coastguard Worker   ASSERT_NO_FATAL_FAILURE(Fork());
67*eb293b8fSAndroid Build Coastguard Worker 
68*eb293b8fSAndroid Build Coastguard Worker   MemoryRemote remote(pid_);
69*eb293b8fSAndroid Build Coastguard Worker 
70*eb293b8fSAndroid Build Coastguard Worker   EXPECT_EQ(1, remote.ReadTag(mapping));
71*eb293b8fSAndroid Build Coastguard Worker   EXPECT_EQ(0, remote.ReadTag(mapping + 16));
72*eb293b8fSAndroid Build Coastguard Worker }
73*eb293b8fSAndroid Build Coastguard Worker 
TEST_F(MemoryMteTest,local_read_tag)74*eb293b8fSAndroid Build Coastguard Worker TEST_F(MemoryMteTest, local_read_tag) {
75*eb293b8fSAndroid Build Coastguard Worker #if !defined(__aarch64__)
76*eb293b8fSAndroid Build Coastguard Worker   GTEST_SKIP() << "Requires aarch64";
77*eb293b8fSAndroid Build Coastguard Worker #else
78*eb293b8fSAndroid Build Coastguard Worker   if (!mte_supported()) {
79*eb293b8fSAndroid Build Coastguard Worker     GTEST_SKIP() << "Requires MTE";
80*eb293b8fSAndroid Build Coastguard Worker   }
81*eb293b8fSAndroid Build Coastguard Worker #endif
82*eb293b8fSAndroid Build Coastguard Worker 
83*eb293b8fSAndroid Build Coastguard Worker   uintptr_t mapping = CreateTagMapping();
84*eb293b8fSAndroid Build Coastguard Worker   ASSERT_NE(0U, mapping);
85*eb293b8fSAndroid Build Coastguard Worker 
86*eb293b8fSAndroid Build Coastguard Worker   MemoryLocal local;
87*eb293b8fSAndroid Build Coastguard Worker 
88*eb293b8fSAndroid Build Coastguard Worker   EXPECT_EQ(1, local.ReadTag(mapping));
89*eb293b8fSAndroid Build Coastguard Worker   EXPECT_EQ(0, local.ReadTag(mapping + 16));
90*eb293b8fSAndroid Build Coastguard Worker }
91*eb293b8fSAndroid Build Coastguard Worker 
92*eb293b8fSAndroid Build Coastguard Worker }  // namespace unwindstack
93