/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef _LIBUNWINDSTACK_ELF_H
#define _LIBUNWINDSTACK_ELF_H
#include <stddef.h>
#include <memory>
#include <mutex>
#include <string>
#include <unordered_map>
#include <utility>
#include <unwindstack/ElfInterface.h>
#include <unwindstack/Memory.h>
#if !defined(EM_AARCH64)
#define EM_AARCH64 183
#endif
namespace unwindstack {
// Forward declaration.
struct MapInfo;
class Regs;
enum ArchEnum : uint8_t {
ARCH_UNKNOWN = 0,
ARCH_ARM,
ARCH_ARM64,
ARCH_X86,
ARCH_X86_64,
ARCH_MIPS,
ARCH_MIPS64,
};
class Elf {
public:
Elf(Memory* memory) : memory_(memory) {}
virtual ~Elf() = default;
bool Init();
void InitGnuDebugdata();
void Invalidate();
std::string GetSoname();
bool GetFunctionName(uint64_t addr, std::string* name, uint64_t* func_offset);
bool GetGlobalVariable(const std::string& name, uint64_t* memory_address);
uint64_t GetRelPc(uint64_t pc, const MapInfo* map_info);
bool StepIfSignalHandler(uint64_t rel_pc, Regs* regs, Memory* process_memory);
bool Step(uint64_t rel_pc, Regs* regs, Memory* process_memory, bool* finished);
ElfInterface* CreateInterfaceFromMemory(Memory* memory);
std::string GetBuildID();
uint64_t GetLoadBias() { return load_bias_; }
bool IsValidPc(uint64_t pc);
void GetLastError(ErrorData* data);
ErrorCode GetLastErrorCode();
uint64_t GetLastErrorAddress();
bool valid() { return valid_; }
uint32_t machine_type() { return machine_type_; }
uint8_t class_type() { return class_type_; }
ArchEnum arch() { return arch_; }
Memory* memory() { return memory_.get(); }
ElfInterface* interface() { return interface_.get(); }
ElfInterface* gnu_debugdata_interface() { return gnu_debugdata_interface_.get(); }
static bool IsValidElf(Memory* memory);
static bool GetInfo(Memory* memory, uint64_t* size);
static uint64_t GetLoadBias(Memory* memory);
static std::string GetBuildID(Memory* memory);
static void SetCachingEnabled(bool enable);
static bool CachingEnabled() { return cache_enabled_; }
static void CacheLock();
static void CacheUnlock();
static void CacheAdd(MapInfo* info);
static bool CacheGet(MapInfo* info);
static bool CacheAfterCreateMemory(MapInfo* info);
protected:
bool valid_ = false;
uint64_t load_bias_ = 0;
std::unique_ptr<ElfInterface> interface_;
std::unique_ptr<Memory> memory_;
uint32_t machine_type_;
uint8_t class_type_;
ArchEnum arch_;
// Protect calls that can modify internal state of the interface object.
std::mutex lock_;
std::unique_ptr<Memory> gnu_debugdata_memory_;
std::unique_ptr<ElfInterface> gnu_debugdata_interface_;
static bool cache_enabled_;
static std::unordered_map<std::string, std::pair<std::shared_ptr<Elf>, bool>>* cache_;
static std::mutex* cache_lock_;
};
} // namespace unwindstack
#endif // _LIBUNWINDSTACK_ELF_H