cool
This commit is contained in:
@@ -8,6 +8,7 @@ option(BUILD_SHARED_LIBS OFF)
|
|||||||
|
|
||||||
include_directories(external/ELFIO)
|
include_directories(external/ELFIO)
|
||||||
include_directories(external/capstone/include)
|
include_directories(external/capstone/include)
|
||||||
|
include_directories(external)
|
||||||
|
|
||||||
if(WIN32)
|
if(WIN32)
|
||||||
add_compile_definitions(_CRT_SECURE_NO_WARNINGS)
|
add_compile_definitions(_CRT_SECURE_NO_WARNINGS)
|
||||||
|
|||||||
Vendored
+20
-20
@@ -1,5 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <types.hpp>
|
#include <ircolib/types.hpp>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
@@ -8,36 +8,36 @@ namespace fs = std::filesystem;
|
|||||||
|
|
||||||
namespace ircolib {
|
namespace ircolib {
|
||||||
static inline std::vector<u8> ReadFileBinary(const std::string &path) {
|
static inline std::vector<u8> ReadFileBinary(const std::string &path) {
|
||||||
std::ifstream file(path, std::ios::binary);
|
std::ifstream file(path, std::ios::binary);
|
||||||
return {std::istreambuf_iterator{file}, {}};
|
return {std::istreambuf_iterator{file}, {}};
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void WriteFileBinary(const std::vector<u8> &data, const std::string &path) {
|
static inline void WriteFileBinary(const std::vector<u8> &data, const std::string &path) {
|
||||||
std::ofstream file(path, std::ios::binary);
|
std::ofstream file(path, std::ios::binary);
|
||||||
std::copy(data.begin(), data.end(), std::ostreambuf_iterator{file});
|
std::copy(data.begin(), data.end(), std::ostreambuf_iterator{file});
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void WriteFileBinary(const u8 *data, const size_t size, const std::string &path) {
|
static inline void WriteFileBinary(const u8 *data, const size_t size, const std::string &path) {
|
||||||
FILE *out = fopen(path.c_str(), "wb");
|
FILE *out = fopen(path.c_str(), "wb");
|
||||||
fwrite(data, size, 1, out);
|
fwrite(data, size, 1, out);
|
||||||
fclose(out);
|
fclose(out);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <size_t Size>
|
template <size_t Size>
|
||||||
static inline void WriteFileBinary(const std::array<u8, Size> &data, const std::string &path) {
|
static inline void WriteFileBinary(const std::array<u8, Size> &data, const std::string &path) {
|
||||||
std::ofstream file(path, std::ios::binary);
|
std::ofstream file(path, std::ios::binary);
|
||||||
std::copy(data.begin(), data.end(), std::ostreambuf_iterator{file});
|
std::copy(data.begin(), data.end(), std::ostreambuf_iterator{file});
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline size_t NextPow2(size_t num) {
|
static inline size_t NextPow2(size_t num) {
|
||||||
// Taken from "Bit Twiddling Hacks" by Sean Anderson:
|
// Taken from "Bit Twiddling Hacks" by Sean Anderson:
|
||||||
// https://graphics.stanford.edu/~seander/bithacks.html#RoundUpPowerOf2
|
// https://graphics.stanford.edu/~seander/bithacks.html#RoundUpPowerOf2
|
||||||
--num;
|
--num;
|
||||||
num |= num >> 1;
|
num |= num >> 1;
|
||||||
num |= num >> 2;
|
num |= num >> 2;
|
||||||
num |= num >> 4;
|
num |= num >> 4;
|
||||||
num |= num >> 8;
|
num |= num >> 8;
|
||||||
num |= num >> 16;
|
num |= num >> 16;
|
||||||
return num + 1;
|
return num + 1;
|
||||||
}
|
}
|
||||||
} // namespace Util
|
} // namespace ircolib
|
||||||
|
|||||||
Vendored
+46
-49
@@ -1,5 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <types.hpp>
|
#include <ircolib/types.hpp>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <bit>
|
#include <bit>
|
||||||
@@ -9,18 +9,17 @@
|
|||||||
|
|
||||||
namespace ircolib {
|
namespace ircolib {
|
||||||
static inline std::vector<u8> IntegralToBuffer(const std::integral auto &val) {
|
static inline std::vector<u8> IntegralToBuffer(const std::integral auto &val) {
|
||||||
std::vector<u8> ret{};
|
std::vector<u8> ret{};
|
||||||
ret.resize(sizeof(val));
|
ret.resize(sizeof(val));
|
||||||
|
|
||||||
memcpy(ret.data(), &val, sizeof(val));
|
memcpy(ret.data(), &val, sizeof(val));
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline constexpr bool IsInsideRange(const std::integral auto& addr,
|
static inline constexpr bool IsInsideRange(const std::integral auto &addr, const std::integral auto &start,
|
||||||
const std::integral auto& start,
|
const std::integral auto &end) {
|
||||||
const std::integral auto& end) {
|
return addr >= start && addr <= end;
|
||||||
return addr >= start && addr <= end;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
@@ -39,97 +38,97 @@ static constexpr inline void WriteAccess(std::array<u8, Size> &data, const u32 i
|
|||||||
|
|
||||||
template <>
|
template <>
|
||||||
constexpr inline u64 ReadAccess(const u8 *data, const u32 index) {
|
constexpr inline u64 ReadAccess(const u8 *data, const u32 index) {
|
||||||
u32 hi = *reinterpret_cast<const u32 *>(&data[index + 0]);
|
u32 hi = *reinterpret_cast<const u32 *>(&data[index + 0]);
|
||||||
u32 lo = *reinterpret_cast<const u32 *>(&data[index + 4]);
|
u32 lo = *reinterpret_cast<const u32 *>(&data[index + 4]);
|
||||||
const auto& result = static_cast<u64>(hi) << 32 | static_cast<u64>(lo);
|
const auto &result = static_cast<u64>(hi) << 32 | static_cast<u64>(lo);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static constexpr inline T ReadAccess(const u8 *data, const u32 index) {
|
static constexpr inline T ReadAccess(const u8 *data, const u32 index) {
|
||||||
return *reinterpret_cast<const T *>(&data[index]);
|
return *reinterpret_cast<const T *>(&data[index]);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template <>
|
||||||
constexpr inline u64 ReadAccess(const std::vector<u8> &data, const u32 index) {
|
constexpr inline u64 ReadAccess(const std::vector<u8> &data, const u32 index) {
|
||||||
u32 hi = *reinterpret_cast<const u32 *>(&data[index + 0]);
|
u32 hi = *reinterpret_cast<const u32 *>(&data[index + 0]);
|
||||||
u32 lo = *reinterpret_cast<const u32 *>(&data[index + 4]);
|
u32 lo = *reinterpret_cast<const u32 *>(&data[index + 4]);
|
||||||
return (static_cast<u64>(hi) << 32) | static_cast<u64>(lo);
|
return (static_cast<u64>(hi) << 32) | static_cast<u64>(lo);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static constexpr inline T ReadAccess(const std::vector<u8> &data, const u32 index) {
|
static constexpr inline T ReadAccess(const std::vector<u8> &data, const u32 index) {
|
||||||
return *reinterpret_cast<const T *>(&data[index]);
|
return *reinterpret_cast<const T *>(&data[index]);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <size_t Size>
|
template <size_t Size>
|
||||||
constexpr inline u64 ReadAccess(const std::array<u8, Size> &data, const u32 index) {
|
constexpr inline u64 ReadAccess(const std::array<u8, Size> &data, const u32 index) {
|
||||||
u32 hi = *reinterpret_cast<const u32 *>(&data[index + 0]);
|
u32 hi = *reinterpret_cast<const u32 *>(&data[index + 0]);
|
||||||
u32 lo = *reinterpret_cast<const u32 *>(&data[index + 4]);
|
u32 lo = *reinterpret_cast<const u32 *>(&data[index + 4]);
|
||||||
return static_cast<u64>(hi) << 32 | static_cast<u64>(lo);
|
return static_cast<u64>(hi) << 32 | static_cast<u64>(lo);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, size_t Size>
|
template <typename T, size_t Size>
|
||||||
static constexpr inline T ReadAccess(const std::array<u8, Size> &data, const u32 index) {
|
static constexpr inline T ReadAccess(const std::array<u8, Size> &data, const u32 index) {
|
||||||
return *reinterpret_cast<const T *>(&data[index]);
|
return *reinterpret_cast<const T *>(&data[index]);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <size_t Size>
|
template <size_t Size>
|
||||||
constexpr inline void WriteAccess(std::array<u8, Size> &data, const u32 index, const u64 val) {
|
constexpr inline void WriteAccess(std::array<u8, Size> &data, const u32 index, const u64 val) {
|
||||||
const u32 hi = val >> 32;
|
const u32 hi = val >> 32;
|
||||||
const u32 lo = val;
|
const u32 lo = val;
|
||||||
|
|
||||||
*reinterpret_cast<u32 *>(&data[index + 0]) = hi;
|
*reinterpret_cast<u32 *>(&data[index + 0]) = hi;
|
||||||
*reinterpret_cast<u32 *>(&data[index + 4]) = lo;
|
*reinterpret_cast<u32 *>(&data[index + 4]) = lo;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, size_t Size>
|
template <typename T, size_t Size>
|
||||||
static constexpr inline void WriteAccess(std::array<u8, Size> &data, const u32 index, const T val) {
|
static constexpr inline void WriteAccess(std::array<u8, Size> &data, const u32 index, const T val) {
|
||||||
*reinterpret_cast<T *>(&data[index]) = val;
|
*reinterpret_cast<T *>(&data[index]) = val;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template <>
|
||||||
constexpr inline void WriteAccess(std::vector<u8> &data, const u32 index, const u64 val) {
|
constexpr inline void WriteAccess(std::vector<u8> &data, const u32 index, const u64 val) {
|
||||||
const u32 hi = val >> 32;
|
const u32 hi = val >> 32;
|
||||||
const u32 lo = val;
|
const u32 lo = val;
|
||||||
|
|
||||||
*reinterpret_cast<u32 *>(&data[index + 0]) = hi;
|
*reinterpret_cast<u32 *>(&data[index + 0]) = hi;
|
||||||
*reinterpret_cast<u32 *>(&data[index + 4]) = lo;
|
*reinterpret_cast<u32 *>(&data[index + 4]) = lo;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static constexpr inline void WriteAccess(std::vector<u8> &data, const u32 index, const T val) {
|
static constexpr inline void WriteAccess(std::vector<u8> &data, const u32 index, const T val) {
|
||||||
*reinterpret_cast<T *>(&data[index]) = val;
|
*reinterpret_cast<T *>(&data[index]) = val;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template <>
|
||||||
constexpr inline void WriteAccess(u8 *data, const u32 index, const u64 val) {
|
constexpr inline void WriteAccess(u8 *data, const u32 index, const u64 val) {
|
||||||
const u32 hi = val >> 32;
|
const u32 hi = val >> 32;
|
||||||
const u32 lo = val;
|
const u32 lo = val;
|
||||||
|
|
||||||
*reinterpret_cast<u32 *>(&data[index + 0]) = hi;
|
*reinterpret_cast<u32 *>(&data[index + 0]) = hi;
|
||||||
*reinterpret_cast<u32 *>(&data[index + 4]) = lo;
|
*reinterpret_cast<u32 *>(&data[index + 4]) = lo;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static constexpr inline void WriteAccess(u8 *data, const u32 index, const T val) {
|
static constexpr inline void WriteAccess(u8 *data, const u32 index, const T val) {
|
||||||
*reinterpret_cast<T *>(&data[index]) = val;
|
*reinterpret_cast<T *>(&data[index]) = val;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static constexpr inline void SwapBuffer(std::vector<u8> &data) {
|
static constexpr inline void SwapBuffer(std::vector<u8> &data) {
|
||||||
for (size_t i = 0; i < data.size(); i += sizeof(T)) {
|
for (size_t i = 0; i < data.size(); i += sizeof(T)) {
|
||||||
const T original = *reinterpret_cast<T *>(&data[i]);
|
const T original = *reinterpret_cast<T *>(&data[i]);
|
||||||
*reinterpret_cast<T *>(&data[i]) = std::byteswap(original);
|
*reinterpret_cast<T *>(&data[i]) = std::byteswap(original);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, size_t Size>
|
template <typename T, size_t Size>
|
||||||
static constexpr inline void SwapBuffer(std::array<u8, Size> &data) {
|
static constexpr inline void SwapBuffer(std::array<u8, Size> &data) {
|
||||||
for (size_t i = 0; i < data.size(); i += sizeof(T)) {
|
for (size_t i = 0; i < data.size(); i += sizeof(T)) {
|
||||||
const T original = *reinterpret_cast<T *>(&data[i]);
|
const T original = *reinterpret_cast<T *>(&data[i]);
|
||||||
*reinterpret_cast<T *>(&data[i]) = std::byteswap(original);
|
*reinterpret_cast<T *>(&data[i]) = std::byteswap(original);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
@@ -137,10 +136,8 @@ inline void *aligned_alloc(const size_t alignment, const size_t size) { return _
|
|||||||
|
|
||||||
inline void aligned_free(void *ptr) { _aligned_free(ptr); }
|
inline void aligned_free(void *ptr) { _aligned_free(ptr); }
|
||||||
#else
|
#else
|
||||||
inline void *aligned_alloc(const size_t alignment, const size_t size) {
|
inline void *aligned_alloc(const size_t alignment, const size_t size) { return std::aligned_alloc(alignment, size); }
|
||||||
return std::aligned_alloc(alignment, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void aligned_free(void *ptr) { std::free(ptr); }
|
inline void aligned_free(void *ptr) { std::free(ptr); }
|
||||||
#endif
|
#endif
|
||||||
} // namespace Util
|
} // namespace ircolib
|
||||||
|
|||||||
@@ -1,21 +1,19 @@
|
|||||||
#include <print>
|
#include <print>
|
||||||
#include <elfio/elfio.hpp>
|
#include <elfio/elfio.hpp>
|
||||||
|
#include <ircolib/mem_access.hpp>
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
ELFIO::elfio reader;
|
ELFIO::elfio reader;
|
||||||
if (!reader.load("tests/elf/application.elf"))
|
if (!reader.load("tests/elf/application.elf"))
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
for (ELFIO::Elf_Half i = 0; i < reader.segments.size(); i++) {
|
for (const auto §ion : reader.sections) {
|
||||||
const auto &segment = reader.segments[i];
|
for (const auto &segment : reader.segments) {
|
||||||
std::println(R"(Segment type {} @ 0x{:08X} -> 0x{:08X})", segment->get_type(), segment->get_virtual_address(),
|
if (ircolib::IsInsideRange(section->get_address(), segment->get_virtual_address(),
|
||||||
segment->get_virtual_address() + segment->get_memory_size() - 1);
|
segment->get_virtual_address() + segment->get_memory_size() - 1))
|
||||||
}
|
std::println("Found section {} @ 0x{:08X} -> 0x{:08X}", section->get_index(), section->get_address(),
|
||||||
|
section->get_address() + section->get_size() - 1);
|
||||||
for (ELFIO::Elf_Half i = 1; i < reader.sections.size(); i++) {
|
}
|
||||||
const auto §ion = reader.sections[i];
|
|
||||||
std::println(R"(Section n.{} "{}": 0x{:08X} -> 0x{:08X})", i, section->get_name(), section->get_address(),
|
|
||||||
section->get_address() + section->get_size() - 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|||||||
Reference in New Issue
Block a user