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