a whole lot more
This commit is contained in:
parent
338549f9cd
commit
97c79ff771
21 changed files with 1921 additions and 1153 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -1,2 +1,3 @@
|
|||
.vscode/
|
||||
test/test
|
||||
obj/
|
||||
bin/
|
||||
|
|
324
bench/bench_window.cpp
Normal file
324
bench/bench_window.cpp
Normal file
|
@ -0,0 +1,324 @@
|
|||
#include <fstream>
|
||||
|
||||
#include "bench_window.hpp"
|
||||
#include "main.hpp"
|
||||
|
||||
bench_window::bench_window()
|
||||
: runner_stop_now(false),
|
||||
runner_stop_on_death(true),
|
||||
runner_stop_on_win(true),
|
||||
runner_update_ui(true),
|
||||
control_box(Gtk::Orientation::VERTICAL),
|
||||
new_round_button("new round"),
|
||||
single_step_button("single step"),
|
||||
start_button("start"),
|
||||
stop_button("stop"),
|
||||
draw_each_step_toggle("draw each step"),
|
||||
pause_on_death_toggle("pause on death"),
|
||||
pause_on_win_toggle("pause on win"),
|
||||
add_warrior_button("add warrior"),
|
||||
remove_warrior_button("remove warrior"),
|
||||
runner({}),
|
||||
runner_active(false) {
|
||||
|
||||
warrior_list_store = Gtk::TreeStore::create(warrior_list_columns);
|
||||
instructions_store = Gtk::TreeStore::create(instructions_columns);
|
||||
|
||||
warrior_list_view.set_model(warrior_list_store);
|
||||
instructions_view.set_model(instructions_store);
|
||||
|
||||
warrior_list_view.get_selection()->set_mode(Gtk::SelectionMode::SINGLE);
|
||||
instructions_view.get_selection()->set_mode(Gtk::SelectionMode::NONE);
|
||||
|
||||
warrior_list_view.get_selection()->signal_changed().connect(sigc::mem_fun(*this, &bench_window::update_buttons));
|
||||
|
||||
warrior_list_view.append_column("name", warrior_list_columns.warrior_name);
|
||||
warrior_list_view.append_column("#p", warrior_list_columns.processes);
|
||||
warrior_list_view.append_column("pc", warrior_list_columns.next_pc);
|
||||
|
||||
instructions_view.append_column("", instructions_columns.address);
|
||||
instructions_view.append_column("", instructions_columns.instruction);
|
||||
|
||||
for (lib94::number_t i = 0; i < LIB94_CORE_SIZE; ++i) {
|
||||
auto row = instructions_store->append();
|
||||
(*row)[instructions_columns.address] = i;
|
||||
}
|
||||
|
||||
update_ui();
|
||||
|
||||
draw_each_step_toggle.activate();
|
||||
pause_on_death_toggle.activate();
|
||||
pause_on_win_toggle.activate();
|
||||
|
||||
new_round_button.signal_clicked().connect(sigc::mem_fun(*this, &bench_window::on_click_new_round));
|
||||
single_step_button.signal_clicked().connect(sigc::mem_fun(*this, &bench_window::on_click_single_step));
|
||||
start_button.signal_clicked().connect(sigc::mem_fun(*this, &bench_window::on_click_start));
|
||||
stop_button.signal_clicked().connect(sigc::mem_fun(*this, &bench_window::on_click_stop));
|
||||
add_warrior_button.signal_clicked().connect(sigc::mem_fun(*this, &bench_window::on_click_add_warrior));
|
||||
remove_warrior_button.signal_clicked().connect(sigc::mem_fun(*this, &bench_window::on_click_remove_warrior));
|
||||
|
||||
draw_each_step_toggle.signal_toggled().connect(sigc::mem_fun(*this, &bench_window::on_toggle_draw_each_step));
|
||||
pause_on_death_toggle.signal_toggled().connect(sigc::mem_fun(*this, &bench_window::on_toggle_pause_on_death));
|
||||
pause_on_win_toggle.signal_toggled().connect(sigc::mem_fun(*this, &bench_window::on_toggle_pause_on_win));
|
||||
|
||||
control_box.append(new_round_button);
|
||||
control_box.append(single_step_button);
|
||||
control_box.append(start_button);
|
||||
control_box.append(stop_button);
|
||||
control_box.append(draw_each_step_toggle);
|
||||
control_box.append(pause_on_death_toggle);
|
||||
control_box.append(pause_on_win_toggle);
|
||||
control_box.append(add_warrior_button);
|
||||
control_box.append(remove_warrior_button);
|
||||
|
||||
warrior_list_scroll.set_child(warrior_list_view);
|
||||
warrior_list_scroll.set_policy(Gtk::PolicyType::NEVER, Gtk::PolicyType::AUTOMATIC);
|
||||
warrior_list_scroll.set_vexpand();
|
||||
|
||||
control_box.append(warrior_list_scroll);
|
||||
|
||||
control_box.append(core_rate_label);
|
||||
control_box.append(core_render_label);
|
||||
|
||||
instructions_scroll.set_child(instructions_view);
|
||||
instructions_scroll.set_policy(Gtk::PolicyType::NEVER, Gtk::PolicyType::AUTOMATIC);
|
||||
|
||||
control_box.set_size_request(200, -1);
|
||||
core.set_expand();
|
||||
instructions_scroll.set_size_request(200, -1);
|
||||
|
||||
main_box.append(control_box);
|
||||
main_box.append(core);
|
||||
main_box.append(instructions_scroll);
|
||||
|
||||
set_child(main_box);
|
||||
set_title("lib94 bench");
|
||||
|
||||
runner_update_ui_dispatcher.connect(sigc::mem_fun(*this, &bench_window::update_ui));
|
||||
runner_stopping_dispatcher.connect(sigc::mem_fun(*this, &bench_window::on_runner_stopping));
|
||||
}
|
||||
|
||||
void bench_window::add_modified_for_instruction_view(std::set<lib94::number_t> the_set) {
|
||||
for (lib94::number_t n : the_set)
|
||||
modified_addresses_for_instructions_view.insert(n);
|
||||
}
|
||||
|
||||
const lib94::warrior *bench_window::do_step() {
|
||||
auto time = std::chrono::system_clock::now();
|
||||
last_core_step_distance = time - last_core_step_start;
|
||||
last_core_step_start = time;
|
||||
|
||||
const lib94::warrior *result = lib94::single_step();
|
||||
|
||||
core.mut.lock();
|
||||
core.age_all();
|
||||
core.add_new_writes(lib94::get_written_addresses());
|
||||
core.add_new_reads(lib94::get_read_addresses());
|
||||
core.add_new_executions(lib94::get_executed_addresses());
|
||||
core.mut.unlock();
|
||||
|
||||
add_modified_for_instruction_view(lib94::get_written_addresses());
|
||||
add_modified_for_instruction_view(lib94::get_read_addresses());
|
||||
add_modified_for_instruction_view(lib94::get_executed_addresses());
|
||||
|
||||
lib94::clear_address_sets();
|
||||
return result;
|
||||
}
|
||||
|
||||
void bench_window::runner_main() {
|
||||
std::chrono::system_clock::time_point last_step = std::chrono::system_clock::now();
|
||||
|
||||
core_mutex.lock();
|
||||
while (!runner_stop_now) {
|
||||
bool death = do_step() != 0;
|
||||
|
||||
if (runner_stop_on_death && death)
|
||||
break;
|
||||
if (runner_stop_on_win && death && lib94::alive_warrior_count() == 1)
|
||||
break;
|
||||
if (runner_update_ui)
|
||||
runner_update_ui_dispatcher.emit();
|
||||
|
||||
core_mutex.unlock();
|
||||
std::this_thread::sleep_until(last_step + time_between_steps);
|
||||
last_step = std::chrono::system_clock::now();
|
||||
core_mutex.lock();
|
||||
}
|
||||
|
||||
runner_stopping_dispatcher.emit();
|
||||
core_mutex.unlock();
|
||||
}
|
||||
|
||||
void bench_window::on_click_single_step() {
|
||||
do_step();
|
||||
update_ui();
|
||||
}
|
||||
|
||||
void bench_window::on_click_new_round() {
|
||||
lib94::clear_core({
|
||||
.op = lib94::DAT,
|
||||
.mod = lib94::F,
|
||||
.amode = lib94::DIRECT,
|
||||
.bmode = lib94::DIRECT,
|
||||
.anumber = 0,
|
||||
.bnumber = 0
|
||||
});
|
||||
|
||||
lib94::init_round(warriors.data(), warriors.size());
|
||||
|
||||
core.mut.lock();
|
||||
core.age_scale = std::pow(2.0 / 3.0, 1.0 / (float)warriors.size());
|
||||
core.clear_all();
|
||||
core.mut.unlock();
|
||||
|
||||
modified_addresses_for_instructions_view.clear();
|
||||
for (lib94::number_t i = 0; i < LIB94_CORE_SIZE; ++i)
|
||||
instructions_store->children()[i][instructions_columns.instruction]
|
||||
= lib94::instruction_to_string(lib94::get_instruction(i));
|
||||
|
||||
update_ui();
|
||||
}
|
||||
|
||||
void bench_window::on_click_start() {
|
||||
runner_active = true;
|
||||
update_ui();
|
||||
runner_stop_now = false;
|
||||
runner = std::thread(sigc::mem_fun(*this, &bench_window::runner_main));
|
||||
}
|
||||
|
||||
void bench_window::on_click_stop() {
|
||||
runner_stop_now = true;
|
||||
}
|
||||
|
||||
void bench_window::on_toggle_draw_each_step() {
|
||||
runner_update_ui = draw_each_step_toggle.get_active();
|
||||
}
|
||||
|
||||
void bench_window::on_toggle_pause_on_death() {
|
||||
runner_stop_on_death = pause_on_death_toggle.get_active();
|
||||
}
|
||||
|
||||
void bench_window::on_toggle_pause_on_win() {
|
||||
runner_stop_on_win = pause_on_win_toggle.get_active();
|
||||
}
|
||||
|
||||
void bench_window::on_add_warrior_dialog_response(int response_id, Gtk::FileChooserDialog *dialog) {
|
||||
if (response_id == Gtk::ResponseType::OK) {
|
||||
|
||||
Glib::RefPtr<Gio::File> file = dialog->get_file();
|
||||
char *contents;
|
||||
gsize length;
|
||||
file->load_contents(contents, length);
|
||||
|
||||
auto w = lib94::compile_warrior(std::string(contents, length));
|
||||
|
||||
delete contents;
|
||||
|
||||
if (std::holds_alternative<lib94::warrior *>(w)) {
|
||||
warriors.push_back(std::get<lib94::warrior *>(w));
|
||||
on_click_new_round();
|
||||
}
|
||||
|
||||
else {
|
||||
Gtk::MessageDialog *md = new Gtk::MessageDialog(std::string("Failed to compile: ") + std::get<std::string>(w));
|
||||
md->set_transient_for(*this);
|
||||
md->set_modal();
|
||||
md->signal_response().connect([md](int) {delete md;});
|
||||
md->show();
|
||||
}
|
||||
}
|
||||
|
||||
delete dialog;
|
||||
}
|
||||
|
||||
void bench_window::on_click_add_warrior() {
|
||||
Gtk::FileChooserDialog *dialog = new Gtk::FileChooserDialog("select a warrior");
|
||||
|
||||
dialog->set_transient_for(*this);
|
||||
dialog->set_modal();
|
||||
dialog->signal_response().connect(sigc::bind(sigc::mem_fun(*this, &bench_window::on_add_warrior_dialog_response), dialog));
|
||||
dialog->add_button("add", Gtk::ResponseType::OK);
|
||||
dialog->add_button("cancel", Gtk::ResponseType::CANCEL);
|
||||
|
||||
Glib::RefPtr<Gtk::FileFilter> text_filter = Gtk::FileFilter::create();
|
||||
text_filter->add_mime_type("text/plain");
|
||||
text_filter->set_name("text files");
|
||||
dialog->add_filter(text_filter);
|
||||
|
||||
Glib::RefPtr<Gtk::FileFilter> any_filter = Gtk::FileFilter::create();
|
||||
any_filter->add_pattern("*");
|
||||
any_filter->set_name("all files");
|
||||
dialog->add_filter(any_filter);
|
||||
|
||||
dialog->show();
|
||||
}
|
||||
|
||||
void bench_window::on_click_remove_warrior() {
|
||||
auto iter = warrior_list_view.get_selection()->get_selected();
|
||||
if (iter) {
|
||||
|
||||
while (!(*iter)[warrior_list_columns.warrior])
|
||||
iter = iter->parent();
|
||||
|
||||
const lib94::warrior *w = (*iter)[warrior_list_columns.warrior];
|
||||
|
||||
for (auto i = warriors.begin(); i != warriors.end(); ++i)
|
||||
if (*i == w) {
|
||||
warriors.erase(i);
|
||||
break;
|
||||
}
|
||||
|
||||
on_click_new_round();
|
||||
|
||||
delete w;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void bench_window::update_buttons() {
|
||||
new_round_button.set_sensitive(!runner_active && warriors.size() > 0);
|
||||
single_step_button.set_sensitive(!runner_active && lib94::alive_warrior_count() > 0);
|
||||
start_button.set_sensitive(!runner_active && lib94::alive_warrior_count() > 0);
|
||||
stop_button.set_sensitive(runner_active);
|
||||
add_warrior_button.set_sensitive(!runner_active);
|
||||
remove_warrior_button.set_sensitive(!runner_active && warrior_list_view.get_selection()->get_selected());
|
||||
}
|
||||
|
||||
void bench_window::update_ui() {
|
||||
if (runner_active)
|
||||
core_mutex.lock();
|
||||
|
||||
update_buttons();
|
||||
|
||||
core_render_label.set_text("core render: " + ns_to_string(core.last_draw_time));
|
||||
core_rate_label.set_text("core rate: " + hz_to_string(1000000000.0 / (double)last_core_step_distance.count()));
|
||||
|
||||
core.queue_draw();
|
||||
|
||||
warrior_list_store->clear();
|
||||
|
||||
for (const lib94::warrior *w : warriors) {
|
||||
auto w_row = warrior_list_store->append();
|
||||
(*w_row)[warrior_list_columns.warrior] = w;
|
||||
(*w_row)[warrior_list_columns.warrior_name] = w->name;
|
||||
auto procs = lib94::get_processes(w);
|
||||
(*w_row)[warrior_list_columns.processes] = procs.size();
|
||||
(*w_row)[warrior_list_columns.next_pc] = procs.size() ? std::to_string(procs[0]) : "";
|
||||
}
|
||||
|
||||
//warrior_list_view.expand_all();
|
||||
|
||||
for (lib94::number_t i : modified_addresses_for_instructions_view)
|
||||
instructions_store->children()[i][instructions_columns.instruction]
|
||||
= lib94::instruction_to_string(lib94::get_instruction(i));
|
||||
modified_addresses_for_instructions_view.clear();
|
||||
|
||||
if (runner_active)
|
||||
core_mutex.unlock();
|
||||
}
|
||||
|
||||
void bench_window::on_runner_stopping() {
|
||||
runner.join();
|
||||
runner_active = false;
|
||||
update_ui();
|
||||
}
|
112
bench/bench_window.hpp
Normal file
112
bench/bench_window.hpp
Normal file
|
@ -0,0 +1,112 @@
|
|||
#ifndef LIB94_BENCH_BENCH_WINDOW_HPP
|
||||
#define LIB94_BENCH_BENCH_WINDOW_HPP
|
||||
|
||||
#include <gtkmm.h>
|
||||
#include <thread>
|
||||
|
||||
#include "core_widget.hpp"
|
||||
|
||||
class bench_window : public Gtk::Window {
|
||||
public:
|
||||
bench_window();
|
||||
|
||||
Glib::Dispatcher runner_update_ui_dispatcher;
|
||||
Glib::Dispatcher runner_stopping_dispatcher;
|
||||
|
||||
bool runner_stop_now;
|
||||
bool runner_stop_on_death;
|
||||
bool runner_stop_on_win;
|
||||
bool runner_update_ui;
|
||||
|
||||
core_widget core;
|
||||
|
||||
private:
|
||||
class warrior_list_columns_record : public Gtk::TreeModelColumnRecord {
|
||||
public:
|
||||
Gtk::TreeModelColumn<Glib::ustring> warrior_name;
|
||||
Gtk::TreeModelColumn<size_t> processes;
|
||||
Gtk::TreeModelColumn<Glib::ustring> next_pc;
|
||||
Gtk::TreeModelColumn<const lib94::warrior *> warrior;
|
||||
|
||||
warrior_list_columns_record() {
|
||||
add(warrior_name);
|
||||
add(processes);
|
||||
add(next_pc);
|
||||
add(warrior);
|
||||
}
|
||||
};
|
||||
|
||||
class instructions_columns_record : public Gtk::TreeModelColumnRecord {
|
||||
public:
|
||||
Gtk::TreeModelColumn<lib94::number_t> address;
|
||||
Gtk::TreeModelColumn<Glib::ustring> instruction;
|
||||
|
||||
instructions_columns_record() {
|
||||
add(address);
|
||||
add(instruction);
|
||||
}
|
||||
};
|
||||
|
||||
Gtk::ScrolledWindow warrior_list_scroll;
|
||||
Gtk::ScrolledWindow instructions_scroll;
|
||||
|
||||
warrior_list_columns_record warrior_list_columns;
|
||||
instructions_columns_record instructions_columns;
|
||||
|
||||
Glib::RefPtr<Gtk::TreeStore> warrior_list_store;
|
||||
Glib::RefPtr<Gtk::TreeStore> instructions_store;
|
||||
|
||||
Gtk::TreeView warrior_list_view;
|
||||
Gtk::TreeView instructions_view;
|
||||
|
||||
std::set<lib94::number_t> modified_addresses_for_instructions_view;
|
||||
|
||||
void add_modified_for_instruction_view(std::set<lib94::number_t> the_set);
|
||||
|
||||
Gtk::Box main_box;
|
||||
Gtk::Box control_box;
|
||||
|
||||
Gtk::Button new_round_button;
|
||||
Gtk::Button single_step_button;
|
||||
Gtk::Button start_button;
|
||||
Gtk::Button stop_button;
|
||||
|
||||
Gtk::CheckButton draw_each_step_toggle;
|
||||
Gtk::CheckButton pause_on_death_toggle;
|
||||
Gtk::CheckButton pause_on_win_toggle;
|
||||
|
||||
Gtk::Button add_warrior_button;
|
||||
Gtk::Button remove_warrior_button;
|
||||
|
||||
std::chrono::system_clock::time_point last_core_step_start;
|
||||
std::chrono::nanoseconds last_core_step_distance;
|
||||
Gtk::Label core_rate_label;
|
||||
Gtk::Label core_render_label;
|
||||
|
||||
const lib94::warrior *do_step();
|
||||
void runner_main();
|
||||
|
||||
void on_click_new_round();
|
||||
void on_click_single_step();
|
||||
void on_click_start();
|
||||
void on_click_stop();
|
||||
|
||||
void on_toggle_draw_each_step();
|
||||
void on_toggle_pause_on_death();
|
||||
void on_toggle_pause_on_win();
|
||||
|
||||
void on_click_add_warrior();
|
||||
void on_click_remove_warrior();
|
||||
|
||||
std::thread runner;
|
||||
bool runner_active;
|
||||
|
||||
void update_buttons();
|
||||
void update_ui();
|
||||
|
||||
void on_runner_stopping();
|
||||
|
||||
void on_add_warrior_dialog_response(int response_id, Gtk::FileChooserDialog *dialog);
|
||||
};
|
||||
|
||||
#endif
|
162
bench/core_widget.cpp
Normal file
162
bench/core_widget.cpp
Normal file
|
@ -0,0 +1,162 @@
|
|||
#include "core_widget.hpp"
|
||||
|
||||
core_widget::core_widget() {
|
||||
clear_all();
|
||||
}
|
||||
|
||||
void core_widget::clear_all() {
|
||||
for (int i = 0; i < LIB94_CORE_SIZE; ++i) {
|
||||
|
||||
if (write_values[i] != 0)
|
||||
to_draw_write.insert(i);
|
||||
write_values[i] = 0;
|
||||
|
||||
if (read_values[i] != 0)
|
||||
to_draw_read.insert(i);
|
||||
read_values[i] = 0;
|
||||
|
||||
if (execute_values[i] != 0)
|
||||
to_draw_execute.insert(i);
|
||||
execute_values[i] = 0;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void core_widget::age_all() {
|
||||
for (int i = 0; i < LIB94_CORE_SIZE; ++i) {
|
||||
|
||||
if (write_values[i] != 0) {
|
||||
to_draw_write.insert(i);
|
||||
write_values[i] = (uint8_t)((float)write_values[i] * age_scale);
|
||||
}
|
||||
|
||||
if (read_values[i] != 0) {
|
||||
to_draw_read.insert(i);
|
||||
read_values[i] = (uint8_t)((float)read_values[i] * age_scale);
|
||||
}
|
||||
|
||||
if (execute_values[i] != 0) {
|
||||
to_draw_execute.insert(i);
|
||||
execute_values[i] = (uint8_t)((float)execute_values[i] * age_scale);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void core_widget::add_new_writes(const std::set<lib94::number_t> &writes) {
|
||||
for (lib94::number_t n : writes) {
|
||||
write_values[n] = 255;
|
||||
to_draw_write.insert(n);
|
||||
}
|
||||
}
|
||||
|
||||
void core_widget::add_new_reads(const std::set<lib94::number_t> &reads) {
|
||||
for (lib94::number_t n : reads) {
|
||||
read_values[n] = 255;
|
||||
to_draw_read.insert(n);
|
||||
}
|
||||
}
|
||||
|
||||
void core_widget::add_new_executions(const std::set<lib94::number_t> &executions) {
|
||||
for (lib94::number_t n : executions) {
|
||||
execute_values[n] = 255;
|
||||
to_draw_execute.insert(n);
|
||||
}
|
||||
}
|
||||
|
||||
void core_widget::measure_vfunc(Gtk::Orientation, int for_size, int &minimum, int &natural, int &minimum_baseline, int &natural_baseline) const {
|
||||
minimum = (LIB94_CORE_SIZE - 1) / for_size + 1;
|
||||
natural = (LIB94_CORE_SIZE - 1) / for_size + 1;
|
||||
minimum_baseline = -1;
|
||||
natural_baseline = -1;
|
||||
}
|
||||
|
||||
void core_widget::draw(int width, int height) {
|
||||
mut.lock();
|
||||
|
||||
if (width == last_width && height == last_height) {
|
||||
|
||||
uint8_t *buffer = (uint8_t *)pixbuf->property_pixels().get_value();
|
||||
|
||||
for (lib94::number_t i : to_draw_write) {
|
||||
int cy = i / cpr * scale + ypad;
|
||||
int cx = i % cpr * scale + xpad;
|
||||
for (int dy = 0; dy < scale; ++dy)
|
||||
for (int dx = 0; dx < scale; ++dx)
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4] = write_values[i];
|
||||
}
|
||||
|
||||
for (lib94::number_t i : to_draw_read) {
|
||||
int cy = i / cpr * scale + ypad;
|
||||
int cx = i % cpr * scale + xpad;
|
||||
for (int dy = 0; dy < scale; ++dy)
|
||||
for (int dx = 0; dx < scale; ++dx)
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4 + 2] = read_values[i];
|
||||
}
|
||||
|
||||
for (lib94::number_t i : to_draw_execute) {
|
||||
int cy = i / cpr * scale + ypad;
|
||||
int cx = i % cpr * scale + xpad;
|
||||
for (int dy = 0; dy < scale; ++dy)
|
||||
for (int dx = 0; dx < scale; ++dx)
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4 + 1] = execute_values[i];
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
else {
|
||||
|
||||
pixbuf = Gdk::Pixbuf::create(Gdk::Colorspace::RGB, true, 8, width, height);
|
||||
last_width = width;
|
||||
last_height = height;
|
||||
|
||||
scale = 1;
|
||||
while (true) {
|
||||
++scale;
|
||||
if ((width / scale) * (height / scale) < LIB94_CORE_SIZE) {
|
||||
--scale;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
cpr = width / scale;
|
||||
xpad = (width % scale) / 2;
|
||||
ypad = (height % scale) / 2;
|
||||
|
||||
uint8_t *buffer = (uint8_t *)pixbuf->property_pixels().get_value();
|
||||
|
||||
for (lib94::number_t i = 0; i < LIB94_CORE_SIZE; ++i) {
|
||||
int cy = i / cpr * scale + ypad;
|
||||
int cx = i % cpr * scale + xpad;
|
||||
for (int dy = 0; dy < scale; ++dy)
|
||||
for (int dx = 0; dx < scale; ++dx) {
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4] = write_values[i];
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4 + 2] = read_values[i];
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4 + 1] = execute_values[i];
|
||||
buffer[(cy + dy) * pixbuf->property_rowstride() + (cx + dx) * 4 + 3] = 255;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
to_draw_write.clear();
|
||||
to_draw_read.clear();
|
||||
to_draw_execute.clear();
|
||||
|
||||
mut.unlock();
|
||||
}
|
||||
|
||||
void core_widget::snapshot_vfunc(const Glib::RefPtr<Gtk::Snapshot> &snapshot) {
|
||||
auto start_time = std::chrono::system_clock::now();
|
||||
|
||||
Gtk::Allocation allocation = get_allocation();
|
||||
Gdk::Rectangle allocation_rect(0, 0, allocation.get_width(), allocation.get_height());
|
||||
|
||||
draw(allocation.get_width(), allocation.get_height());
|
||||
|
||||
auto texture = Gdk::Texture::create_for_pixbuf(pixbuf);
|
||||
snapshot->append_texture(texture, allocation_rect);
|
||||
|
||||
auto end_time = std::chrono::system_clock::now();
|
||||
last_draw_time = end_time - start_time;
|
||||
}
|
46
bench/core_widget.hpp
Normal file
46
bench/core_widget.hpp
Normal file
|
@ -0,0 +1,46 @@
|
|||
#ifndef LIB94_BENCH_CORE_WIDGET_HPP
|
||||
#define LIB94_BENCH_CORE_WIDGET_HPP
|
||||
|
||||
#include <lib94/lib94.hpp>
|
||||
#include <gtkmm.h>
|
||||
|
||||
class core_widget : public Gtk::Widget {
|
||||
public:
|
||||
core_widget();
|
||||
void clear_all();
|
||||
void age_all();
|
||||
void add_new_writes(const std::set<lib94::number_t> &writes);
|
||||
void add_new_reads(const std::set<lib94::number_t> &reads);
|
||||
void add_new_executions(const std::set<lib94::number_t> &executions);
|
||||
|
||||
void measure_vfunc(Gtk::Orientation orientation, int for_size, int &minimum, int &natural, int &minimum_baseline, int &natural_baseline) const override;
|
||||
void snapshot_vfunc(const Glib::RefPtr<Gtk::Snapshot> &snapshot);
|
||||
|
||||
float age_scale;
|
||||
|
||||
std::chrono::nanoseconds last_draw_time;
|
||||
|
||||
std::mutex mut;
|
||||
|
||||
private:
|
||||
uint8_t write_values[LIB94_CORE_SIZE];
|
||||
uint8_t read_values[LIB94_CORE_SIZE];
|
||||
uint8_t execute_values[LIB94_CORE_SIZE];
|
||||
|
||||
std::set<lib94::number_t> to_draw_write;
|
||||
std::set<lib94::number_t> to_draw_read;
|
||||
std::set<lib94::number_t> to_draw_execute;
|
||||
|
||||
Glib::RefPtr<Gdk::Pixbuf> pixbuf;
|
||||
|
||||
int last_width;
|
||||
int last_height;
|
||||
int scale;
|
||||
int cpr;
|
||||
int xpad;
|
||||
int ypad;
|
||||
|
||||
void draw(int width, int height);
|
||||
};
|
||||
|
||||
#endif
|
31
bench/main.cpp
Normal file
31
bench/main.cpp
Normal file
|
@ -0,0 +1,31 @@
|
|||
#include "bench_window.hpp"
|
||||
#include "main.hpp"
|
||||
|
||||
std::vector<lib94::warrior *> warriors;
|
||||
std::chrono::milliseconds time_between_steps(50);
|
||||
std::mutex core_mutex;
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
auto app = Gtk::Application::create("net.benjidial.lib94.bench");
|
||||
return app->make_window_and_run<bench_window>(argc, argv);
|
||||
}
|
||||
|
||||
std::string ns_to_string(std::chrono::nanoseconds dur) {
|
||||
if (dur.count() >= 10000000000)
|
||||
return std::to_string((dur.count() + 500000000) / 1000000000) + "s";
|
||||
if (dur.count() >= 10000000)
|
||||
return std::to_string((dur.count() + 500000) / 1000000) + "ms";
|
||||
if (dur.count() >= 10000)
|
||||
return std::to_string((dur.count() + 500) / 1000) + "μs";
|
||||
return std::to_string(dur.count()) + "ns";
|
||||
}
|
||||
|
||||
std::string hz_to_string(double rate) {
|
||||
if (rate >= 10000000.0)
|
||||
return std::to_string((int)std::round(rate / 1000000.0)) + "MHz";
|
||||
if (rate >= 10000.0)
|
||||
return std::to_string((int)std::round(rate / 1000.0)) + "kHz";
|
||||
if (rate >= 10.0)
|
||||
return std::to_string((int)std::round(rate)) + "Hz";
|
||||
return std::to_string((int)std::round(rate * 1000.0)) + "mHz";
|
||||
}
|
16
bench/main.hpp
Normal file
16
bench/main.hpp
Normal file
|
@ -0,0 +1,16 @@
|
|||
#ifndef LIB94_BENCH_MAIN_HPP
|
||||
#define LIB94_BENCH_MAIN_HPP
|
||||
|
||||
#include <lib94/lib94.hpp>
|
||||
#include <chrono>
|
||||
#include <vector>
|
||||
#include <mutex>
|
||||
|
||||
extern std::vector<lib94::warrior *> warriors;
|
||||
extern std::chrono::milliseconds time_between_steps;
|
||||
extern std::mutex core_mutex;
|
||||
|
||||
std::string ns_to_string(std::chrono::nanoseconds dur);
|
||||
std::string hz_to_string(double rate);
|
||||
|
||||
#endif
|
|
@ -1,5 +1,5 @@
|
|||
#ifndef LIB94_HPP
|
||||
#define LIB94_HPP
|
||||
#ifndef LIB94_LIB94_HPP
|
||||
#define LIB94_LIB94_HPP
|
||||
|
||||
#include <filesystem>
|
||||
#include <optional>
|
||||
|
@ -7,82 +7,88 @@
|
|||
#include <string>
|
||||
#include <random>
|
||||
#include <vector>
|
||||
#include <queue>
|
||||
#include <deque>
|
||||
#include <set>
|
||||
|
||||
#ifndef CORE_SIZE
|
||||
#define CORE_SIZE 8000
|
||||
#ifndef LIB94_CORE_SIZE
|
||||
#define LIB94_CORE_SIZE 8000
|
||||
#endif
|
||||
|
||||
typedef uint32_t number_t;
|
||||
namespace lib94 {
|
||||
|
||||
enum opcode : uint8_t {
|
||||
DAT, MOV, ADD, SUB,
|
||||
MUL, DIV, MOD, JMP,
|
||||
JMZ, JMN, DJN, SEQ,
|
||||
SNE, SLT, SPL, NOP
|
||||
};
|
||||
typedef int_least32_t number_t;
|
||||
|
||||
enum modifier : uint8_t {
|
||||
A, B, AB, BA,
|
||||
F, X, I
|
||||
};
|
||||
enum opcode : uint8_t {
|
||||
DAT, MOV, ADD, SUB,
|
||||
MUL, DIV, MOD, JMP,
|
||||
JMZ, JMN, DJN, SEQ,
|
||||
SNE, SLT, SPL, NOP
|
||||
};
|
||||
|
||||
enum mode : uint8_t {
|
||||
IMMEDIATE, DIRECT,
|
||||
A_INDIRECT, B_INDIRECT,
|
||||
A_DECREMENT, B_DECREMENT,
|
||||
A_INCREMENT, B_INCREMENT
|
||||
};
|
||||
enum modifier : uint8_t {
|
||||
A, B, AB, BA,
|
||||
F, X, I
|
||||
};
|
||||
|
||||
struct instruction {
|
||||
opcode op;
|
||||
modifier mod;
|
||||
mode amode;
|
||||
mode bmode;
|
||||
number_t anumber;
|
||||
number_t bnumber;
|
||||
};
|
||||
enum mode : uint8_t {
|
||||
IMMEDIATE, DIRECT,
|
||||
A_INDIRECT, B_INDIRECT,
|
||||
A_DECREMENT, B_DECREMENT,
|
||||
A_INCREMENT, B_INCREMENT
|
||||
};
|
||||
|
||||
struct warrior {
|
||||
std::string name;
|
||||
std::string author;
|
||||
struct instruction {
|
||||
opcode op;
|
||||
modifier mod;
|
||||
mode amode;
|
||||
mode bmode;
|
||||
number_t anumber;
|
||||
number_t bnumber;
|
||||
};
|
||||
|
||||
number_t org;
|
||||
struct warrior {
|
||||
std::string name;
|
||||
std::string author;
|
||||
|
||||
std::vector<instruction> instructions;
|
||||
};
|
||||
number_t org;
|
||||
|
||||
void seed_prng(uint_fast64_t seed);
|
||||
std::vector<instruction> instructions;
|
||||
};
|
||||
|
||||
std::variant<warrior *, std::string> compile_warrior(std::string source);
|
||||
bool save_warrior(const warrior &w, const std::filesystem::path &to);
|
||||
std::optional<warrior *> load_warrior(const std::filesystem::path &from);
|
||||
void seed_prng(uint_fast64_t seed);
|
||||
|
||||
void clear_core(const instruction &background);
|
||||
void clear_core_random();
|
||||
std::string instruction_to_string(const instruction &instr);
|
||||
|
||||
//warrior pointers need to remain valid for other
|
||||
//functions to return valid things during the round
|
||||
bool init_round(const std::vector<const warrior *> &warriors);
|
||||
std::variant<warrior *, std::string> compile_warrior(std::string source);
|
||||
bool save_warrior(const warrior &w, const std::filesystem::path &to);
|
||||
std::optional<warrior *> load_warrior(const std::filesystem::path &from);
|
||||
|
||||
size_t alive_warrior_count();
|
||||
void clear_core(const instruction &background);
|
||||
void clear_core_random();
|
||||
|
||||
//asserts that there is a next warrior
|
||||
const warrior *get_next_warrior();
|
||||
const std::queue<number_t> &get_processes(const warrior *for_warrior);
|
||||
//asserts that there is a next process
|
||||
number_t get_next_process(const warrior *for_warrior);
|
||||
//warrior pointers need to remain valid for other
|
||||
//functions to return valid things during the round
|
||||
bool init_round(const warrior *const *warriors, size_t count);
|
||||
|
||||
const std::set<number_t> &get_modified_addresses();
|
||||
void clear_modified_addresses();
|
||||
size_t alive_warrior_count();
|
||||
|
||||
const instruction &get_instruction(number_t address);
|
||||
//asserts that there is a next warrior
|
||||
const warrior *get_next_warrior();
|
||||
const std::deque<number_t> &get_processes(const warrior *for_warrior);
|
||||
//asserts that there is a next process
|
||||
number_t get_next_process(const warrior *for_warrior);
|
||||
|
||||
//asserts that there is a next warrior
|
||||
void single_step();
|
||||
const std::set<number_t> &get_written_addresses();
|
||||
const std::set<number_t> &get_read_addresses();
|
||||
const std::set<number_t> &get_executed_addresses();
|
||||
void clear_address_sets();
|
||||
|
||||
const warrior *run_until_warrior_death();
|
||||
const warrior *run_until_winner();
|
||||
const instruction &get_instruction(number_t address);
|
||||
|
||||
//assumes that there is a next warrior
|
||||
//returns a warrior if it dies
|
||||
const warrior *single_step();
|
||||
|
||||
}
|
||||
|
||||
#endif
|
||||
|
|
546
lib94/core.cpp
546
lib94/core.cpp
|
@ -2,293 +2,303 @@
|
|||
#include <functional>
|
||||
#include <cassert>
|
||||
#include <random>
|
||||
#include <queue>
|
||||
#include <deque>
|
||||
#include <set>
|
||||
|
||||
static std::mt19937_64 prng;
|
||||
instruction core[CORE_SIZE];
|
||||
namespace lib94 {
|
||||
|
||||
void seed_prng(uint_fast64_t seed) {
|
||||
prng.seed(seed);
|
||||
}
|
||||
static std::mt19937_64 prng;
|
||||
instruction core[LIB94_CORE_SIZE];
|
||||
|
||||
void clear_core(const instruction &background) {
|
||||
for (instruction &i : core)
|
||||
i = background;
|
||||
}
|
||||
|
||||
void clear_core_random() {
|
||||
std::uniform_int_distribution<uint8_t> op(0, 15);
|
||||
std::uniform_int_distribution<uint8_t> mod(0, 6);
|
||||
std::uniform_int_distribution<uint8_t> modes(0, 7);
|
||||
std::uniform_int_distribution<number_t> number(0, CORE_SIZE - 1);
|
||||
|
||||
for (instruction &i : core)
|
||||
i = {
|
||||
.op = (opcode)op(prng),
|
||||
.mod = (modifier)mod(prng),
|
||||
.amode = (mode)modes(prng),
|
||||
.bmode = (mode)modes(prng),
|
||||
.anumber = number(prng),
|
||||
.bnumber = number(prng)
|
||||
};
|
||||
}
|
||||
|
||||
static std::set<number_t> modified_addresses;
|
||||
|
||||
struct warrior_info {
|
||||
const warrior *the_warrior;
|
||||
std::queue<number_t> processes;
|
||||
};
|
||||
|
||||
static std::vector<warrior_info> warrior_infos;
|
||||
std::queue<warrior_info *> alive_warriors;
|
||||
|
||||
bool init_round(const std::vector<const warrior *> &warriors) {
|
||||
modified_addresses.clear();
|
||||
warrior_infos.clear();
|
||||
alive_warriors = std::queue<warrior_info *>();
|
||||
|
||||
std::uniform_int_distribution<number_t> number(0, CORE_SIZE - 1);
|
||||
std::vector<std::pair<number_t, number_t>> placements;
|
||||
|
||||
for (const warrior *w : warriors) {
|
||||
unsigned tries = 0;
|
||||
|
||||
new_place_at:
|
||||
if (tries > 1000)
|
||||
return false;
|
||||
++tries;
|
||||
|
||||
number_t place_at = number(prng);
|
||||
|
||||
for (std::pair<number_t, number_t> &other : placements)
|
||||
//there has to be a better way
|
||||
for (number_t i = 0; i < w->instructions.size(); ++i)
|
||||
if (((place_at + i) % CORE_SIZE >= other.first && (place_at + i) % CORE_SIZE < other.first + other.second) ||
|
||||
((place_at + i) % CORE_SIZE + CORE_SIZE >= other.first && (place_at + i) % CORE_SIZE + CORE_SIZE < other.first + other.second))
|
||||
goto new_place_at;
|
||||
|
||||
placements.push_back(std::make_pair<>(place_at, w->instructions.size()));
|
||||
|
||||
for (number_t i = 0; i < w->instructions.size(); ++i)
|
||||
core[(place_at + i) % CORE_SIZE] = w->instructions[i];
|
||||
|
||||
warrior_infos.push_back({});
|
||||
warrior_info *wi = &warrior_infos.back();
|
||||
wi->the_warrior = w;
|
||||
wi->processes.push((place_at + w->org) % CORE_SIZE);
|
||||
void seed_prng(uint_fast64_t seed) {
|
||||
prng.seed(seed);
|
||||
}
|
||||
|
||||
std::shuffle(warrior_infos.begin(), warrior_infos.end(), prng);
|
||||
void clear_core(const instruction &background) {
|
||||
for (instruction &i : core)
|
||||
i = background;
|
||||
}
|
||||
|
||||
for (warrior_info &wi : warrior_infos)
|
||||
alive_warriors.push(&wi);
|
||||
void clear_core_random() {
|
||||
std::uniform_int_distribution<uint8_t> op(0, 15);
|
||||
std::uniform_int_distribution<uint8_t> mod(0, 6);
|
||||
std::uniform_int_distribution<uint8_t> modes(0, 7);
|
||||
std::uniform_int_distribution<number_t> number(0, LIB94_CORE_SIZE - 1);
|
||||
|
||||
return true;
|
||||
}
|
||||
for (instruction &i : core)
|
||||
i = {
|
||||
.op = (opcode)op(prng),
|
||||
.mod = (modifier)mod(prng),
|
||||
.amode = (mode)modes(prng),
|
||||
.bmode = (mode)modes(prng),
|
||||
.anumber = number(prng),
|
||||
.bnumber = number(prng)
|
||||
};
|
||||
}
|
||||
|
||||
size_t alive_warrior_count() {
|
||||
return alive_warriors.size();
|
||||
}
|
||||
static std::set<number_t> written_addresses;
|
||||
static std::set<number_t> read_addresses;
|
||||
static std::set<number_t> executed_addresses;
|
||||
|
||||
const warrior *get_next_warrior() {
|
||||
assert(alive_warriors.size() > 0);
|
||||
return alive_warriors.front()->the_warrior;
|
||||
}
|
||||
void add_written_instruction(const instruction *instr) {
|
||||
written_addresses.insert(instr - core);
|
||||
}
|
||||
|
||||
const std::queue<number_t> &get_processes(const warrior *for_warrior) {
|
||||
for (const warrior_info &wi : warrior_infos)
|
||||
if (wi.the_warrior == for_warrior)
|
||||
return wi.processes;
|
||||
assert(false);
|
||||
}
|
||||
void add_read_instruction(const instruction *instr) {
|
||||
read_addresses.insert(instr - core);
|
||||
}
|
||||
|
||||
number_t get_next_process(const warrior *for_warrior) {
|
||||
for (const warrior_info &wi : warrior_infos)
|
||||
if (wi.the_warrior == for_warrior) {
|
||||
assert(wi.processes.size() > 0);
|
||||
return wi.processes.front();
|
||||
void add_executed_instruction(const instruction *instr) {
|
||||
executed_addresses.insert(instr - core);
|
||||
}
|
||||
|
||||
struct warrior_info {
|
||||
const warrior *the_warrior;
|
||||
std::deque<number_t> processes;
|
||||
};
|
||||
|
||||
static std::vector<warrior_info> warrior_infos;
|
||||
std::deque<warrior_info *> alive_warriors;
|
||||
|
||||
bool init_round(const warrior *const *warriors, size_t count) {
|
||||
clear_address_sets();
|
||||
warrior_infos.clear();
|
||||
alive_warriors = std::deque<warrior_info *>();
|
||||
|
||||
std::uniform_int_distribution<number_t> number(0, LIB94_CORE_SIZE - 1);
|
||||
std::vector<std::pair<number_t, number_t>> placements;
|
||||
|
||||
for (size_t i = 0; i < count; ++i) {
|
||||
const warrior *w = warriors[i];
|
||||
unsigned tries = 0;
|
||||
|
||||
new_place_at:
|
||||
if (tries > 1000)
|
||||
return false;
|
||||
++tries;
|
||||
|
||||
number_t place_at = i == 0 ? 0 : number(prng);
|
||||
|
||||
for (std::pair<number_t, number_t> &other : placements)
|
||||
//there has to be a better way
|
||||
for (number_t i = 0; i < (number_t)w->instructions.size(); ++i)
|
||||
if (((place_at + i) % LIB94_CORE_SIZE >= other.first && (place_at + i) % LIB94_CORE_SIZE < other.first + other.second) ||
|
||||
((place_at + i) % LIB94_CORE_SIZE + LIB94_CORE_SIZE >= other.first && (place_at + i) % LIB94_CORE_SIZE + LIB94_CORE_SIZE < other.first + other.second))
|
||||
goto new_place_at;
|
||||
|
||||
placements.push_back(std::make_pair<>(place_at, w->instructions.size()));
|
||||
|
||||
for (number_t i = 0; i < (number_t)w->instructions.size(); ++i) {
|
||||
core[(place_at + i) % LIB94_CORE_SIZE] = w->instructions[i];
|
||||
add_written_instruction(core + (place_at + i) % LIB94_CORE_SIZE);
|
||||
}
|
||||
|
||||
warrior_infos.push_back({});
|
||||
warrior_info *wi = &warrior_infos.back();
|
||||
wi->the_warrior = w;
|
||||
wi->processes.push_back((place_at + w->org) % LIB94_CORE_SIZE);
|
||||
}
|
||||
assert(false);
|
||||
}
|
||||
|
||||
const std::set<number_t> &get_modified_addresses() {
|
||||
return modified_addresses;
|
||||
}
|
||||
std::shuffle(warrior_infos.begin(), warrior_infos.end(), prng);
|
||||
|
||||
void clear_modified_addresses() {
|
||||
modified_addresses.clear();
|
||||
}
|
||||
for (warrior_info &wi : warrior_infos)
|
||||
alive_warriors.push_back(&wi);
|
||||
|
||||
void add_modified_instruction(const instruction *instr) {
|
||||
modified_addresses.insert(instr - core);
|
||||
}
|
||||
|
||||
const instruction &get_instruction(number_t address) {
|
||||
return core[address];
|
||||
}
|
||||
|
||||
static const warrior *single_step_return_dead();
|
||||
|
||||
void single_step() {
|
||||
single_step_return_dead();
|
||||
}
|
||||
|
||||
const warrior *run_until_warrior_death() {
|
||||
while (true) {
|
||||
const warrior *w = single_step_return_dead();
|
||||
if (w)
|
||||
return w;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
const warrior *run_until_winner() {
|
||||
while (true) {
|
||||
const warrior *w = single_step_return_dead();
|
||||
if (w && alive_warriors.size() == 1)
|
||||
for (const warrior_info &wi : warrior_infos)
|
||||
if (wi.processes.size() > 0)
|
||||
return wi.the_warrior;
|
||||
size_t alive_warrior_count() {
|
||||
return alive_warriors.size();
|
||||
}
|
||||
}
|
||||
|
||||
number_t program_counter;
|
||||
static instruction instruction_register;
|
||||
|
||||
static number_t a_pointer, b_pointer;
|
||||
instruction a_instruction, b_instruction;
|
||||
|
||||
instruction *a_instruction_writable;
|
||||
instruction *b_instruction_writable;
|
||||
|
||||
static void evaluate_operand(mode m, number_t n, number_t &ptr, instruction &instr) {
|
||||
instruction *secondary;
|
||||
|
||||
switch (m) {
|
||||
|
||||
case IMMEDIATE:
|
||||
ptr = 0;
|
||||
instr = core[program_counter];
|
||||
break;
|
||||
|
||||
case DIRECT:
|
||||
ptr = n;
|
||||
instr = core[(program_counter + n) % CORE_SIZE];
|
||||
break;
|
||||
|
||||
case A_INDIRECT:
|
||||
secondary = core + (program_counter + n) % CORE_SIZE;
|
||||
ptr = (n + secondary->anumber) % CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % CORE_SIZE];
|
||||
break;
|
||||
|
||||
case B_INDIRECT:
|
||||
secondary = core + (program_counter + n) % CORE_SIZE;
|
||||
ptr = (n + secondary->bnumber) % CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % CORE_SIZE];
|
||||
break;
|
||||
|
||||
case A_DECREMENT:
|
||||
secondary = core + (program_counter + n) % CORE_SIZE;
|
||||
ptr = (n + --secondary->anumber) % CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % CORE_SIZE];
|
||||
add_modified_instruction(secondary);
|
||||
break;
|
||||
|
||||
case B_DECREMENT:
|
||||
secondary = core + (program_counter + n) % CORE_SIZE;
|
||||
ptr = (n + --secondary->bnumber) % CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % CORE_SIZE];
|
||||
add_modified_instruction(secondary);
|
||||
break;
|
||||
|
||||
case A_INCREMENT:
|
||||
secondary = core + (program_counter + n) % CORE_SIZE;
|
||||
ptr = (n + secondary->anumber) % CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % CORE_SIZE];
|
||||
++secondary->anumber;
|
||||
add_modified_instruction(secondary);
|
||||
break;
|
||||
|
||||
case B_INCREMENT:
|
||||
secondary = core + (program_counter + n) % CORE_SIZE;
|
||||
ptr = (n + secondary->bnumber) % CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % CORE_SIZE];
|
||||
++secondary->bnumber;
|
||||
add_modified_instruction(secondary);
|
||||
break;
|
||||
|
||||
const warrior *get_next_warrior() {
|
||||
assert(alive_warriors.size() > 0);
|
||||
return alive_warriors.front()->the_warrior;
|
||||
}
|
||||
}
|
||||
|
||||
template<opcode op, modifier mod>
|
||||
bool execute();
|
||||
|
||||
#define EXECUTOR_DECL_LINE(mod) \
|
||||
extern template bool execute<DAT, mod>(); extern template bool execute<MOV, mod>(); \
|
||||
extern template bool execute<ADD, mod>(); extern template bool execute<SUB, mod>(); \
|
||||
extern template bool execute<MUL, mod>(); extern template bool execute<DIV, mod>(); \
|
||||
extern template bool execute<MOD, mod>(); extern template bool execute<JMP, mod>(); \
|
||||
extern template bool execute<JMZ, mod>(); extern template bool execute<JMN, mod>(); \
|
||||
extern template bool execute<DJN, mod>(); extern template bool execute<SEQ, mod>(); \
|
||||
extern template bool execute<SNE, mod>(); extern template bool execute<SLT, mod>(); \
|
||||
extern template bool execute<SPL, mod>(); extern template bool execute<NOP, mod>();
|
||||
|
||||
EXECUTOR_DECL_LINE(A)
|
||||
EXECUTOR_DECL_LINE(B)
|
||||
EXECUTOR_DECL_LINE(AB)
|
||||
EXECUTOR_DECL_LINE(BA)
|
||||
EXECUTOR_DECL_LINE(F)
|
||||
EXECUTOR_DECL_LINE(X)
|
||||
EXECUTOR_DECL_LINE(I)
|
||||
|
||||
#define EXECUTOR_REF_LINE(mod) \
|
||||
&execute<DAT, mod>, &execute<MOV, mod>, \
|
||||
&execute<ADD, mod>, &execute<SUB, mod>, \
|
||||
&execute<MUL, mod>, &execute<DIV, mod>, \
|
||||
&execute<MOD, mod>, &execute<JMP, mod>, \
|
||||
&execute<JMZ, mod>, &execute<JMN, mod>, \
|
||||
&execute<DJN, mod>, &execute<SEQ, mod>, \
|
||||
&execute<SNE, mod>, &execute<SLT, mod>, \
|
||||
&execute<SPL, mod>, &execute<NOP, mod>
|
||||
|
||||
static const std::function<bool ()> executors[] = {
|
||||
EXECUTOR_REF_LINE(A), EXECUTOR_REF_LINE(B),
|
||||
EXECUTOR_REF_LINE(AB), EXECUTOR_REF_LINE(BA),
|
||||
EXECUTOR_REF_LINE(F), EXECUTOR_REF_LINE(X),
|
||||
EXECUTOR_REF_LINE(I)
|
||||
};
|
||||
|
||||
static warrior_info *this_warrior;
|
||||
|
||||
void enqueue_process(number_t pc) {
|
||||
this_warrior->processes.push(pc);
|
||||
}
|
||||
|
||||
static const warrior *single_step_return_dead() {
|
||||
this_warrior = alive_warriors.front();
|
||||
alive_warriors.pop();
|
||||
|
||||
program_counter = this_warrior->processes.front();
|
||||
this_warrior->processes.pop();
|
||||
|
||||
instruction_register = core[program_counter];
|
||||
|
||||
evaluate_operand(instruction_register.amode, instruction_register.anumber, a_pointer, a_instruction);
|
||||
evaluate_operand(instruction_register.bmode, instruction_register.bnumber, b_pointer, b_instruction);
|
||||
|
||||
a_instruction_writable = core + (program_counter + a_pointer) % CORE_SIZE;
|
||||
b_instruction_writable = core + (program_counter + b_pointer) % CORE_SIZE;
|
||||
|
||||
bool enqueue_process = executors[instruction_register.op + instruction_register.mod * 16]();
|
||||
|
||||
if (enqueue_process)
|
||||
this_warrior->processes.push((program_counter + 1) % CORE_SIZE);
|
||||
|
||||
else if (this_warrior->processes.size() == 0)
|
||||
return this_warrior->the_warrior;
|
||||
|
||||
alive_warriors.push(this_warrior);
|
||||
return 0;
|
||||
|
||||
const std::deque<number_t> &get_processes(const warrior *for_warrior) {
|
||||
for (const warrior_info &wi : warrior_infos)
|
||||
if (wi.the_warrior == for_warrior)
|
||||
return wi.processes;
|
||||
assert(false);
|
||||
}
|
||||
|
||||
number_t get_next_process(const warrior *for_warrior) {
|
||||
for (const warrior_info &wi : warrior_infos)
|
||||
if (wi.the_warrior == for_warrior) {
|
||||
assert(wi.processes.size() > 0);
|
||||
return wi.processes.front();
|
||||
}
|
||||
assert(false);
|
||||
}
|
||||
|
||||
const std::set<number_t> &get_written_addresses() {
|
||||
return written_addresses;
|
||||
}
|
||||
|
||||
const std::set<number_t> &get_read_addresses() {
|
||||
return read_addresses;
|
||||
}
|
||||
|
||||
const std::set<number_t> &get_executed_addresses() {
|
||||
return executed_addresses;
|
||||
}
|
||||
|
||||
void clear_address_sets() {
|
||||
written_addresses.clear();
|
||||
read_addresses.clear();
|
||||
executed_addresses.clear();
|
||||
}
|
||||
|
||||
const instruction &get_instruction(number_t address) {
|
||||
return core[address];
|
||||
}
|
||||
|
||||
number_t program_counter;
|
||||
static instruction instruction_register;
|
||||
|
||||
static number_t a_pointer, b_pointer;
|
||||
instruction a_instruction, b_instruction;
|
||||
|
||||
instruction *a_instruction_writable;
|
||||
instruction *b_instruction_writable;
|
||||
|
||||
static void evaluate_operand(mode m, number_t n, number_t &ptr, instruction &instr) {
|
||||
instruction *secondary;
|
||||
|
||||
switch (m) {
|
||||
|
||||
case IMMEDIATE:
|
||||
ptr = 0;
|
||||
instr = core[program_counter];
|
||||
break;
|
||||
|
||||
case DIRECT:
|
||||
ptr = n;
|
||||
instr = core[(program_counter + n) % LIB94_CORE_SIZE];
|
||||
break;
|
||||
|
||||
case A_INDIRECT:
|
||||
secondary = core + (program_counter + n) % LIB94_CORE_SIZE;
|
||||
ptr = (n + secondary->anumber) % LIB94_CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % LIB94_CORE_SIZE];
|
||||
add_read_instruction(secondary);
|
||||
break;
|
||||
|
||||
case B_INDIRECT:
|
||||
secondary = core + (program_counter + n) % LIB94_CORE_SIZE;
|
||||
ptr = (n + secondary->bnumber) % LIB94_CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % LIB94_CORE_SIZE];
|
||||
add_read_instruction(secondary);
|
||||
break;
|
||||
|
||||
case A_DECREMENT:
|
||||
secondary = core + (program_counter + n) % LIB94_CORE_SIZE;
|
||||
ptr = (n + --secondary->anumber) % LIB94_CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % LIB94_CORE_SIZE];
|
||||
add_read_instruction(secondary);
|
||||
add_written_instruction(secondary);
|
||||
break;
|
||||
|
||||
case B_DECREMENT:
|
||||
secondary = core + (program_counter + n) % LIB94_CORE_SIZE;
|
||||
ptr = (n + --secondary->bnumber) % LIB94_CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % LIB94_CORE_SIZE];
|
||||
add_read_instruction(secondary);
|
||||
add_written_instruction(secondary);
|
||||
break;
|
||||
|
||||
case A_INCREMENT:
|
||||
secondary = core + (program_counter + n) % LIB94_CORE_SIZE;
|
||||
ptr = (n + secondary->anumber) % LIB94_CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % LIB94_CORE_SIZE];
|
||||
++secondary->anumber;
|
||||
add_read_instruction(secondary);
|
||||
add_written_instruction(secondary);
|
||||
break;
|
||||
|
||||
case B_INCREMENT:
|
||||
secondary = core + (program_counter + n) % LIB94_CORE_SIZE;
|
||||
ptr = (n + secondary->bnumber) % LIB94_CORE_SIZE;
|
||||
instr = core[(program_counter + ptr) % LIB94_CORE_SIZE];
|
||||
++secondary->bnumber;
|
||||
add_read_instruction(secondary);
|
||||
add_written_instruction(secondary);
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
template<opcode op, modifier mod>
|
||||
bool execute();
|
||||
|
||||
#define EXECUTOR_DECL_LINE(mod) \
|
||||
extern template bool execute<DAT, mod>(); extern template bool execute<MOV, mod>(); \
|
||||
extern template bool execute<ADD, mod>(); extern template bool execute<SUB, mod>(); \
|
||||
extern template bool execute<MUL, mod>(); extern template bool execute<DIV, mod>(); \
|
||||
extern template bool execute<MOD, mod>(); extern template bool execute<JMP, mod>(); \
|
||||
extern template bool execute<JMZ, mod>(); extern template bool execute<JMN, mod>(); \
|
||||
extern template bool execute<DJN, mod>(); extern template bool execute<SEQ, mod>(); \
|
||||
extern template bool execute<SNE, mod>(); extern template bool execute<SLT, mod>(); \
|
||||
extern template bool execute<SPL, mod>(); extern template bool execute<NOP, mod>();
|
||||
|
||||
EXECUTOR_DECL_LINE(A)
|
||||
EXECUTOR_DECL_LINE(B)
|
||||
EXECUTOR_DECL_LINE(AB)
|
||||
EXECUTOR_DECL_LINE(BA)
|
||||
EXECUTOR_DECL_LINE(F)
|
||||
EXECUTOR_DECL_LINE(X)
|
||||
EXECUTOR_DECL_LINE(I)
|
||||
|
||||
#define EXECUTOR_REF_LINE(mod) \
|
||||
&execute<DAT, mod>, &execute<MOV, mod>, \
|
||||
&execute<ADD, mod>, &execute<SUB, mod>, \
|
||||
&execute<MUL, mod>, &execute<DIV, mod>, \
|
||||
&execute<MOD, mod>, &execute<JMP, mod>, \
|
||||
&execute<JMZ, mod>, &execute<JMN, mod>, \
|
||||
&execute<DJN, mod>, &execute<SEQ, mod>, \
|
||||
&execute<SNE, mod>, &execute<SLT, mod>, \
|
||||
&execute<SPL, mod>, &execute<NOP, mod>
|
||||
|
||||
static const std::function<bool ()> executors[] = {
|
||||
EXECUTOR_REF_LINE(A), EXECUTOR_REF_LINE(B),
|
||||
EXECUTOR_REF_LINE(AB), EXECUTOR_REF_LINE(BA),
|
||||
EXECUTOR_REF_LINE(F), EXECUTOR_REF_LINE(X),
|
||||
EXECUTOR_REF_LINE(I)
|
||||
};
|
||||
|
||||
static warrior_info *this_warrior;
|
||||
|
||||
void enqueue_process(number_t pc) {
|
||||
this_warrior->processes.push_back(pc);
|
||||
}
|
||||
|
||||
const warrior *single_step() {
|
||||
this_warrior = alive_warriors.front();
|
||||
alive_warriors.pop_front();
|
||||
|
||||
program_counter = this_warrior->processes.front();
|
||||
this_warrior->processes.pop_front();
|
||||
|
||||
executed_addresses.insert(program_counter);
|
||||
instruction_register = core[program_counter];
|
||||
|
||||
evaluate_operand(instruction_register.amode, instruction_register.anumber, a_pointer, a_instruction);
|
||||
evaluate_operand(instruction_register.bmode, instruction_register.bnumber, b_pointer, b_instruction);
|
||||
|
||||
a_instruction_writable = core + (program_counter + a_pointer) % LIB94_CORE_SIZE;
|
||||
b_instruction_writable = core + (program_counter + b_pointer) % LIB94_CORE_SIZE;
|
||||
|
||||
bool should_endeque = executors[instruction_register.op + instruction_register.mod * 16]();
|
||||
|
||||
if (should_endeque)
|
||||
this_warrior->processes.push_back((program_counter + 1) % LIB94_CORE_SIZE);
|
||||
|
||||
else if (this_warrior->processes.size() == 0)
|
||||
return this_warrior->the_warrior;
|
||||
|
||||
alive_warriors.push_back(this_warrior);
|
||||
return 0;
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -2,331 +2,364 @@
|
|||
#include <functional>
|
||||
#include <cassert>
|
||||
|
||||
void enqueue_process(number_t pc);
|
||||
#include <cstdio>
|
||||
|
||||
extern instruction core[CORE_SIZE];
|
||||
extern number_t program_counter;
|
||||
extern instruction a_instruction, b_instruction;
|
||||
extern instruction *a_instruction_writable;
|
||||
extern instruction *b_instruction_writable;
|
||||
namespace lib94 {
|
||||
|
||||
void add_modified_instruction(const instruction *instr);
|
||||
void enqueue_process(number_t pc);
|
||||
|
||||
struct single_target {
|
||||
number_t *ptr;
|
||||
extern instruction core[LIB94_CORE_SIZE];
|
||||
extern number_t program_counter;
|
||||
extern instruction a_instruction, b_instruction;
|
||||
extern instruction *a_instruction_writable;
|
||||
extern instruction *b_instruction_writable;
|
||||
|
||||
inline bool assign(const single_target &left, const single_target &right,
|
||||
std::function<std::optional<number_t> (number_t, number_t)> f) {
|
||||
void add_written_instruction(const instruction *instr);
|
||||
void add_read_instruction(const instruction *instr);
|
||||
|
||||
std::optional<number_t> result = f(*left.ptr, *right.ptr);
|
||||
if (result.has_value()) {
|
||||
*ptr = result.value();
|
||||
struct single_target {
|
||||
number_t *ptr;
|
||||
|
||||
inline bool assign(const single_target &left, const single_target &right,
|
||||
std::function<std::optional<number_t> (number_t, number_t)> f) {
|
||||
|
||||
std::optional<number_t> result = f(*left.ptr, *right.ptr);
|
||||
if (result.has_value()) {
|
||||
*ptr = result.value();
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
|
||||
}
|
||||
|
||||
inline bool is_zero() {
|
||||
return *ptr == 0;
|
||||
}
|
||||
|
||||
inline bool dec_not_zero() {
|
||||
return --*ptr != 0;
|
||||
}
|
||||
|
||||
inline bool equal_to(const single_target &other) {
|
||||
return *ptr == *other.ptr;
|
||||
}
|
||||
|
||||
inline bool less_than(const single_target &other) {
|
||||
return *ptr < *other.ptr;
|
||||
}
|
||||
};
|
||||
|
||||
struct pair_target {
|
||||
number_t *ptr1;
|
||||
number_t *ptr2;
|
||||
|
||||
inline bool assign(const pair_target &left, const pair_target &right,
|
||||
std::function<std::optional<number_t> (number_t, number_t)> f) {
|
||||
bool to_return = true;
|
||||
|
||||
std::optional<number_t> result1 = f(*left.ptr1, *right.ptr1);
|
||||
if (result1.has_value())
|
||||
*ptr1 = result1.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
std::optional<number_t> result2 = f(*left.ptr2, *right.ptr2);
|
||||
if (result2.has_value())
|
||||
*ptr2 = result2.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
return to_return;
|
||||
|
||||
}
|
||||
|
||||
inline bool is_zero() {
|
||||
return *ptr1 == 0 && *ptr2 == 0;
|
||||
}
|
||||
|
||||
inline bool dec_not_zero() {
|
||||
--*ptr1;
|
||||
--*ptr2;
|
||||
return *ptr1 != 0 || *ptr2 != 0;
|
||||
}
|
||||
|
||||
inline bool equal_to(const pair_target &other) {
|
||||
return *ptr1 == *other.ptr1 &&
|
||||
*ptr2 == *other.ptr2;
|
||||
}
|
||||
|
||||
inline bool less_than(const pair_target &other) {
|
||||
return *ptr1 < *other.ptr1 &&
|
||||
*ptr2 < *other.ptr2;
|
||||
}
|
||||
};
|
||||
|
||||
struct instruction_target {
|
||||
instruction *instr;
|
||||
|
||||
inline bool assign(const instruction_target &left, const instruction_target &right,
|
||||
std::function<std::optional<number_t> (number_t, number_t)> f) {
|
||||
bool to_return = true;
|
||||
|
||||
std::optional<number_t> result1 = f(left.instr->anumber, right.instr->anumber);
|
||||
if (result1.has_value())
|
||||
instr->anumber = result1.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
std::optional<number_t> result2 = f(left.instr->bnumber, right.instr->bnumber);
|
||||
if (result2.has_value())
|
||||
instr->bnumber = result2.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
return to_return;
|
||||
|
||||
}
|
||||
|
||||
inline bool is_zero() {
|
||||
return instr->anumber == 0 && instr->bnumber == 0;
|
||||
}
|
||||
|
||||
inline bool dec_not_zero() {
|
||||
--instr->anumber;
|
||||
--instr->bnumber;
|
||||
return instr->anumber != 0 || instr->bnumber != 0;
|
||||
}
|
||||
|
||||
inline bool equal_to(const instruction_target &other) {
|
||||
return instr->op == other.instr->op &&
|
||||
instr->mod == other.instr->mod &&
|
||||
instr->amode == other.instr->amode &&
|
||||
instr->anumber == other.instr->anumber &&
|
||||
instr->bmode == other.instr->bmode &&
|
||||
instr->bnumber == other.instr->bnumber;
|
||||
}
|
||||
|
||||
inline bool less_than(const instruction_target &other) {
|
||||
return instr->anumber < other.instr->anumber &&
|
||||
instr->bnumber < other.instr->bnumber;
|
||||
}
|
||||
};
|
||||
|
||||
template<modifier mod>
|
||||
struct target;
|
||||
|
||||
template<> struct target<A> : single_target {};
|
||||
template<> struct target<B> : single_target {};
|
||||
template<> struct target<AB> : single_target {};
|
||||
template<> struct target<BA> : single_target {};
|
||||
template<> struct target<F> : pair_target {};
|
||||
template<> struct target<X> : pair_target {};
|
||||
template<> struct target<I> : instruction_target {};
|
||||
|
||||
template<opcode op, modifier mod>
|
||||
bool execute() {
|
||||
target<mod> dest_out;
|
||||
target<mod> dest_in;
|
||||
target<mod> src_in;
|
||||
|
||||
if constexpr (mod == A) {
|
||||
dest_out.ptr = &b_instruction_writable->anumber;
|
||||
dest_in.ptr = &b_instruction.anumber;
|
||||
src_in.ptr = &a_instruction.anumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == B) {
|
||||
dest_out.ptr = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr = &b_instruction.bnumber;
|
||||
src_in.ptr = &a_instruction.bnumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == AB) {
|
||||
dest_out.ptr = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr = &b_instruction.bnumber;
|
||||
src_in.ptr = &a_instruction.anumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == BA) {
|
||||
dest_out.ptr = &b_instruction_writable->anumber;
|
||||
dest_in.ptr = &b_instruction.anumber;
|
||||
src_in.ptr = &a_instruction.bnumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == F) {
|
||||
dest_out.ptr1 = &b_instruction_writable->anumber;
|
||||
dest_out.ptr2 = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr1 = &b_instruction.anumber;
|
||||
dest_in.ptr2 = &b_instruction.bnumber;
|
||||
src_in.ptr1 = &a_instruction.anumber;
|
||||
src_in.ptr2 = &a_instruction.bnumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == X) {
|
||||
dest_out.ptr1 = &b_instruction_writable->anumber;
|
||||
dest_out.ptr2 = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr1 = &b_instruction.anumber;
|
||||
dest_in.ptr2 = &b_instruction.bnumber;
|
||||
src_in.ptr1 = &a_instruction.bnumber;
|
||||
src_in.ptr2 = &a_instruction.anumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == I) {
|
||||
dest_out.instr = b_instruction_writable;
|
||||
dest_in.instr = &b_instruction;
|
||||
src_in.instr = &a_instruction;
|
||||
}
|
||||
|
||||
else assert(false);
|
||||
|
||||
if constexpr (op == DAT)
|
||||
return false;
|
||||
|
||||
else if constexpr (op == MOV) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
if constexpr (mod == I) {
|
||||
dest_out.instr->op = src_in.instr->op;
|
||||
dest_out.instr->mod = src_in.instr->mod;
|
||||
dest_out.instr->amode = src_in.instr->amode;
|
||||
dest_out.instr->bmode = src_in.instr->bmode;
|
||||
}
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
(void)a;
|
||||
return b;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == ADD) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return (a + b) % LIB94_CORE_SIZE;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == SUB) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return (a + LIB94_CORE_SIZE - b) % LIB94_CORE_SIZE;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == MUL) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return (a * b) % LIB94_CORE_SIZE;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == DIV) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
if (b == 0)
|
||||
return {};
|
||||
return a / b;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == MOD) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
if (b == 0)
|
||||
return {};
|
||||
return a % b;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == JMP) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
program_counter = (a_instruction_writable - core + LIB94_CORE_SIZE - 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
|
||||
}
|
||||
|
||||
inline bool is_zero() {
|
||||
return *ptr == 0;
|
||||
}
|
||||
|
||||
inline bool dec_not_zero() {
|
||||
return --*ptr != 0;
|
||||
}
|
||||
|
||||
inline bool equal_to(const single_target &other) {
|
||||
return *ptr == *other.ptr;
|
||||
}
|
||||
|
||||
inline bool less_than(const single_target &other) {
|
||||
return *ptr < *other.ptr;
|
||||
}
|
||||
};
|
||||
|
||||
struct pair_target {
|
||||
number_t *ptr1;
|
||||
number_t *ptr2;
|
||||
|
||||
inline bool assign(const pair_target &left, const pair_target &right,
|
||||
std::function<std::optional<number_t> (number_t, number_t)> f) {
|
||||
bool to_return = true;
|
||||
|
||||
std::optional<number_t> result1 = f(*left.ptr1, *right.ptr1);
|
||||
if (result1.has_value())
|
||||
*ptr1 = result1.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
std::optional<number_t> result2 = f(*left.ptr2, *right.ptr2);
|
||||
if (result2.has_value())
|
||||
*ptr2 = result2.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
return to_return;
|
||||
|
||||
}
|
||||
|
||||
inline bool is_zero() {
|
||||
return *ptr1 == 0 && *ptr2 == 0;
|
||||
}
|
||||
|
||||
inline bool dec_not_zero() {
|
||||
--*ptr1;
|
||||
--*ptr2;
|
||||
return *ptr1 != 0 || *ptr2 != 0;
|
||||
}
|
||||
|
||||
inline bool equal_to(const pair_target &other) {
|
||||
return *ptr1 == *other.ptr1 &&
|
||||
*ptr2 == *other.ptr2;
|
||||
}
|
||||
|
||||
inline bool less_than(const pair_target &other) {
|
||||
return *ptr1 < *other.ptr1 &&
|
||||
*ptr2 < *other.ptr2;
|
||||
}
|
||||
};
|
||||
|
||||
struct instruction_target {
|
||||
instruction *instr;
|
||||
|
||||
inline bool assign(const instruction_target &left, const instruction_target &right,
|
||||
std::function<std::optional<number_t> (number_t, number_t)> f) {
|
||||
bool to_return = true;
|
||||
|
||||
std::optional<number_t> result1 = f(left.instr->anumber, right.instr->anumber);
|
||||
if (result1.has_value())
|
||||
instr->anumber = result1.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
std::optional<number_t> result2 = f(left.instr->bnumber, right.instr->bnumber);
|
||||
if (result2.has_value())
|
||||
instr->bnumber = result2.value();
|
||||
else
|
||||
to_return = false;
|
||||
|
||||
return to_return;
|
||||
|
||||
}
|
||||
|
||||
inline bool is_zero() {
|
||||
return instr->anumber == 0 && instr->bnumber == 0;
|
||||
}
|
||||
|
||||
inline bool dec_not_zero() {
|
||||
--instr->anumber;
|
||||
--instr->bnumber;
|
||||
return instr->anumber != 0 || instr->bnumber != 0;
|
||||
}
|
||||
|
||||
inline bool equal_to(const instruction_target &other) {
|
||||
return instr->op == other.instr->op &&
|
||||
instr->mod == other.instr->mod &&
|
||||
instr->amode == other.instr->amode &&
|
||||
instr->anumber == other.instr->anumber &&
|
||||
instr->bmode == other.instr->bmode &&
|
||||
instr->bnumber == other.instr->bnumber;
|
||||
}
|
||||
|
||||
inline bool less_than(const instruction_target &other) {
|
||||
return instr->anumber < other.instr->anumber &&
|
||||
instr->bnumber < other.instr->bnumber;
|
||||
}
|
||||
};
|
||||
|
||||
template<modifier mod>
|
||||
struct target;
|
||||
|
||||
template<> struct target<A> : single_target {};
|
||||
template<> struct target<B> : single_target {};
|
||||
template<> struct target<AB> : single_target {};
|
||||
template<> struct target<BA> : single_target {};
|
||||
template<> struct target<F> : pair_target {};
|
||||
template<> struct target<X> : pair_target {};
|
||||
template<> struct target<I> : instruction_target {};
|
||||
|
||||
template<opcode op, modifier mod>
|
||||
bool execute() {
|
||||
target<mod> dest_out;
|
||||
target<mod> dest_in;
|
||||
target<mod> src_in;
|
||||
|
||||
if constexpr (mod == A) {
|
||||
dest_out.ptr = &b_instruction_writable->anumber;
|
||||
dest_in.ptr = &b_instruction.anumber;
|
||||
src_in.ptr = &a_instruction.anumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == B) {
|
||||
dest_out.ptr = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr = &b_instruction.bnumber;
|
||||
src_in.ptr = &a_instruction.bnumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == AB) {
|
||||
dest_out.ptr = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr = &b_instruction.bnumber;
|
||||
src_in.ptr = &a_instruction.anumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == BA) {
|
||||
dest_out.ptr = &b_instruction_writable->anumber;
|
||||
dest_in.ptr = &b_instruction.anumber;
|
||||
src_in.ptr = &a_instruction.bnumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == F) {
|
||||
dest_out.ptr1 = &b_instruction_writable->anumber;
|
||||
dest_out.ptr2 = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr1 = &b_instruction.anumber;
|
||||
dest_in.ptr2 = &b_instruction.bnumber;
|
||||
src_in.ptr1 = &a_instruction.anumber;
|
||||
src_in.ptr2 = &a_instruction.bnumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == X) {
|
||||
dest_out.ptr1 = &b_instruction_writable->anumber;
|
||||
dest_out.ptr2 = &b_instruction_writable->bnumber;
|
||||
dest_in.ptr1 = &b_instruction.anumber;
|
||||
dest_in.ptr2 = &b_instruction.bnumber;
|
||||
src_in.ptr1 = &a_instruction.bnumber;
|
||||
src_in.ptr2 = &a_instruction.anumber;
|
||||
}
|
||||
|
||||
else if constexpr (mod == I) {
|
||||
dest_out.instr = b_instruction_writable;
|
||||
dest_in.instr = &b_instruction;
|
||||
src_in.instr = &a_instruction;
|
||||
}
|
||||
|
||||
else assert(false);
|
||||
|
||||
if constexpr (op == DAT)
|
||||
return false;
|
||||
|
||||
else if constexpr (op == MOV) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
if constexpr (mod == I) {
|
||||
dest_out.instr->op = src_in.instr->op;
|
||||
dest_out.instr->mod = src_in.instr->mod;
|
||||
dest_out.instr->amode = src_in.instr->amode;
|
||||
dest_out.instr->bmode = src_in.instr->bmode;
|
||||
else if constexpr (op == JMZ) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
if (dest_in.is_zero())
|
||||
program_counter = (a_instruction_writable - core + LIB94_CORE_SIZE - 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return b;
|
||||
});
|
||||
|
||||
else if constexpr (op == JMN) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
if (!dest_in.is_zero())
|
||||
program_counter = (a_instruction_writable - core + LIB94_CORE_SIZE - 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == DJN) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
add_written_instruction(b_instruction_writable);
|
||||
if (dest_in.dec_not_zero())
|
||||
program_counter = (a_instruction_writable - core + LIB94_CORE_SIZE - 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SEQ) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
if (src_in.equal_to(dest_in))
|
||||
program_counter = (program_counter + 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SNE) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
if (!src_in.equal_to(dest_in))
|
||||
program_counter = (program_counter + 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SLT) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
add_read_instruction(b_instruction_writable);
|
||||
if (src_in.less_than(dest_in))
|
||||
program_counter = (program_counter + 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SPL) {
|
||||
add_read_instruction(a_instruction_writable);
|
||||
enqueue_process((program_counter + 1) % LIB94_CORE_SIZE);
|
||||
program_counter = (a_instruction_writable - core + LIB94_CORE_SIZE - 1) % LIB94_CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == NOP)
|
||||
return true;
|
||||
|
||||
else assert(false);
|
||||
}
|
||||
|
||||
else if constexpr (op == ADD) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return (a + b) % CORE_SIZE;
|
||||
});
|
||||
}
|
||||
#define EXECUTOR_INST_LINE(mod) \
|
||||
template bool execute<DAT, mod>(); template bool execute<MOV, mod>(); \
|
||||
template bool execute<ADD, mod>(); template bool execute<SUB, mod>(); \
|
||||
template bool execute<MUL, mod>(); template bool execute<DIV, mod>(); \
|
||||
template bool execute<MOD, mod>(); template bool execute<JMP, mod>(); \
|
||||
template bool execute<JMZ, mod>(); template bool execute<JMN, mod>(); \
|
||||
template bool execute<DJN, mod>(); template bool execute<SEQ, mod>(); \
|
||||
template bool execute<SNE, mod>(); template bool execute<SLT, mod>(); \
|
||||
template bool execute<SPL, mod>(); template bool execute<NOP, mod>();
|
||||
|
||||
else if constexpr (op == SUB) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return (a + CORE_SIZE - b) % CORE_SIZE;
|
||||
});
|
||||
}
|
||||
EXECUTOR_INST_LINE(A)
|
||||
EXECUTOR_INST_LINE(B)
|
||||
EXECUTOR_INST_LINE(AB)
|
||||
EXECUTOR_INST_LINE(BA)
|
||||
EXECUTOR_INST_LINE(F)
|
||||
EXECUTOR_INST_LINE(X)
|
||||
EXECUTOR_INST_LINE(I)
|
||||
|
||||
else if constexpr (op == MUL) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
return (a * b) % CORE_SIZE;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == DIV) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
if (b == 0)
|
||||
return {};
|
||||
return a / b;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == MOD) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
return dest_out.assign(dest_in, src_in, [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
if (b == 0)
|
||||
return {};
|
||||
return a % b;
|
||||
});
|
||||
}
|
||||
|
||||
else if constexpr (op == JMP) {
|
||||
program_counter = (a_instruction_writable - core + CORE_SIZE - 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == JMZ) {
|
||||
if (dest_in.is_zero())
|
||||
program_counter = (a_instruction_writable - core + CORE_SIZE - 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == JMN) {
|
||||
if (!dest_in.is_zero())
|
||||
program_counter = (a_instruction_writable - core + CORE_SIZE - 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == DJN) {
|
||||
add_modified_instruction(b_instruction_writable);
|
||||
if (dest_in.dec_not_zero())
|
||||
program_counter = (a_instruction_writable - core + CORE_SIZE - 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SEQ) {
|
||||
if (src_in.equal_to(dest_in))
|
||||
program_counter = (program_counter + 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SNE) {
|
||||
if (!src_in.equal_to(dest_in))
|
||||
program_counter = (program_counter + 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SLT) {
|
||||
if (src_in.less_than(dest_in))
|
||||
program_counter = (program_counter + 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == SPL) {
|
||||
enqueue_process((program_counter + 1) % CORE_SIZE);
|
||||
program_counter = (a_instruction_writable - core + CORE_SIZE - 1) % CORE_SIZE;
|
||||
return true;
|
||||
}
|
||||
|
||||
else if constexpr (op == NOP)
|
||||
return true;
|
||||
|
||||
else assert(false);
|
||||
}
|
||||
|
||||
#define EXECUTOR_INST_LINE(mod) \
|
||||
template bool execute<DAT, mod>(); template bool execute<MOV, mod>(); \
|
||||
template bool execute<ADD, mod>(); template bool execute<SUB, mod>(); \
|
||||
template bool execute<MUL, mod>(); template bool execute<DIV, mod>(); \
|
||||
template bool execute<MOD, mod>(); template bool execute<JMP, mod>(); \
|
||||
template bool execute<JMZ, mod>(); template bool execute<JMN, mod>(); \
|
||||
template bool execute<DJN, mod>(); template bool execute<SEQ, mod>(); \
|
||||
template bool execute<SNE, mod>(); template bool execute<SLT, mod>(); \
|
||||
template bool execute<SPL, mod>(); template bool execute<NOP, mod>();
|
||||
|
||||
EXECUTOR_INST_LINE(A)
|
||||
EXECUTOR_INST_LINE(B)
|
||||
EXECUTOR_INST_LINE(AB)
|
||||
EXECUTOR_INST_LINE(BA)
|
||||
EXECUTOR_INST_LINE(F)
|
||||
EXECUTOR_INST_LINE(X)
|
||||
EXECUTOR_INST_LINE(I)
|
||||
|
|
|
@ -6,471 +6,512 @@
|
|||
#include <memory>
|
||||
#include <map>
|
||||
|
||||
struct number_expr {
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) = 0;
|
||||
};
|
||||
namespace lib94 {
|
||||
|
||||
struct label_number_expr : public number_expr {
|
||||
std::string label;
|
||||
static const std::string opcode_strings[] = {
|
||||
"dat", "mov", "add", "sub", "mul", "div", "mod", "jmp",
|
||||
"jmz", "jmn", "djn", "seq", "sne", "slt", "spl", "nop"
|
||||
};
|
||||
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) {
|
||||
auto result = label_offsets.find(label);
|
||||
if (result == label_offsets.end())
|
||||
static const std::string modifier_strings[] = {
|
||||
"a", "b", "ab", "ba", "f", "x", "i"
|
||||
};
|
||||
|
||||
static const char mode_chars[] = "#$*@{<}>";
|
||||
|
||||
std::string instruction_to_string(const instruction &instr) {
|
||||
return
|
||||
opcode_strings[instr.op] + '.' + modifier_strings[instr.mod] + ' ' +
|
||||
mode_chars[instr.amode] + std::to_string(instr.anumber) + ", " +
|
||||
mode_chars[instr.bmode] + std::to_string(instr.bnumber);
|
||||
}
|
||||
|
||||
number_t real_mod(number_t input) {
|
||||
return (input % LIB94_CORE_SIZE + LIB94_CORE_SIZE) % LIB94_CORE_SIZE;
|
||||
}
|
||||
|
||||
struct number_expr {
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) = 0;
|
||||
};
|
||||
|
||||
struct label_number_expr : public number_expr {
|
||||
std::string label;
|
||||
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) {
|
||||
auto result = label_offsets.find(label);
|
||||
if (result == label_offsets.end())
|
||||
return {};
|
||||
return real_mod(result->second - this_offset);
|
||||
}
|
||||
};
|
||||
|
||||
struct number_number_expr : public number_expr {
|
||||
number_t value;
|
||||
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) {
|
||||
(void)label_offsets;
|
||||
(void)this_offset;
|
||||
return real_mod(value);
|
||||
}
|
||||
};
|
||||
|
||||
struct op_number_expr : public number_expr {
|
||||
std::unique_ptr<number_expr> left;
|
||||
std::unique_ptr<number_expr> right;
|
||||
std::function<std::optional<number_t> (number_t, number_t)> op;
|
||||
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) {
|
||||
std::optional<number_t> left_result = left->to_number(label_offsets, this_offset);
|
||||
std::optional<number_t> right_result = right->to_number(label_offsets, this_offset);
|
||||
if (left_result.has_value() && right_result.has_value()) {
|
||||
std::optional<number_t> op_result = op(left_result.value(), right_result.value());
|
||||
if (op_result.has_value())
|
||||
return real_mod(op_result.value());
|
||||
}
|
||||
return {};
|
||||
return result->second - this_offset;
|
||||
}
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
struct number_number_expr : public number_expr {
|
||||
number_t value;
|
||||
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) {
|
||||
return value;
|
||||
}
|
||||
};
|
||||
|
||||
struct op_number_expr : public number_expr {
|
||||
std::unique_ptr<number_expr> left;
|
||||
std::unique_ptr<number_expr> right;
|
||||
std::function<std::optional<number_t> (number_t, number_t)> op;
|
||||
|
||||
virtual std::optional<number_t> to_number(const std::map<std::string, number_t> &label_offsets, number_t this_offset) {
|
||||
std::optional<number_t> left_result = left->to_number(label_offsets, this_offset);
|
||||
std::optional<number_t> right_result = right->to_number(label_offsets, this_offset);
|
||||
if (left_result.has_value() && right_result.has_value())
|
||||
return op(left_result.value(), right_result.value());
|
||||
return {};
|
||||
}
|
||||
};
|
||||
|
||||
bool valid_label(std::string candidate) {
|
||||
if (candidate == "")
|
||||
return false;
|
||||
|
||||
if (!isalpha(candidate[0]) && candidate[0] != '_')
|
||||
return false;
|
||||
|
||||
for (char ch : candidate)
|
||||
if (!isalnum(ch) && ch != '_')
|
||||
bool valid_label(std::string candidate) {
|
||||
if (candidate == "")
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
if (!isalpha(candidate[0]) && candidate[0] != '_')
|
||||
return false;
|
||||
|
||||
size_t find_respecting_parentheses(std::string part, std::string candidates) {
|
||||
size_t layers = 0;
|
||||
for (char ch : candidate)
|
||||
if (!isalnum(ch) && ch != '_')
|
||||
return false;
|
||||
|
||||
for (size_t i = 0; i < part.size(); ++i)
|
||||
if (layers == 0 && candidates.find(part[i]) != std::string::npos)
|
||||
return i;
|
||||
else if (part[i] == '(')
|
||||
++layers;
|
||||
else if (part[i] == ')')
|
||||
--layers;
|
||||
|
||||
return std::string::npos;
|
||||
}
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> to_number_expr(std::string part);
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> make_op_expr(std::string part, size_t split) {
|
||||
std::string left = part.substr(0, split);
|
||||
std::string right = part.substr(split + 1);
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> left_expr = to_number_expr(left);
|
||||
std::optional<std::unique_ptr<number_expr>> right_expr = to_number_expr(right);
|
||||
if (!left_expr.has_value() || !right_expr.has_value())
|
||||
return {};
|
||||
|
||||
auto expr = std::make_unique<op_number_expr>();
|
||||
expr->left = std::move(left_expr.value());
|
||||
expr->right = std::move(right_expr.value());
|
||||
|
||||
switch (part[split]) {
|
||||
case '+': expr->op = [](number_t a, number_t b) {return a + b;}; break;
|
||||
case '-': expr->op = [](number_t a, number_t b) {return a - b;}; break;
|
||||
case '*': expr->op = [](number_t a, number_t b) {return a * b;}; break;
|
||||
case '/': expr->op = [](number_t a, number_t b) {return a / b;}; break;
|
||||
case '%': expr->op = [](number_t a, number_t b) {return a % b;}; break;
|
||||
return true;
|
||||
}
|
||||
|
||||
return expr;
|
||||
}
|
||||
size_t find_respecting_parentheses(std::string part, std::string candidates) {
|
||||
size_t layers = 0;
|
||||
|
||||
std::string trim_spaces(std::string str) {
|
||||
size_t start = str.find_first_not_of(' ');
|
||||
size_t end = str.find_last_not_of(' ') + 1;
|
||||
if (start == std::string::npos)
|
||||
return "";
|
||||
return str.substr(start, end - start);
|
||||
}
|
||||
for (int i = part.size() - 1; i >= 0; --i)
|
||||
if (layers == 0 && candidates.find(part[i]) != std::string::npos)
|
||||
return i;
|
||||
else if (part[i] == ')')
|
||||
++layers;
|
||||
else if (part[i] == '(')
|
||||
--layers;
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> to_number_expr(std::string part) {
|
||||
part = trim_spaces(part);
|
||||
if (part == "")
|
||||
return {};
|
||||
|
||||
size_t split = find_respecting_parentheses(part, "+-");
|
||||
if (split == std::string::npos)
|
||||
split = find_respecting_parentheses(part, "*/%");
|
||||
if (split != std::string::npos)
|
||||
return make_op_expr(part, split);
|
||||
|
||||
if (part[0] == '(' && part[part.size() - 1] == ')')
|
||||
return to_number_expr(part.substr(1, part.size() - 2));
|
||||
|
||||
size_t count;
|
||||
number_t number = std::stoul(part, &count);
|
||||
if (count == part.size()) {
|
||||
std::unique_ptr<number_number_expr> expr = std::make_unique<number_number_expr>();
|
||||
expr->value = number;
|
||||
return expr;
|
||||
return std::string::npos;
|
||||
}
|
||||
|
||||
if (valid_label(part)) {
|
||||
std::unique_ptr<label_number_expr> expr = std::make_unique<label_number_expr>();
|
||||
expr->label = part;
|
||||
return expr;
|
||||
}
|
||||
std::optional<std::unique_ptr<number_expr>> to_number_expr(std::string part);
|
||||
|
||||
return {};
|
||||
}
|
||||
std::optional<std::unique_ptr<number_expr>> make_op_expr(std::string part, size_t split) {
|
||||
std::string left = part.substr(0, split);
|
||||
std::string right = part.substr(split + 1);
|
||||
|
||||
static const std::map<char, mode> mode_symbols = {
|
||||
{'#', IMMEDIATE}, {'$', DIRECT},
|
||||
{'*', A_INDIRECT}, {'@', B_INDIRECT},
|
||||
{'{', A_DECREMENT}, {'<', B_DECREMENT},
|
||||
{'}', A_INCREMENT}, {'>', B_INCREMENT}
|
||||
};
|
||||
std::optional<std::unique_ptr<number_expr>> left_expr = to_number_expr(left);
|
||||
std::optional<std::unique_ptr<number_expr>> right_expr = to_number_expr(right);
|
||||
if (!left_expr.has_value() || !right_expr.has_value())
|
||||
return {};
|
||||
|
||||
std::optional<std::pair<mode, std::unique_ptr<number_expr>>> to_field(std::string part) {
|
||||
if (part == "")
|
||||
return {};
|
||||
auto expr = std::make_unique<op_number_expr>();
|
||||
expr->left = std::move(left_expr.value());
|
||||
expr->right = std::move(right_expr.value());
|
||||
|
||||
mode m = DIRECT;
|
||||
|
||||
auto result = mode_symbols.find(part[0]);
|
||||
if (result != mode_symbols.end()) {
|
||||
m = result->second;
|
||||
part = trim_spaces(part.substr(1));
|
||||
}
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> expr = to_number_expr(part);
|
||||
if (expr.has_value())
|
||||
return std::make_pair<>(m, std::move(expr.value()));
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
static const std::map<std::string, opcode> opcode_names = {
|
||||
{"dat", DAT}, {"mov", MOV}, {"add", ADD}, {"sub", SUB},
|
||||
{"mul", MUL}, {"div", DIV}, {"mod", MOD}, {"jmp", JMP},
|
||||
{"jmz", JMZ}, {"jmn", JMN}, {"djn", DJN}, {"seq", SEQ},
|
||||
{"sne", SNE}, {"slt", SLT}, {"spl", SPL}, {"nop", NOP},
|
||||
{"cmp", SEQ}
|
||||
};
|
||||
|
||||
struct future_instruction {
|
||||
size_t source_line;
|
||||
opcode op;
|
||||
modifier mod;
|
||||
mode amode;
|
||||
mode bmode;
|
||||
std::unique_ptr<number_expr> anumber;
|
||||
std::unique_ptr<number_expr> bnumber;
|
||||
};
|
||||
|
||||
std::variant<warrior *, std::string> compile_warrior(std::string source) {
|
||||
for (char &ch : source)
|
||||
if (ch == '\t' || ch == '\r')
|
||||
ch = ' ';
|
||||
|
||||
std::unique_ptr<warrior> w = std::make_unique<warrior>();
|
||||
|
||||
std::vector<future_instruction> instructions;
|
||||
std::map<std::string, number_t> label_offsets;
|
||||
std::optional<std::unique_ptr<number_expr>> org = {};
|
||||
number_t org_offset;
|
||||
size_t org_source_line;
|
||||
|
||||
size_t on_line = 0;
|
||||
|
||||
while (true) {
|
||||
if (source == "")
|
||||
break;
|
||||
|
||||
++on_line;
|
||||
|
||||
size_t newline = source.find('\n');
|
||||
if (newline == std::string::npos)
|
||||
newline = source.size();
|
||||
|
||||
std::string line = source.substr(0, newline);
|
||||
source = source.substr(newline + 1);
|
||||
|
||||
size_t semicolon = line.find(';');
|
||||
|
||||
if (semicolon != std::string::npos) {
|
||||
std::string comment = trim_spaces(line.substr(semicolon + 1));
|
||||
line = line.substr(0, semicolon);
|
||||
|
||||
if (comment.starts_with("name "))
|
||||
w->name = trim_spaces(comment.substr(5));
|
||||
else if (comment.starts_with("author "))
|
||||
w->author = trim_spaces(comment.substr(7));
|
||||
switch (part[split]) {
|
||||
case '+': expr->op = [](number_t a, number_t b) {return a + b;}; break;
|
||||
case '-': expr->op = [](number_t a, number_t b) {return a - b;}; break;
|
||||
case '*': expr->op = [](number_t a, number_t b) {return a * b;}; break;
|
||||
case '/': expr->op = [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
if (b == 0)
|
||||
return {};
|
||||
return a / b;
|
||||
}; break;
|
||||
case '%': expr->op = [](number_t a, number_t b) -> std::optional<number_t> {
|
||||
if (b == 0)
|
||||
return {};
|
||||
return a % b;
|
||||
}; break;
|
||||
}
|
||||
|
||||
for (char &ch : line)
|
||||
ch = tolower(ch);
|
||||
return expr;
|
||||
}
|
||||
|
||||
std::string trim_spaces(std::string str) {
|
||||
size_t start = str.find_first_not_of(' ');
|
||||
size_t end = str.find_last_not_of(' ') + 1;
|
||||
if (start == std::string::npos)
|
||||
return "";
|
||||
return str.substr(start, end - start);
|
||||
}
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> to_number_expr(std::string part) {
|
||||
part = trim_spaces(part);
|
||||
if (part == "")
|
||||
return {};
|
||||
|
||||
size_t split = find_respecting_parentheses(part, "+-");
|
||||
if (split == std::string::npos)
|
||||
split = find_respecting_parentheses(part, "*/%");
|
||||
if (split != std::string::npos)
|
||||
return make_op_expr(part, split);
|
||||
|
||||
if (part[0] == '(' && part[part.size() - 1] == ')')
|
||||
return to_number_expr(part.substr(1, part.size() - 2));
|
||||
|
||||
try {
|
||||
size_t count;
|
||||
number_t number = std::stoul(part, &count);
|
||||
if (count == part.size()) {
|
||||
std::unique_ptr<number_number_expr> expr = std::make_unique<number_number_expr>();
|
||||
expr->value = number;
|
||||
return expr;
|
||||
}
|
||||
}
|
||||
catch (const std::exception &e) {}
|
||||
|
||||
if (valid_label(part)) {
|
||||
std::unique_ptr<label_number_expr> expr = std::make_unique<label_number_expr>();
|
||||
expr->label = part;
|
||||
return expr;
|
||||
}
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
static const std::map<char, mode> mode_symbols = {
|
||||
{'#', IMMEDIATE}, {'$', DIRECT},
|
||||
{'*', A_INDIRECT}, {'@', B_INDIRECT},
|
||||
{'{', A_DECREMENT}, {'<', B_DECREMENT},
|
||||
{'}', A_INCREMENT}, {'>', B_INCREMENT}
|
||||
};
|
||||
|
||||
std::optional<std::pair<mode, std::unique_ptr<number_expr>>> to_field(std::string part) {
|
||||
if (part == "")
|
||||
return {};
|
||||
|
||||
mode m = DIRECT;
|
||||
|
||||
auto result = mode_symbols.find(part[0]);
|
||||
if (result != mode_symbols.end()) {
|
||||
m = result->second;
|
||||
part = trim_spaces(part.substr(1));
|
||||
}
|
||||
|
||||
std::optional<std::unique_ptr<number_expr>> expr = to_number_expr(part);
|
||||
if (expr.has_value())
|
||||
return std::make_pair<>(m, std::move(expr.value()));
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
static const std::map<std::string, opcode> opcode_names = {
|
||||
{"dat", DAT}, {"mov", MOV}, {"add", ADD}, {"sub", SUB},
|
||||
{"mul", MUL}, {"div", DIV}, {"mod", MOD}, {"jmp", JMP},
|
||||
{"jmz", JMZ}, {"jmn", JMN}, {"djn", DJN}, {"seq", SEQ},
|
||||
{"sne", SNE}, {"slt", SLT}, {"spl", SPL}, {"nop", NOP},
|
||||
{"cmp", SEQ}, {"jnz", JMN}
|
||||
};
|
||||
|
||||
struct future_instruction {
|
||||
size_t source_line;
|
||||
opcode op;
|
||||
modifier mod;
|
||||
mode amode;
|
||||
mode bmode;
|
||||
std::unique_ptr<number_expr> anumber;
|
||||
std::unique_ptr<number_expr> bnumber;
|
||||
};
|
||||
|
||||
std::variant<warrior *, std::string> compile_warrior(std::string source) {
|
||||
for (char &ch : source)
|
||||
if (ch == '\t' || ch == '\r')
|
||||
ch = ' ';
|
||||
|
||||
std::unique_ptr<warrior> w = std::make_unique<warrior>();
|
||||
|
||||
std::vector<future_instruction> instructions;
|
||||
std::map<std::string, number_t> label_offsets;
|
||||
std::optional<std::unique_ptr<number_expr>> org = {};
|
||||
number_t org_offset = -1;
|
||||
size_t org_source_line = -1;
|
||||
|
||||
size_t on_line = 0;
|
||||
|
||||
while (true) {
|
||||
size_t colon = line.find(':');
|
||||
if (colon == std::string::npos)
|
||||
if (source == "")
|
||||
break;
|
||||
|
||||
std::string label = line.substr(0, colon);
|
||||
line = line.substr(colon + 1);
|
||||
++on_line;
|
||||
|
||||
if (!valid_label(label))
|
||||
return std::string("bad label on line ") + std::to_string(on_line);
|
||||
if (label_offsets.contains(label))
|
||||
return std::string("duplicate label on line ") + std::to_string(on_line);
|
||||
size_t newline = source.find('\n');
|
||||
if (newline == std::string::npos)
|
||||
newline = source.size();
|
||||
|
||||
label_offsets[label] = instructions.size();
|
||||
}
|
||||
std::string line = source.substr(0, newline);
|
||||
source = source.substr(newline + 1);
|
||||
|
||||
line = trim_spaces(line);
|
||||
if (line == "")
|
||||
continue;
|
||||
size_t semicolon = line.find(';');
|
||||
|
||||
future_instruction instr;
|
||||
instr.source_line = on_line;
|
||||
|
||||
std::string opcode_str = line.substr(0, 3);
|
||||
line = trim_spaces(line.substr(3));
|
||||
|
||||
if (opcode_str == "org") {
|
||||
if (org.has_value())
|
||||
return std::string("duplicate org on line ") + std::to_string(on_line);
|
||||
|
||||
org = to_number_expr(line);
|
||||
|
||||
if (!org.has_value())
|
||||
return std::string("bad expression on line ") + std::to_string(on_line);
|
||||
|
||||
org_offset = instructions.size();
|
||||
org_source_line = on_line;
|
||||
continue;
|
||||
}
|
||||
|
||||
auto opcode_it = opcode_names.find(opcode_str);
|
||||
|
||||
if (opcode_it == opcode_names.end())
|
||||
return std::string("unknown opcode on line ") + std::to_string(on_line);
|
||||
|
||||
instr.op = opcode_it->second;
|
||||
|
||||
bool got_mod = false;
|
||||
|
||||
if (line != "" && line[0] == '.') {
|
||||
line = trim_spaces(line.substr(1));
|
||||
|
||||
if (line == "")
|
||||
return std::string("dot with no modifier on line ") + std::to_string(on_line);
|
||||
|
||||
got_mod = true;
|
||||
|
||||
size_t mod_length = 2;
|
||||
|
||||
if (line.starts_with("ab"))
|
||||
instr.mod = AB;
|
||||
else if (line.starts_with("ba"))
|
||||
instr.mod = BA;
|
||||
else {
|
||||
|
||||
mod_length = 1;
|
||||
|
||||
if (line[0] == 'a')
|
||||
instr.mod = A;
|
||||
else if (line[0] == 'b')
|
||||
instr.mod = B;
|
||||
else if (line[0] == 'f')
|
||||
instr.mod = F;
|
||||
else if (line[0] == 'x')
|
||||
instr.mod = X;
|
||||
else if (line[0] == 'i')
|
||||
instr.mod = I;
|
||||
|
||||
else
|
||||
return std::string("unknown opcode modifier on line ") + std::to_string(on_line);
|
||||
if (semicolon != std::string::npos) {
|
||||
std::string comment = trim_spaces(line.substr(semicolon + 1));
|
||||
line = line.substr(0, semicolon);
|
||||
|
||||
if (comment.starts_with("name "))
|
||||
w->name = trim_spaces(comment.substr(5));
|
||||
else if (comment.starts_with("author "))
|
||||
w->author = trim_spaces(comment.substr(7));
|
||||
}
|
||||
|
||||
line = trim_spaces(line.substr(mod_length));
|
||||
}
|
||||
for (char &ch : line)
|
||||
ch = tolower(ch);
|
||||
|
||||
std::optional<std::pair<mode, std::unique_ptr<number_expr>>> afield, bfield;
|
||||
while (true) {
|
||||
size_t colon = line.find(':');
|
||||
if (colon == std::string::npos)
|
||||
break;
|
||||
|
||||
if (line == "") {
|
||||
afield = to_field("0");
|
||||
bfield = to_field("0");
|
||||
}
|
||||
std::string label = line.substr(0, colon);
|
||||
line = line.substr(colon + 1);
|
||||
|
||||
else {
|
||||
size_t comma = line.find(",");
|
||||
if (!valid_label(label))
|
||||
return std::string("bad label on line ") + std::to_string(on_line);
|
||||
if (label_offsets.contains(label))
|
||||
return std::string("duplicate label on line ") + std::to_string(on_line);
|
||||
|
||||
if (comma == std::string::npos) {
|
||||
afield = to_field(line);
|
||||
label_offsets[label] = instructions.size();
|
||||
}
|
||||
|
||||
line = trim_spaces(line);
|
||||
if (line == "")
|
||||
continue;
|
||||
|
||||
future_instruction instr;
|
||||
instr.source_line = on_line;
|
||||
|
||||
std::string opcode_str = line.substr(0, 3);
|
||||
line = trim_spaces(line.substr(3));
|
||||
|
||||
if (opcode_str == "org") {
|
||||
if (org.has_value())
|
||||
return std::string("duplicate org on line ") + std::to_string(on_line);
|
||||
|
||||
org = to_number_expr(line);
|
||||
|
||||
if (!org.has_value())
|
||||
return std::string("bad expression on line ") + std::to_string(on_line);
|
||||
|
||||
org_offset = instructions.size();
|
||||
org_source_line = on_line;
|
||||
continue;
|
||||
}
|
||||
|
||||
auto opcode_it = opcode_names.find(opcode_str);
|
||||
|
||||
if (opcode_it == opcode_names.end())
|
||||
return std::string("unknown opcode on line ") + std::to_string(on_line);
|
||||
|
||||
instr.op = opcode_it->second;
|
||||
|
||||
bool got_mod = false;
|
||||
|
||||
if (line != "" && line[0] == '.') {
|
||||
line = trim_spaces(line.substr(1));
|
||||
|
||||
if (line == "")
|
||||
return std::string("dot with no modifier on line ") + std::to_string(on_line);
|
||||
|
||||
got_mod = true;
|
||||
|
||||
size_t mod_length = 2;
|
||||
|
||||
if (line.starts_with("ab"))
|
||||
instr.mod = AB;
|
||||
else if (line.starts_with("ba"))
|
||||
instr.mod = BA;
|
||||
else {
|
||||
|
||||
mod_length = 1;
|
||||
|
||||
if (line[0] == 'a')
|
||||
instr.mod = A;
|
||||
else if (line[0] == 'b')
|
||||
instr.mod = B;
|
||||
else if (line[0] == 'f')
|
||||
instr.mod = F;
|
||||
else if (line[0] == 'x')
|
||||
instr.mod = X;
|
||||
else if (line[0] == 'i')
|
||||
instr.mod = I;
|
||||
|
||||
else
|
||||
return std::string("unknown opcode modifier on line ") + std::to_string(on_line);
|
||||
|
||||
}
|
||||
|
||||
line = trim_spaces(line.substr(mod_length));
|
||||
}
|
||||
|
||||
std::optional<std::pair<mode, std::unique_ptr<number_expr>>> afield, bfield;
|
||||
|
||||
if (line == "") {
|
||||
afield = to_field("0");
|
||||
bfield = to_field("0");
|
||||
}
|
||||
|
||||
else {
|
||||
std::string apart = trim_spaces(line.substr(0, comma));
|
||||
std::string bpart = trim_spaces(line.substr(comma + 1));
|
||||
afield = to_field(apart);
|
||||
bfield = to_field(bpart);
|
||||
size_t comma = line.find(",");
|
||||
|
||||
if (comma == std::string::npos) {
|
||||
afield = to_field(line);
|
||||
bfield = to_field("0");
|
||||
}
|
||||
|
||||
else {
|
||||
std::string apart = trim_spaces(line.substr(0, comma));
|
||||
std::string bpart = trim_spaces(line.substr(comma + 1));
|
||||
afield = to_field(apart);
|
||||
bfield = to_field(bpart);
|
||||
}
|
||||
}
|
||||
|
||||
if (!afield.has_value() || !bfield.has_value())
|
||||
return std::string("bad expression on line ") + std::to_string(on_line);
|
||||
|
||||
instr.amode = afield.value().first;
|
||||
instr.bmode = bfield.value().first;
|
||||
instr.anumber = std::move(afield.value().second);
|
||||
instr.bnumber = std::move(bfield.value().second);
|
||||
|
||||
if (!got_mod)
|
||||
switch (instr.op) {
|
||||
|
||||
case DAT:
|
||||
instr.mod = F;
|
||||
break;
|
||||
|
||||
case MOV:
|
||||
case SEQ:
|
||||
case SNE:
|
||||
if (instr.amode == IMMEDIATE)
|
||||
instr.mod = AB;
|
||||
else if (instr.bmode == IMMEDIATE)
|
||||
instr.mod = B;
|
||||
else
|
||||
instr.mod = I;
|
||||
break;
|
||||
|
||||
case ADD:
|
||||
case SUB:
|
||||
case MUL:
|
||||
case DIV:
|
||||
case MOD:
|
||||
if (instr.amode == IMMEDIATE)
|
||||
instr.mod = AB;
|
||||
else if (instr.bmode == IMMEDIATE)
|
||||
instr.mod = B;
|
||||
else
|
||||
instr.mod = F;
|
||||
break;
|
||||
|
||||
case SLT:
|
||||
if (instr.amode == IMMEDIATE)
|
||||
instr.mod = AB;
|
||||
instr.mod = B;
|
||||
break;
|
||||
|
||||
case JMP:
|
||||
case JMZ:
|
||||
case JMN:
|
||||
case DJN:
|
||||
case SPL:
|
||||
case NOP:
|
||||
instr.mod = B;
|
||||
break;
|
||||
|
||||
}
|
||||
|
||||
instructions.push_back(std::move(instr));
|
||||
}
|
||||
|
||||
if (!afield.has_value() || !bfield.has_value())
|
||||
return std::string("bad expression on line ") + std::to_string(on_line);
|
||||
if (w->name == "")
|
||||
return "no name";
|
||||
if (w->author == "")
|
||||
return "no author";
|
||||
|
||||
instr.amode = afield.value().first;
|
||||
instr.bmode = bfield.value().first;
|
||||
instr.anumber = std::move(afield.value().second);
|
||||
instr.bnumber = std::move(bfield.value().second);
|
||||
if (org.has_value()) {
|
||||
std::optional<number_t> org_number = org.value()->to_number(label_offsets, org_offset);
|
||||
if (org_number.has_value())
|
||||
w->org = org_number.value() + org_offset;
|
||||
else
|
||||
return std::string("bad expression on line ") + std::to_string(org_source_line);
|
||||
}
|
||||
|
||||
if (!got_mod)
|
||||
switch (instr.op) {
|
||||
for (size_t i = 0; i < instructions.size(); ++i) {
|
||||
const future_instruction &instr = instructions[i];
|
||||
std::optional<number_t> anumber = instr.anumber->to_number(label_offsets, i);
|
||||
std::optional<number_t> bnumber = instr.bnumber->to_number(label_offsets, i);
|
||||
if (!anumber.has_value() || !bnumber.has_value())
|
||||
return std::string("bad expression on line ") + std::to_string(instr.source_line);
|
||||
w->instructions.push_back({
|
||||
.op = instr.op, .mod = instr.mod, .amode = instr.amode, .bmode = instr.bmode,
|
||||
.anumber = anumber.value(), .bnumber = bnumber.value()
|
||||
});
|
||||
}
|
||||
|
||||
case DAT:
|
||||
instr.mod = F;
|
||||
break;
|
||||
|
||||
case MOV:
|
||||
case SEQ:
|
||||
case SNE:
|
||||
if (instr.amode == IMMEDIATE)
|
||||
instr.mod = AB;
|
||||
else if (instr.bmode == IMMEDIATE)
|
||||
instr.mod = B;
|
||||
else
|
||||
instr.mod = I;
|
||||
break;
|
||||
|
||||
case ADD:
|
||||
case SUB:
|
||||
case MUL:
|
||||
case DIV:
|
||||
case MOD:
|
||||
if (instr.amode == IMMEDIATE)
|
||||
instr.mod = AB;
|
||||
else if (instr.bmode == IMMEDIATE)
|
||||
instr.mod = B;
|
||||
else
|
||||
instr.mod = F;
|
||||
break;
|
||||
|
||||
case SLT:
|
||||
if (instr.amode == IMMEDIATE)
|
||||
instr.mod = AB;
|
||||
instr.mod = B;
|
||||
break;
|
||||
|
||||
case JMP:
|
||||
case JMZ:
|
||||
case JMN:
|
||||
case DJN:
|
||||
case SPL:
|
||||
case NOP:
|
||||
instr.mod = B;
|
||||
break;
|
||||
|
||||
}
|
||||
|
||||
instructions.push_back(std::move(instr));
|
||||
return w.release();
|
||||
}
|
||||
|
||||
if (w->name == "")
|
||||
return "no name";
|
||||
if (w->author == "")
|
||||
return "no author";
|
||||
|
||||
if (org.has_value()) {
|
||||
std::optional<number_t> org_number = org.value()->to_number(label_offsets, org_offset);
|
||||
if (org_number.has_value())
|
||||
w->org = org_number.value() + org_offset;
|
||||
else
|
||||
return std::string("bad expression on line ") + std::to_string(org_source_line);
|
||||
}
|
||||
|
||||
for (int i = 0; i < instructions.size(); ++i) {
|
||||
const future_instruction &instr = instructions[i];
|
||||
std::optional<number_t> anumber = instr.anumber->to_number(label_offsets, i);
|
||||
std::optional<number_t> bnumber = instr.bnumber->to_number(label_offsets, i);
|
||||
if (!anumber.has_value() || !bnumber.has_value())
|
||||
return std::string("bad expression on line ") + std::to_string(instr.source_line);
|
||||
w->instructions.push_back({
|
||||
.op = instr.op, .mod = instr.mod, .amode = instr.amode, .bmode = instr.bmode,
|
||||
.anumber = ((anumber.value() % CORE_SIZE) + CORE_SIZE) % CORE_SIZE,
|
||||
.bnumber = ((bnumber.value() % CORE_SIZE) + CORE_SIZE) % CORE_SIZE
|
||||
});
|
||||
}
|
||||
|
||||
return w.release();
|
||||
}
|
||||
|
||||
struct wheader {
|
||||
size_t name_size;
|
||||
size_t author_size;
|
||||
size_t instructions_size;
|
||||
number_t org;
|
||||
};
|
||||
|
||||
bool save_warrior(const warrior &w, const std::filesystem::path &to) {
|
||||
FILE *f = fopen(to.c_str(), "wb");
|
||||
if (!f)
|
||||
return false;
|
||||
|
||||
wheader wh = {
|
||||
.name_size = w.name.size(), .author_size = w.author.size(),
|
||||
.instructions_size = w.instructions.size(), .org = w.org
|
||||
struct wheader {
|
||||
size_t name_size;
|
||||
size_t author_size;
|
||||
size_t instructions_size;
|
||||
number_t org;
|
||||
};
|
||||
|
||||
fwrite(&wh, sizeof(wheader), 1, f);
|
||||
fwrite(w.name.c_str(), w.name.size(), 1, f);
|
||||
fwrite(w.author.c_str(), w.author.size(), 1, f);
|
||||
fwrite(w.instructions.data(), sizeof(instruction) * w.instructions.size(), 1, f);
|
||||
bool save_warrior(const warrior &w, const std::filesystem::path &to) {
|
||||
FILE *f = fopen(to.c_str(), "wb");
|
||||
if (!f)
|
||||
return false;
|
||||
|
||||
fclose(f);
|
||||
return true;
|
||||
}
|
||||
wheader wh = {
|
||||
.name_size = w.name.size(), .author_size = w.author.size(),
|
||||
.instructions_size = w.instructions.size(), .org = w.org
|
||||
};
|
||||
|
||||
std::optional<warrior *> load_warrior(const std::filesystem::path &from) {
|
||||
FILE *f = fopen(from.c_str(), "rb");
|
||||
if (!f)
|
||||
return {};
|
||||
fwrite(&wh, sizeof(wheader), 1, f);
|
||||
fwrite(w.name.c_str(), w.name.size(), 1, f);
|
||||
fwrite(w.author.c_str(), w.author.size(), 1, f);
|
||||
fwrite(w.instructions.data(), sizeof(instruction) * w.instructions.size(), 1, f);
|
||||
|
||||
std::unique_ptr<warrior> w = std::make_unique<warrior>();
|
||||
fclose(f);
|
||||
return true;
|
||||
}
|
||||
|
||||
wheader wh;
|
||||
fread(&wh, sizeof(wheader), 1, f);
|
||||
|
||||
w->name.resize(wh.name_size);
|
||||
w->author.resize(wh.author_size);
|
||||
w->instructions.resize(wh.instructions_size);
|
||||
w->org = wh.org;
|
||||
|
||||
fread(w->name.data(), wh.name_size, 1, f);
|
||||
fread(w->author.data(), wh.author_size, 1, f);
|
||||
fread(w->instructions.data(), wh.instructions_size, 1, f);
|
||||
|
||||
fclose(f);
|
||||
|
||||
for (const instruction &i : w->instructions)
|
||||
if (i.op > NOP || i.mod > I || i.amode > B_INCREMENT || i.bmode > B_INCREMENT ||
|
||||
i.anumber >= CORE_SIZE || i.bnumber >= CORE_SIZE)
|
||||
std::optional<warrior *> load_warrior(const std::filesystem::path &from) {
|
||||
FILE *f = fopen(from.c_str(), "rb");
|
||||
if (!f)
|
||||
return {};
|
||||
|
||||
return w.release();
|
||||
std::unique_ptr<warrior> w = std::make_unique<warrior>();
|
||||
|
||||
wheader wh;
|
||||
fread(&wh, sizeof(wheader), 1, f);
|
||||
|
||||
w->name.resize(wh.name_size);
|
||||
w->author.resize(wh.author_size);
|
||||
w->instructions.resize(wh.instructions_size);
|
||||
w->org = wh.org;
|
||||
|
||||
fread(w->name.data(), wh.name_size, 1, f);
|
||||
fread(w->author.data(), wh.author_size, 1, f);
|
||||
fread(w->instructions.data(), wh.instructions_size, 1, f);
|
||||
|
||||
fclose(f);
|
||||
|
||||
for (const instruction &i : w->instructions)
|
||||
if (i.op > NOP || i.mod > I || i.amode > B_INCREMENT || i.bmode > B_INCREMENT ||
|
||||
i.anumber >= LIB94_CORE_SIZE || i.bnumber >= LIB94_CORE_SIZE)
|
||||
return {};
|
||||
|
||||
return w.release();
|
||||
}
|
||||
|
||||
}
|
||||
|
|
23
makefile
Normal file
23
makefile
Normal file
|
@ -0,0 +1,23 @@
|
|||
CPP_ARGS = -std=c++20 -O2 -Wall -Wextra -Iinclude
|
||||
GTKMM_CPP_ARGS = $(shell pkg-config --cflags gtkmm-4.0)
|
||||
GTKMM_LD_ARGS = $(shell pkg-config --libs gtkmm-4.0)
|
||||
|
||||
default: bin/bench
|
||||
|
||||
clean:
|
||||
rm -r obj bin
|
||||
|
||||
bin/bench: obj/bench/main.o obj/bench/bench_window.o obj/bench/core_widget.o obj/lib94.o
|
||||
@mkdir -p $(dir $@)
|
||||
g++ ${CPP_ARGS} $^ ${GTKMM_LD_ARGS} -o $@
|
||||
|
||||
obj/lib94.o: obj/lib94/core.o obj/lib94/executors.o obj/lib94/warrior.o
|
||||
ld -r $^ -o $@
|
||||
|
||||
obj/bench/%.o: bench/%.cpp
|
||||
@mkdir -p $(dir $@)
|
||||
g++ -c ${CPP_ARGS} ${GTKMM_CPP_ARGS} $^ -o $@
|
||||
|
||||
obj/%.o: %.cpp
|
||||
@mkdir -p $(dir $@)
|
||||
g++ -c ${CPP_ARGS} $^ -o $@
|
|
@ -1 +0,0 @@
|
|||
g++ -O2 --std=c++20 -DCORE_SIZE=16 test.cpp ../lib94/*.cpp -I ../include -o test
|
|
@ -1,10 +0,0 @@
|
|||
;author Standard
|
||||
;name Dwarf
|
||||
|
||||
org 1
|
||||
|
||||
dat 1, 1 ;not 0, 0 so i can see it change
|
||||
|
||||
mov 0-1, 3
|
||||
add.ab #4, 0-1
|
||||
jmp 0-2
|
|
@ -1,4 +0,0 @@
|
|||
;author Standard
|
||||
;name Imp
|
||||
|
||||
mov 0, 1
|
|
@ -1,85 +0,0 @@
|
|||
#include <lib94/lib94.hpp>
|
||||
#include <iostream>
|
||||
#include <cassert>
|
||||
#include <fstream>
|
||||
#include <chrono>
|
||||
#include <thread>
|
||||
|
||||
const char *const opcodes[] = {
|
||||
"dat", "mov", "add", "sub",
|
||||
"mul", "div", "mod", "jmp",
|
||||
"jmz", "jmn", "djn", "seq",
|
||||
"sne", "slt", "spl", "nop"
|
||||
};
|
||||
|
||||
const char *const modifiers[] = {
|
||||
"a ", "b ", "ab", "ba", "f ", "x ", "i "
|
||||
};
|
||||
|
||||
const char *modes = "#$*@{<}>";
|
||||
|
||||
int main(int argc, const char **argv) {
|
||||
assert(argc > 1);
|
||||
|
||||
seed_prng(time(0));
|
||||
|
||||
std::vector<const warrior *> ws;
|
||||
|
||||
for (int i = 1; i < argc; ++i) {
|
||||
std::string source;
|
||||
std::ifstream f(argv[i]);
|
||||
std::getline(f, source, '\0');
|
||||
f.close();
|
||||
|
||||
auto result = compile_warrior(source);
|
||||
if (std::holds_alternative<std::string>(result)) {
|
||||
std::cout << "Error in " << argv[i] << ": " << std::get<std::string>(result) << std::endl;
|
||||
return 0;
|
||||
}
|
||||
|
||||
ws.push_back(std::get<warrior *>(result));
|
||||
}
|
||||
|
||||
std::vector<number_t> wps;
|
||||
wps.resize(ws.size());
|
||||
|
||||
clear_core({.op = DAT, .mod = F, .amode = DIRECT, .bmode = DIRECT, .anumber = 0, .bnumber = 0});
|
||||
init_round(ws);
|
||||
|
||||
while (true) {
|
||||
for (int i = 0; i < ws.size(); ++i)
|
||||
if (get_processes(ws[i]).size() == 0)
|
||||
wps[i] = -1;
|
||||
else
|
||||
wps[i] = get_next_process(ws[i]);
|
||||
|
||||
auto mod = get_modified_addresses();
|
||||
|
||||
printf("\x1b[H\x1b[2J");
|
||||
|
||||
for (int a = 0; a < CORE_SIZE; ++a) {
|
||||
const instruction &i = get_instruction(a);
|
||||
|
||||
printf("%d: %s.%s %c%d, %c%d",
|
||||
a, opcodes[i.op], modifiers[i.mod],
|
||||
modes[i.amode], i.anumber,
|
||||
modes[i.bmode], i.bnumber);
|
||||
|
||||
if (mod.contains(a))
|
||||
printf(" *");
|
||||
|
||||
for (int w = 0; w < ws.size(); ++w)
|
||||
if (a == wps[w])
|
||||
printf(" < %s", ws[w]->name.c_str());
|
||||
|
||||
putchar('\n');
|
||||
}
|
||||
|
||||
std::this_thread::sleep_for(std::chrono::milliseconds(250));
|
||||
|
||||
clear_modified_addresses();
|
||||
single_step();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
20
warriors/big-nothing.red
Normal file
20
warriors/big-nothing.red
Normal file
|
@ -0,0 +1,20 @@
|
|||
;author blah
|
||||
;name big-nothing
|
||||
|
||||
start:
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
nop
|
||||
jmp start
|
6
warriors/dwarf.red
Normal file
6
warriors/dwarf.red
Normal file
|
@ -0,0 +1,6 @@
|
|||
;author standard
|
||||
;name dwarf
|
||||
|
||||
mov 0-1, 3
|
||||
add.ab #4, 0-1
|
||||
jmp 0-2
|
4
warriors/imp.red
Normal file
4
warriors/imp.red
Normal file
|
@ -0,0 +1,4 @@
|
|||
;author standard
|
||||
;name imp
|
||||
|
||||
mov 0, 1
|
5
warriors/splitter.red
Normal file
5
warriors/splitter.red
Normal file
|
@ -0,0 +1,5 @@
|
|||
;author blah
|
||||
;name splitter
|
||||
|
||||
spl 0
|
||||
jmp 0-1
|
28
warriors/trap.red
Normal file
28
warriors/trap.red
Normal file
|
@ -0,0 +1,28 @@
|
|||
;author benji
|
||||
;name trap
|
||||
|
||||
org start
|
||||
|
||||
trap:
|
||||
spl 1
|
||||
jmp trap
|
||||
|
||||
start:
|
||||
mov.i bomb, bomb + 8
|
||||
add.ab #8, start
|
||||
sub.ab #8, bomb
|
||||
sne.ab #bomb - 8 - start, start
|
||||
jmp core_clear
|
||||
jmp start
|
||||
|
||||
bomb:
|
||||
jmp trap - 8
|
||||
|
||||
core_clear:
|
||||
mov background, background + 1
|
||||
add.ab #1, core_clear
|
||||
jnz.b core_clear, core_clear
|
||||
mov.ab #background + 1 - core_clear, core_clear
|
||||
jmp core_clear
|
||||
|
||||
background:
|
Loading…
Add table
Reference in a new issue