lots and lots of userspace stuff
This commit is contained in:
parent
9af5588c30
commit
fbfc078e9f
74 changed files with 2324 additions and 1280 deletions
|
@ -6,7 +6,7 @@ build/%.cpp.o: source/%.cpp
|
|||
$(HILBERT_CC) -c $^ -o $@
|
||||
|
||||
build/goldman.elf: $(SOURCES:%=build/%.o)
|
||||
$(HILBERT_CC) $^ -o $@
|
||||
$(HILBERT_CC) $^ -ldaguerre -o $@
|
||||
|
||||
clean:
|
||||
rm -rf build
|
||||
|
|
|
@ -1,4 +1,79 @@
|
|||
int main(int, char **) {
|
||||
while (1)
|
||||
;
|
||||
#include <daguerre/framebuffer.hpp>
|
||||
#include <daguerre/ppm.hpp>
|
||||
|
||||
daguerre::hilbert_color trans_color;
|
||||
|
||||
void convert_pointer(
|
||||
daguerre::hilbert_color &dest, const daguerre::hilbert_color &src) {
|
||||
if (src != trans_color)
|
||||
dest = src;
|
||||
}
|
||||
|
||||
int main(int, char **) {
|
||||
|
||||
trans_color = euler::syscall::encode_color(0xff, 0x00, 0xff);
|
||||
|
||||
daguerre::image<daguerre::hilbert_color> framebuffer =
|
||||
daguerre::get_hilbert_framebuffer();
|
||||
|
||||
int fw = framebuffer.width;
|
||||
int fh = framebuffer.height;
|
||||
|
||||
daguerre::image<daguerre::hilbert_color> double_buffer(fw, fh);
|
||||
|
||||
std::optional<daguerre::image<daguerre::hilbert_color>>
|
||||
background_original = daguerre::try_load_ppm("/assets/background.ppm");
|
||||
|
||||
daguerre::image<daguerre::hilbert_color> background;
|
||||
|
||||
if (background_original.has_value())
|
||||
background = background_original->stretch(fw, fh);
|
||||
else {
|
||||
background = daguerre::image<daguerre::hilbert_color>(fw, fh);
|
||||
background.fill(euler::syscall::encode_color(0, 0, 0));
|
||||
}
|
||||
|
||||
std::optional<daguerre::image<daguerre::hilbert_color>>
|
||||
pointer_original = daguerre::try_load_ppm("/assets/pointer.ppm");
|
||||
|
||||
if (!pointer_original.has_value())
|
||||
//TODO
|
||||
while (1)
|
||||
;
|
||||
|
||||
daguerre::image<daguerre::hilbert_color>
|
||||
pointer = std::move(*pointer_original);
|
||||
|
||||
int mouse_x = fw / 2;
|
||||
int mouse_y = fh / 2;
|
||||
|
||||
while (true) {
|
||||
|
||||
double_buffer.copy_from(background, 0, 0);
|
||||
double_buffer.convert_from(pointer, mouse_x, mouse_y, 0, 0,
|
||||
std::min(pointer. width, double_buffer. width - mouse_x),
|
||||
std::min(pointer.height, double_buffer.height - mouse_y),
|
||||
&convert_pointer);
|
||||
|
||||
framebuffer.copy_from(double_buffer, 0, 0);
|
||||
|
||||
auto result = euler::syscall::get_input_packet();
|
||||
if (std::holds_alternative<euler::syscall::mouse_packet>(result)) {
|
||||
const auto &packet = std::get<euler::syscall::mouse_packet>(result);
|
||||
mouse_x += packet.x_changed;
|
||||
mouse_y += packet.y_changed;
|
||||
if (mouse_x < 0)
|
||||
mouse_x = 0;
|
||||
else if (mouse_x >= fw)
|
||||
mouse_x = fw - 1;
|
||||
if (mouse_y < 0)
|
||||
mouse_y = 0;
|
||||
else if (mouse_y >= fh)
|
||||
mouse_y = fh - 1;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
}
|
||||
|
|
|
@ -1,19 +1,11 @@
|
|||
#include <euler/start_process.hpp>
|
||||
#include <euler/syscall.hpp>
|
||||
|
||||
//this does not keep track of the processes or whether they have started
|
||||
//successfully, nor does it set their argc, argv, stdin, stdout, or stderr.
|
||||
|
||||
int main(int, char **) {
|
||||
|
||||
__euler_process_handle dummy;
|
||||
|
||||
euler::start_process wm_process("/bin/compositor");
|
||||
wm_process.add_env_variable("ARGC", "1");
|
||||
wm_process.add_env_variable("ARGV0", "/bin/compositor");
|
||||
wm_process.start(dummy);
|
||||
|
||||
euler::start_process hello_process("/bin/hello");
|
||||
hello_process.add_env_variable("ARGC", "1");
|
||||
hello_process.add_env_variable("ARGV0", "/bin/hello");
|
||||
hello_process.start(dummy);
|
||||
|
||||
euler::syscall::process_handle dummy;
|
||||
euler::syscall::start_process("/bin/compositor", {}, {}, dummy);
|
||||
euler::syscall::start_process("/bin/hello", {}, {}, dummy);
|
||||
return 0;
|
||||
|
||||
}
|
||||
|
|
|
@ -13,7 +13,7 @@ data types:
|
|||
|
||||
color:
|
||||
opaque dword (result of encode color system call).
|
||||
from c++, use __euler_encode_color in euler/syscall.hpp.
|
||||
from c++, use euler::syscall::encode_color in euler/syscall.hpp.
|
||||
|
||||
color rectangle:
|
||||
multiple hilbert colors, top to bottom by row, left to right within row
|
||||
|
|
|
@ -1,32 +0,0 @@
|
|||
this file documents dynamic memory allocation and deallocation in userspace.
|
||||
the unused areas of a process's usable mapped memory are divided into "chunks"
|
||||
with a start and a length, satisfying the following properties:
|
||||
a) the length of a chunk is a positive power of 2
|
||||
b) the start of a chunk is a multiple of its length
|
||||
c) let s be a power of 2 and k be an integer. there are never two chunks with
|
||||
length s and starts s * (2 * k) and s * (2 * k + 1). if ever an operation
|
||||
would result in two such chunks, they are combined into one chunk with
|
||||
length 2 * s and start 2 * s * k.
|
||||
|
||||
a "chunk info page" is divided into 512 64-bit integers describing up to 255
|
||||
chunks and a pointer to another chunk info page. for each n from 0 to 254:
|
||||
if the (2 * n)'th integer (where the first integer is the 0th one) is 0:
|
||||
the (2 * n + 1)'th integer is unused.
|
||||
if the (2 * n)'th integer is not 0:
|
||||
the (2 * n)'th integer describes the length of a chunk
|
||||
the (2 * n + 1)'th integer describes the start of the same chunk
|
||||
the 510th integer is a pointer to the next chunk info page, and the 511th
|
||||
integer is never used.
|
||||
|
||||
when a program calls new or malloc with needed size s, we find a free chunk of
|
||||
length at least s + 8. we then remove that chunk from the list of free chunks,
|
||||
and add back in whatever is left after the first s + 8 bytes, if anything.
|
||||
in the first 8 bytes of the original chunk, we store the value s + 8. the
|
||||
remainder of the orginal chunk is returned to the program.
|
||||
|
||||
during that process, if there isn't a chunk with the needed size, one or more
|
||||
new pages are requested from the kernel to create such a chunk.
|
||||
|
||||
when a program calls delete or free with pointer ptr, we read the integer in
|
||||
the 8 bytes starting at ptr - 8 into a variable s. we then add the region
|
||||
starting at ptr - 8 with length s to the free memory.
|
|
@ -20,6 +20,7 @@ stream result:
|
|||
12 = other end closed
|
||||
13 = already exists
|
||||
14 = not sized
|
||||
15 = not readable
|
||||
|
||||
encode color:
|
||||
rax in: 0
|
||||
|
@ -173,3 +174,27 @@ get other end process handle:
|
|||
rdi in: stream handle
|
||||
rax out: stream result
|
||||
rdi out: process handle (if rax = success)
|
||||
|
||||
start thread:
|
||||
rax in: 20
|
||||
rdi in: entry point
|
||||
rsi in: argument
|
||||
at entry, rsp is set to the top of a new stack, rdi is set
|
||||
to the argument, and the other registers are undefined.
|
||||
|
||||
clear socket read queue:
|
||||
rax in: 21
|
||||
rdi in: stream handle
|
||||
rax out: number of bytes cleared (0 on bad handle or not a socket)
|
||||
|
||||
get environment variable length:
|
||||
rax in: 22
|
||||
rdi in: pointer to variable name
|
||||
rsi in: variable name length
|
||||
rax out: variable value length, or -1 if unset
|
||||
|
||||
get environment variable value:
|
||||
rax in: 23
|
||||
rdi in: pointer to variable name
|
||||
rsi in: variable name length
|
||||
rdx in: pointer to buffer for variable value
|
||||
|
|
|
@ -3,6 +3,7 @@ panic. the following are the defined colors so far:
|
|||
#48a6ed - failed to get root node of initfs
|
||||
#5f8860 - no initfs module was given
|
||||
#7e874d - failed to look up /bin/init
|
||||
#8077a6 - pure virtual function called (i think this should never happen?)
|
||||
#9af5e6 - an unimplemented path in the kernel
|
||||
#ba40bb - cpu exception occurred
|
||||
#c39db3 - failed to parse /bin/init
|
||||
|
|
|
@ -1,12 +0,0 @@
|
|||
#pragma once
|
||||
|
||||
namespace std {
|
||||
|
||||
template <class t>
|
||||
void swap(t &a, t &b) {
|
||||
t tmp = a;
|
||||
a = b;
|
||||
b = tmp;
|
||||
}
|
||||
|
||||
}
|
3
euler/include/cctype
Normal file
3
euler/include/cctype
Normal file
|
@ -0,0 +1,3 @@
|
|||
#pragma once
|
||||
|
||||
extern "C" int isspace(int ch);
|
|
@ -1,20 +1,16 @@
|
|||
#pragma once
|
||||
|
||||
#include <euler/stream.hpp>
|
||||
#include <stddef.h>
|
||||
#include <cstddef>
|
||||
|
||||
namespace std {
|
||||
typedef euler::stream FILE;
|
||||
|
||||
typedef euler::stream FILE;
|
||||
extern "C" FILE *fopen(const char *filename, const char *mode);
|
||||
extern "C" void fclose(FILE *stream);
|
||||
|
||||
FILE *fopen(const char *filename, const char *mode);
|
||||
int fclose(FILE *stream);
|
||||
#define SEEK_CUR 2
|
||||
#define SEEK_END 1
|
||||
#define SEEK_SET 0
|
||||
|
||||
int fseek(FILE *stream, long offset, int origin);
|
||||
#define SEEK_SET 0
|
||||
#define SEEK_CUR 2
|
||||
#define SEEK_END 1
|
||||
|
||||
size_t fread(void *buffer, size_t size, size_t count, FILE *stream);
|
||||
|
||||
}
|
||||
extern "C" int fseek(FILE *stream, long offset, int origin);
|
||||
extern "C" size_t fread(void *buffer, size_t size, size_t count, FILE *stream);
|
||||
|
|
|
@ -1,8 +1,9 @@
|
|||
#pragma once
|
||||
|
||||
#include <stddef.h>
|
||||
#include <cstddef>
|
||||
|
||||
namespace std {
|
||||
size_t strlen(const char *str);
|
||||
void *memcpy(void *dest, const void *src, size_t count);
|
||||
}
|
||||
extern "C" void *memset(void *dest, int ch, size_t count);
|
||||
extern "C" void *memcpy(void *dest, const void *src, size_t count);
|
||||
|
||||
extern "C" int strcmp(const char *lhs, const char *rhs);
|
||||
extern "C" size_t strlen(const char *str);
|
||||
|
|
|
@ -1,8 +1,10 @@
|
|||
#pragma once
|
||||
|
||||
#include <stdint.h>
|
||||
#include <cstddef>
|
||||
|
||||
namespace euler::heap {
|
||||
|
||||
void *get_block(size_t length);
|
||||
void return_block(void *start, size_t length);
|
||||
|
||||
namespace euler {
|
||||
void *alloc(uint64_t bytes);
|
||||
void dealloc(void *start, uint64_t bytes);
|
||||
}
|
||||
|
|
|
@ -1,30 +0,0 @@
|
|||
#pragma once
|
||||
|
||||
#include <euler/syscall.hpp>
|
||||
#include <vector>
|
||||
|
||||
namespace euler {
|
||||
|
||||
struct start_process {
|
||||
|
||||
private:
|
||||
const char *path;
|
||||
uint64_t path_len;
|
||||
std::vector<__euler_env_var_spec> env_var_specs;
|
||||
std::vector<__euler_gift_stream_spec> gift_stream_specs;
|
||||
|
||||
public:
|
||||
//path needs to stay valid through any call to start
|
||||
start_process(const char *path);
|
||||
|
||||
//name and value need to stay valid through any call to start
|
||||
void add_env_variable(const char *name, const char *value);
|
||||
|
||||
void gift_stream(
|
||||
__euler_stream_handle to_gifter, __euler_stream_handle to_giftee);
|
||||
|
||||
__euler_stream_result start(__euler_process_handle &handle_out);
|
||||
|
||||
};
|
||||
|
||||
}
|
|
@ -7,41 +7,42 @@ namespace euler {
|
|||
class stream {
|
||||
|
||||
public:
|
||||
virtual ~stream();
|
||||
virtual bool try_read(void *into, uint64_t bytes) = 0;
|
||||
virtual bool try_seek(__euler_seek_from from, int64_t offset) = 0;
|
||||
virtual ~stream() {}
|
||||
|
||||
virtual syscall::stream_result seek(
|
||||
euler::syscall::seek_from from, int64_t offset) = 0;
|
||||
|
||||
virtual std::pair<uint64_t, syscall::stream_result>
|
||||
read(uint64_t bytes, void *into) = 0;
|
||||
|
||||
virtual void close() = 0;
|
||||
|
||||
};
|
||||
|
||||
class file_stream : public stream {
|
||||
|
||||
private:
|
||||
syscall::stream_handle handle;
|
||||
bool may_read;
|
||||
|
||||
bool buffer_loaded;
|
||||
uint64_t buffer_start;
|
||||
uint8_t buffer[1024];
|
||||
|
||||
__euler_stream_handle handle;
|
||||
bool is_readable;
|
||||
bool is_writable;
|
||||
uint64_t length;
|
||||
uint64_t offset;
|
||||
|
||||
uint8_t *buffer;
|
||||
uint64_t buffer_offset;
|
||||
uint64_t buffer_size;
|
||||
bool buffer_dirty;
|
||||
|
||||
void write_buffer();
|
||||
uint64_t position;
|
||||
|
||||
public:
|
||||
|
||||
bool good;
|
||||
|
||||
file_stream(
|
||||
__euler_stream_handle handle, bool is_readable, bool is_writable,
|
||||
bool clear, bool seek_to_end);
|
||||
syscall::stream_handle handle, bool may_read,
|
||||
uint64_t length, uint64_t position);
|
||||
|
||||
~file_stream();
|
||||
virtual syscall::stream_result seek(
|
||||
syscall::seek_from from, int64_t offset) override;
|
||||
|
||||
bool try_read(void *into, uint64_t bytes) override;
|
||||
bool try_seek(__euler_seek_from from, int64_t offset) override;
|
||||
virtual std::pair<uint64_t, syscall::stream_result>
|
||||
read(uint64_t bytes, void *into) override;
|
||||
|
||||
virtual void close() override;
|
||||
|
||||
};
|
||||
|
||||
|
|
|
@ -1,102 +1,136 @@
|
|||
#pragma once
|
||||
|
||||
#include <stdint.h>
|
||||
#include <optional>
|
||||
#include <cstdint>
|
||||
#include <utility>
|
||||
#include <variant>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
enum __euler_stream_result : uint64_t {
|
||||
__EULER_SR_SUCCESS = 0,
|
||||
__EULER_SR_BAD_HANDLE,
|
||||
__EULER_SR_IO_ERROR,
|
||||
__EULER_SR_OUT_OF_BOUNDS,
|
||||
__EULER_SR_DOES_NOT_EXIST,
|
||||
__EULER_SR_NOT_A_REGULAR_FILE,
|
||||
__EULER_SR_NOT_AN_EXECUTABLE,
|
||||
__EULER_SR_NOT_WRITABLE,
|
||||
__EULER_SR_NOT_SEEKABLE,
|
||||
__EULER_SR_SOCKET_ID_ALREADY_IN_USE,
|
||||
__EULER_SR_SOCKET_ID_NOT_IN_USE,
|
||||
__EULER_SR_SOCKET_LISTENER_CLOSED,
|
||||
__EULER_SR_OTHER_END_CLOSED,
|
||||
__EULER_SR_ALREADY_EXISTS,
|
||||
__EULER_SR_NOT_SIZED
|
||||
};
|
||||
namespace euler::syscall {
|
||||
|
||||
enum __euler_seek_from : uint8_t {
|
||||
__EULER_SF_BEGINNING = 0,
|
||||
__EULER_SF_END,
|
||||
__EULER_SF_CURRENT_POSITION
|
||||
};
|
||||
enum class stream_result : uint64_t {
|
||||
success,
|
||||
bad_handle,
|
||||
io_error,
|
||||
out_of_bounds,
|
||||
does_not_exist,
|
||||
not_a_regular_file,
|
||||
not_an_executable,
|
||||
not_writable,
|
||||
not_seekable,
|
||||
socket_id_already_used,
|
||||
socket_id_not_in_use,
|
||||
socket_listener_closed,
|
||||
other_end_closed,
|
||||
already_exists,
|
||||
not_sized,
|
||||
not_readable
|
||||
};
|
||||
|
||||
typedef uint64_t __euler_stream_handle;
|
||||
typedef uint64_t __euler_process_handle;
|
||||
enum class seek_from : uint8_t {
|
||||
beginning,
|
||||
end,
|
||||
current_position
|
||||
};
|
||||
|
||||
extern "C" __euler_stream_result __euler_open_file(
|
||||
const char *path, uint64_t path_len, __euler_stream_handle &handle_out,
|
||||
bool allow_creation, bool only_allow_creation);
|
||||
struct mouse_packet {
|
||||
bool left_button_down;
|
||||
bool right_button_down;
|
||||
bool middle_button_down;
|
||||
int16_t x_changed;
|
||||
int16_t y_changed;
|
||||
};
|
||||
|
||||
extern "C" [[noreturn]] void __euler_end_this_thread(int32_t exit_code);
|
||||
struct key_packet {
|
||||
bool was_key_up_event;
|
||||
bool num_lock;
|
||||
bool caps_lock;
|
||||
bool right_win;
|
||||
bool left_win;
|
||||
bool right_alt;
|
||||
bool left_alt;
|
||||
bool right_ctrl;
|
||||
bool left_ctrl;
|
||||
bool right_shift;
|
||||
bool left_shift;
|
||||
uint8_t key_code;
|
||||
};
|
||||
|
||||
extern "C" __euler_stream_result __euler_seek_stream(
|
||||
__euler_stream_handle handle, __euler_seek_from from, int64_t offset);
|
||||
typedef uint32_t encoded_color;
|
||||
typedef uint64_t stream_handle;
|
||||
typedef uint64_t listener_handle;
|
||||
typedef uint64_t process_handle;
|
||||
|
||||
extern "C" __euler_stream_result __euler_set_stream_length(
|
||||
__euler_stream_handle handle, uint64_t new_length);
|
||||
encoded_color encode_color(
|
||||
uint8_t r, uint8_t g, uint8_t b);
|
||||
|
||||
extern "C" void __euler_close_stream(__euler_stream_handle handle);
|
||||
void get_framebuffer(
|
||||
encoded_color *&buffer_out, uint32_t &width_out,
|
||||
uint32_t &height_out, uint32_t &pitch_out);
|
||||
|
||||
extern "C" __euler_stream_result __euler_get_stream_length(
|
||||
__euler_stream_handle handle, uint64_t &length_out);
|
||||
stream_result open_file(
|
||||
const std::string &file_path, bool allow_creation,
|
||||
bool only_allow_creation, stream_handle &handle_out);
|
||||
|
||||
extern "C" void *__euler_get_new_pages(uint64_t count);
|
||||
[[noreturn]] void end_this_thread(int32_t exit_code);
|
||||
|
||||
extern "C" __euler_stream_result __euler_write_to_stream(
|
||||
__euler_stream_handle handle, uint64_t count, const void *buffer);
|
||||
void *get_new_pages(uint64_t n_pages);
|
||||
|
||||
extern "C" __euler_stream_result __euler_read_from_stream(
|
||||
__euler_stream_handle handle, uint64_t count, void *buffer);
|
||||
std::variant<mouse_packet, key_packet> get_input_packet();
|
||||
|
||||
extern "C" uint32_t *__euler_get_framebuffer(
|
||||
uint32_t &width_out, uint32_t &height_out, uint32_t &pitch_out);
|
||||
void create_private_socket(
|
||||
stream_handle &end_1_out, stream_handle &end_2_out);
|
||||
|
||||
extern "C" uint32_t __euler_encode_color(uint8_t r, uint8_t g, uint8_t b);
|
||||
stream_result create_socket_listener(
|
||||
const std::string &id, listener_handle &handle_out);
|
||||
|
||||
enum __euler_mouse_buttons : uint8_t {
|
||||
__EULER_MB_LEFT = 1,
|
||||
__EULER_MB_RIGHT = 2,
|
||||
__EULER_MB_MIDDLE = 4
|
||||
};
|
||||
void stop_socket_listener(listener_handle handle);
|
||||
|
||||
enum __euler_input_packet_type : uint8_t {
|
||||
__EULER_IPT_MOUSE = 1,
|
||||
__EULER_IPT_KEYBOARD = 2
|
||||
};
|
||||
stream_result accept_socket_connection(
|
||||
listener_handle listener, stream_handle &stream_out);
|
||||
|
||||
extern "C" __euler_input_packet_type __euler_get_input_packet(
|
||||
__euler_mouse_buttons &buttons_out, int16_t &x_change_out,
|
||||
int16_t &y_change_out, uint32_t &keyboard_packet_out);
|
||||
stream_result connect_to_socket(
|
||||
const std::string &id, stream_handle &handle_out);
|
||||
|
||||
struct [[gnu::packed]] __euler_env_var_spec {
|
||||
uint64_t name_len;
|
||||
const char *name;
|
||||
uint64_t value_len;
|
||||
const char *value;
|
||||
};
|
||||
void close_stream(stream_handle handle);
|
||||
|
||||
struct [[gnu::packed]] __euler_gift_stream_spec {
|
||||
__euler_stream_handle stream_handle_to_gifter;
|
||||
__euler_stream_handle stream_handle_to_giftee;
|
||||
};
|
||||
stream_result seek_stream(
|
||||
stream_handle handle, seek_from from, int64_t offset);
|
||||
|
||||
struct [[gnu::packed]] __euler_process_start_info {
|
||||
uint64_t file_path_length;
|
||||
const char *file_path;
|
||||
uint64_t env_var_count;
|
||||
const __euler_env_var_spec *env_vars;
|
||||
uint64_t gift_stream_count;
|
||||
const __euler_gift_stream_spec *gift_streams;
|
||||
};
|
||||
stream_result read_from_stream(
|
||||
stream_handle handle, uint64_t bytes, void *into);
|
||||
|
||||
extern "C" __euler_stream_result __euler_start_process(
|
||||
const __euler_process_start_info &info, __euler_process_handle &handle_out);
|
||||
stream_result write_to_stream(
|
||||
stream_handle handle, uint64_t bytes, const void *from);
|
||||
|
||||
extern "C" __euler_stream_result __euler_get_other_end_process_handle(
|
||||
__euler_stream_handle handle_in, __euler_process_handle &handle_out);
|
||||
stream_result get_stream_length(
|
||||
stream_handle handle, uint64_t &length_out);
|
||||
|
||||
stream_result start_process(
|
||||
const std::string &file_path,
|
||||
const std::vector<std::pair<std::string, std::string>> &environment_variables,
|
||||
const std::vector<std::pair<stream_handle, stream_result>> &gifted_streams,
|
||||
process_handle &handle_out);
|
||||
|
||||
[[noreturn]] void end_this_process(int32_t exit_code);
|
||||
|
||||
stream_result set_stream_length(
|
||||
stream_handle handle, uint64_t new_length);
|
||||
|
||||
stream_result get_other_end_process_handle(
|
||||
stream_handle stream, process_handle &process_out);
|
||||
|
||||
//entry_point must not return
|
||||
void start_thread(void (*entry_point)(uint64_t), uint64_t arg);
|
||||
|
||||
//entry_point must not return
|
||||
void start_thread(void (*entry_point)());
|
||||
|
||||
//return value is number of bytes cleared
|
||||
uint64_t clear_socket_read_queue(stream_handle handle);
|
||||
|
||||
std::optional<std::string> try_get_environment_variable(
|
||||
const std::string &name);
|
||||
|
||||
}
|
||||
|
|
30
euler/include/std/allocator.hpp
Normal file
30
euler/include/std/allocator.hpp
Normal file
|
@ -0,0 +1,30 @@
|
|||
#pragma once
|
||||
|
||||
#include <euler/heap.hpp>
|
||||
|
||||
namespace std {
|
||||
|
||||
template <class T>
|
||||
struct allocator {
|
||||
|
||||
using value_type = T;
|
||||
|
||||
constexpr allocator() noexcept {}
|
||||
constexpr allocator(const allocator &other) noexcept {}
|
||||
|
||||
template <class U>
|
||||
constexpr allocator(const allocator<U> &other) noexcept {}
|
||||
|
||||
constexpr ~allocator() {}
|
||||
|
||||
constexpr T *allocate(size_t n) {
|
||||
return (T *)euler::heap::get_block(n * sizeof(T));
|
||||
}
|
||||
|
||||
constexpr void deallocate(T *p, size_t n) {
|
||||
euler::heap::return_block(p, n * sizeof(T));
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
80
euler/include/std/string.hpp
Normal file
80
euler/include/std/string.hpp
Normal file
|
@ -0,0 +1,80 @@
|
|||
#pragma once
|
||||
|
||||
#include <cstddef>
|
||||
#include <vector>
|
||||
|
||||
namespace std {
|
||||
|
||||
class string {
|
||||
|
||||
std::vector<char> characters;
|
||||
|
||||
public:
|
||||
constexpr string() : characters({'\0'}) {}
|
||||
|
||||
constexpr string(const string &other)
|
||||
: characters(other.characters) {}
|
||||
|
||||
constexpr string(string &&other) noexcept
|
||||
: characters(std::move(other.characters)) {
|
||||
other.characters.push_back('\0');
|
||||
}
|
||||
|
||||
constexpr string(const char *s) {
|
||||
size_t count = 0;
|
||||
while (s[count] != '\0')
|
||||
++count;
|
||||
characters.resize(count + 1);
|
||||
for (size_t i = 0; i <= count; ++i)
|
||||
characters[i] = s[i];
|
||||
}
|
||||
|
||||
constexpr ~string() {}
|
||||
|
||||
constexpr string &operator =(const string &str) {
|
||||
characters = str.characters;
|
||||
return *this;
|
||||
}
|
||||
|
||||
constexpr string &operator =(string &&str) noexcept {
|
||||
characters = std::move(str.characters);
|
||||
str.characters.push_back('\0');
|
||||
return *this;
|
||||
}
|
||||
|
||||
constexpr size_t size() const noexcept {
|
||||
return characters.size() - 1;
|
||||
}
|
||||
|
||||
constexpr void resize(size_t count) {
|
||||
if (count < characters.size() - 1) {
|
||||
characters.resize(count + 1);
|
||||
characters[count] = '\0';
|
||||
}
|
||||
else if (count > characters.size() - 1)
|
||||
characters.resize(count + 1);
|
||||
}
|
||||
|
||||
constexpr void clear() noexcept {
|
||||
resize(0);
|
||||
}
|
||||
|
||||
constexpr const char *data() const noexcept {
|
||||
return characters.data();
|
||||
}
|
||||
|
||||
constexpr char *data() noexcept {
|
||||
return characters.data();
|
||||
}
|
||||
|
||||
constexpr char &operator[](size_t pos) {
|
||||
return characters[pos];
|
||||
}
|
||||
|
||||
constexpr const char &operator[](size_t pos) const {
|
||||
return characters[pos];
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
191
euler/include/std/vector.hpp
Normal file
191
euler/include/std/vector.hpp
Normal file
|
@ -0,0 +1,191 @@
|
|||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
|
||||
namespace std {
|
||||
|
||||
template <class T, class Allocator = std::allocator<T>>
|
||||
class vector {
|
||||
|
||||
public:
|
||||
using size_type = size_t;
|
||||
using value_type = T;
|
||||
using reference = value_type &;
|
||||
using const_reference = const value_type &;
|
||||
|
||||
private:
|
||||
Allocator _alloc;
|
||||
size_type _capacity;
|
||||
size_type _size;
|
||||
T *_data;
|
||||
|
||||
constexpr void expand_capacity_to(size_type at_least) {
|
||||
|
||||
size_type new_capacity = _capacity == 0 ? 1 : _capacity;
|
||||
while (new_capacity < at_least)
|
||||
new_capacity *= 2;
|
||||
if (new_capacity == _capacity)
|
||||
return;
|
||||
|
||||
T *new_data = _alloc.allocate(new_capacity);
|
||||
for (size_type i = 0; i < _size; ++i) {
|
||||
std::construct_at(new_data + i, std::move(_data[i]));
|
||||
std::destroy_at(_data + i);
|
||||
}
|
||||
if (_capacity > 0)
|
||||
_alloc.deallocate(_data, _capacity);
|
||||
|
||||
_capacity = new_capacity;
|
||||
_data = new_data;
|
||||
|
||||
}
|
||||
|
||||
public:
|
||||
constexpr vector() noexcept(noexcept(Allocator()))
|
||||
: _alloc(), _capacity(0), _size(0), _data(0) {}
|
||||
|
||||
constexpr vector(const vector &other)
|
||||
: _alloc(), _capacity(other._size), _size(other._size),
|
||||
_data(_alloc.allocate(_capacity)) {
|
||||
for (size_type i = 0; i < _size; ++i)
|
||||
std::construct_at(_data + i, other._data[i]);
|
||||
}
|
||||
|
||||
constexpr vector(vector &&other) noexcept
|
||||
: _alloc(std::move(other._alloc)), _capacity(other._capacity),
|
||||
_size(other._size), _data(other._data) {
|
||||
other._capacity = 0;
|
||||
other._size = 0;
|
||||
}
|
||||
|
||||
constexpr vector(
|
||||
std::initializer_list<T> init, const Allocator &alloc = Allocator())
|
||||
: _alloc(alloc), _capacity(init.size()),
|
||||
_size(init.size()), _data(_alloc.allocate(_capacity)) {
|
||||
for (size_type i = 0; i < _size; ++i)
|
||||
std::construct_at(_data + i, init.begin()[i]);
|
||||
}
|
||||
|
||||
explicit vector(size_type count)
|
||||
: _alloc(), _capacity(count), _size(count),
|
||||
_data(_alloc.allocate(_capacity)) {
|
||||
for (size_type i = 0; i < _size; ++i)
|
||||
std::construct_at(_data + i);
|
||||
}
|
||||
|
||||
constexpr ~vector() {
|
||||
if (_capacity > 0) {
|
||||
for (size_type i = 0; i < _size; ++i)
|
||||
std::destroy_at(_data + i);
|
||||
_alloc.deallocate(_data, _capacity);
|
||||
}
|
||||
}
|
||||
|
||||
constexpr vector &operator =(const vector &other) {
|
||||
|
||||
if (&other == this)
|
||||
return *this;
|
||||
|
||||
expand_capacity_to(other._size);
|
||||
|
||||
for (size_type i = 0; i < _size; ++i)
|
||||
std::destroy_at(_data + i);
|
||||
|
||||
for (size_type i = 0; i < other._size; ++i) {
|
||||
std::construct_at(_data + i, std::move(other._data[i]));
|
||||
std::destroy_at(other._data + i);
|
||||
}
|
||||
|
||||
_size = other._size;
|
||||
return *this;
|
||||
|
||||
}
|
||||
|
||||
constexpr vector &operator =(vector &&other)
|
||||
noexcept(
|
||||
std::allocator_traits<Allocator>::
|
||||
propagate_on_container_move_assignment::value ||
|
||||
std::allocator_traits<Allocator>::
|
||||
is_always_equal::value) {
|
||||
|
||||
if (&other == this)
|
||||
return *this;
|
||||
|
||||
if (_capacity > 0) {
|
||||
for (size_type i = 0; i < _size; ++i)
|
||||
std::destroy_at(_data + i);
|
||||
_alloc.deallocate(_data, _capacity);
|
||||
}
|
||||
|
||||
_alloc = std::move(other._alloc);
|
||||
_capacity = other._capacity;
|
||||
_size = other._size;
|
||||
_data = other._data;
|
||||
|
||||
other._capacity = 0;
|
||||
other._size = 0;
|
||||
|
||||
return *this;
|
||||
|
||||
}
|
||||
|
||||
constexpr size_type size() const noexcept {
|
||||
return _size;
|
||||
}
|
||||
|
||||
constexpr T *data() noexcept {
|
||||
return _data;
|
||||
}
|
||||
|
||||
constexpr const T *data() const noexcept {
|
||||
return _data;
|
||||
}
|
||||
|
||||
constexpr reference operator[](size_type pos) {
|
||||
return _data[pos];
|
||||
}
|
||||
|
||||
constexpr const_reference operator[](size_type pos) const {
|
||||
return _data[pos];
|
||||
}
|
||||
|
||||
constexpr reference back() {
|
||||
return _data[_size - 1];
|
||||
}
|
||||
|
||||
constexpr const_reference back() const {
|
||||
return _data[_size - 1];
|
||||
}
|
||||
|
||||
constexpr void resize(size_type count) {
|
||||
|
||||
if (count < _size) {
|
||||
for (size_type i = count; i < _size; ++i)
|
||||
std::destroy_at(_data + i);
|
||||
_size = count;
|
||||
}
|
||||
|
||||
else if (count > _size) {
|
||||
expand_capacity_to(count);
|
||||
for (size_type i = _size; i < count; ++i)
|
||||
std::construct_at(_data + i);
|
||||
_size = count;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
constexpr void push_back(const T &value) {
|
||||
expand_capacity_to(_size + 1);
|
||||
std::construct_at(_data + _size, value);
|
||||
++_size;
|
||||
}
|
||||
|
||||
constexpr void push_back(T &&value) {
|
||||
expand_capacity_to(_size + 1);
|
||||
std::construct_at(_data + _size, std::move(value));
|
||||
++_size;
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
}
|
13
euler/include/string
Normal file
13
euler/include/string
Normal file
|
@ -0,0 +1,13 @@
|
|||
#pragma once
|
||||
|
||||
#include <std/string.hpp>
|
||||
|
||||
namespace std {
|
||||
|
||||
int stoi(const std::string &str, size_t *pos = 0, int base = 10);
|
||||
|
||||
std::string to_string(int value);
|
||||
|
||||
std::string operator +(std::string &&lhs, std::string &&rhs);
|
||||
|
||||
}
|
|
@ -1,23 +0,0 @@
|
|||
#pragma once
|
||||
|
||||
namespace std {
|
||||
|
||||
template <class t>
|
||||
struct remove_reference {
|
||||
typedef t type;
|
||||
};
|
||||
|
||||
template <class t>
|
||||
struct remove_reference<t &> {
|
||||
typedef t type;
|
||||
};
|
||||
|
||||
template <class t>
|
||||
struct remove_reference<t &&> {
|
||||
typedef t type;
|
||||
};
|
||||
|
||||
template <class t>
|
||||
using remove_reference_t = typename remove_reference<t>::type;
|
||||
|
||||
}
|
|
@ -1,12 +0,0 @@
|
|||
#pragma once
|
||||
|
||||
#include <type_traits>
|
||||
|
||||
namespace std {
|
||||
|
||||
template <class t>
|
||||
constexpr std::remove_reference_t<t> &&move(t &&x) {
|
||||
return static_cast<std::remove_reference_t<t> &&>(x);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,62 +1,3 @@
|
|||
#pragma once
|
||||
|
||||
#include <stddef.h>
|
||||
#include <utility>
|
||||
|
||||
namespace std {
|
||||
|
||||
template <class t>
|
||||
class vector {
|
||||
|
||||
t *buffer;
|
||||
size_t buffer_length;//always positive
|
||||
size_t count;
|
||||
|
||||
public:
|
||||
vector() : buffer(new t[16]), buffer_length(16), count(0) {}
|
||||
|
||||
~vector() {
|
||||
delete[] buffer;
|
||||
}
|
||||
|
||||
t &operator[](size_t pos) {
|
||||
return buffer[pos];
|
||||
}
|
||||
|
||||
const t &operator[](size_t pos) const {
|
||||
return buffer[pos];
|
||||
}
|
||||
|
||||
t *data() {
|
||||
return buffer;
|
||||
}
|
||||
|
||||
const t *data() const {
|
||||
return buffer;
|
||||
}
|
||||
|
||||
size_t size() const {
|
||||
return count;
|
||||
}
|
||||
|
||||
void reserve(size_t new_length) {
|
||||
if (new_length <= buffer_length)
|
||||
return;
|
||||
t *new_buffer = new t[new_length];
|
||||
for (size_t i = 0; i < count; ++i)
|
||||
new_buffer[i] = std::move(buffer[i]);
|
||||
delete[] buffer;
|
||||
buffer = new_buffer;
|
||||
buffer_length = new_length;
|
||||
}
|
||||
|
||||
void push_back(t &&value) {
|
||||
if (count == buffer_length)
|
||||
reserve(count * 2);
|
||||
buffer[count++] = std::move(value);
|
||||
}
|
||||
|
||||
//TODO
|
||||
};
|
||||
|
||||
}
|
||||
#include <std/vector.hpp>
|
||||
|
|
|
@ -1,7 +1,6 @@
|
|||
LIBSTDCPP_SOURCES = euler/stream.cpp strings/strlen.cpp euler/syscall.asm \
|
||||
euler/entry.cpp io/fopen.cpp euler/gcc.asm memory/delete.cpp euler/heap.cpp \
|
||||
memory/new.cpp io/fclose.cpp io/fread.cpp strings/memcpy.cpp io/fseek.cpp \
|
||||
euler/start_process.cpp
|
||||
LIBC_SOURCES = \
|
||||
entry.cpp std/string.cpp std/cstring.cpp syscall.cpp std/cstdlib.cpp \
|
||||
heap.cpp syscall.asm std/cctype.cpp std/cstdio.cpp stream.cpp
|
||||
|
||||
clean:
|
||||
rm -rf build
|
||||
|
@ -12,12 +11,12 @@ build/%.asm.o: source/%.asm
|
|||
|
||||
build/%.cpp.o: source/%.cpp
|
||||
@mkdir -p $(@D)
|
||||
$(HILBERT_CC) -c $^ -o $@
|
||||
$(HILBERT_CC) -ffreestanding -c $^ -o $@
|
||||
|
||||
build/crt0.o: build/empty.asm.o
|
||||
cp $^ $@
|
||||
|
||||
build/libc.a: build/empty.asm.o
|
||||
build/libc.a: ${LIBC_SOURCES:%=build/%.o}
|
||||
$(HILBERT_AR) rcs $@ $^
|
||||
|
||||
build/libg.a: build/empty.asm.o
|
||||
|
@ -25,6 +24,3 @@ build/libg.a: build/empty.asm.o
|
|||
|
||||
build/libm.a: build/empty.asm.o
|
||||
$(HILBERT_AR) rcs $@ $^
|
||||
|
||||
build/libstdc++.a: ${LIBSTDCPP_SOURCES:%=build/%.o}
|
||||
$(HILBERT_AR) rcs $@ $^
|
||||
|
|
28
euler/source/entry.cpp
Normal file
28
euler/source/entry.cpp
Normal file
|
@ -0,0 +1,28 @@
|
|||
#include <euler/syscall.hpp>
|
||||
#include <cstdlib>
|
||||
#include <string>
|
||||
|
||||
int main(int argc, char **argv);
|
||||
|
||||
extern "C" [[noreturn]] void _start() {
|
||||
|
||||
auto argc_raw = euler::syscall::try_get_environment_variable("ARGC");
|
||||
int argc = argc_raw.has_value() ? std::stoi(argc_raw.value()) : 0;
|
||||
|
||||
std::vector<std::string> argv;
|
||||
|
||||
for (int i = 0; i < argc; ++i) {
|
||||
std::string arg_name = std::string("ARGV") + std::to_string(i);
|
||||
auto arg_raw = euler::syscall::try_get_environment_variable(arg_name);
|
||||
argv.push_back(arg_raw.has_value() ? arg_raw.value() : "");
|
||||
}
|
||||
|
||||
std::vector<char *> c_argv(argc);
|
||||
for (int i = 0; i < argc; ++i)
|
||||
c_argv[i] = argv[i].data();
|
||||
|
||||
int exit_code = main(argc, c_argv.data());
|
||||
|
||||
euler::syscall::end_this_process(exit_code);
|
||||
|
||||
}
|
|
@ -1,11 +0,0 @@
|
|||
#include <euler/syscall.hpp>
|
||||
|
||||
int main(int argc, char **argv);
|
||||
|
||||
extern "C" [[noreturn]] void _start() {
|
||||
//TODO: call global constructors
|
||||
//TODO: get argc, argv from environment
|
||||
int exit_code = main(0, 0);
|
||||
//TODO: call global destructors
|
||||
__euler_end_this_thread(exit_code);
|
||||
}
|
|
@ -1,53 +0,0 @@
|
|||
bits 64
|
||||
|
||||
global strlen
|
||||
|
||||
section .text
|
||||
|
||||
strlen:
|
||||
xor rax, rax
|
||||
.loop:
|
||||
mov rdx, qword [rdi]
|
||||
test edx, 0xff
|
||||
jz .plus0
|
||||
test edx, 0xff00
|
||||
jz .plus1
|
||||
test edx, 0xff0000
|
||||
jz .plus2
|
||||
test edx, 0xff000000
|
||||
jz .plus3
|
||||
shr rdx, 32
|
||||
test edx, 0xff
|
||||
jz .plus4
|
||||
test edx, 0xff00
|
||||
jz .plus5
|
||||
test edx, 0xff0000
|
||||
jz .plus6
|
||||
test edx, 0xff000000
|
||||
jz .plus7
|
||||
add rax, 8
|
||||
add rdi, 8
|
||||
jmp .loop
|
||||
.plus0:
|
||||
ret
|
||||
.plus1:
|
||||
add rax, 1
|
||||
ret
|
||||
.plus2:
|
||||
add rax, 2
|
||||
ret
|
||||
.plus3:
|
||||
add rax, 3
|
||||
ret
|
||||
.plus4:
|
||||
add rax, 4
|
||||
ret
|
||||
.plus5:
|
||||
add rax, 5
|
||||
ret
|
||||
.plus6:
|
||||
add rax, 6
|
||||
ret
|
||||
.plus7:
|
||||
add rax, 7
|
||||
ret
|
|
@ -1,66 +0,0 @@
|
|||
#include <euler/syscall.hpp>
|
||||
#include <euler/heap.hpp>
|
||||
|
||||
namespace euler {
|
||||
|
||||
static uint64_t *last_chunk_info_page = 0;
|
||||
|
||||
void add_record(uint64_t start, uint64_t length) {
|
||||
uint64_t dual_start = start ^ length;
|
||||
for (uint64_t *cip = last_chunk_info_page; cip; cip = (uint64_t *)cip[510])
|
||||
for (int i = 0; i < 255; ++i)
|
||||
if (cip[i * 2] == length && cip[i * 2 + 1] == dual_start) {
|
||||
cip[i * 2] = length * 2;
|
||||
cip[i * 2 + 1] &= start;
|
||||
return;
|
||||
}
|
||||
for (uint64_t *cip = last_chunk_info_page; cip; cip = (uint64_t *)cip[510])
|
||||
for (int i = 0; i < 255; ++i)
|
||||
if (cip[i * 2] == 0) {
|
||||
cip[i * 2] = length;
|
||||
cip[i * 2 + 1] = start;
|
||||
return;
|
||||
}
|
||||
uint64_t *new_cip = (uint64_t *)__euler_get_new_pages(1);
|
||||
new_cip[0] = length;
|
||||
new_cip[1] = start;
|
||||
for (int i = 1; i < 255; ++i)
|
||||
new_cip[i * 2] = 0;
|
||||
new_cip[510] = (uint64_t)last_chunk_info_page;
|
||||
last_chunk_info_page = new_cip;
|
||||
}
|
||||
|
||||
void *alloc(uint64_t bytes) {
|
||||
for (uint64_t *cip = last_chunk_info_page; cip; cip = (uint64_t *)cip[510])
|
||||
for (int i = 0; i < 255; ++i)
|
||||
if (cip[i * 2] >= bytes) {
|
||||
uint64_t start = cip[i * 2 + 1];
|
||||
uint64_t extra_length = cip[i * 2] - bytes;
|
||||
cip[i * 2] = 0;
|
||||
if (extra_length > 0)
|
||||
dealloc((void *)(start + bytes), extra_length);
|
||||
return (void *)start;
|
||||
}
|
||||
uint64_t pages_needed = (bytes - 1) / 4096 + 1;
|
||||
uint64_t start = (uint64_t)__euler_get_new_pages(pages_needed);
|
||||
if (bytes != pages_needed * 4096)
|
||||
dealloc((void *)(start + bytes), pages_needed * 4096 - bytes);
|
||||
return (void *)start;
|
||||
}
|
||||
|
||||
//this implementation is a little lazy
|
||||
void dealloc(void *start, uint64_t bytes) {
|
||||
uint64_t s = (uint64_t)start;
|
||||
while (bytes > 0) {
|
||||
uint64_t l = 1;
|
||||
do
|
||||
l *= 2;
|
||||
while (l <= bytes && s % l == 0);
|
||||
l /= 2;
|
||||
add_record(s, l);
|
||||
s += l;
|
||||
bytes -= l;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -1,35 +0,0 @@
|
|||
#include <euler/start_process.hpp>
|
||||
#include <cstring>
|
||||
|
||||
namespace euler {
|
||||
|
||||
start_process::start_process(const char *path)
|
||||
: path(path), path_len(std::strlen(path)) {}
|
||||
|
||||
void start_process::add_env_variable(const char *name, const char *value) {
|
||||
env_var_specs.push_back({
|
||||
.name_len = std::strlen(name), .name = name,
|
||||
.value_len = std::strlen(value), .value = value });
|
||||
}
|
||||
|
||||
void start_process::gift_stream(
|
||||
__euler_stream_handle to_gifter, __euler_stream_handle to_giftee) {
|
||||
gift_stream_specs.push_back({
|
||||
.stream_handle_to_gifter = to_gifter,
|
||||
.stream_handle_to_giftee = to_giftee });
|
||||
}
|
||||
|
||||
__euler_stream_result start_process::start(
|
||||
__euler_process_handle &handle_out) {
|
||||
__euler_process_start_info info = {
|
||||
.file_path_length = path_len,
|
||||
.file_path = path,
|
||||
.env_var_count = env_var_specs.size(),
|
||||
.env_vars = env_var_specs.data(),
|
||||
.gift_stream_count = gift_stream_specs.size(),
|
||||
.gift_streams = gift_stream_specs.data(),
|
||||
};
|
||||
return __euler_start_process(info, handle_out);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,151 +0,0 @@
|
|||
#include <euler/stream.hpp>
|
||||
#include <cstring>
|
||||
|
||||
namespace euler {
|
||||
|
||||
stream::~stream() {}
|
||||
|
||||
void file_stream::write_buffer() {
|
||||
if (__euler_write_to_stream(handle, buffer_size, buffer)
|
||||
!= __EULER_SR_SUCCESS)
|
||||
good = false;//TODO: more precise error reporting
|
||||
else
|
||||
buffer_dirty = false;
|
||||
}
|
||||
|
||||
file_stream::file_stream(
|
||||
__euler_stream_handle handle, bool is_readable, bool is_writable,
|
||||
bool clear, bool seek_to_end)
|
||||
|
||||
: handle(handle), is_readable(is_readable), is_writable(is_writable),
|
||||
buffer(0), good(true) {
|
||||
|
||||
if (clear) {
|
||||
if (__euler_set_stream_length(handle, 0) != __EULER_SR_SUCCESS) {
|
||||
good = false;
|
||||
return;
|
||||
}
|
||||
length = 0;
|
||||
}
|
||||
|
||||
else if (__euler_get_stream_length(handle, length) != __EULER_SR_SUCCESS) {
|
||||
good = false;
|
||||
return;
|
||||
}
|
||||
|
||||
if (seek_to_end) {
|
||||
if (__euler_seek_stream(handle, __EULER_SF_END, 0)
|
||||
!= __EULER_SR_SUCCESS) {
|
||||
good = false;
|
||||
return;
|
||||
}
|
||||
offset = length;
|
||||
}
|
||||
|
||||
else
|
||||
offset = 0;
|
||||
|
||||
}
|
||||
|
||||
file_stream::~file_stream() {
|
||||
if (buffer) {
|
||||
if (buffer_dirty)
|
||||
write_buffer();
|
||||
delete[] buffer;
|
||||
}
|
||||
__euler_close_stream(handle);
|
||||
}
|
||||
|
||||
bool file_stream::try_read(void *into, uint64_t bytes) {
|
||||
|
||||
if (bytes == 0)
|
||||
return true;
|
||||
|
||||
if (offset + bytes > length)
|
||||
return false;
|
||||
|
||||
if (buffer) {
|
||||
uint64_t start = offset > buffer_offset ? offset : buffer_offset;
|
||||
uint64_t end = offset + bytes < buffer_offset + buffer_size
|
||||
? offset + bytes : buffer_offset + buffer_size;
|
||||
if (end > start) {
|
||||
|
||||
uint64_t real_end = offset + bytes;
|
||||
|
||||
std::memcpy(
|
||||
(uint8_t *)into + start - offset, buffer + start - buffer_offset,
|
||||
end - start);
|
||||
if (start != offset)
|
||||
if (!try_read(into, start - offset))
|
||||
return false;
|
||||
|
||||
if (end != real_end) {
|
||||
if (!try_seek(__EULER_SF_BEGINNING, end))
|
||||
return false;
|
||||
if (!try_read((uint8_t *)into + end, real_end - end))
|
||||
return false;
|
||||
}
|
||||
|
||||
else if (offset != real_end)
|
||||
if (!try_seek(__EULER_SF_BEGINNING, real_end))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
if (buffer_dirty) {
|
||||
write_buffer();
|
||||
if (!good)
|
||||
return false;
|
||||
}
|
||||
|
||||
//1024 is arbitrary
|
||||
buffer_size = length - offset < 1024 ? length - offset : 1024;
|
||||
|
||||
if (buffer != 0)
|
||||
delete[] buffer;
|
||||
|
||||
buffer = new uint8_t[buffer_size];
|
||||
buffer_dirty = false;
|
||||
buffer_offset = offset;
|
||||
|
||||
if (__euler_read_from_stream(handle, buffer_size, buffer)
|
||||
!= __EULER_SR_SUCCESS) {
|
||||
delete[] buffer;
|
||||
buffer = 0;
|
||||
return false;
|
||||
}
|
||||
|
||||
uint64_t read_this_buffer = buffer_size < bytes ? buffer_size : bytes;
|
||||
std::memcpy(into, buffer, read_this_buffer);
|
||||
offset += read_this_buffer;
|
||||
|
||||
if (read_this_buffer == bytes)
|
||||
return true;
|
||||
|
||||
return try_read(
|
||||
(uint8_t *)into + read_this_buffer, bytes - read_this_buffer);
|
||||
|
||||
}
|
||||
|
||||
bool file_stream::try_seek(__euler_seek_from from, int64_t offset) {
|
||||
if (__euler_seek_stream(handle, from, offset) != __EULER_SR_SUCCESS)
|
||||
return false;
|
||||
switch (from) {
|
||||
case __EULER_SF_BEGINNING:
|
||||
this->offset = offset;
|
||||
return true;
|
||||
case __EULER_SF_END:
|
||||
this->offset = length + offset;
|
||||
return true;
|
||||
case __EULER_SF_CURRENT_POSITION:
|
||||
this->offset += offset;
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -1,144 +0,0 @@
|
|||
bits 64
|
||||
|
||||
global __euler_open_file
|
||||
global __euler_close_stream
|
||||
global __euler_get_new_pages
|
||||
global __euler_write_to_stream
|
||||
global __euler_seek_stream
|
||||
global __euler_get_stream_length
|
||||
global __euler_set_stream_length
|
||||
global __euler_end_this_thread
|
||||
global __euler_read_from_stream
|
||||
global __euler_get_framebuffer
|
||||
global __euler_encode_color
|
||||
global __euler_get_input_packet
|
||||
global __euler_start_process
|
||||
global __euler_get_other_end_process_handle
|
||||
|
||||
section .text
|
||||
|
||||
__euler_open_file:
|
||||
mov rax, 2
|
||||
push rdx
|
||||
xor rdx, rdx
|
||||
or rdx, r8
|
||||
shl r9, 1
|
||||
or rdx, r9
|
||||
syscall
|
||||
pop rdx
|
||||
mov qword [rdx], rdi
|
||||
ret
|
||||
|
||||
__euler_close_stream:
|
||||
mov rax, 11
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_get_new_pages:
|
||||
mov rax, 4
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_write_to_stream:
|
||||
mov rax, 14
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_seek_stream:
|
||||
mov rax, 12
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_get_stream_length:
|
||||
push rsi
|
||||
mov rax, 15
|
||||
syscall
|
||||
pop rsi
|
||||
mov qword [rsi], rdi
|
||||
ret
|
||||
|
||||
__euler_set_stream_length:
|
||||
mov rax, 18
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_end_this_thread:
|
||||
mov rax, 3
|
||||
syscall
|
||||
|
||||
__euler_read_from_stream:
|
||||
mov rax, 13
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_get_framebuffer:
|
||||
push rdi
|
||||
push rsi
|
||||
push rdx
|
||||
mov rax, 1
|
||||
syscall
|
||||
pop rdx
|
||||
mov dword [rdx], esi
|
||||
pop rdx
|
||||
pop rsi
|
||||
mov dword [rsi], edi
|
||||
shr rdi, 32
|
||||
mov dword [rdx], edi
|
||||
ret
|
||||
|
||||
__euler_encode_color:
|
||||
xor ah, ah
|
||||
mov al, dl
|
||||
shl ax, 8
|
||||
mov al, sil
|
||||
shl eax, 8
|
||||
mov al, dil
|
||||
mov edi, eax
|
||||
xor rax, rax
|
||||
syscall
|
||||
ret
|
||||
|
||||
__euler_get_input_packet:
|
||||
push rdi
|
||||
push rsi
|
||||
push rdx
|
||||
push rcx
|
||||
mov rax, 5
|
||||
syscall
|
||||
|
||||
test al, 0x80
|
||||
jnz .mouse_packet
|
||||
|
||||
pop rcx
|
||||
mov dword [rcx], edi
|
||||
add rsp, 24
|
||||
mov al, 2
|
||||
ret
|
||||
|
||||
.mouse_packet:
|
||||
add rsp, 8
|
||||
pop rdx
|
||||
mov word [rdx], si
|
||||
pop rdx
|
||||
mov word [rdx], di
|
||||
pop rdx
|
||||
and al, 0x7f
|
||||
mov byte [rdx], al
|
||||
mov al, 1
|
||||
ret
|
||||
|
||||
__euler_start_process:
|
||||
push rsi
|
||||
mov rax, 16
|
||||
syscall
|
||||
pop rsi
|
||||
mov qword [rsi], rdi
|
||||
ret
|
||||
|
||||
__euler_get_other_end_process_handle:
|
||||
push rsi
|
||||
mov rax, 19
|
||||
syscall
|
||||
pop rsi
|
||||
mov qword [rsi], rdi
|
||||
ret
|
155
euler/source/heap.cpp
Normal file
155
euler/source/heap.cpp
Normal file
|
@ -0,0 +1,155 @@
|
|||
#include <euler/syscall.hpp>
|
||||
#include <euler/heap.hpp>
|
||||
#include <cstdint>
|
||||
|
||||
namespace euler::heap {
|
||||
|
||||
struct heap_aligned_entry {
|
||||
size_t start_vaddr;
|
||||
size_t length;
|
||||
heap_aligned_entry *prev;
|
||||
heap_aligned_entry *next;
|
||||
};
|
||||
|
||||
struct unused_heap_entries_page {
|
||||
heap_aligned_entry *unused[510];
|
||||
unused_heap_entries_page *prev;
|
||||
uint16_t unused_in_this_page;
|
||||
};
|
||||
|
||||
heap_aligned_entry *first_entry = 0;
|
||||
heap_aligned_entry *last_entry = 0;
|
||||
unused_heap_entries_page *last_page = 0;
|
||||
|
||||
void mark_entry_unused(heap_aligned_entry *entry) {
|
||||
|
||||
for (unused_heap_entries_page *page = last_page; page != 0; page = page->prev) {
|
||||
if (page->unused_in_this_page == 510)
|
||||
continue;
|
||||
for (uint16_t i = 0; i < 510; ++i)
|
||||
if (page->unused[i] == 0) {
|
||||
page->unused[i] = entry;
|
||||
++page->unused_in_this_page;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
unused_heap_entries_page *page = (unused_heap_entries_page *)euler::syscall::get_new_pages(1);
|
||||
page->prev = last_page;
|
||||
last_page = page;
|
||||
|
||||
page->unused_in_this_page = 1;
|
||||
page->unused[0] = entry;
|
||||
|
||||
for (uint16_t i = 1; i < 510; ++i)
|
||||
page->unused[i] = 0;
|
||||
|
||||
}
|
||||
|
||||
void remove_entry(heap_aligned_entry *entry) {
|
||||
|
||||
if (entry->prev)
|
||||
entry->prev->next = entry->next;
|
||||
else
|
||||
first_entry = entry->next;
|
||||
if (entry->next)
|
||||
entry->next->prev = entry->prev;
|
||||
else
|
||||
last_entry = entry->prev;
|
||||
|
||||
mark_entry_unused(entry);
|
||||
|
||||
}
|
||||
|
||||
heap_aligned_entry *get_new_entry() {
|
||||
|
||||
for (unused_heap_entries_page *page = last_page; page != 0; page = page->prev) {
|
||||
if (page->unused_in_this_page == 0)
|
||||
continue;
|
||||
for (uint16_t i = 0; i < 510; ++i)
|
||||
if (page->unused[i] != 0) {
|
||||
heap_aligned_entry *entry = page->unused[i];
|
||||
page->unused[i] = 0;
|
||||
--page->unused_in_this_page;
|
||||
return entry;
|
||||
}
|
||||
}
|
||||
|
||||
heap_aligned_entry *new_entries = (heap_aligned_entry *)euler::syscall::get_new_pages(1);
|
||||
|
||||
for (uint8_t i = 1; i < 128; ++i)
|
||||
mark_entry_unused(new_entries + i);
|
||||
|
||||
return new_entries;
|
||||
|
||||
}
|
||||
|
||||
void *get_block(size_t length) {
|
||||
|
||||
for (heap_aligned_entry *entry = first_entry; entry != 0; entry = entry->next) {
|
||||
if (entry->length == length) {
|
||||
void *start = (void *)entry->start_vaddr;
|
||||
remove_entry(entry);
|
||||
return start;
|
||||
}
|
||||
if (entry->length > length) {
|
||||
void *start = (void *)entry->start_vaddr;
|
||||
entry->start_vaddr += length;
|
||||
entry->length -= length;
|
||||
return start;
|
||||
}
|
||||
}
|
||||
|
||||
size_t pages = (length - 1) / 4096 + 1;
|
||||
void *new_memory = euler::syscall::get_new_pages(pages);
|
||||
if (pages * 4096 != length)
|
||||
return_block((uint8_t *)new_memory + length, pages * 4096 - length);
|
||||
return new_memory;
|
||||
|
||||
}
|
||||
|
||||
void return_block(void *start, size_t length) {
|
||||
|
||||
heap_aligned_entry *after = first_entry;
|
||||
while (after != 0 && after->start_vaddr < (size_t)start + length)
|
||||
after = after->next;
|
||||
|
||||
heap_aligned_entry *before;
|
||||
if (after == 0)
|
||||
before = last_entry;
|
||||
else
|
||||
before = after->prev;
|
||||
|
||||
heap_aligned_entry *us;
|
||||
|
||||
if (after && after->start_vaddr == (size_t)start + length) {
|
||||
after->start_vaddr -= length;
|
||||
after->length += length;
|
||||
us = after;
|
||||
}
|
||||
|
||||
else {
|
||||
us = get_new_entry();
|
||||
us->prev = before;
|
||||
us->next = after;
|
||||
us->start_vaddr = (size_t)start;
|
||||
us->length = length;
|
||||
if (before)
|
||||
before->next = us;
|
||||
else
|
||||
first_entry = us;
|
||||
if (after)
|
||||
after->prev = us;
|
||||
else
|
||||
last_entry = us;
|
||||
}
|
||||
|
||||
if (before && before->start_vaddr + before->length == (size_t)start) {
|
||||
us->start_vaddr -= before->length;
|
||||
us->length += before->length;
|
||||
remove_entry(before);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
|
@ -1,10 +0,0 @@
|
|||
#include <cstdio>
|
||||
|
||||
namespace std {
|
||||
|
||||
int fclose(FILE *stream) {
|
||||
delete stream;
|
||||
return 0;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,54 +0,0 @@
|
|||
#include <stdint.h>
|
||||
#include <cstring>
|
||||
#include <cstdio>
|
||||
|
||||
namespace std {
|
||||
|
||||
FILE *fopen(const char *filename, const char *mode) {
|
||||
|
||||
bool read = false;
|
||||
bool write = false;
|
||||
bool append = false;
|
||||
bool extended = false;
|
||||
bool create = false;
|
||||
|
||||
for (const char *p = mode; *p; ++p)
|
||||
switch (*p) {
|
||||
case 'r':
|
||||
read = true;
|
||||
continue;
|
||||
case 'w':
|
||||
write = true;
|
||||
continue;
|
||||
case 'a':
|
||||
append = true;
|
||||
continue;
|
||||
case '+':
|
||||
extended = true;
|
||||
continue;
|
||||
case 'x':
|
||||
create = true;
|
||||
continue;
|
||||
default:
|
||||
continue;
|
||||
}
|
||||
|
||||
__euler_stream_handle handle;
|
||||
__euler_stream_result res = __euler_open_file(
|
||||
filename, strlen(filename), handle, write || append, create);
|
||||
|
||||
if (res != __EULER_SR_SUCCESS)
|
||||
return 0;
|
||||
|
||||
euler::file_stream *f = new euler::file_stream(handle, read || extended,
|
||||
write || extended, write && !append, append);
|
||||
|
||||
if (f->good)
|
||||
return f;
|
||||
|
||||
delete f;
|
||||
return 0;
|
||||
|
||||
}
|
||||
|
||||
}
|
|
@ -1,9 +0,0 @@
|
|||
#include <cstdio>
|
||||
|
||||
namespace std {
|
||||
|
||||
size_t fread(void *buffer, size_t size, size_t count, FILE *stream) {
|
||||
return stream->try_read(buffer, size * count) ? count : 0;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,11 +0,0 @@
|
|||
#include <cstdio>
|
||||
|
||||
namespace std {
|
||||
|
||||
int fseek(FILE *stream, long offset, int origin) {
|
||||
if (origin < 0 || origin > 2)
|
||||
return 1;
|
||||
return stream->try_seek((__euler_seek_from)origin, offset) ? 0 : 2;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,17 +0,0 @@
|
|||
#include <euler/heap.hpp>
|
||||
|
||||
void operator delete(void *ptr) {
|
||||
euler::dealloc((uint8_t *)ptr - 8, *(uint64_t *)((uint8_t *)ptr - 8));
|
||||
}
|
||||
|
||||
void operator delete(void *ptr, uint64_t) {
|
||||
euler::dealloc((uint8_t *)ptr - 8, *(uint64_t *)((uint8_t *)ptr - 8));
|
||||
}
|
||||
|
||||
void operator delete[](void *ptr) {
|
||||
euler::dealloc((uint8_t *)ptr - 8, *(uint64_t *)((uint8_t *)ptr - 8));
|
||||
}
|
||||
|
||||
void operator delete[](void *ptr, uint64_t) {
|
||||
euler::dealloc((uint8_t *)ptr - 8, *(uint64_t *)((uint8_t *)ptr - 8));
|
||||
}
|
|
@ -1,13 +0,0 @@
|
|||
#include <euler/heap.hpp>
|
||||
|
||||
void *operator new(uint64_t size) {
|
||||
void *ptr = euler::alloc(size + 8);
|
||||
*(uint64_t *)ptr = size + 8;
|
||||
return (uint8_t *)ptr + 8;
|
||||
}
|
||||
|
||||
void *operator new[](uint64_t size) {
|
||||
void *ptr = euler::alloc(size + 8);
|
||||
*(uint64_t *)ptr = size + 8;
|
||||
return (uint8_t *)ptr + 8;
|
||||
}
|
7
euler/source/std/cctype.cpp
Normal file
7
euler/source/std/cctype.cpp
Normal file
|
@ -0,0 +1,7 @@
|
|||
#include <cctype>
|
||||
|
||||
extern "C" int isspace(int ch) {
|
||||
return
|
||||
ch == ' ' || ch == '\f' || ch == '\n' ||
|
||||
ch == '\r' || ch == '\t' || ch == '\v';
|
||||
}
|
59
euler/source/std/cstdio.cpp
Normal file
59
euler/source/std/cstdio.cpp
Normal file
|
@ -0,0 +1,59 @@
|
|||
#include <cstdio>
|
||||
|
||||
extern "C" FILE *fopen(const char *filename, const char *mode) {
|
||||
|
||||
bool r = false, w = false, a = false, p = false, x = false;
|
||||
for (size_t i = 0; mode[i] != '\0'; ++i)
|
||||
if (mode[i] == 'r') r = true;
|
||||
else if (mode[i] == 'w') w = true;
|
||||
else if (mode[i] == 'a') a = true;
|
||||
else if (mode[i] == 'p') p = true;
|
||||
else if (mode[i] == 'x') x = true;
|
||||
|
||||
euler::syscall::stream_handle handle;
|
||||
if (euler::syscall::open_file(filename, !r, x, handle) !=
|
||||
euler::syscall::stream_result::success)
|
||||
return 0;
|
||||
|
||||
if (w && euler::syscall::set_stream_length(handle, 0) !=
|
||||
euler::syscall::stream_result::success)
|
||||
return 0;
|
||||
|
||||
if (a && euler::syscall::seek_stream(
|
||||
handle, euler::syscall::seek_from::end, 0) !=
|
||||
euler::syscall::stream_result::success) {
|
||||
euler::syscall::close_stream(handle);
|
||||
return 0;
|
||||
}
|
||||
|
||||
uint64_t length;
|
||||
if (euler::syscall::get_stream_length(handle, length) !=
|
||||
euler::syscall::stream_result::success) {
|
||||
euler::syscall::close_stream(handle);
|
||||
return 0;
|
||||
}
|
||||
|
||||
return new euler::file_stream(handle, r || p, length, a ? length : 0);
|
||||
|
||||
}
|
||||
|
||||
extern "C" void fclose(FILE *stream) {
|
||||
stream->close();
|
||||
delete stream;
|
||||
}
|
||||
|
||||
extern "C" int fseek(FILE *stream, long offset, int origin) {
|
||||
|
||||
if (origin < 0 || origin > 2)
|
||||
return -1;
|
||||
|
||||
return (int)stream->seek((euler::syscall::seek_from)origin, offset);
|
||||
|
||||
}
|
||||
|
||||
extern "C" size_t fread(
|
||||
void *buffer, size_t size, size_t count, FILE *stream) {
|
||||
|
||||
return (stream->read(size * count, buffer).first - 1) / size + 1;
|
||||
|
||||
}
|
19
euler/source/std/cstdlib.cpp
Normal file
19
euler/source/std/cstdlib.cpp
Normal file
|
@ -0,0 +1,19 @@
|
|||
#include <euler/heap.hpp>
|
||||
#include <cstdlib>
|
||||
|
||||
extern "C" [[noreturn]] void abort() noexcept {
|
||||
//TODO
|
||||
while (1)
|
||||
;
|
||||
}
|
||||
|
||||
extern "C" void *malloc(size_t size) {
|
||||
size_t *block = (size_t *)euler::heap::get_block(size + 8);
|
||||
*block = size;
|
||||
return block + 1;
|
||||
}
|
||||
|
||||
extern "C" void free(void *ptr) {
|
||||
size_t *block = (size_t *)ptr - 1;
|
||||
euler::heap::return_block(block, *block + 8);
|
||||
}
|
36
euler/source/std/cstring.cpp
Normal file
36
euler/source/std/cstring.cpp
Normal file
|
@ -0,0 +1,36 @@
|
|||
#include <cstring>
|
||||
|
||||
extern "C" void *memset(void *dest, int ch, size_t count) {
|
||||
unsigned char c = static_cast<unsigned char>(ch);
|
||||
unsigned char *d = (unsigned char *)dest;
|
||||
for (size_t i = 0; i < count; ++i)
|
||||
d[i] = c;
|
||||
return dest;
|
||||
}
|
||||
|
||||
extern "C" void *memcpy(void *dest, const void *src, size_t count) {
|
||||
unsigned char *d = (unsigned char *)dest;
|
||||
const unsigned char *s = (const unsigned char *)src;
|
||||
for (size_t i = 0; i < count; ++i)
|
||||
d[i] = s[i];
|
||||
return dest;
|
||||
}
|
||||
|
||||
extern "C" int strcmp(const char *lhs, const char *rhs) {
|
||||
const unsigned char *l = (const unsigned char *)lhs;
|
||||
const unsigned char *r = (const unsigned char *)rhs;
|
||||
while (*l == *r) {
|
||||
if (*l == 0)
|
||||
return 0;
|
||||
++l;
|
||||
++r;
|
||||
}
|
||||
return *l < *r ? -1 : 1;
|
||||
}
|
||||
|
||||
extern "C" size_t strlen(const char *str) {
|
||||
size_t len = 0;
|
||||
while (str[len] != '\0')
|
||||
++len;
|
||||
return len;
|
||||
}
|
86
euler/source/std/string.cpp
Normal file
86
euler/source/std/string.cpp
Normal file
|
@ -0,0 +1,86 @@
|
|||
#include <cctype>
|
||||
#include <string>
|
||||
|
||||
namespace std {
|
||||
|
||||
int stoi(const std::string &str, size_t *pos, int base) {
|
||||
//TODO: exceptions
|
||||
|
||||
size_t i = 0;
|
||||
while (isspace(str[i]))
|
||||
++i;
|
||||
|
||||
bool is_negative = false;
|
||||
if (str[i] == '-') {
|
||||
is_negative = true;
|
||||
++i;
|
||||
}
|
||||
else if (str[i] == '+')
|
||||
++i;
|
||||
|
||||
if ((base == 16 || base == 0) && str[i] == '0' &&
|
||||
(str[i + 1] == 'x' || str[i + 1] == 'X')) {
|
||||
base = 16;
|
||||
i += 2;
|
||||
}
|
||||
|
||||
else if ((base == 8 || base == 0) && str[i] == '0') {
|
||||
base = 8;
|
||||
++i;
|
||||
}
|
||||
|
||||
else if (base == 0)
|
||||
base = 10;
|
||||
|
||||
int value = 0;
|
||||
|
||||
while (true) {
|
||||
char c = str[i];
|
||||
if (c >= '0' && c < '0' + base)
|
||||
value = value * base + c - '0';
|
||||
else if (c >= 'a' && c < 'a' + base - 10)
|
||||
value = value * base + c - 'a' + 10;
|
||||
else if (c >= 'A' && c < 'A' + base - 10)
|
||||
value = value * base + c - 'A' + 10;
|
||||
else
|
||||
break;
|
||||
}
|
||||
|
||||
if (pos != 0)
|
||||
*pos = i;
|
||||
|
||||
return is_negative ? -value : value;
|
||||
|
||||
}
|
||||
|
||||
std::string to_string(int value) {
|
||||
|
||||
int max_place = 1;
|
||||
int places = 1;
|
||||
while (max_place <= value / 10) {
|
||||
max_place *= 10;
|
||||
++places;
|
||||
}
|
||||
|
||||
std::string s;
|
||||
s.resize(places);
|
||||
|
||||
for (int i = 0; i < places; ++i) {
|
||||
s[i] = (value / max_place) % 10 + '0';
|
||||
max_place /= 10;
|
||||
}
|
||||
|
||||
return s;
|
||||
|
||||
}
|
||||
|
||||
std::string operator +(std::string &&lhs, std::string &&rhs) {
|
||||
std::string s = std::move(lhs);
|
||||
s.resize(lhs.size() + rhs.size());
|
||||
for (size_t i = 0; i < rhs.size(); ++i)
|
||||
s[lhs.size() + i] = rhs[i];
|
||||
rhs.clear();
|
||||
return s;
|
||||
}
|
||||
|
||||
}
|
106
euler/source/stream.cpp
Normal file
106
euler/source/stream.cpp
Normal file
|
@ -0,0 +1,106 @@
|
|||
#include <euler/stream.hpp>
|
||||
#include <cstring>
|
||||
|
||||
namespace euler {
|
||||
|
||||
file_stream::file_stream(
|
||||
syscall::stream_handle handle, bool may_read,
|
||||
uint64_t length, uint64_t position)
|
||||
: handle(handle), may_read(may_read), buffer_loaded(false),
|
||||
length(length), position(position) {}
|
||||
|
||||
syscall::stream_result file_stream::seek(
|
||||
syscall::seek_from from, int64_t offset) {
|
||||
|
||||
int64_t new_position = offset +
|
||||
(from == syscall::seek_from::beginning ? 0 :
|
||||
from == syscall::seek_from::end ? length : position);
|
||||
|
||||
if (new_position < 0 || (uint64_t)new_position > length)
|
||||
return syscall::stream_result::out_of_bounds;
|
||||
|
||||
position = new_position;
|
||||
return syscall::stream_result::success;
|
||||
|
||||
}
|
||||
|
||||
std::pair<uint64_t, syscall::stream_result>
|
||||
file_stream::read(uint64_t bytes, void *into) {
|
||||
|
||||
if (!may_read)
|
||||
return {0, syscall::stream_result::not_readable};
|
||||
|
||||
uint64_t have_read = 0;
|
||||
syscall::stream_result result = syscall::stream_result::success;
|
||||
|
||||
uint64_t end = position + bytes;
|
||||
|
||||
if (end > length) {
|
||||
end = length;
|
||||
result = syscall::stream_result::out_of_bounds;
|
||||
}
|
||||
|
||||
uint64_t block_start = (position / 1024) * 1024;
|
||||
|
||||
while (position < end) {
|
||||
|
||||
uint64_t length_in_this_block =
|
||||
std::min(end, block_start + 1024) - position;
|
||||
|
||||
if (buffer_loaded && buffer_start == block_start)
|
||||
memcpy(into, buffer + position - block_start, length_in_this_block);
|
||||
|
||||
else if (length_in_this_block == 1024) {
|
||||
|
||||
syscall::stream_result seek_result =
|
||||
syscall::seek_stream(
|
||||
handle, syscall::seek_from::beginning, block_start);
|
||||
if (seek_result != syscall::stream_result::success)
|
||||
return {have_read, seek_result};
|
||||
|
||||
syscall::stream_result read_result =
|
||||
syscall::read_from_stream(handle, 1024, into);
|
||||
if (read_result != syscall::stream_result::success)
|
||||
return {have_read, read_result};
|
||||
|
||||
}
|
||||
|
||||
else {
|
||||
|
||||
syscall::stream_result seek_result =
|
||||
syscall::seek_stream(
|
||||
handle, syscall::seek_from::beginning, block_start);
|
||||
if (seek_result != syscall::stream_result::success)
|
||||
return {have_read, seek_result};
|
||||
|
||||
uint64_t buffer_length = std::min(1024UL, length - block_start);
|
||||
|
||||
syscall::stream_result read_result =
|
||||
syscall::read_from_stream(handle, buffer_length, buffer);
|
||||
if (read_result != syscall::stream_result::success) {
|
||||
buffer_loaded = false;
|
||||
return {have_read, read_result};
|
||||
}
|
||||
|
||||
buffer_loaded = true;
|
||||
buffer_start = block_start;
|
||||
memcpy(into, buffer + position - block_start, length_in_this_block);
|
||||
|
||||
}
|
||||
|
||||
into = (uint8_t *)into + length_in_this_block;
|
||||
have_read += length_in_this_block;
|
||||
position += length_in_this_block;
|
||||
block_start += 1024;
|
||||
|
||||
}
|
||||
|
||||
return {have_read, result};
|
||||
|
||||
}
|
||||
|
||||
void file_stream::close() {
|
||||
syscall::close_stream(handle);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,14 +0,0 @@
|
|||
#include <stdint.h>
|
||||
#include <cstring>
|
||||
|
||||
namespace std {
|
||||
|
||||
void *memcpy(void *dest, const void *src, size_t count) {
|
||||
uint8_t *d8 = (uint8_t *)dest;
|
||||
const uint8_t *s8 = (const uint8_t *)src;
|
||||
for (size_t i = 0; i < count; ++i)
|
||||
d8[i] = s8[i];
|
||||
return dest;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,12 +0,0 @@
|
|||
#include <cstring>
|
||||
|
||||
namespace std {
|
||||
|
||||
size_t strlen(const char *str) {
|
||||
size_t len = 0;
|
||||
while (str[len])
|
||||
++len;
|
||||
return len;
|
||||
}
|
||||
|
||||
}
|
32
euler/source/syscall.asm
Normal file
32
euler/source/syscall.asm
Normal file
|
@ -0,0 +1,32 @@
|
|||
bits 64
|
||||
|
||||
section .text
|
||||
|
||||
global __euler_do_syscall
|
||||
__euler_do_syscall:
|
||||
|
||||
push rdi
|
||||
push rsi
|
||||
push rdx
|
||||
push rcx
|
||||
|
||||
mov rax, qword [rdi]
|
||||
mov rdi, qword [rsi]
|
||||
mov rsi, qword [rdx]
|
||||
mov rdx, qword [rcx]
|
||||
|
||||
syscall
|
||||
|
||||
pop rcx
|
||||
mov qword [rcx], rdx
|
||||
|
||||
pop rdx
|
||||
mov qword [rdx], rsi
|
||||
|
||||
pop rsi
|
||||
mov qword [rsi], rdi
|
||||
|
||||
pop rdi
|
||||
mov qword [rdi], rax
|
||||
|
||||
ret
|
397
euler/source/syscall.cpp
Normal file
397
euler/source/syscall.cpp
Normal file
|
@ -0,0 +1,397 @@
|
|||
#include <euler/syscall.hpp>
|
||||
|
||||
extern "C" void __euler_do_syscall(
|
||||
uint64_t &rax, uint64_t &rdi, uint64_t &rsi, uint64_t &rdx);
|
||||
|
||||
namespace euler::syscall {
|
||||
|
||||
encoded_color encode_color(
|
||||
uint8_t r, uint8_t g, uint8_t b) {
|
||||
|
||||
uint64_t rax = 0;
|
||||
uint64_t rdi = (uint32_t)r | ((uint32_t)g << 8) | ((uint32_t)b << 16);
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return (encoded_color)(rax & 0xffffffff);
|
||||
|
||||
}
|
||||
|
||||
void get_framebuffer(
|
||||
encoded_color *&buffer_out, uint32_t &width_out,
|
||||
uint32_t &height_out, uint32_t &pitch_out) {
|
||||
|
||||
uint64_t rax = 1;
|
||||
uint64_t rdi;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
buffer_out = (encoded_color *)rax;
|
||||
width_out = rdi & 0xffffffff;
|
||||
height_out = rdi >> 32;
|
||||
pitch_out = rsi & 0xffffffff;
|
||||
|
||||
}
|
||||
|
||||
stream_result open_file(
|
||||
const std::string &file_path, bool allow_creation,
|
||||
bool only_allow_creation, stream_handle &handle_out) {
|
||||
|
||||
uint64_t rax = 2;
|
||||
uint64_t rdi = (uint64_t)file_path.data();
|
||||
uint64_t rsi = file_path.size();
|
||||
uint64_t rdx =
|
||||
( allow_creation ? 0x1 : 0x0) |
|
||||
(only_allow_creation ? 0x2 : 0x0);
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
handle_out = rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
[[noreturn]] void end_this_thread(int32_t exit_code) {
|
||||
|
||||
uint64_t rax = 3;
|
||||
uint64_t rdi = (uint32_t)exit_code;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
__builtin_unreachable();
|
||||
|
||||
}
|
||||
|
||||
void *get_new_pages(uint64_t n_pages) {
|
||||
|
||||
uint64_t rax = 4;
|
||||
uint64_t rdi = n_pages;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return (void *)rax;
|
||||
|
||||
}
|
||||
|
||||
std::variant<mouse_packet, key_packet> get_input_packet() {
|
||||
|
||||
uint64_t rax = 5;
|
||||
uint64_t rdi;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
if (rax & 0x80)
|
||||
return (mouse_packet){
|
||||
. left_button_down = (rax & 0x01) != 0,
|
||||
. right_button_down = (rax & 0x02) != 0,
|
||||
.middle_button_down = (rax & 0x04) != 0,
|
||||
.x_changed = (int16_t)(rdi & 0xffff),
|
||||
.y_changed = (int16_t)(rsi & 0xffff)
|
||||
};
|
||||
|
||||
return (key_packet){
|
||||
.was_key_up_event = (rdi & 0x40000) != 0,
|
||||
. num_lock = (rdi & 0x20000) != 0,
|
||||
. caps_lock = (rdi & 0x10000) != 0,
|
||||
. right_win = (rdi & 0x8000) != 0,
|
||||
. left_win = (rdi & 0x4000) != 0,
|
||||
. right_alt = (rdi & 0x2000) != 0,
|
||||
. left_alt = (rdi & 0x1000) != 0,
|
||||
. right_ctrl = (rdi & 0x800) != 0,
|
||||
. left_ctrl = (rdi & 0x400) != 0,
|
||||
. right_shift = (rdi & 0x200) != 0,
|
||||
. left_shift = (rdi & 0x100) != 0,
|
||||
. key_code = (uint8_t)(rdi & 0xff)
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
void create_private_socket(
|
||||
stream_handle &end_1_out, stream_handle &end_2_out) {
|
||||
|
||||
uint64_t rax = 6;
|
||||
uint64_t rdi;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
end_1_out = rax;
|
||||
end_2_out = rdi;
|
||||
|
||||
}
|
||||
|
||||
stream_result create_socket_listener(
|
||||
const std::string &id, listener_handle &handle_out) {
|
||||
|
||||
uint64_t rax = 7;
|
||||
uint64_t rdi = (uint64_t)id.data();
|
||||
uint64_t rsi = id.size();
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
handle_out = rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
void stop_socket_listener(listener_handle handle) {
|
||||
|
||||
uint64_t rax = 8;
|
||||
uint64_t rdi = handle;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
}
|
||||
|
||||
stream_result accept_socket_connection(
|
||||
listener_handle listener, stream_handle &stream_out) {
|
||||
|
||||
uint64_t rax = 9;
|
||||
uint64_t rdi = listener;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
stream_out = rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
stream_result connect_to_socket(
|
||||
const std::string &id, stream_handle &handle_out) {
|
||||
|
||||
uint64_t rax = 10;
|
||||
uint64_t rdi = (uint64_t)id.data();
|
||||
uint64_t rsi = id.size();
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
handle_out = rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
void close_stream(stream_handle handle) {
|
||||
|
||||
uint64_t rax = 11;
|
||||
uint64_t rdi = handle;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
}
|
||||
|
||||
stream_result seek_stream(
|
||||
stream_handle handle, seek_from from, int64_t offset) {
|
||||
|
||||
uint64_t rax = 12;
|
||||
uint64_t rdi = (uint64_t)handle;
|
||||
uint64_t rsi = (uint8_t)from;
|
||||
uint64_t rdx = (uint64_t)offset;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
stream_result read_from_stream(
|
||||
stream_handle handle, uint64_t bytes, void *into) {
|
||||
|
||||
uint64_t rax = 13;
|
||||
uint64_t rdi = (uint64_t)handle;
|
||||
uint64_t rsi = bytes;
|
||||
uint64_t rdx = (uint64_t)into;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
stream_result write_to_stream(
|
||||
stream_handle handle, uint64_t bytes, const void *from) {
|
||||
|
||||
uint64_t rax = 14;
|
||||
uint64_t rdi = (uint64_t)handle;
|
||||
uint64_t rsi = bytes;
|
||||
uint64_t rdx = (uint64_t)from;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
stream_result get_stream_length(
|
||||
stream_handle handle, uint64_t &length_out) {
|
||||
|
||||
uint64_t rax = 15;
|
||||
uint64_t rdi = (uint64_t)handle;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
length_out = rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
stream_result start_process(
|
||||
const std::string &file_path,
|
||||
const std::vector<std::pair<std::string, std::string>> &environment_variables,
|
||||
const std::vector<std::pair<stream_handle, stream_result>> &gifted_streams,
|
||||
process_handle &handle_out) {
|
||||
|
||||
std::vector<uint64_t> ev_structs(environment_variables.size() * 4);
|
||||
for (size_t i = 0; i < environment_variables.size(); ++i) {
|
||||
ev_structs[i * 4] = environment_variables[i]. first.size();
|
||||
ev_structs[i * 4 + 1] = (uint64_t)environment_variables[i]. first.data();
|
||||
ev_structs[i * 4 + 2] = environment_variables[i].second.size();
|
||||
ev_structs[i * 4 + 3] = (uint64_t)environment_variables[i].second.data();
|
||||
}
|
||||
|
||||
std::vector<uint64_t> gs_structs(gifted_streams.size() * 2);
|
||||
for (size_t i = 0; i < environment_variables.size(); ++i) {
|
||||
gs_structs[i * 2] = (uint64_t)gifted_streams[i]. first;
|
||||
gs_structs[i * 2 + 1] = (uint64_t)gifted_streams[i].second;
|
||||
}
|
||||
|
||||
uint64_t psi_struct[] = {
|
||||
file_path.size(), (uint64_t) file_path.data(),
|
||||
environment_variables.size(), (uint64_t)ev_structs.data(),
|
||||
gifted_streams.size(), (uint64_t)gs_structs.data()
|
||||
};
|
||||
|
||||
uint64_t rax = 16;
|
||||
uint64_t rdi = (uint64_t)psi_struct;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
handle_out = rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
[[noreturn]] void end_this_process(int32_t exit_code) {
|
||||
|
||||
uint64_t rax = 17;
|
||||
uint64_t rdi = (uint32_t)exit_code;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
__builtin_unreachable();
|
||||
|
||||
}
|
||||
|
||||
stream_result set_stream_length(
|
||||
stream_handle handle, uint64_t new_length) {
|
||||
|
||||
uint64_t rax = 18;
|
||||
uint64_t rdi = (uint64_t)handle;
|
||||
uint64_t rsi = new_length;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
stream_result get_other_end_process_handle(
|
||||
stream_handle stream, process_handle &process_out) {
|
||||
|
||||
uint64_t rax = 19;
|
||||
uint64_t rdi = (uint64_t)stream;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
process_out = (process_handle)rdi;
|
||||
return (stream_result)rax;
|
||||
|
||||
}
|
||||
|
||||
//entry_point must not return
|
||||
void start_thread(void (*entry_point)(uint64_t), uint64_t arg) {
|
||||
|
||||
uint64_t rax = 20;
|
||||
uint64_t rdi = (uint64_t)entry_point;
|
||||
uint64_t rsi = arg;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
}
|
||||
|
||||
//entry_point must not return
|
||||
void start_thread(void (*entry_point)()) {
|
||||
|
||||
uint64_t rax = 20;
|
||||
uint64_t rdi = (uint64_t)entry_point;
|
||||
uint64_t rsi = 0;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
}
|
||||
|
||||
//return value is number of bytes cleared
|
||||
uint64_t clear_socket_read_queue(stream_handle handle) {
|
||||
|
||||
uint64_t rax = 21;
|
||||
uint64_t rdi = handle;
|
||||
uint64_t rsi;
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
return rax;
|
||||
|
||||
}
|
||||
|
||||
//this function has a race condition if the variable is changed
|
||||
//or unset by another thread between the two syscalls.
|
||||
std::optional<std::string> try_get_environment_variable(
|
||||
const std::string &name) {
|
||||
|
||||
uint64_t rax = 22;
|
||||
uint64_t rdi = (uint64_t)name.data();
|
||||
uint64_t rsi = name.size();
|
||||
uint64_t rdx;
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
|
||||
if (rax == (uint64_t)-1)
|
||||
return {};
|
||||
|
||||
std::string s;
|
||||
s.resize(rax);
|
||||
|
||||
rax = 23;
|
||||
rdi = (uint64_t)name.data();
|
||||
rsi = name.size();
|
||||
rdx = (uint64_t)s.data();
|
||||
|
||||
__euler_do_syscall(rax, rdi, rsi, rdx);
|
||||
//do i need to tell gcc that s is modified?
|
||||
|
||||
return s;
|
||||
|
||||
}
|
||||
|
||||
}
|
|
@ -22,7 +22,8 @@ namespace hilbert::kernel::application {
|
|||
socket_listener_closed,
|
||||
other_end_closed,
|
||||
already_exists,
|
||||
not_sized
|
||||
not_sized,
|
||||
not_readable
|
||||
};
|
||||
|
||||
struct [[gnu::packed]] cpu_state {
|
||||
|
@ -57,10 +58,10 @@ namespace hilbert::kernel::application {
|
|||
|
||||
class process;
|
||||
class thread;
|
||||
class file_stream;
|
||||
class socket;
|
||||
class socket_stream_end;
|
||||
class socket_listener;
|
||||
struct file_stream;
|
||||
struct socket;
|
||||
struct socket_stream_end;
|
||||
struct socket_listener;
|
||||
|
||||
struct generic_stream_ptr {
|
||||
bool is_socket;
|
||||
|
@ -93,7 +94,7 @@ namespace hilbert::kernel::application {
|
|||
|
||||
//saves running thread state, and switches to new task. when the thread is
|
||||
//resumed, returns to caller.
|
||||
extern "C" void yield();
|
||||
extern "C" void yield(cpu_state &save_state_into);
|
||||
|
||||
extern "C" [[noreturn]] void resume_next_thread();
|
||||
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
#pragma once
|
||||
|
||||
namespace hilbert::kernel {
|
||||
[[noreturn]] void panic(uint32_t code);
|
||||
extern "C" [[noreturn]] void panic(uint32_t code);
|
||||
}
|
||||
|
|
|
@ -381,6 +381,10 @@ namespace hilbert::kernel::utility {
|
|||
count = written;
|
||||
}
|
||||
|
||||
void clear() {
|
||||
count = 0;
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
template <class t>
|
||||
|
|
|
@ -10,7 +10,8 @@ build/%.asm.o: source/%.asm
|
|||
|
||||
build/%.cpp.o: source/%.cpp
|
||||
@mkdir -p $(@D)
|
||||
$(HILBERT_CC) -c -ffreestanding -mcmodel=kernel -I ${LIMINE_DIR} $^ -o $@
|
||||
$(HILBERT_CC) -c -ffreestanding -fno-exceptions -fno-rtti \
|
||||
-mcmodel=kernel -I ${LIMINE_DIR} -I ${MINTSUKI_HEADERS_DIR} $^ -o $@
|
||||
|
||||
build/kernel.elf: $(SOURCES:%=build/%.o)
|
||||
$(HILBERT_LD) -T link.ld $^ -o $@
|
||||
|
|
|
@ -140,6 +140,7 @@ extern copy_syscall_stack
|
|||
;returns: pointer to copy
|
||||
|
||||
extern resume_next_thread
|
||||
extern running_thread
|
||||
|
||||
global yield
|
||||
yield:
|
||||
|
|
|
@ -239,7 +239,7 @@ namespace hilbert::kernel::application {
|
|||
void thread::wait_for_socket_stream(socket_stream_end *the_socket_stream) {
|
||||
waiting_for_socket_stream = the_socket_stream;
|
||||
the_socket_stream->waiting_to_read.insert(this);
|
||||
yield();
|
||||
yield(saved_state);
|
||||
waiting_for_socket_stream = 0;
|
||||
}
|
||||
|
||||
|
@ -247,7 +247,7 @@ namespace hilbert::kernel::application {
|
|||
socket_listener *the_socket_listener) {
|
||||
waiting_to_accept_from = the_socket_listener;
|
||||
the_socket_listener->waiting_to_accept.insert(this);
|
||||
yield();
|
||||
yield(saved_state);
|
||||
waiting_to_accept_from = 0;
|
||||
return new_socket_stream_id;
|
||||
}
|
||||
|
@ -256,15 +256,15 @@ namespace hilbert::kernel::application {
|
|||
socket_listener *the_socket_listener) {
|
||||
waiting_to_connect_to = the_socket_listener;
|
||||
the_socket_listener->waiting_to_connect.insert(this);
|
||||
yield();
|
||||
yield(saved_state);
|
||||
waiting_to_connect_to = 0;
|
||||
return new_socket_stream_id;
|
||||
}
|
||||
|
||||
void thread::wait_for_input() {
|
||||
waiting_for_input = false;
|
||||
waiting_for_input = true;
|
||||
input::waiting_for_input->insert(this);
|
||||
yield();
|
||||
yield(saved_state);
|
||||
waiting_for_input = false;
|
||||
}
|
||||
|
||||
|
|
|
@ -700,10 +700,44 @@ namespace hilbert::kernel::syscall {
|
|||
|
||||
}
|
||||
|
||||
typedef void (*syscall_handler)(
|
||||
uint64_t &rax, uint64_t &rdi, uint64_t &rsi, uint64_t &rdx);
|
||||
void start_thread_syscall(
|
||||
uint64_t &rax, uint64_t &rdi, uint64_t &rsi, uint64_t &rdx) {
|
||||
|
||||
uint64_t entry_point = rdi;
|
||||
uint64_t argument = rsi;
|
||||
set_zero(rax, rdi, rsi, rdx);
|
||||
|
||||
auto *p = application::running_thread->owner;
|
||||
auto *t = new application::thread(p, entry_point);
|
||||
t->saved_state.rdi = argument;
|
||||
p->add_thread(t);
|
||||
application::paused_threads->insert(t);
|
||||
|
||||
}
|
||||
|
||||
void clear_socket_read_queue_syscall(
|
||||
uint64_t &rax, uint64_t &rdi, uint64_t &rsi, uint64_t &rdx) {
|
||||
|
||||
unsigned handle = (unsigned)rdi;
|
||||
set_zero(rax, rdi, rsi, rdx);
|
||||
|
||||
auto stream = application::running_thread->owner->get_stream(handle);
|
||||
|
||||
if (stream.is_null() || !stream.is_socket) {
|
||||
rax = 0;
|
||||
return;
|
||||
}
|
||||
|
||||
auto &queue = stream.as_socket_stream->read_queue;
|
||||
|
||||
rax = queue.count;
|
||||
queue.clear();
|
||||
|
||||
}
|
||||
|
||||
void (*handlers[])(
|
||||
uint64_t &rax, uint64_t &rdi, uint64_t &rsi, uint64_t &rdx) = {
|
||||
|
||||
syscall_handler handlers[] = {
|
||||
&encode_color_syscall,
|
||||
&get_framebuffer_syscall,
|
||||
&open_file_syscall,
|
||||
|
@ -723,10 +757,13 @@ namespace hilbert::kernel::syscall {
|
|||
&start_process_syscall,
|
||||
&end_this_process_syscall,
|
||||
&set_stream_length_syscall,
|
||||
&get_other_end_process_handle_syscall
|
||||
&get_other_end_process_handle_syscall,
|
||||
&start_thread_syscall,
|
||||
&clear_socket_read_queue_syscall
|
||||
|
||||
};
|
||||
|
||||
static constexpr int max_syscall_number = 19;
|
||||
static constexpr int max_syscall_number = 20;
|
||||
|
||||
}
|
||||
|
||||
|
|
|
@ -1,209 +0,0 @@
|
|||
#pragma once
|
||||
|
||||
#include <algorithm>
|
||||
#include <stdint.h>
|
||||
#include <cstring>
|
||||
#include <cstdio>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
typedef uint32_t hilbert_color;
|
||||
|
||||
struct rgb24 {
|
||||
uint8_t r;
|
||||
uint8_t g;
|
||||
uint8_t b;
|
||||
};
|
||||
|
||||
template <class color_t>
|
||||
static inline void default_overlay(color_t &dest, const color_t &src) {
|
||||
dest = src;
|
||||
}
|
||||
|
||||
static inline void default_overlay(hilbert_color &dest, const rgb24 &src) {
|
||||
dest = __euler_encode_color(src.r, src.g, src.b);
|
||||
}
|
||||
|
||||
static inline void default_overlay(rgb24 &dest, const bool &src) {
|
||||
dest.r = src ? 255 : 0;
|
||||
dest.g = src ? 255 : 0;
|
||||
dest.b = src ? 255 : 0;
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
class image {
|
||||
|
||||
public:
|
||||
bool delete_buffer_on_destruct;
|
||||
color_t *buffer;
|
||||
unsigned width;
|
||||
unsigned height;
|
||||
unsigned pitch;//in sizeof(color_t)
|
||||
|
||||
image()
|
||||
: delete_buffer_on_destruct(false), buffer(0), width(0), height(0),
|
||||
pitch(0) {}
|
||||
|
||||
image(unsigned width, unsigned height)
|
||||
: delete_buffer_on_destruct(true), buffer(new color_t[width * height]),
|
||||
width(width), height(height), pitch(width) {}
|
||||
|
||||
image(
|
||||
color_t *buffer, unsigned width, unsigned height, unsigned pitch,
|
||||
bool delete_buffer_on_destruct)
|
||||
: delete_buffer_on_destruct(delete_buffer_on_destruct), buffer(buffer),
|
||||
width(width), height(height), pitch(pitch) {}
|
||||
|
||||
~image() {
|
||||
if (delete_buffer_on_destruct && buffer)
|
||||
delete[] buffer;
|
||||
}
|
||||
|
||||
template <class other_color_t,
|
||||
void overlay(color_t &, const other_color_t &) = default_overlay>
|
||||
image(const image<other_color_t> &other)
|
||||
: delete_buffer_on_destruct(true),
|
||||
buffer(new color_t[other.width * other.height]), width(other.width),
|
||||
height(other.height), pitch(other.width) {
|
||||
overlay_from<other_color_t, overlay>(other, 0, 0, 0, 0, width, height);
|
||||
}
|
||||
|
||||
image(image<color_t> &&other)
|
||||
: delete_buffer_on_destruct(other.delete_buffer_on_destruct),
|
||||
buffer(other.buffer), width(other.width), height(other.height),
|
||||
pitch(other.pitch) {
|
||||
other.buffer = 0;
|
||||
}
|
||||
|
||||
template <class other_color_t,
|
||||
void overlay(color_t &, const other_color_t &) = default_overlay>
|
||||
image<color_t> &operator =(const image<other_color_t> &other) {
|
||||
if (delete_buffer_on_destruct && buffer)
|
||||
delete[] buffer;
|
||||
delete_buffer_on_destruct = true;
|
||||
width = other.width;
|
||||
height = other.height;
|
||||
pitch = width;
|
||||
buffer = new color_t[width * height];
|
||||
overlay_from<other_color_t, overlay>(other, 0, 0, 0, 0, width, height);
|
||||
return *this;
|
||||
}
|
||||
|
||||
image<color_t> &operator =(image<color_t> &&other) {
|
||||
if (delete_buffer_on_destruct && buffer)
|
||||
delete[] buffer;
|
||||
delete_buffer_on_destruct = other.delete_buffer_on_destruct;
|
||||
buffer = other.buffer;
|
||||
width = other.width;
|
||||
height = other.height;
|
||||
pitch = other.pitch;
|
||||
other.buffer = 0;
|
||||
return *this;
|
||||
}
|
||||
|
||||
color_t get(unsigned x, unsigned y) const {
|
||||
return buffer[y * pitch + x];
|
||||
}
|
||||
|
||||
void set(unsigned x, unsigned y, const color_t &c) {
|
||||
buffer[y * pitch + x] = c;
|
||||
}
|
||||
|
||||
void copy_from(
|
||||
const image<color_t> &other, unsigned to_x, unsigned to_y,
|
||||
unsigned from_x, unsigned from_y, unsigned width, unsigned height) {
|
||||
|
||||
color_t *to_start = buffer + pitch * to_y + to_x;
|
||||
const color_t *from_start = other.buffer + other.pitch * from_y + from_x;
|
||||
|
||||
for (unsigned y = 0; y < height; ++y)
|
||||
std::memcpy(
|
||||
to_start + pitch * y, from_start + other.pitch * y,
|
||||
width * sizeof(color_t));
|
||||
|
||||
}
|
||||
|
||||
template <class other_color_t,
|
||||
void overlay(color_t &, const other_color_t &) = default_overlay>
|
||||
void overlay_from(
|
||||
const image<other_color_t> &other, unsigned to_x, unsigned to_y,
|
||||
unsigned from_x, unsigned from_y, unsigned width, unsigned height) {
|
||||
|
||||
color_t *to_start = buffer + pitch * to_y + to_x;
|
||||
const other_color_t *from_start =
|
||||
other.buffer + other.pitch * from_y + from_x;
|
||||
|
||||
for (unsigned y = 0; y < height; ++y)
|
||||
for (unsigned x = 0; x < width; ++x)
|
||||
overlay(to_start[pitch * y + x], from_start[other.pitch * y + x]);
|
||||
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
template <class color_t>
|
||||
void swap(image<color_t> &a, image<color_t> &b) {
|
||||
std::swap(a.delete_buffer_on_destruct, b.delete_buffer_on_destruct);
|
||||
std::swap(a.buffer, b.buffer);
|
||||
std::swap(a.width, b.width);
|
||||
std::swap(a.height, b.height);
|
||||
std::swap(a.pitch, b.pitch);
|
||||
}
|
||||
|
||||
image<hilbert_color> get_hilbert_framebuffer();
|
||||
|
||||
bool try_load_ppm(std::FILE *input, image<rgb24> &into);
|
||||
|
||||
static inline bool try_load_ppm(const char *path, image<rgb24> &into) {
|
||||
std::FILE *f = std::fopen(path, "r");
|
||||
if (!f)
|
||||
return false;
|
||||
bool success = try_load_ppm(f, into);
|
||||
std::fclose(f);
|
||||
return success;
|
||||
}
|
||||
|
||||
//TODO: unicode
|
||||
template <class color_t>
|
||||
class fixed_bitmap_font {
|
||||
|
||||
public:
|
||||
unsigned width;
|
||||
unsigned height;
|
||||
image<color_t> glyphs[128];
|
||||
|
||||
template <class target_color_t,
|
||||
void overlay(target_color_t &dest, const color_t &src) = default_overlay>
|
||||
void overlay_text(
|
||||
image<target_color_t> &target, unsigned x,
|
||||
unsigned y, const char *text) const {
|
||||
|
||||
while (1) {
|
||||
uint8_t ch = (uint8_t)*text;
|
||||
if (ch == 0)
|
||||
return;
|
||||
if (ch < 128) {
|
||||
target.template overlay_from<color_t, overlay>(
|
||||
glyphs[ch], x, y, 0, 0, width, height);
|
||||
x += width;
|
||||
}
|
||||
++text;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
bool try_load_psf(std::FILE *input, fixed_bitmap_font<bool> &into);
|
||||
|
||||
static inline bool try_load_psf(
|
||||
const char *path, fixed_bitmap_font<bool> &into) {
|
||||
std::FILE *f = std::fopen(path, "r");
|
||||
if (!f)
|
||||
return false;
|
||||
bool success = try_load_psf(f, into);
|
||||
std::fclose(f);
|
||||
return success;
|
||||
}
|
||||
|
||||
}
|
57
libraries/daguerre/include/daguerre/fixed-font.hpp
Normal file
57
libraries/daguerre/include/daguerre/fixed-font.hpp
Normal file
|
@ -0,0 +1,57 @@
|
|||
#pragma once
|
||||
|
||||
#include <daguerre/general.hpp>
|
||||
|
||||
//TODO: support more than just ascii
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
//forward declare since these depend on each other
|
||||
template <class color_t>
|
||||
class image;
|
||||
|
||||
template <class color_t>
|
||||
class fixed_font {
|
||||
|
||||
public:
|
||||
int glyph_width;
|
||||
int glyph_height;
|
||||
image<color_t> glyphs[128];
|
||||
|
||||
//initializes every glyph to an empty image
|
||||
fixed_font();
|
||||
|
||||
//initializes each glyph to have the right dimensions and
|
||||
//a valid buffer, but does nothing to their contents.
|
||||
fixed_font(int glyph_width, int glyph_height);
|
||||
|
||||
//moves the other font's glyphs here, then
|
||||
//sets the other font to have empty images.
|
||||
fixed_font(fixed_font<color_t> &&other);
|
||||
|
||||
//moves the other font's glyphs here, then
|
||||
//sets the other font to have empty images.
|
||||
fixed_font<color_t> &operator =(fixed_font<color_t> &&other);
|
||||
|
||||
//use one of the two constructors below instead
|
||||
fixed_font(const fixed_font<color_t> &other) = delete;
|
||||
|
||||
//copies the other font's images here, passing them through conversion.
|
||||
template <class other_color_t>
|
||||
fixed_font(const fixed_font<other_color_t> &other,
|
||||
converter_t<color_t, other_color_t> *conversion = &default_conversion);
|
||||
|
||||
//directly calls memcpy on the images' buffers. the second argument
|
||||
//does nothing except distinguish this from the other copy constructor.
|
||||
fixed_font(const fixed_font<color_t> &other, bool);
|
||||
|
||||
//use copy constructor and move assignment instead
|
||||
fixed_font<color_t> &operator =(const fixed_font<color_t> &other) = delete;
|
||||
|
||||
~fixed_font() = default;
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#include <daguerre/impl/fixed-font.hpp>
|
9
libraries/daguerre/include/daguerre/framebuffer.hpp
Normal file
9
libraries/daguerre/include/daguerre/framebuffer.hpp
Normal file
|
@ -0,0 +1,9 @@
|
|||
#pragma once
|
||||
|
||||
#include <daguerre/image.hpp>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
image<hilbert_color> get_hilbert_framebuffer();
|
||||
|
||||
}
|
59
libraries/daguerre/include/daguerre/general.hpp
Normal file
59
libraries/daguerre/include/daguerre/general.hpp
Normal file
|
@ -0,0 +1,59 @@
|
|||
#pragma once
|
||||
|
||||
#include <euler/syscall.hpp>
|
||||
#include <stdint.h>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
typedef euler::syscall::encoded_color hilbert_color;
|
||||
|
||||
struct rgb24 {
|
||||
uint8_t r;
|
||||
uint8_t g;
|
||||
uint8_t b;
|
||||
};
|
||||
|
||||
template <class dest_t, class src_t>
|
||||
using converter_t = void (dest_t &, const src_t &);
|
||||
|
||||
//copies src to dest
|
||||
template <class color_t>
|
||||
static inline void default_conversion(color_t &dest, const color_t &src) {
|
||||
dest = src;
|
||||
}
|
||||
|
||||
//encodes src and copies that to dest
|
||||
static inline void default_conversion(hilbert_color &dest, const rgb24 &src) {
|
||||
dest = euler::syscall::encode_color(src.r, src.g, src.b);
|
||||
}
|
||||
|
||||
template <class dest_t, class src_t, class param_t>
|
||||
using param_converter_t = void (dest_t &, const src_t &, const param_t &);
|
||||
|
||||
//if src is true, copies param to dest.
|
||||
//if src is false, does nothing.
|
||||
template <class color_t>
|
||||
static inline void default_conversion(
|
||||
color_t &dest, const bool &src, const color_t ¶m) {
|
||||
if (src)
|
||||
dest = param;
|
||||
}
|
||||
|
||||
//intersects [to, to + length) with [min, max) at stores the result in
|
||||
//[to, to + length). if anything was removed from either end, the same
|
||||
//amount is removed from the same end(s) of [from, from + length)
|
||||
static inline void make_safe(
|
||||
int &to, int &from, int &length, int min, int max) {
|
||||
|
||||
if (to < min) {
|
||||
from += min - to;
|
||||
length -= min - to;
|
||||
to = min;
|
||||
}
|
||||
|
||||
if (to + length > max)
|
||||
length = max - to;
|
||||
|
||||
}
|
||||
|
||||
}
|
118
libraries/daguerre/include/daguerre/image.hpp
Normal file
118
libraries/daguerre/include/daguerre/image.hpp
Normal file
|
@ -0,0 +1,118 @@
|
|||
#pragma once
|
||||
|
||||
#include <daguerre/general.hpp>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
//forward declare since these depend on each other
|
||||
template <class color_t>
|
||||
class fixed_font;
|
||||
|
||||
template <class color_t>
|
||||
class image {
|
||||
|
||||
public:
|
||||
bool delete_buffer_on_destruct;
|
||||
int width;
|
||||
int height;
|
||||
color_t *buffer;
|
||||
int buffer_pitch;//in sizeof(color_t)
|
||||
|
||||
//makes empty image
|
||||
image();
|
||||
|
||||
//make image with specified dimensions, contents uninitialized
|
||||
image(int width, int height);
|
||||
|
||||
//make image with specified dimensions and buffer
|
||||
image(
|
||||
int width, int height, color_t *buffer,
|
||||
int buffer_pitch, bool delete_buffer_on_destruct);
|
||||
|
||||
//moves the other image's data here, and
|
||||
//then sets the other image to an empty one.
|
||||
image(image<color_t> &&other);
|
||||
|
||||
//moves the other image's data here, and
|
||||
//then sets the other image to an empty one.
|
||||
image<color_t> &operator =(image<color_t> &&other);
|
||||
|
||||
//use one of the two constructors below instead
|
||||
image(const image<color_t> &other) = delete;
|
||||
|
||||
//copies the other image's data here, passing it through conversion
|
||||
template <class other_color_t>
|
||||
image(const image<other_color_t> &other,
|
||||
converter_t<color_t, other_color_t> *conversion = &default_conversion);
|
||||
|
||||
//directly calls memcpy on the buffer. the second argument does
|
||||
//nothing except distinguish this from the other copy constructor.
|
||||
image(const image<color_t> &other, bool);
|
||||
|
||||
//use copy constructor and move assignment instead
|
||||
image<color_t> &operator =(const image<color_t> &other) = delete;
|
||||
|
||||
~image();
|
||||
|
||||
void fill(const color_t &color);
|
||||
|
||||
//does not check bounds
|
||||
color_t &at(int x, int y);
|
||||
|
||||
//does not check bounds
|
||||
const color_t &at(int x, int y) const;
|
||||
|
||||
image<color_t> stretch(int new_width, int new_height);
|
||||
|
||||
//copies the region here via memcpy. does not check bounds.
|
||||
void copy_from(
|
||||
const image<color_t> &other, int to_x, int to_y,
|
||||
int from_x, int from_y, int width, int height);
|
||||
|
||||
//copies the entire image here via memcpy. does not check bounds.
|
||||
void copy_from(const image<color_t> &other, int to_x, int to_y);
|
||||
|
||||
//copies the region here through conversion. does not check bounds.
|
||||
template <class other_color_t>
|
||||
void convert_from(
|
||||
const image<other_color_t> &other, int to_x, int to_y,
|
||||
int from_x, int from_y, int width, int height,
|
||||
converter_t<color_t, other_color_t> *conversion = &default_conversion);
|
||||
|
||||
//copies the entire image here through conversion. does not check bounds.
|
||||
template <class other_color_t>
|
||||
void convert_from(
|
||||
const image<other_color_t> &other, int to_x, int to_y,
|
||||
converter_t<color_t, other_color_t> *conversion = &default_conversion);
|
||||
|
||||
//copies the region here through conversion. does not check bounds.
|
||||
template <class other_color_t, class param_t>
|
||||
void convert_from(
|
||||
const param_t ¶m, const image<other_color_t> &other, int to_x,
|
||||
int to_y, int from_x, int from_y, int width, int height,
|
||||
param_converter_t<color_t, other_color_t, param_t> *conversion =
|
||||
&default_conversion);
|
||||
|
||||
//copies the entire image here through conversion. does not check bounds.
|
||||
template <class other_color_t, class param_t>
|
||||
void convert_from(const param_t ¶m,
|
||||
const image<other_color_t> &other, int to_x, int to_y,
|
||||
param_converter_t<color_t, other_color_t, param_t> *conversion =
|
||||
&default_conversion);
|
||||
|
||||
//does not check bounds or wrap text
|
||||
template <class font_color_t, class param_t>
|
||||
void render_text(
|
||||
const fixed_font<font_color_t> &font,
|
||||
const param_t ¶rm, int x, int y, const char *text,
|
||||
param_converter_t<color_t, font_color_t, param_t> *conversion =
|
||||
&default_conversion);
|
||||
|
||||
};
|
||||
|
||||
template <class color_t>
|
||||
void swap(image<color_t> &a, image<color_t> &b);
|
||||
|
||||
}
|
||||
|
||||
#include <daguerre/impl/image.hpp>
|
59
libraries/daguerre/include/daguerre/impl/fixed-font.hpp
Normal file
59
libraries/daguerre/include/daguerre/impl/fixed-font.hpp
Normal file
|
@ -0,0 +1,59 @@
|
|||
#pragma once
|
||||
|
||||
//this file should only be included from <daguerre/fixed-font.hpp>
|
||||
|
||||
#include <daguerre/image.hpp>
|
||||
#include <utility>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
template<class color_t>
|
||||
fixed_font<color_t>::fixed_font() : glyph_width(0), glyph_height(0) {}
|
||||
|
||||
template <class color_t>
|
||||
fixed_font<color_t>::fixed_font(int glyph_width, int glyph_height)
|
||||
: glyph_width(glyph_width), glyph_height(glyph_height) {
|
||||
for (int i = 0; i < 128; ++i)
|
||||
glyphs[i] = image<color_t>(glyph_width, glyph_height);
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
fixed_font<color_t>::fixed_font(fixed_font<color_t> &&other)
|
||||
: glyph_width(other.glyph_width), glyph_height(other.glyph_height) {
|
||||
for (int i = 0; 0 < 128; ++i)
|
||||
glyphs[i] = std::move(other.glyphs[i]);
|
||||
other.glyph_width = 0;
|
||||
other.glyph_height = 0;
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
fixed_font<color_t> &fixed_font<color_t>::operator =(
|
||||
fixed_font<color_t> &&other) {
|
||||
glyph_width = other.glyph_width;
|
||||
glyph_height = other.glyph_height;
|
||||
for (int i = 0; 0 < 128; ++i)
|
||||
glyphs[i] = std::move(other.glyphs[i]);
|
||||
other.glyph_width = 0;
|
||||
other.glyph_height = 0;
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class other_color_t>
|
||||
fixed_font<color_t>::fixed_font(
|
||||
const fixed_font<other_color_t> &other,
|
||||
converter_t<color_t, other_color_t> *conversion)
|
||||
: glyph_width(other.glyph_width), glyph_height(other.glyph_height) {
|
||||
for (int i = 0; i < 128; ++i)
|
||||
glyphs[i] = image<color_t>(other.glyphs[i], conversion);
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
fixed_font<color_t>::fixed_font(const fixed_font<color_t> &other, bool)
|
||||
: glyph_width(other.glyph_width), glyph_height(other.glyph_height) {
|
||||
for (int i = 0; i < 128; ++i)
|
||||
glyphs[i] = image<color_t>(other.glyphs[i], true);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
#include <daguerre/image.hpp>
|
217
libraries/daguerre/include/daguerre/impl/image.hpp
Normal file
217
libraries/daguerre/include/daguerre/impl/image.hpp
Normal file
|
@ -0,0 +1,217 @@
|
|||
#pragma once
|
||||
|
||||
//this file should only be included from <daguerre/image.hpp>
|
||||
|
||||
#include <daguerre/fixed-font.hpp>
|
||||
#include <algorithm>
|
||||
#include <cstring>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
template <class color_t>
|
||||
image<color_t>::image()
|
||||
: delete_buffer_on_destruct(false), width(0),
|
||||
height(0), buffer(0), buffer_pitch(0) {}
|
||||
|
||||
template <class color_t>
|
||||
image<color_t>::image(int width, int height)
|
||||
: delete_buffer_on_destruct(true), width(width), height(height),
|
||||
buffer(new color_t[width * height]), buffer_pitch(width) {}
|
||||
|
||||
template <class color_t>
|
||||
image<color_t>::image(
|
||||
int width, int height, color_t *buffer,
|
||||
int buffer_pitch, bool delete_buffer_on_destruct)
|
||||
: delete_buffer_on_destruct(delete_buffer_on_destruct), width(width),
|
||||
height(height), buffer(buffer), buffer_pitch(buffer_pitch) {}
|
||||
|
||||
template <class color_t>
|
||||
image<color_t>::image(image<color_t> &&other)
|
||||
: delete_buffer_on_destruct(other.delete_buffer_on_destruct),
|
||||
width(other.width), height(other.height), buffer(other.buffer),
|
||||
buffer_pitch(other.buffer_pitch) {
|
||||
|
||||
other.width = 0;
|
||||
other.height = 0;
|
||||
other.buffer = 0;
|
||||
|
||||
}
|
||||
|
||||
template<class color_t>
|
||||
image<color_t> &image<color_t>::operator =(image<color_t> &&other) {
|
||||
|
||||
if (delete_buffer_on_destruct && buffer)
|
||||
delete[] buffer;
|
||||
|
||||
delete_buffer_on_destruct = other.delete_buffer_on_destruct;
|
||||
width = other.width;
|
||||
height = other.height;
|
||||
buffer = other.buffer;
|
||||
buffer_pitch = other.buffer_pitch;
|
||||
|
||||
other.width = 0;
|
||||
other.height = 0;
|
||||
other.buffer = 0;
|
||||
|
||||
return *this;
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class other_color_t>
|
||||
image<color_t>::image(const image<other_color_t> &other,
|
||||
converter_t<color_t, other_color_t> *conversion)
|
||||
: delete_buffer_on_destruct(true), width(other.width),
|
||||
height(other.height), buffer(new color_t[other.width * other.height]),
|
||||
buffer_pitch(other.width) {
|
||||
|
||||
for (int y = 0; y < height; ++y)
|
||||
for (int x = 0; x < width; ++x)
|
||||
conversion(
|
||||
buffer[y * buffer_pitch + x],
|
||||
other.buffer[y * other.buffer_pitch + x]);
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
image<color_t>::image(const image<color_t> &other, bool)
|
||||
: delete_buffer_on_destruct(true), width(width), height(other.height),
|
||||
buffer(new color_t[other.width * other.height]),
|
||||
buffer_pitch(other.width) {
|
||||
|
||||
if (buffer_pitch == other.buffer_pitch)
|
||||
memcpy(buffer, other.buffer, (height - 1) * buffer_pitch + width);
|
||||
|
||||
else
|
||||
for (int y = 0; y < height; ++y)
|
||||
memcpy(buffer + y * buffer_pitch,
|
||||
other.buffer + y * other.buffer_pitch,
|
||||
sizeof(color_t) * width);
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
image<color_t>::~image() {
|
||||
if (delete_buffer_on_destruct && buffer)
|
||||
delete[] buffer;
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
void image<color_t>::fill(const color_t &color) {
|
||||
for (int y = 0; y < height; ++y)
|
||||
for (int x = 0; x < width; ++x)
|
||||
buffer[y * buffer_pitch + x] = color;
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
color_t &image<color_t>::at(int x, int y) {
|
||||
return buffer[y * buffer_pitch + x];
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
const color_t &image<color_t>::at(int x, int y) const {
|
||||
return buffer[y * buffer_pitch + x];
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
image<color_t> image<color_t>::stretch(int new_width, int new_height) {
|
||||
image<color_t> im(new_width, new_height);
|
||||
for (int y = 0; y < new_height; ++y)
|
||||
for (int x = 0; x < new_width; ++x)
|
||||
im.at(x, y) = at(x * width / new_width, y * height / new_height);
|
||||
return im;
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
void image<color_t>::copy_from(
|
||||
const image<color_t> &other, int to_x, int to_y,
|
||||
int from_x, int from_y, int width, int height) {
|
||||
|
||||
color_t *to = buffer + to_y * buffer_pitch + to_x;
|
||||
const color_t *from = other.buffer + from_y * other.buffer_pitch + from_x;
|
||||
|
||||
for (int y = 0; y < height; ++y)
|
||||
memcpy(
|
||||
to + y * buffer_pitch,
|
||||
from + y * other.buffer_pitch,
|
||||
width * sizeof(color_t));
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
void image<color_t>::copy_from(
|
||||
const image<color_t> &other, int to_x, int to_y) {
|
||||
copy_from(other, to_x, to_y, 0, 0, other.width, other.height);
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class other_color_t>
|
||||
void image<color_t>::convert_from(
|
||||
const image<other_color_t> &other, int to_x, int to_y,
|
||||
int from_x, int from_y, int width, int height,
|
||||
converter_t<color_t, other_color_t> *conversion) {
|
||||
|
||||
for (int y = 0; y < height; ++y)
|
||||
for (int x = 0; x < width; ++x)
|
||||
conversion(at(to_x + x, to_y + y), other.at(from_x + x, from_y + y));
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class other_color_t>
|
||||
void image<color_t>::convert_from(
|
||||
const image<other_color_t> &other, int to_x, int to_y,
|
||||
converter_t<color_t, other_color_t> *conversion) {
|
||||
convert_from(other, to_x, to_y, 0, 0, other.width, other.y, conversion);
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class other_color_t, class param_t>
|
||||
void image<color_t>::convert_from(
|
||||
const param_t ¶m, const image<other_color_t> &other, int to_x,
|
||||
int to_y, int from_x, int from_y, int width, int height,
|
||||
param_converter_t<color_t, other_color_t, param_t> *conversion) {
|
||||
|
||||
for (int y = 0; y < height; ++y)
|
||||
for (int x = 0; x < width; ++x)
|
||||
conversion(
|
||||
at(to_x + x, to_y + y), other.at(from_x + x, from_y + y), param);
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class other_color_t, class param_t>
|
||||
void image<color_t>::convert_from(
|
||||
const param_t ¶m, const image<other_color_t> &other, int to_x,
|
||||
int to_y, param_converter_t<color_t, other_color_t, param_t> *conversion) {
|
||||
convert_from(
|
||||
param, other, to_x, to_y, 0, 0, other.width, other.y, conversion);
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
template <class font_color_t, class param_t>
|
||||
void image<color_t>::render_text(
|
||||
const fixed_font<font_color_t> &font,
|
||||
const param_t ¶m, int x, int y, const char *text,
|
||||
param_converter_t<color_t, font_color_t, param_t> *conversion) {
|
||||
|
||||
while (*text) {
|
||||
int ch = *text;
|
||||
if (ch >= 0 && ch < 128)
|
||||
convert_from(param, font.glyphs[ch], x, y, conversion);
|
||||
++text;
|
||||
x += font.glyph_width;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
template <class color_t>
|
||||
void swap(image<color_t> &a, image<color_t> &b) {
|
||||
std::swap(a.delete_buffer_on_destruct, b.delete_buffer_on_destruct);
|
||||
std::swap(a.width, b.width);
|
||||
std::swap(a.height, b.height);
|
||||
std::swap(a.buffer, b.buffer);
|
||||
std::swap(a.buffer_pitch, b.buffer_pitch);
|
||||
}
|
||||
|
||||
}
|
20
libraries/daguerre/include/daguerre/ppm.hpp
Normal file
20
libraries/daguerre/include/daguerre/ppm.hpp
Normal file
|
@ -0,0 +1,20 @@
|
|||
#pragma once
|
||||
|
||||
#include <daguerre/image.hpp>
|
||||
#include <optional>
|
||||
#include <cstdio>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
std::optional<image<rgb24>> try_load_ppm(FILE *input);
|
||||
|
||||
static inline std::optional<image<rgb24>> try_load_ppm(const char *path) {
|
||||
FILE *f = fopen(path, "r");
|
||||
if (!f)
|
||||
return {};
|
||||
auto result = try_load_ppm(f);
|
||||
fclose(f);
|
||||
return result;
|
||||
}
|
||||
|
||||
}
|
20
libraries/daguerre/include/daguerre/psf.hpp
Normal file
20
libraries/daguerre/include/daguerre/psf.hpp
Normal file
|
@ -0,0 +1,20 @@
|
|||
#pragma once
|
||||
|
||||
#include <daguerre/fixed-font.hpp>
|
||||
#include <optional>
|
||||
#include <cstdio>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
std::optional<fixed_font<bool>> try_load_psf(FILE *input);
|
||||
|
||||
static inline std::optional<fixed_font<bool>> try_load_psf(const char *path) {
|
||||
FILE *f = fopen(path, "r");
|
||||
if (!f)
|
||||
return {};
|
||||
auto result = try_load_psf(f);
|
||||
fclose(f);
|
||||
return result;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,5 +1,5 @@
|
|||
SOURCES = \
|
||||
daguerre.cpp
|
||||
framebuffer.cpp ppm.cpp psf.cpp
|
||||
|
||||
build/%.cpp.o: source/%.cpp
|
||||
@mkdir -p $(@D)
|
||||
|
|
|
@ -1,110 +0,0 @@
|
|||
#include <daguerre.hpp>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
image<hilbert_color> get_hilbert_framebuffer() {
|
||||
uint32_t width, height, pitch;
|
||||
hilbert_color *ptr = __euler_get_framebuffer(width, height, pitch);
|
||||
return image<hilbert_color>(ptr, width, height, pitch, false);
|
||||
}
|
||||
|
||||
//TODO: make this more robust
|
||||
unsigned read_text_int(std::FILE *input) {
|
||||
unsigned n = 0;
|
||||
char ch;
|
||||
std::fread(&ch, 1, 1, input);
|
||||
if (ch == '#') {
|
||||
do
|
||||
std::fread(&ch, 1, 1, input);
|
||||
while (ch != '\n');
|
||||
std::fread(&ch, 1, 1, input);
|
||||
}
|
||||
do {
|
||||
n = n * 10 + ch - '0';
|
||||
std::fread(&ch, 1, 1, input);
|
||||
} while (ch >= '0' && ch <= '9');
|
||||
return n;
|
||||
}
|
||||
|
||||
//only supports p6 format
|
||||
bool try_load_ppm(std::FILE *input, image<rgb24> &into) {
|
||||
|
||||
char header[3];
|
||||
if (std::fread(header, 1, 3, input) != 3)
|
||||
return false;
|
||||
|
||||
if (header[0] != 'P' || header[1] != '6' || header[2] != '\n')
|
||||
return false;
|
||||
|
||||
unsigned width = read_text_int(input);
|
||||
unsigned height = read_text_int(input);
|
||||
unsigned max = read_text_int(input);
|
||||
|
||||
into = image<rgb24>(width, height);
|
||||
|
||||
for (unsigned y = 0; y < height; ++y)
|
||||
for (unsigned x = 0; x < width; ++x) {
|
||||
if (std::fread(&into.buffer[y * width + x].r, 1, 1, input) != 1)
|
||||
return false;
|
||||
if (std::fread(&into.buffer[y * width + x].g, 1, 1, input) != 1)
|
||||
return false;
|
||||
if (std::fread(&into.buffer[y * width + x].b, 1, 1, input) != 1)
|
||||
return false;
|
||||
}
|
||||
|
||||
if (max != 255)
|
||||
for (unsigned v = 0; v < width * height; ++v) {
|
||||
into.buffer[v].r = ((uint16_t)into.buffer[v].r * 255) / max;
|
||||
into.buffer[v].g = ((uint16_t)into.buffer[v].g * 255) / max;
|
||||
into.buffer[v].b = ((uint16_t)into.buffer[v].b * 255) / max;
|
||||
}
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
//assumes the font is in psf2 format, and has a unicode table
|
||||
bool try_load_psf(std::FILE *input, fixed_bitmap_font<bool> &into) {
|
||||
|
||||
uint32_t header[8];
|
||||
if (std::fread(header, 4, 8, input) != 8)
|
||||
return false;
|
||||
|
||||
const uint32_t glyphs_start = header[2];
|
||||
const uint32_t glyph_count = header[4];
|
||||
const uint32_t glyph_length = header[5];
|
||||
into.height = header[6];
|
||||
into.width = header[7];
|
||||
|
||||
const uint32_t unicode_start = glyphs_start + glyph_count * glyph_length;
|
||||
std::fseek(input, unicode_start, SEEK_SET);
|
||||
|
||||
uint32_t indices[128];
|
||||
|
||||
for (uint32_t index = 0; index < glyph_count; ++index) {
|
||||
uint8_t ch;
|
||||
std::fread(&ch, 1, 1, input);
|
||||
if (ch < 128)
|
||||
indices[ch] = index;
|
||||
do
|
||||
std::fread(&ch, 1, 1, input);
|
||||
while (ch != 0xff);
|
||||
}
|
||||
|
||||
for (uint8_t ch = 0; ch < 128; ++ch) {
|
||||
std::fseek(input, glyphs_start + glyph_length * indices[ch], SEEK_SET);
|
||||
into.glyphs[ch] = image<bool>(into.width, into.height);
|
||||
for (unsigned h = 0; h < into.height; ++h)
|
||||
for (unsigned wb = 0; wb < into.width; wb += 8) {
|
||||
uint8_t byte;
|
||||
std::fread(&byte, 1, 1, input);
|
||||
for (unsigned x = 0; x < 8 && wb + x < into.width; ++x)
|
||||
into.glyphs[ch].set(wb + x, h, (byte >> (7 - x)) & 1);
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
|
||||
}
|
||||
|
||||
}
|
12
libraries/daguerre/source/framebuffer.cpp
Normal file
12
libraries/daguerre/source/framebuffer.cpp
Normal file
|
@ -0,0 +1,12 @@
|
|||
#include <daguerre/framebuffer.hpp>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
image<hilbert_color> get_hilbert_framebuffer() {
|
||||
hilbert_color *ptr;
|
||||
uint32_t width, height, pitch;
|
||||
euler::syscall::get_framebuffer(ptr, width, height, pitch);
|
||||
return image<hilbert_color>(width, height, ptr, pitch, false);
|
||||
}
|
||||
|
||||
}
|
60
libraries/daguerre/source/ppm.cpp
Normal file
60
libraries/daguerre/source/ppm.cpp
Normal file
|
@ -0,0 +1,60 @@
|
|||
#include <daguerre/ppm.hpp>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
//TODO: make this more robust
|
||||
static unsigned read_text_int(FILE *input) {
|
||||
unsigned n = 0;
|
||||
char ch;
|
||||
fread(&ch, 1, 1, input);
|
||||
if (ch == '#') {
|
||||
do
|
||||
fread(&ch, 1, 1, input);
|
||||
while (ch != '\n');
|
||||
fread(&ch, 1, 1, input);
|
||||
}
|
||||
do {
|
||||
n = n * 10 + ch - '0';
|
||||
fread(&ch, 1, 1, input);
|
||||
} while (ch >= '0' && ch <= '9');
|
||||
return n;
|
||||
}
|
||||
|
||||
//TODO: this only supports p6 format, and assumes max < 256
|
||||
std::optional<image<rgb24>> try_load_ppm(FILE *input) {
|
||||
|
||||
char header[3];
|
||||
if (fread(header, 1, 3, input) != 3)
|
||||
return {};
|
||||
|
||||
if (header[0] != 'P' || header[1] != '6' || header[2] != '\n')
|
||||
return {};
|
||||
|
||||
unsigned width = read_text_int(input);
|
||||
unsigned height = read_text_int(input);
|
||||
unsigned max = read_text_int(input);
|
||||
|
||||
image<rgb24> im(width, height);
|
||||
|
||||
for (unsigned y = 0; y < height; ++y)
|
||||
for (unsigned x = 0; x < width; ++x) {
|
||||
if (fread(&im.buffer[y * width + x].r, 1, 1, input) != 1)
|
||||
return {};
|
||||
if (fread(&im.buffer[y * width + x].g, 1, 1, input) != 1)
|
||||
return {};
|
||||
if (fread(&im.buffer[y * width + x].b, 1, 1, input) != 1)
|
||||
return {};
|
||||
}
|
||||
|
||||
if (max != 255)
|
||||
for (unsigned v = 0; v < width * height; ++v) {
|
||||
im.buffer[v].r = ((uint16_t)im.buffer[v].r * 255) / max;
|
||||
im.buffer[v].g = ((uint16_t)im.buffer[v].g * 255) / max;
|
||||
im.buffer[v].b = ((uint16_t)im.buffer[v].b * 255) / max;
|
||||
}
|
||||
|
||||
return im;
|
||||
|
||||
}
|
||||
|
||||
}
|
51
libraries/daguerre/source/psf.cpp
Normal file
51
libraries/daguerre/source/psf.cpp
Normal file
|
@ -0,0 +1,51 @@
|
|||
#include <daguerre/psf.hpp>
|
||||
|
||||
namespace daguerre {
|
||||
|
||||
//TODO: this assumes the font is in psf2 format, and has a unicode table
|
||||
std::optional<fixed_font<bool>> try_load_psf(FILE *input) {
|
||||
|
||||
uint32_t header[8];
|
||||
if (fread(header, 4, 8, input) != 8)
|
||||
return {};
|
||||
|
||||
const uint32_t glyphs_start = header[2];
|
||||
const uint32_t glyph_count = header[4];
|
||||
const uint32_t glyph_length = header[5];
|
||||
const uint32_t height = header[6];
|
||||
const uint32_t width = header[7];
|
||||
|
||||
fixed_font<bool> font(width, height);
|
||||
|
||||
const uint32_t unicode_start = glyphs_start + glyph_count * glyph_length;
|
||||
fseek(input, unicode_start, SEEK_SET);
|
||||
|
||||
uint32_t indices[128];
|
||||
|
||||
for (uint32_t index = 0; index < glyph_count; ++index) {
|
||||
uint8_t ch;
|
||||
fread(&ch, 1, 1, input);
|
||||
if (ch < 128)
|
||||
indices[ch] = index;
|
||||
do
|
||||
if (fread(&ch, 1, 1, input) != 1)
|
||||
return {};
|
||||
while (ch != 0xff);
|
||||
}
|
||||
|
||||
for (uint8_t ch = 0; ch < 128; ++ch) {
|
||||
fseek(input, glyphs_start + glyph_length * indices[ch], SEEK_SET);
|
||||
for (unsigned h = 0; h < height; ++h)
|
||||
for (unsigned wb = 0; wb < width; wb += 8) {
|
||||
uint8_t byte;
|
||||
fread(&byte, 1, 1, input);
|
||||
for (unsigned x = 0; x < 8 && wb + x < width; ++x)
|
||||
font.glyphs[ch].at(wb + x, h) = (byte >> (7 - x)) & 1;
|
||||
}
|
||||
}
|
||||
|
||||
return font;
|
||||
|
||||
}
|
||||
|
||||
}
|
49
makefile
49
makefile
|
@ -1,29 +1,32 @@
|
|||
LIMINE_DIR = $(abspath dependencies/limine)
|
||||
MINTSUKI_HEADERS_DIR = $(abspath dependencies/mintsuki-headers)
|
||||
TOOLCHAIN_DIR = $(abspath toolchain)
|
||||
|
||||
EXTRA_CC_ARGS = -Wall -Wextra -Og -ggdb -fno-exceptions
|
||||
|
||||
HILBERT_NASM = nasm -f elf64
|
||||
HILBERT_CC = ${TOOLCHAIN_DIR}/usr/bin/x86_64-elf-c++ -std=c++17 -Wall \
|
||||
-Wextra -O3 -ggdb -static -fno-exceptions -fno-rtti -mno-sse -I include \
|
||||
-I $(abspath dependencies/mintsuki-headers) -I $(abspath euler/include) \
|
||||
-I $(abspath libraries/daguerre/include)
|
||||
HILBERT_CC = ${TOOLCHAIN_DIR}/usr/bin/x86_64-elf-c++ -std=c++20 \
|
||||
${EXTRA_CC_ARGS} -static -mno-sse -I include -I $(abspath euler/include) \
|
||||
-I $(abspath libraries/daguerre/include) -I ${MINTSUKI_HEADERS_DIR}
|
||||
HILBERT_AR = ${TOOLCHAIN_DIR}/usr/bin/x86_64-elf-ar
|
||||
HILBERT_LD = ${TOOLCHAIN_DIR}/usr/bin/x86_64-elf-ld -z noexecstack
|
||||
|
||||
.EXPORT_ALL_VARIABLES:
|
||||
|
||||
LIB_DIR = toolchain/usr/x86_64-elf/lib/
|
||||
LIB_DIR = ${TOOLCHAIN_DIR}/usr/x86_64-elf/lib
|
||||
|
||||
LIMINE_DEP = dependencies/.limine-done
|
||||
MINTSUKI_HEADERS_DEP = dependencies/.mintsuki-headers-done
|
||||
BINUTILS_DEP = toolchain/.binutils-done
|
||||
GCC_DEP = toolchain/.gcc-done
|
||||
LIBGCC_DEP = toolchain/.libgcc-done
|
||||
LIBSTDCPP_DEP = toolchain/.libstdcpp-done
|
||||
|
||||
EULER_DEP = toolchain/.euler-done
|
||||
DAGUERRE_DEP = toolchain/.daguerre-done
|
||||
|
||||
APP_DEPS = ${GCC_DEP} ${LIBGCC_DEP} ${EULER_DEP} ${MINTSUKI_HEADERS_DEP}
|
||||
LIBRARY_DEPS = ${GCC_DEP} ${MINTSUKI_HEADERS_DEP}
|
||||
APP_DEPS = ${GCC_DEP} ${LIBGCC_DEP} ${LIBSTDCPP_DEP} ${EULER_DEP}
|
||||
LIBRARY_DEPS = ${GCC_DEP} ${LIBSTDCPP_DEP}
|
||||
|
||||
.PHONY: default run clean clean-dependencies
|
||||
|
||||
|
@ -53,7 +56,10 @@ ${LIMINE_DEP}:
|
|||
|
||||
${MINTSUKI_HEADERS_DEP}:
|
||||
mkdir -p dependencies
|
||||
test -e dependencies/mintsuki-headers || git clone --depth 1 https://github.com/mintsuki/freestanding-headers dependencies/mintsuki-headers
|
||||
test -e dependencies/mintsuki-headers || git clone --depth 1 https://github.com/osdev0/freestanding-headers dependencies/mintsuki-headers
|
||||
cd dependencies/mintsuki-headers && git fetch --depth=1 origin dd3abd2d7147efc4170dff478d3b7730bed14147
|
||||
cd dependencies/mintsuki-headers && git checkout dd3abd2d7147efc4170dff478d3b7730bed14147
|
||||
patch dependencies/mintsuki-headers/stddef.h patches/mintsuki-stddef.patch
|
||||
touch $@
|
||||
|
||||
${BINUTILS_DEP}:
|
||||
|
@ -71,8 +77,10 @@ ${GCC_DEP}: ${BINUTILS_DEP}
|
|||
test -e dependencies/gcc || git clone --depth 1 -b releases/gcc-14.1.0 https://gcc.gnu.org/git/gcc dependencies/gcc
|
||||
mkdir -p dependencies/gcc/build
|
||||
cd dependencies/gcc/build && ../configure --disable-fixed-point \
|
||||
--disable-gcov --disable-multilib --disable-shared --enable-languages=c++ \
|
||||
--target=x86_64-elf --prefix=${TOOLCHAIN_DIR}/usr --without-headers
|
||||
--disable-gcov --disable-multilib --disable-shared \
|
||||
--disable-hosted-libstdcxx \
|
||||
--enable-languages=c++ --target=x86_64-elf --enable-cstdio=stdio_pure \
|
||||
--prefix=${TOOLCHAIN_DIR}/usr --without-headers --enable-cxx-flags=-mno-sse
|
||||
+make -C dependencies/gcc/build all-gcc
|
||||
+make -C dependencies/gcc/build install-gcc
|
||||
touch $@
|
||||
|
@ -82,17 +90,22 @@ ${LIBGCC_DEP}: ${GCC_DEP}
|
|||
+make -C dependencies/gcc/build install-target-libgcc
|
||||
touch $@
|
||||
|
||||
${EULER_DEP}: ${GCC_DEP} ${MINTSUKI_HEADERS_DEP}
|
||||
+make -C euler build/crt0.o build/libc.a build/libg.a build/libm.a \
|
||||
build/libstdc++.a
|
||||
mkdir -p toolchain/usr/x86_64-elf/lib
|
||||
cp euler/build/crt0.o euler/build/libc.a euler/build/libg.a \
|
||||
euler/build/libm.a euler/build/libstdc++.a ${LIB_DIR}
|
||||
${LIBSTDCPP_DEP}: ${GCC_DEP}
|
||||
+make -C dependencies/gcc/build all-target-libstdc++-v3
|
||||
+make -C dependencies/gcc/build install-target-libstdc++-v3
|
||||
patch toolchain/usr/x86_64-elf/include/c++/14.1.0/memory patches/gcc-memory.patch
|
||||
touch $@
|
||||
|
||||
${EULER_DEP}: ${LIBRARY_DEPS}
|
||||
+make -C euler build/crt0.o build/libc.a build/libg.a build/libm.a
|
||||
mkdir -p ${LIB_DIR}
|
||||
cp euler/build/crt0.o euler/build/libc.a \
|
||||
euler/build/libg.a euler/build/libm.a ${LIB_DIR}/
|
||||
touch $@
|
||||
|
||||
${DAGUERRE_DEP}: ${LIBRARY_DEPS}
|
||||
+make -C libraries/daguerre build/libdaguerre.a
|
||||
cp libraries/daguerre/build/libdaguerre.a ${LIB_DIR}
|
||||
cp libraries/daguerre/build/libdaguerre.a ${LIB_DIR}/
|
||||
touch $@
|
||||
|
||||
kernel/build/kernel.elf: ${GCC_DEP} ${MINTSUKI_HEADERS_DEP} ${LIMINE_DEP}
|
||||
|
@ -101,7 +114,7 @@ kernel/build/kernel.elf: ${GCC_DEP} ${MINTSUKI_HEADERS_DEP} ${LIMINE_DEP}
|
|||
applications/init/build/init.elf: ${APP_DEPS}
|
||||
+make -C applications/init build/init.elf
|
||||
|
||||
applications/goldman/build/goldman.elf: ${APP_DEPS}
|
||||
applications/goldman/build/goldman.elf: ${APP_DEPS} ${DAGUERRE_DEP}
|
||||
+make -C applications/goldman build/goldman.elf
|
||||
|
||||
build/initfs.tgz: applications/init/build/init.elf \
|
||||
|
|
2
patches/gcc-memory.patch
Normal file
2
patches/gcc-memory.patch
Normal file
|
@ -0,0 +1,2 @@
|
|||
67a68
|
||||
> #include <std/allocator.hpp>
|
14
patches/mintsuki-stddef.patch
Normal file
14
patches/mintsuki-stddef.patch
Normal file
|
@ -0,0 +1,14 @@
|
|||
diff --git a/stddef.h b/stddef.h
|
||||
index 376eb75..fb1e99d 100644
|
||||
--- a/stddef.h
|
||||
+++ b/stddef.h
|
||||
@@ -50,4 +50,9 @@ typedef decltype(nullptr) nullptr_t;
|
||||
# define unreachable() __builtin_unreachable()
|
||||
#endif
|
||||
|
||||
+typedef struct {
|
||||
+ long long ll __attribute__((__aligned__(__alignof__(long long))));
|
||||
+ long double ld __attribute__((__aligned__(__alignof__(long double))));
|
||||
+} max_align_t;
|
||||
+
|
||||
#endif
|
1
qemu.gdb
1
qemu.gdb
|
@ -1,5 +1,6 @@
|
|||
target remote | qemu-system-x86_64 -gdb stdio -cdrom build/disk.iso -boot d
|
||||
symbol-file kernel/build/kernel.elf
|
||||
add-symbol-file applications/goldman/build/goldman.elf
|
||||
set disassembly-flavor intel
|
||||
set print asm-demangle on
|
||||
layout src
|
||||
|
|
15
readme.txt
15
readme.txt
|
@ -22,6 +22,8 @@ acknowledgements (any under "dependencies" are downloaded during build):
|
|||
dependencies/gcc/COPYING3 (gnu gpl v3)
|
||||
dependencies/gcc/COPYING.RUNTIME (gcc runtime library exception v3.1)
|
||||
homepage: https://gcc.gnu.org/
|
||||
i patch the output <memory> header file from libstdc++ with
|
||||
patches/gcc-memory.patch to include my std::allocator implementation.
|
||||
|
||||
- dependencies/limine (limine bootloader v7.5.1)
|
||||
copyright 2019 - 2024 mintsuki and contributors
|
||||
|
@ -31,7 +33,11 @@ acknowledgements (any under "dependencies" are downloaded during build):
|
|||
- dependencies/minstuki-headers
|
||||
copyright 2022 - 2024 mintsuki and contributors
|
||||
license: dependencies/mintsuki-headers/LICENSE (bsd zero-clause)
|
||||
homepage: https://github.com/mintsuki/freestanding-headers/
|
||||
homepage: https://github.com/osdev0/freestanding-headers/
|
||||
i patch the <stddef.h> header file with patches/minstuki-stddef.patch
|
||||
to add a max_align_t type to make libstdc++ happy. i use the commit
|
||||
dd3abd2d7147efc4170dff478d3b7730bed14147 so i don't have to worry
|
||||
about that file changing in a future commit.
|
||||
|
||||
- skeleton/assets/burden.ppm
|
||||
("selective focus photography snowflakes" by aaron burden)
|
||||
|
@ -71,8 +77,8 @@ project structure:
|
|||
some descriptions of things that have not been created yet.
|
||||
|
||||
- euler:
|
||||
(a minimal start to) a c/c++ standard library and runtime. the plan is to
|
||||
follow the c++17 standard, and only add things as i need them.
|
||||
(a minimal start to) a c++20 standard library, plus some hilbert
|
||||
specific functions. this uses the freestanding part of libstdc++.
|
||||
|
||||
- kernel:
|
||||
the kernel.
|
||||
|
@ -80,5 +86,8 @@ project structure:
|
|||
- libraries/daguerre:
|
||||
an image loading / rendering library.
|
||||
|
||||
- patches:
|
||||
a couple patches that are applied to dependencies
|
||||
|
||||
- skeleton:
|
||||
files that are copied directly to the initfs.
|
||||
|
|
1
skeleton/assets/background.ppm
Symbolic link
1
skeleton/assets/background.ppm
Symbolic link
|
@ -0,0 +1 @@
|
|||
burden.ppm
|
Reference in a new issue