xref: /aosp_15_r20/art/libartbase/base/mem_map_fuchsia.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2018 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #include "mem_map.h"
18*795d594fSAndroid Build Coastguard Worker #include "logging.h"
19*795d594fSAndroid Build Coastguard Worker #include "mman.h"
20*795d594fSAndroid Build Coastguard Worker 
21*795d594fSAndroid Build Coastguard Worker #include <zircon/process.h>
22*795d594fSAndroid Build Coastguard Worker #include <zircon/syscalls.h>
23*795d594fSAndroid Build Coastguard Worker 
24*795d594fSAndroid Build Coastguard Worker namespace art {
25*795d594fSAndroid Build Coastguard Worker 
26*795d594fSAndroid Build Coastguard Worker static zx_handle_t fuchsia_lowmem_vmar = ZX_HANDLE_INVALID;
27*795d594fSAndroid Build Coastguard Worker static zx_vaddr_t fuchsia_lowmem_base = 0;
28*795d594fSAndroid Build Coastguard Worker static size_t fuchsia_lowmem_size = 0;
29*795d594fSAndroid Build Coastguard Worker 
30*795d594fSAndroid Build Coastguard Worker static const char map_name[] = "mmap-android";
31*795d594fSAndroid Build Coastguard Worker static constexpr uintptr_t FUCHSIA_LOWER_MEM_START = 0x80000000;
32*795d594fSAndroid Build Coastguard Worker static constexpr uintptr_t FUCHSIA_LOWER_MEM_SIZE  = 0x60000000;
33*795d594fSAndroid Build Coastguard Worker 
TargetMMapInit()34*795d594fSAndroid Build Coastguard Worker void MemMap::TargetMMapInit() {
35*795d594fSAndroid Build Coastguard Worker   if (fuchsia_lowmem_vmar != ZX_HANDLE_INVALID) {
36*795d594fSAndroid Build Coastguard Worker     return;
37*795d594fSAndroid Build Coastguard Worker   }
38*795d594fSAndroid Build Coastguard Worker 
39*795d594fSAndroid Build Coastguard Worker   zx_info_vmar_t vmarinfo;
40*795d594fSAndroid Build Coastguard Worker   CHECK_EQ(zx_object_get_info(zx_vmar_root_self(),
41*795d594fSAndroid Build Coastguard Worker                               ZX_INFO_VMAR,
42*795d594fSAndroid Build Coastguard Worker                               &vmarinfo,
43*795d594fSAndroid Build Coastguard Worker                               sizeof(vmarinfo),
44*795d594fSAndroid Build Coastguard Worker                               nullptr,
45*795d594fSAndroid Build Coastguard Worker                               nullptr), ZX_OK) << "could not find info from root vmar";
46*795d594fSAndroid Build Coastguard Worker 
47*795d594fSAndroid Build Coastguard Worker   uintptr_t lower_mem_start = FUCHSIA_LOWER_MEM_START - vmarinfo.base;
48*795d594fSAndroid Build Coastguard Worker   fuchsia_lowmem_size = FUCHSIA_LOWER_MEM_SIZE;
49*795d594fSAndroid Build Coastguard Worker   uint32_t allocflags = ZX_VM_FLAG_CAN_MAP_READ |
50*795d594fSAndroid Build Coastguard Worker                         ZX_VM_FLAG_CAN_MAP_WRITE |
51*795d594fSAndroid Build Coastguard Worker                         ZX_VM_FLAG_CAN_MAP_EXECUTE |
52*795d594fSAndroid Build Coastguard Worker                         ZX_VM_FLAG_SPECIFIC;
53*795d594fSAndroid Build Coastguard Worker   CHECK_EQ(zx_vmar_allocate(zx_vmar_root_self(),
54*795d594fSAndroid Build Coastguard Worker                             lower_mem_start,
55*795d594fSAndroid Build Coastguard Worker                             fuchsia_lowmem_size,
56*795d594fSAndroid Build Coastguard Worker                             allocflags,
57*795d594fSAndroid Build Coastguard Worker                             &fuchsia_lowmem_vmar,
58*795d594fSAndroid Build Coastguard Worker                             &fuchsia_lowmem_base), ZX_OK) << "could not allocate lowmem vmar";
59*795d594fSAndroid Build Coastguard Worker }
60*795d594fSAndroid Build Coastguard Worker 
TargetMMap(void * start,size_t len,int prot,int flags,int fd,off_t fd_off)61*795d594fSAndroid Build Coastguard Worker void* MemMap::TargetMMap(void* start, size_t len, int prot, int flags, int fd, off_t fd_off) {
62*795d594fSAndroid Build Coastguard Worker   zx_status_t status;
63*795d594fSAndroid Build Coastguard Worker   uintptr_t mem = 0;
64*795d594fSAndroid Build Coastguard Worker 
65*795d594fSAndroid Build Coastguard Worker   bool mmap_lower = (flags & MAP_32BIT) != 0;
66*795d594fSAndroid Build Coastguard Worker 
67*795d594fSAndroid Build Coastguard Worker   // for file-based mapping use system library
68*795d594fSAndroid Build Coastguard Worker   if ((flags & MAP_ANONYMOUS) == 0) {
69*795d594fSAndroid Build Coastguard Worker     if (start != nullptr) {
70*795d594fSAndroid Build Coastguard Worker       flags |= MAP_FIXED;
71*795d594fSAndroid Build Coastguard Worker     }
72*795d594fSAndroid Build Coastguard Worker     CHECK(!mmap_lower) << "cannot map files into low memory for Fuchsia";
73*795d594fSAndroid Build Coastguard Worker     return mmap(start, len, prot, flags, fd, fd_off);
74*795d594fSAndroid Build Coastguard Worker   }
75*795d594fSAndroid Build Coastguard Worker 
76*795d594fSAndroid Build Coastguard Worker   uint32_t vmarflags = 0;
77*795d594fSAndroid Build Coastguard Worker   if ((prot & PROT_READ) != 0) {
78*795d594fSAndroid Build Coastguard Worker     vmarflags |= ZX_VM_FLAG_PERM_READ;
79*795d594fSAndroid Build Coastguard Worker   }
80*795d594fSAndroid Build Coastguard Worker   if ((prot & PROT_WRITE) != 0) {
81*795d594fSAndroid Build Coastguard Worker     vmarflags |= ZX_VM_FLAG_PERM_WRITE;
82*795d594fSAndroid Build Coastguard Worker   }
83*795d594fSAndroid Build Coastguard Worker   if ((prot & PROT_EXEC) != 0) {
84*795d594fSAndroid Build Coastguard Worker     vmarflags |= ZX_VM_FLAG_PERM_EXECUTE;
85*795d594fSAndroid Build Coastguard Worker   }
86*795d594fSAndroid Build Coastguard Worker 
87*795d594fSAndroid Build Coastguard Worker   if (len == 0) {
88*795d594fSAndroid Build Coastguard Worker     errno = EINVAL;
89*795d594fSAndroid Build Coastguard Worker     return MAP_FAILED;
90*795d594fSAndroid Build Coastguard Worker   }
91*795d594fSAndroid Build Coastguard Worker 
92*795d594fSAndroid Build Coastguard Worker   zx_info_vmar_t vmarinfo;
93*795d594fSAndroid Build Coastguard Worker   size_t vmaroffset = 0;
94*795d594fSAndroid Build Coastguard Worker   if (start != nullptr) {
95*795d594fSAndroid Build Coastguard Worker     vmarflags |= ZX_VM_FLAG_SPECIFIC;
96*795d594fSAndroid Build Coastguard Worker     status = zx_object_get_info((mmap_lower ? fuchsia_lowmem_vmar : zx_vmar_root_self()),
97*795d594fSAndroid Build Coastguard Worker                                 ZX_INFO_VMAR,
98*795d594fSAndroid Build Coastguard Worker                                 &vmarinfo,
99*795d594fSAndroid Build Coastguard Worker                                 sizeof(vmarinfo),
100*795d594fSAndroid Build Coastguard Worker                                 nullptr,
101*795d594fSAndroid Build Coastguard Worker                                 nullptr);
102*795d594fSAndroid Build Coastguard Worker     if (status < 0 || reinterpret_cast<uintptr_t>(start) < vmarinfo.base) {
103*795d594fSAndroid Build Coastguard Worker       errno = EINVAL;
104*795d594fSAndroid Build Coastguard Worker       return MAP_FAILED;
105*795d594fSAndroid Build Coastguard Worker     }
106*795d594fSAndroid Build Coastguard Worker     vmaroffset = reinterpret_cast<uintptr_t>(start) - vmarinfo.base;
107*795d594fSAndroid Build Coastguard Worker   }
108*795d594fSAndroid Build Coastguard Worker 
109*795d594fSAndroid Build Coastguard Worker   zx_handle_t vmo;
110*795d594fSAndroid Build Coastguard Worker   if (zx_vmo_create(len, 0, &vmo) < 0) {
111*795d594fSAndroid Build Coastguard Worker     errno = ENOMEM;
112*795d594fSAndroid Build Coastguard Worker     return MAP_FAILED;
113*795d594fSAndroid Build Coastguard Worker   }
114*795d594fSAndroid Build Coastguard Worker   zx_vmo_get_size(vmo, &len);
115*795d594fSAndroid Build Coastguard Worker   zx_object_set_property(vmo, ZX_PROP_NAME, map_name, strlen(map_name));
116*795d594fSAndroid Build Coastguard Worker 
117*795d594fSAndroid Build Coastguard Worker   if (mmap_lower) {
118*795d594fSAndroid Build Coastguard Worker     status = zx_vmar_map(fuchsia_lowmem_vmar, vmaroffset, vmo, fd_off, len, vmarflags, &mem);
119*795d594fSAndroid Build Coastguard Worker   } else {
120*795d594fSAndroid Build Coastguard Worker     status = zx_vmar_map(zx_vmar_root_self(), vmaroffset, vmo, fd_off, len, vmarflags, &mem);
121*795d594fSAndroid Build Coastguard Worker   }
122*795d594fSAndroid Build Coastguard Worker   zx_handle_close(vmo);
123*795d594fSAndroid Build Coastguard Worker   if (status != ZX_OK) {
124*795d594fSAndroid Build Coastguard Worker     return MAP_FAILED;
125*795d594fSAndroid Build Coastguard Worker   }
126*795d594fSAndroid Build Coastguard Worker 
127*795d594fSAndroid Build Coastguard Worker   return reinterpret_cast<void *>(mem);
128*795d594fSAndroid Build Coastguard Worker }
129*795d594fSAndroid Build Coastguard Worker 
TargetMUnmap(void * start,size_t len)130*795d594fSAndroid Build Coastguard Worker int MemMap::TargetMUnmap(void* start, size_t len) {
131*795d594fSAndroid Build Coastguard Worker   uintptr_t addr = reinterpret_cast<uintptr_t>(start);
132*795d594fSAndroid Build Coastguard Worker   zx_handle_t alloc_vmar = zx_vmar_root_self();
133*795d594fSAndroid Build Coastguard Worker   if (addr >= fuchsia_lowmem_base && addr < fuchsia_lowmem_base + fuchsia_lowmem_size) {
134*795d594fSAndroid Build Coastguard Worker     alloc_vmar = fuchsia_lowmem_vmar;
135*795d594fSAndroid Build Coastguard Worker   }
136*795d594fSAndroid Build Coastguard Worker   zx_status_t status = zx_vmar_unmap(alloc_vmar, addr, len);
137*795d594fSAndroid Build Coastguard Worker   if (status < 0) {
138*795d594fSAndroid Build Coastguard Worker     errno = EINVAL;
139*795d594fSAndroid Build Coastguard Worker     return -1;
140*795d594fSAndroid Build Coastguard Worker   }
141*795d594fSAndroid Build Coastguard Worker   return 0;
142*795d594fSAndroid Build Coastguard Worker }
143*795d594fSAndroid Build Coastguard Worker 
144*795d594fSAndroid Build Coastguard Worker }  // namespace art
145