slige/runtime/main.cpp

194 lines
4.8 KiB
C++
Raw Normal View History

2024-11-19 04:06:27 +00:00
#include "arch.hpp"
#include "json.hpp"
#include "rpc_server.hpp"
2024-11-19 04:06:27 +00:00
#include "vm.hpp"
2024-11-08 11:22:42 +00:00
#include <format>
#include <iostream>
2024-11-20 11:54:03 +00:00
#include <string>
2024-11-19 04:06:27 +00:00
#include <utility>
#include <variant>
2024-11-08 11:22:42 +00:00
2024-11-20 11:54:03 +00:00
enum class AsmLineType {
Op,
Lit,
Loc,
Ref,
};
struct Loc {
explicit Loc(std::string value)
: value(value)
{
}
std::string value;
};
struct Ref {
explicit Ref(std::string value)
: value(value)
{
}
std::string value;
};
struct AsmLine {
/* clang-format off */
AsmLine(sliger::Op value) : type(AsmLineType::Op), value(value) {}
AsmLine(uint32_t value) : type(AsmLineType::Lit), value(value) {}
AsmLine(Loc value) : type(AsmLineType::Loc), value(value) {}
AsmLine(Ref value) : type(AsmLineType::Ref), value(value) {}
/* clang-format on */
AsmLineType type;
std::variant<sliger::Op, uint32_t, Loc, Ref> value;
};
auto compile_asm(const std::vector<AsmLine>& lines) -> std::vector<uint32_t>
{
size_t ip = 0;
auto output = std::vector<uint32_t>();
auto locs = std::unordered_map<std::string, size_t>();
auto refs = std::unordered_map<size_t, std::string>();
for (const auto& line : lines) {
switch (line.type) {
case AsmLineType::Op: {
output.push_back(
std::to_underlying(std::get<sliger::Op>(line.value)));
ip += 1;
break;
}
case AsmLineType::Lit: {
output.push_back(std::get<uint32_t>(line.value));
ip += 1;
break;
}
case AsmLineType::Loc: {
2024-11-20 13:46:19 +00:00
locs.insert_or_assign(std::get<Loc>(line.value).value, ip);
2024-11-20 11:54:03 +00:00
break;
}
case AsmLineType::Ref: {
output.push_back(0);
refs.insert_or_assign(ip, std::get<Ref>(line.value).value);
ip += 1;
break;
}
}
}
for (size_t i = 0; i < output.size(); ++i) {
2024-11-20 13:46:19 +00:00
if (!refs.contains(i)) {
continue;
2024-11-20 11:54:03 +00:00
}
2024-11-20 13:46:19 +00:00
if (!locs.contains(refs.at(i))) {
std::cerr << std::format(
"error: label \"{}\" used at {} not defined\n", refs.at(i), i);
continue;
}
output.at(i) = static_cast<uint32_t>(locs.at(refs.at(i)));
2024-11-20 11:54:03 +00:00
}
return output;
}
auto execute_action(std::unique_ptr<sliger::json::Value> req,
std::unique_ptr<slige_rpc::BufferedWriter> writer) -> void
{
}
2024-11-08 11:22:42 +00:00
int main()
{
2024-11-20 11:54:03 +00:00
using R = Ref;
using L = Loc;
using enum sliger::Op;
2024-11-19 04:06:27 +00:00
auto rpc = slige_rpc::RpcServer(
[&](std::unique_ptr<sliger::json::Value> req,
std::unique_ptr<slige_rpc::BufferedWriter> writer) {
execute_action(std::move(req), std::move(writer));
});
rpc.listen();
2024-11-19 04:06:27 +00:00
// fn add(a, b) {
// + a b
// }
2024-11-08 11:22:42 +00:00
//
2024-11-20 13:46:19 +00:00
// fn main() {
// let result = 0;
// let i = 0;
// loop {
// if i >= 10 {
// break;
// }
// result = add(result, 5);
// i = + i 1;
2024-11-19 04:06:27 +00:00
// }
2024-11-20 13:46:19 +00:00
// result
2024-11-19 04:06:27 +00:00
// }
2024-11-20 11:54:03 +00:00
auto program_asm = std::vector<AsmLine> {
// clang-format off
SourceMap, 0, 0, 0,
PushPtr, R("main"),
2024-11-20 13:46:19 +00:00
Call, 0,
2024-11-20 11:54:03 +00:00
PushPtr, R("_exit"),
Jump,
Pop,
L("add"),
SourceMap, 19, 2, 5,
Add,
Return,
2024-11-20 13:46:19 +00:00
L("main"),
2024-11-20 11:54:03 +00:00
SourceMap, 28, 5, 1,
PushInt, 0,
2024-11-20 13:46:19 +00:00
PushInt, 0,
2024-11-20 11:54:03 +00:00
SourceMap, 44, 6, 1,
PushInt, 0,
SourceMap, 55, 7, 1,
2024-11-20 13:46:19 +00:00
L("0"),
2024-11-20 11:54:03 +00:00
SourceMap, 66, 8, 5,
2024-11-20 13:46:19 +00:00
LoadLocal, 2,
PushInt, 10,
2024-11-20 11:54:03 +00:00
LessThan,
Not,
PushPtr, R("1"),
JumpIfFalse,
SourceMap, 87, 9, 9,
PushPtr, R("2"),
Jump,
2024-11-20 13:46:19 +00:00
L("1"),
2024-11-20 11:54:03 +00:00
SourceMap, 104, 11, 5,
2024-11-20 13:46:19 +00:00
LoadLocal, 1,
2024-11-20 11:54:03 +00:00
PushInt, 5,
PushPtr, R("add"),
Call, 2,
2024-11-20 13:46:19 +00:00
StoreLocal, 1,
2024-11-20 11:54:03 +00:00
SourceMap, 133, 12, 5,
2024-11-20 13:46:19 +00:00
LoadLocal, 2,
2024-11-20 11:54:03 +00:00
PushInt, 1,
Add,
2024-11-20 13:46:19 +00:00
StoreLocal, 2,
2024-11-20 11:54:03 +00:00
PushPtr, R("0"),
Jump,
L("2"),
2024-11-20 13:46:19 +00:00
LoadLocal, 1,
StoreLocal, 0,
Pop,
2024-11-20 11:54:03 +00:00
Pop,
Return,
L("_exit"),
SourceMap, 147, 15, 1
// clang-format on
2024-11-19 04:06:27 +00:00
};
2024-11-20 11:54:03 +00:00
auto program = compile_asm(program_asm);
2024-11-19 04:06:27 +00:00
auto vm = sliger::VM(program,
{
.flame_graph = true,
.code_coverage = true,
});
vm.run_until_done();
2024-11-20 14:07:39 +00:00
std::cout << std::format("done\n{}\n", vm.stack_repr_string(4));
auto flame_graph = vm.flame_graph_json();
std::cout << std::format("flame graph: {}\n", flame_graph);
2024-11-21 03:12:07 +00:00
auto code_coverage = vm.code_coverage_json();
std::cout << std::format("code coverage: {}\n", code_coverage);
2024-11-08 11:22:42 +00:00
}