Initial commit
This commit is contained in:
commit
ef40bd03c5
14 changed files with 276 additions and 0 deletions
1
.envrc
Normal file
1
.envrc
Normal file
|
|
@ -0,0 +1 @@
|
|||
use flake
|
||||
8
.gitignore
vendored
Normal file
8
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,8 @@
|
|||
# Nix
|
||||
result*
|
||||
|
||||
# Misc
|
||||
.direnv
|
||||
|
||||
.cache/
|
||||
build/
|
||||
10
CMakeLists.txt
Executable file
10
CMakeLists.txt
Executable file
|
|
@ -0,0 +1,10 @@
|
|||
cmake_minimum_required(VERSION 3.18)
|
||||
|
||||
set(CMAKE_CXX_STANDARD 23)
|
||||
|
||||
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
|
||||
# set(CMAKE_CXX_CLANG_TIDY clang-tidy -checks=-*,bugprone-*,clang-analyzer-*,concurrency-*,misc-*,performance-*,portability-*,readability-*)
|
||||
|
||||
project(executable_memory)
|
||||
|
||||
add_subdirectory(src)
|
||||
13
CMakePresets.json
Executable file
13
CMakePresets.json
Executable file
|
|
@ -0,0 +1,13 @@
|
|||
{
|
||||
"version": 1,
|
||||
"configurePresets": [
|
||||
{
|
||||
"name": "ninja-release",
|
||||
"binaryDir": "${sourceDir}/build/${presetName}",
|
||||
"generator": "Ninja",
|
||||
"cacheVariables": {
|
||||
"CMAKE_BUILD_TYPE": "Release"
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
62
flake.lock
generated
Normal file
62
flake.lock
generated
Normal file
|
|
@ -0,0 +1,62 @@
|
|||
{
|
||||
"nodes": {
|
||||
"flake-utils": {
|
||||
"inputs": {
|
||||
"systems": [
|
||||
"systems"
|
||||
]
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1731533236,
|
||||
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"nixpkgs": {
|
||||
"locked": {
|
||||
"lastModified": 1760443284,
|
||||
"narHash": "sha256-jDqjCfod2sCnsFSwTrHTU71El67OQ/XR2CDWcMR2JcY=",
|
||||
"owner": "NixOS",
|
||||
"repo": "nixpkgs",
|
||||
"rev": "2ea123939c5cb71d2761896c27e475aeab266364",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"id": "nixpkgs",
|
||||
"type": "indirect"
|
||||
}
|
||||
},
|
||||
"root": {
|
||||
"inputs": {
|
||||
"flake-utils": "flake-utils",
|
||||
"nixpkgs": "nixpkgs",
|
||||
"systems": "systems"
|
||||
}
|
||||
},
|
||||
"systems": {
|
||||
"locked": {
|
||||
"lastModified": 1680978846,
|
||||
"narHash": "sha256-Gtqg8b/v49BFDpDetjclCYXm8mAnTrUzR0JnE2nv5aw=",
|
||||
"owner": "nix-systems",
|
||||
"repo": "x86_64-linux",
|
||||
"rev": "2ecfcac5e15790ba6ce360ceccddb15ad16d08a8",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "nix-systems",
|
||||
"repo": "x86_64-linux",
|
||||
"type": "github"
|
||||
}
|
||||
}
|
||||
},
|
||||
"root": "root",
|
||||
"version": 7
|
||||
}
|
||||
43
flake.nix
Normal file
43
flake.nix
Normal file
|
|
@ -0,0 +1,43 @@
|
|||
{
|
||||
description = "Simple JIT";
|
||||
|
||||
inputs = {
|
||||
nixpkgs.url = "nixpkgs";
|
||||
systems.url = "github:nix-systems/x86_64-linux";
|
||||
flake-utils = {
|
||||
url = "github:numtide/flake-utils";
|
||||
inputs.systems.follows = "systems";
|
||||
};
|
||||
};
|
||||
|
||||
outputs =
|
||||
{ self
|
||||
, nixpkgs
|
||||
, flake-utils
|
||||
, ...
|
||||
}:
|
||||
flake-utils.lib.eachDefaultSystem (system:
|
||||
let
|
||||
pkgs = nixpkgs.legacyPackages.${system};
|
||||
pname = "simple-jit"; #package name
|
||||
version = "0.0.1";
|
||||
src = ./.;
|
||||
buildInputs = with pkgs; [
|
||||
];
|
||||
nativeBuildInputs = with pkgs; [
|
||||
ninja
|
||||
cmake
|
||||
pkg-config
|
||||
clang-tools
|
||||
];
|
||||
in
|
||||
{
|
||||
devShells.default = pkgs.mkShell {
|
||||
inherit buildInputs nativeBuildInputs;
|
||||
};
|
||||
|
||||
packages.default = pkgs.mkShell.override { stdenv = pkgs.clangStdenv; } {
|
||||
inherit buildInputs nativeBuildInputs;
|
||||
};
|
||||
});
|
||||
}
|
||||
11
include/Assembler.h
Executable file
11
include/Assembler.h
Executable file
|
|
@ -0,0 +1,11 @@
|
|||
#include <vector>
|
||||
|
||||
#include "Types.h"
|
||||
|
||||
struct Assembler {
|
||||
explicit Assembler(std::vector<u8>&);
|
||||
|
||||
std::vector<u8>& output;
|
||||
|
||||
void emit8(u8) const;
|
||||
};
|
||||
26
include/Executable.h
Executable file
26
include/Executable.h
Executable file
|
|
@ -0,0 +1,26 @@
|
|||
#include <cstddef>
|
||||
|
||||
#include "Types.h"
|
||||
#include "Noncopyable.h"
|
||||
|
||||
class Executable {
|
||||
MAKE_NONCOPYABLE(Executable);
|
||||
MAKE_NONMOVABLE(Executable);
|
||||
|
||||
public:
|
||||
Executable(void* code, std::size_t code_size);
|
||||
~Executable();
|
||||
|
||||
template<typename Return>
|
||||
Return run() const
|
||||
{
|
||||
using ExecutableFunction = Return();
|
||||
auto executable_function = reinterpret_cast<ExecutableFunction*>(m_code);
|
||||
|
||||
return executable_function();
|
||||
}
|
||||
|
||||
private:
|
||||
void* m_code;
|
||||
std::size_t m_code_size;
|
||||
};
|
||||
9
include/Noncopyable.h
Executable file
9
include/Noncopyable.h
Executable file
|
|
@ -0,0 +1,9 @@
|
|||
#define MAKE_NONCOPYABLE(c) \
|
||||
private: \
|
||||
c(c const&) = delete; \
|
||||
c& operator=(c const&) = delete
|
||||
|
||||
#define MAKE_NONMOVABLE(c) \
|
||||
private: \
|
||||
c(c&&) = delete; \
|
||||
c& operator=(c&&) = delete
|
||||
23
include/Types.h
Executable file
23
include/Types.h
Executable file
|
|
@ -0,0 +1,23 @@
|
|||
using u64 = __UINT64_TYPE__;
|
||||
using u32 = __UINT32_TYPE__;
|
||||
using u16 = __UINT16_TYPE__;
|
||||
using u8 = __UINT8_TYPE__;
|
||||
|
||||
using i64 = __INT64_TYPE__;
|
||||
using i32 = __INT32_TYPE__;
|
||||
using i16 = __INT16_TYPE__;
|
||||
using i8 = __INT8_TYPE__;
|
||||
|
||||
using f32 = float;
|
||||
static_assert(__FLT_MANT_DIG__ == 24 && __FLT_MAX_EXP__ == 128);
|
||||
|
||||
using f64 = double;
|
||||
static_assert(__DBL_MANT_DIG__ == 53 && __DBL_MAX_EXP__ == 1024);
|
||||
|
||||
#if __LDBL_MANT_DIG__ == 64 && __LDBL_MAX__EXP__ == 16384
|
||||
using f80 = long double;
|
||||
#elif __LDBL_MANT_DIG__ == 113 && __LDBL_MAX_EXP__ == 16384
|
||||
using f128 = long double;
|
||||
#endif
|
||||
|
||||
using size_t = __SIZE_TYPE__;
|
||||
13
src/Assembler.cpp
Executable file
13
src/Assembler.cpp
Executable file
|
|
@ -0,0 +1,13 @@
|
|||
#include <Assembler.h>
|
||||
|
||||
#include <vector>
|
||||
|
||||
Assembler::Assembler(std::vector<u8>& output)
|
||||
: output(output)
|
||||
{
|
||||
}
|
||||
|
||||
void Assembler::emit8(u8 value) const
|
||||
{
|
||||
output.push_back(value);
|
||||
}
|
||||
12
src/CMakeLists.txt
Executable file
12
src/CMakeLists.txt
Executable file
|
|
@ -0,0 +1,12 @@
|
|||
set(SOURCES
|
||||
Executable.cpp
|
||||
Assembler.cpp)
|
||||
|
||||
add_library(jit STATIC EXCLUDE_FROM_ALL ${SOURCES})
|
||||
target_include_directories(jit PUBLIC ${CMAKE_SOURCE_DIR}/include)
|
||||
|
||||
add_executable(executable_bytes main.cpp)
|
||||
target_link_libraries(executable_bytes jit)
|
||||
set_target_properties(executable_bytes
|
||||
PROPERTIES
|
||||
RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
|
||||
15
src/Executable.cpp
Executable file
15
src/Executable.cpp
Executable file
|
|
@ -0,0 +1,15 @@
|
|||
#include <Executable.h>
|
||||
|
||||
#include <cstddef>
|
||||
#include <sys/mman.h>
|
||||
|
||||
Executable::Executable(void* code, std::size_t code_size)
|
||||
: m_code(code)
|
||||
, m_code_size(code_size)
|
||||
{
|
||||
}
|
||||
|
||||
Executable::~Executable()
|
||||
{
|
||||
munmap(m_code, m_code_size);
|
||||
}
|
||||
30
src/main.cpp
Executable file
30
src/main.cpp
Executable file
|
|
@ -0,0 +1,30 @@
|
|||
#include <cstring>
|
||||
#include <print>
|
||||
|
||||
#include <sys/mman.h>
|
||||
|
||||
#include <Assembler.h>
|
||||
#include <Executable.h>
|
||||
|
||||
int main()
|
||||
{
|
||||
std::vector<u8> output;
|
||||
auto const assembler = Assembler(output);
|
||||
assembler.emit8(0x55); // push ebp
|
||||
assembler.emit8(0x89); assembler.emit8(0xE5); // mov ebp, esp
|
||||
assembler.emit8(0xB8); assembler.emit8(0x2A); assembler.emit8(0x00); assembler.emit8(0x00); assembler.emit8(0x00); // mov eax, 42
|
||||
assembler.emit8(0x5D); // pop ebp
|
||||
assembler.emit8(0xC3); // ret
|
||||
|
||||
auto* executable_memory = mmap(nullptr, output.size(), PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_PRIVATE, 0, 0);
|
||||
if (executable_memory == MAP_FAILED) {
|
||||
return 1;
|
||||
}
|
||||
std::memcpy(executable_memory, output.data(), output.size());
|
||||
mprotect(executable_memory, output.size(), PROT_READ | PROT_EXEC);
|
||||
|
||||
auto const executable = Executable(executable_memory, output.size());
|
||||
auto const result = executable.run<int>();
|
||||
|
||||
std::println("{}", result);
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue