blob: 39378a3af70f9e6de34109662faf704a3e7b5add [file] [log] [blame]
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -07001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Christopher Ferris5f118512017-09-01 11:17:16 -070017#include <sys/mman.h>
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -070018#include <sys/types.h>
19#include <unistd.h>
20
21#include <memory>
Christopher Ferrisbe788d82017-11-27 14:50:38 -080022#include <mutex>
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -070023#include <string>
24
Christopher Ferrisd226a512017-07-14 10:37:19 -070025#include <unwindstack/Elf.h>
26#include <unwindstack/MapInfo.h>
27#include <unwindstack/Maps.h>
28#include <unwindstack/Memory.h>
29
30namespace unwindstack {
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -070031
Christopher Ferris3f805ac2017-08-30 13:15:19 -070032Memory* MapInfo::GetFileMemory() {
33 std::unique_ptr<MemoryFileAtOffset> memory(new MemoryFileAtOffset);
34 if (offset == 0) {
35 if (memory->Init(name, 0)) {
36 return memory.release();
37 }
38 return nullptr;
39 }
40
41 // There are two possibilities when the offset is non-zero.
42 // - There is an elf file embedded in a file.
43 // - The whole file is an elf file, and the offset needs to be saved.
44 //
45 // Map in just the part of the file for the map. If this is not
46 // a valid elf, then reinit as if the whole file is an elf file.
47 // If the offset is a valid elf, then determine the size of the map
48 // and reinit to that size. This is needed because the dynamic linker
49 // only maps in a portion of the original elf, and never the symbol
50 // file data.
51 uint64_t map_size = end - start;
52 if (!memory->Init(name, offset, map_size)) {
53 return nullptr;
54 }
55
56 bool valid;
57 uint64_t max_size;
58 Elf::GetInfo(memory.get(), &valid, &max_size);
59 if (!valid) {
60 // Init as if the whole file is an elf.
61 if (memory->Init(name, 0)) {
62 elf_offset = offset;
63 return memory.release();
64 }
65 return nullptr;
66 }
67
68 if (max_size > map_size) {
69 if (memory->Init(name, offset, max_size)) {
70 return memory.release();
71 }
72 // Try to reinit using the default map_size.
73 if (memory->Init(name, offset, map_size)) {
74 return memory.release();
75 }
76 return nullptr;
77 }
78 return memory.release();
79}
80
Christopher Ferris5f118512017-09-01 11:17:16 -070081Memory* MapInfo::CreateMemory(const std::shared_ptr<Memory>& process_memory) {
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -070082 if (end <= start) {
83 return nullptr;
84 }
85
86 elf_offset = 0;
87
Christopher Ferris5f118512017-09-01 11:17:16 -070088 // Fail on device maps.
89 if (flags & MAPS_FLAGS_DEVICE_MAP) {
90 return nullptr;
91 }
92
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -070093 // First try and use the file associated with the info.
94 if (!name.empty()) {
Christopher Ferris3f805ac2017-08-30 13:15:19 -070095 Memory* memory = GetFileMemory();
96 if (memory != nullptr) {
97 return memory;
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -070098 }
99 }
100
Christopher Ferris5f118512017-09-01 11:17:16 -0700101 // If the map isn't readable, don't bother trying to read from process memory.
102 if (!(flags & PROT_READ)) {
103 return nullptr;
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -0700104 }
Josh Gao29c53782017-09-26 14:11:37 -0700105 return new MemoryRange(process_memory, start, end - start, 0);
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -0700106}
107
Christopher Ferris5f118512017-09-01 11:17:16 -0700108Elf* MapInfo::GetElf(const std::shared_ptr<Memory>& process_memory, bool init_gnu_debugdata) {
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800109 // Make sure no other thread is trying to add the elf to this map.
110 std::lock_guard<std::mutex> guard(mutex_);
111
Christopher Ferris0b79ae12018-01-25 12:15:56 -0800112 if (elf.get() != nullptr) {
113 return elf.get();
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -0700114 }
115
Christopher Ferris0b79ae12018-01-25 12:15:56 -0800116 bool locked = false;
117 if (Elf::CachingEnabled() && !name.empty()) {
118 Elf::CacheLock();
119 locked = true;
Christopher Ferrisd9575b62018-02-16 13:48:19 -0800120 if (Elf::CacheGet(this)) {
Christopher Ferris0b79ae12018-01-25 12:15:56 -0800121 Elf::CacheUnlock();
122 return elf.get();
123 }
124 }
Christopher Ferrise69f4702017-10-19 16:08:58 -0700125
Christopher Ferris0b79ae12018-01-25 12:15:56 -0800126 Memory* memory = CreateMemory(process_memory);
Christopher Ferrisd9575b62018-02-16 13:48:19 -0800127 if (locked) {
128 if (Elf::CacheAfterCreateMemory(this)) {
Christopher Ferris0b79ae12018-01-25 12:15:56 -0800129 delete memory;
130 Elf::CacheUnlock();
131 return elf.get();
132 }
133 }
134 elf.reset(new Elf(memory));
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -0700135 // If the init fails, keep the elf around as an invalid object so we
136 // don't try to reinit the object.
Christopher Ferris0b79ae12018-01-25 12:15:56 -0800137 elf->Init(init_gnu_debugdata);
138
139 if (locked) {
140 Elf::CacheAdd(this);
141 Elf::CacheUnlock();
142 }
143 return elf.get();
Christopher Ferris0d7cf3e2017-04-19 15:42:19 -0700144}
Christopher Ferrisd226a512017-07-14 10:37:19 -0700145
Christopher Ferrisb7de5f52017-12-01 21:37:37 -0800146uint64_t MapInfo::GetLoadBias(const std::shared_ptr<Memory>& process_memory) {
Christopher Ferrise7b66242017-12-15 11:17:45 -0800147 uint64_t cur_load_bias = load_bias.load();
148 if (cur_load_bias != static_cast<uint64_t>(-1)) {
149 return cur_load_bias;
150 }
151
Christopher Ferrisb7de5f52017-12-01 21:37:37 -0800152 {
153 // Make sure no other thread is trying to add the elf to this map.
154 std::lock_guard<std::mutex> guard(mutex_);
155 if (elf != nullptr) {
156 if (elf->valid()) {
Christopher Ferrise7b66242017-12-15 11:17:45 -0800157 cur_load_bias = elf->GetLoadBias();
158 load_bias = cur_load_bias;
159 return cur_load_bias;
Christopher Ferrisb7de5f52017-12-01 21:37:37 -0800160 } else {
Christopher Ferrise7b66242017-12-15 11:17:45 -0800161 load_bias = 0;
Christopher Ferrisb7de5f52017-12-01 21:37:37 -0800162 return 0;
163 }
164 }
165 }
166
167 // Call lightweight static function that will only read enough of the
168 // elf data to get the load bias.
169 std::unique_ptr<Memory> memory(CreateMemory(process_memory));
Christopher Ferrise7b66242017-12-15 11:17:45 -0800170 cur_load_bias = Elf::GetLoadBias(memory.get());
171 load_bias = cur_load_bias;
172 return cur_load_bias;
Christopher Ferrisb7de5f52017-12-01 21:37:37 -0800173}
174
Christopher Ferrisd226a512017-07-14 10:37:19 -0700175} // namespace unwindstack