| // Copyright 2021 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef SRC_LIB_UNWINDER_MEMORY_H_ |
| #define SRC_LIB_UNWINDER_MEMORY_H_ |
| |
| #include <stdio.h> |
| |
| #include <cerrno> |
| #include <cstdint> |
| #include <cstring> |
| #include <map> |
| #include <memory> |
| #include <vector> |
| |
| #include "sdk/lib/fit/include/lib/fit/function.h" |
| #include "src/lib/unwinder/error.h" |
| |
| namespace unwinder { |
| |
| // Abstract representation of a readable memory space. |
| class Memory { |
| public: |
| virtual ~Memory() = default; |
| virtual Error ReadBytes(uint64_t addr, uint64_t size, void* dst) = 0; |
| |
| template <class Type> |
| [[nodiscard]] Error Read(uint64_t addr, Type& res) { |
| return ReadBytes(addr, sizeof(res), &res); |
| } |
| |
| [[nodiscard]] Error ReadString(uint64_t addr, char* str, size_t max_length) { |
| char ch; |
| size_t i = 0; |
| do { |
| if (i >= max_length) { |
| str[i - 1] = '\0'; |
| break; |
| } |
| if (auto err = Read(addr + i, ch); err.has_err()) { |
| return err; |
| } |
| str[i] = ch; |
| i++; |
| } while (ch != '\0'); |
| return Success(); |
| } |
| |
| // Read an object and advance the addr by the read size. Do not advance if failed. |
| template <class Type> |
| [[nodiscard]] Error ReadAndAdvance(uint64_t& addr, Type& res) { |
| if (auto err = Read(addr, res); err.has_err()) { |
| return err; |
| } |
| addr += sizeof(res); |
| return Success(); |
| } |
| |
| [[nodiscard]] Error ReadSLEB128AndAdvance(uint64_t& addr, int64_t& res); |
| [[nodiscard]] Error ReadULEB128AndAdvance(uint64_t& addr, uint64_t& res); |
| |
| // Read the data in DWARF encoding. data_rel_base is only used in .eh_frame_hdr. |
| [[nodiscard]] Error ReadEncodedAndAdvance(uint64_t& addr, uint64_t& res, uint8_t enc, |
| uint64_t data_rel_base = 0); |
| [[nodiscard]] Error ReadEncoded(uint64_t addr, uint64_t& res, uint8_t enc, |
| uint64_t data_rel_base = 0) { |
| return ReadEncodedAndAdvance(addr, res, enc, data_rel_base); |
| } |
| }; |
| |
| // This interface implementation provides facilities for implementations to inject asynchronous |
| // memory fetching before using the typical |Memory| interface. |
| class AsyncMemory : public Memory { |
| public: |
| class Delegate : public Memory { |
| public: |
| // Perform the asynchronous reads for each (address, size) pair. |cb| should be issued once all |
| // requested memory has been received and is usable with |ReadBytes|. |
| virtual void FetchMemoryRanges(std::vector<std::pair<uint64_t, uint32_t>> ranges, |
| fit::callback<void()> cb) = 0; |
| }; |
| |
| explicit AsyncMemory(Delegate* delegate) : delegate_(delegate) {} |
| |
| // Request |delegate_| to fetch the given address ranges. |done| will be issued only after all |
| // ranges have been fetched, at which point it is guaranteed that ReadBytes will complete |
| // synchronously. |
| void FetchMemoryRanges(std::vector<std::pair<uint64_t, uint32_t>> ranges, |
| fit::callback<void()> done) { |
| delegate_->FetchMemoryRanges(std::move(ranges), std::move(done)); |
| } |
| |
| Error ReadBytes(uint64_t addr, uint64_t size, void* dst) override { |
| return delegate_->ReadBytes(addr, size, dst); |
| } |
| |
| private: |
| Delegate* delegate_; |
| }; |
| |
| class LocalMemory : public Memory { |
| public: |
| Error ReadBytes(uint64_t addr, uint64_t size, void* dst) override { |
| memcpy(dst, reinterpret_cast<void*>(addr), size); // NOLINT(performance-no-int-to-ptr) |
| return Success(); |
| } |
| }; |
| |
| class FileMemory : public Memory { |
| public: |
| explicit FileMemory(const std::string& path) : file_(fopen(path.c_str(), "rb"), &fclose) {} |
| Error ReadBytes(uint64_t offset, uint64_t size, void* dst) override { |
| if (fseek(file_.get(), static_cast<int64_t>(offset), SEEK_SET) != 0) { |
| return Error("bad fseek: %s", strerror(errno)); |
| } |
| |
| if (fread(dst, 1, size, file_.get()) != size) { |
| return Error("didn't read full size!"); |
| } |
| |
| return Success(); |
| } |
| |
| private: |
| std::unique_ptr<FILE, decltype(&fclose)> file_; |
| }; |
| |
| // A memory that fails any reads. |
| class UnavailableMemory : public Memory { |
| public: |
| Error ReadBytes(uint64_t addr, uint64_t size, void* dst) override { |
| return Error("Unavailable memory"); |
| } |
| }; |
| |
| } // namespace unwinder |
| |
| #endif // SRC_LIB_UNWINDER_MEMORY_H_ |