forked from AbleOS/holey-bytes
Reimplemented memory instructions, deal with it.
This commit is contained in:
parent
4dd2052634
commit
7f2676af91
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -73,6 +73,7 @@ dependencies = [
|
||||||
"hashbrown",
|
"hashbrown",
|
||||||
"hbbytecode",
|
"hbbytecode",
|
||||||
"log",
|
"log",
|
||||||
|
"paste",
|
||||||
"static_assertions",
|
"static_assertions",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
|
@ -1,28 +0,0 @@
|
||||||
# Math operations
|
|
||||||
```
|
|
||||||
MATH_OP
|
|
||||||
Add
|
|
||||||
Sub
|
|
||||||
Mul
|
|
||||||
Div
|
|
||||||
Mod
|
|
||||||
```
|
|
||||||
```
|
|
||||||
MATH_TYPE
|
|
||||||
Unsigned
|
|
||||||
Signed
|
|
||||||
FloatingPoint
|
|
||||||
```
|
|
||||||
|
|
||||||
```
|
|
||||||
MATH_OP_SIDES
|
|
||||||
Register Constant
|
|
||||||
Register Register
|
|
||||||
Constant Constant
|
|
||||||
Constant Register
|
|
||||||
```
|
|
||||||
`[MATH_OP] [MATH_OP_SIDES] [MATH_TYPE] [IMM_LHS] [IMM_RHS] [REG]`
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
160
hbasm/src/lib.rs
160
hbasm/src/lib.rs
|
@ -27,7 +27,7 @@ macro_rules! tokendef {
|
||||||
#[regex(
|
#[regex(
|
||||||
"r[0-9]+",
|
"r[0-9]+",
|
||||||
|lexer| match lexer.slice()[1..].parse() {
|
|lexer| match lexer.slice()[1..].parse() {
|
||||||
Ok(n) if n <= 59 => Some(n),
|
Ok(n) => Some(n),
|
||||||
_ => None
|
_ => None
|
||||||
},
|
},
|
||||||
)] Register(u8),
|
)] Register(u8),
|
||||||
|
@ -52,11 +52,10 @@ macro_rules! tokendef {
|
||||||
|
|
||||||
#[rustfmt::skip]
|
#[rustfmt::skip]
|
||||||
tokendef![
|
tokendef![
|
||||||
"nop", "add", "sub", "mul", "and", "or", "xor", "sl", "sr", "srs", "cmp", "cmpu", "not", "neg",
|
"nop", "add", "sub", "mul", "and", "or", "xor", "sl", "sr", "srs", "cmp", "cmpu",
|
||||||
"dir", "addf", "mulf", "dirf", "addi", "muli", "andi", "ori",
|
"dir", "neg", "not", "addi", "muli", "andi", "ori", "xori", "sli", "sri", "srsi",
|
||||||
"xori", "sli", "sri", "srsi", "cmpi", "cmpui", "addfi", "mulfi", "cp", "li", "lb",
|
"cmpi", "cmpui", "cp", "swa", "li", "ld", "st", "bmc", "brc", "jmp", "jeq", "jne",
|
||||||
"ld", "lq", "lo", "sb", "sd", "sq", "so", "jmp", "jeq", "jne", "jlt", "jgt",
|
"jlt", "jgt", "jltu", "jgtu", "ecall", "addf", "mulf", "dirf", "addfi", "mulfi",
|
||||||
"jltu", "jgtu", "ecall",
|
|
||||||
];
|
];
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||||
|
@ -113,11 +112,96 @@ pub fn assembly(code: &str, buf: &mut Vec<u8>) -> Result<(), Error> {
|
||||||
self.buf.push(op);
|
self.buf.push(op);
|
||||||
match op {
|
match op {
|
||||||
NOP | ECALL => Ok(()),
|
NOP | ECALL => Ok(()),
|
||||||
DIR | DIRF => self.rrrr(),
|
DIR | DIRF => {
|
||||||
ADD..=CMPU | ADDF..=MULF => self.rrr(),
|
expect_matches!(
|
||||||
NOT | CP => self.rr(),
|
self,
|
||||||
LI | JMP => self.ri(),
|
Token::Register(r0),
|
||||||
ADDI..=CMPUI | ADDFI..=MULFI | LB..=SO | JEQ..=JGTU => self.rri(),
|
Token::PSep,
|
||||||
|
Token::Register(r1),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r2),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r3),
|
||||||
|
);
|
||||||
|
self.buf.extend([r0, r1, r2, r3]);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
ADD..=CMPU | ADDF..=MULF => {
|
||||||
|
expect_matches!(
|
||||||
|
self,
|
||||||
|
Token::Register(r0),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r1),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r2),
|
||||||
|
);
|
||||||
|
self.buf.extend([r0, r1, r2]);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
BRC => {
|
||||||
|
expect_matches!(
|
||||||
|
self,
|
||||||
|
Token::Register(r0),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r1),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Integer(count),
|
||||||
|
);
|
||||||
|
self.buf.extend([
|
||||||
|
r0,
|
||||||
|
r1,
|
||||||
|
u8::try_from(count).map_err(|_| ErrorKind::UnexpectedToken)?,
|
||||||
|
]);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
NEG..=NOT | CP..=SWA => {
|
||||||
|
expect_matches!(
|
||||||
|
self,
|
||||||
|
Token::Register(r0),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r1),
|
||||||
|
);
|
||||||
|
self.buf.extend([r0, r1]);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
LI | JMP => {
|
||||||
|
expect_matches!(self, Token::Register(r0), Token::PSep);
|
||||||
|
self.buf.push(r0);
|
||||||
|
self.insert_imm()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
ADDI..=CMPUI | BMC | JEQ..=JGTU | ADDFI..=MULFI => {
|
||||||
|
expect_matches!(
|
||||||
|
self,
|
||||||
|
Token::Register(r0),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r1),
|
||||||
|
Token::PSep,
|
||||||
|
);
|
||||||
|
self.buf.extend([r0, r1]);
|
||||||
|
self.insert_imm()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
LD..=ST => {
|
||||||
|
expect_matches!(
|
||||||
|
self,
|
||||||
|
Token::Register(r0),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Register(r1),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Integer(offset),
|
||||||
|
Token::PSep,
|
||||||
|
Token::Integer(len),
|
||||||
|
);
|
||||||
|
self.buf.extend([r0, r1]);
|
||||||
|
self.buf.extend(offset.to_le_bytes());
|
||||||
|
self.buf.extend(
|
||||||
|
u16::try_from(len)
|
||||||
|
.map_err(|_| ErrorKind::InvalidToken)?
|
||||||
|
.to_le_bytes(),
|
||||||
|
);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}?;
|
}?;
|
||||||
match self.next() {
|
match self.next() {
|
||||||
|
@ -154,60 +238,6 @@ pub fn assembly(code: &str, buf: &mut Vec<u8>) -> Result<(), Error> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn rrr(&mut self) -> Result<(), ErrorKind> {
|
|
||||||
expect_matches!(
|
|
||||||
self,
|
|
||||||
Token::Register(r0),
|
|
||||||
Token::PSep,
|
|
||||||
Token::Register(r1),
|
|
||||||
Token::PSep,
|
|
||||||
Token::Register(r2)
|
|
||||||
);
|
|
||||||
self.buf.extend([r0, r1, r2]);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn rr(&mut self) -> Result<(), ErrorKind> {
|
|
||||||
expect_matches!(self, Token::Register(r0), Token::PSep, Token::Register(r1),);
|
|
||||||
self.buf.extend([r0, r1]);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn ri(&mut self) -> Result<(), ErrorKind> {
|
|
||||||
expect_matches!(self, Token::Register(r0), Token::PSep);
|
|
||||||
self.buf.push(r0);
|
|
||||||
self.insert_imm()?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn rri(&mut self) -> Result<(), ErrorKind> {
|
|
||||||
expect_matches!(
|
|
||||||
self,
|
|
||||||
Token::Register(r0),
|
|
||||||
Token::PSep,
|
|
||||||
Token::Register(r1),
|
|
||||||
Token::PSep,
|
|
||||||
);
|
|
||||||
self.buf.extend([r0, r1]);
|
|
||||||
self.insert_imm()?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn rrrr(&mut self) -> Result<(), ErrorKind> {
|
|
||||||
expect_matches!(
|
|
||||||
self,
|
|
||||||
Token::Register(r0),
|
|
||||||
Token::PSep,
|
|
||||||
Token::Register(r1),
|
|
||||||
Token::PSep,
|
|
||||||
Token::Register(r2),
|
|
||||||
Token::PSep,
|
|
||||||
Token::Register(r3),
|
|
||||||
);
|
|
||||||
self.buf.extend([r0, r1, r2, r3]);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn insert_imm(&mut self) -> Result<(), ErrorKind> {
|
fn insert_imm(&mut self) -> Result<(), ErrorKind> {
|
||||||
let imm = match self.next()? {
|
let imm = match self.next()? {
|
||||||
Token::Integer(i) => i.to_le_bytes(),
|
Token::Integer(i) => i.to_le_bytes(),
|
||||||
|
|
60
hbbytecode/hbbytecode.h
Normal file
60
hbbytecode/hbbytecode.h
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
/* HoleyBytes Bytecode representation in C
|
||||||
|
* Requires C23 compiler or better
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
typedef enum hbbc_Opcode: uint8_t {
|
||||||
|
hbbc_Op_NOP, hbbc_Op_ADD, hbbc_Op_MUL, hbbc_Op_AND, hbbc_Op_OR, hbbc_Op_XOR, hbbc_Op_SL,
|
||||||
|
hbbc_Op_SR, hbbc_Op_SRS, hbbc_Op_CMP, hbbc_Op_CMPU, hbbc_Op_DIR, hbbc_Op_NEG, hbbc_Op_NOT,
|
||||||
|
hbbc_Op_ADDI, hbbc_Op_MULI, hbbc_Op_ANDI, hbbc_Op_ORI, hbbc_Op_XORI, hbbc_Op_SLI, hbbc_Op_SRI,
|
||||||
|
hbbc_Op_SRSI, hbbc_Op_CMPI, hbbc_Op_CMPUI, hbbc_Op_CP, hbbc_Op_SWA, hbbc_Op_LI, hbbc_Op_LD,
|
||||||
|
hbbc_Op_ST, hbbc_Op_BMC, hbbc_Op_BRC, hbbc_Op_JMP, hbbc_Op_JEQ, hbbc_Op_JNE, hbbc_Op_JLT,
|
||||||
|
hbbc_Op_JGT, hbbc_Op_JLTU, hbbc_Op_JGTU, hbbc_Op_ECALL, hbbc_Op_ADDF, hbbc_Op_MULF,
|
||||||
|
hbbc_Op_DIRF, hbbc_Op_ADDFI, hbbc_Op_MULFI,
|
||||||
|
} hbbc_Opcode;
|
||||||
|
|
||||||
|
static_assert(sizeof(hbbc_Opcode) == 1);
|
||||||
|
|
||||||
|
#pragma pack(push, 1)
|
||||||
|
typedef struct hbbc_ParamBBBB
|
||||||
|
{ uint8_t _0; uint8_t _1; uint8_t _2; uint8_t _3; }
|
||||||
|
hbbc_ParamBBBB;
|
||||||
|
static_assert(sizeof(hbbc_ParamBBBB) == 4);
|
||||||
|
|
||||||
|
typedef struct hbbc_ParamBBB
|
||||||
|
{ uint8_t _0; uint8_t _1; uint8_t _2; }
|
||||||
|
hbbc_ParamBBB;
|
||||||
|
static_assert(sizeof(hbbc_ParamBBB) == 3);
|
||||||
|
|
||||||
|
typedef struct hbbc_ParamBBDH
|
||||||
|
{ uint8_t _0; uint8_t _1; uint64_t _2; uint16_t _3; }
|
||||||
|
hbbc_ParamBBDH;
|
||||||
|
static_assert(sizeof(hbbc_ParamBBDH) == 12);
|
||||||
|
|
||||||
|
typedef struct hbbc_ParamBBDB
|
||||||
|
{ uint8_t _0; uint8_t _1; uint64_t _2; uint8_t _3; }
|
||||||
|
hbbc_ParamBBDB;
|
||||||
|
static_assert(sizeof(hbbc_ParamBBDB) == 11);
|
||||||
|
|
||||||
|
typedef struct hbbc_ParamBBD
|
||||||
|
{ uint8_t _0; uint8_t _1; uint64_t _2; }
|
||||||
|
hbbc_ParamBBD;
|
||||||
|
static_assert(sizeof(hbbc_ParamBBD) == 10);
|
||||||
|
|
||||||
|
typedef struct hbbc_ParamBB
|
||||||
|
{ uint8_t _0; uint8_t _1; }
|
||||||
|
hbbc_ParamBB;
|
||||||
|
static_assert(sizeof(hbbc_ParamBB) == 2);
|
||||||
|
|
||||||
|
typedef struct hbbc_ParamBD
|
||||||
|
{ uint8_t _0; uint64_t _1; }
|
||||||
|
hbbc_ParamBD;
|
||||||
|
static_assert(sizeof(hbbc_ParamBD) == 9);
|
||||||
|
|
||||||
|
typedef uint64_t hbbc_ParamD;
|
||||||
|
static_assert(sizeof(hbbc_ParamD) == 8);
|
||||||
|
|
||||||
|
#pragma pack(pop)
|
|
@ -20,88 +20,87 @@ constmod!(pub opcode(u8) {
|
||||||
|
|
||||||
NOP = 0, "N; Do nothing";
|
NOP = 0, "N; Do nothing";
|
||||||
|
|
||||||
ADD = 1, "RRR; #0 ← #1 + #2";
|
ADD = 1, "BBB; #0 ← #1 + #2";
|
||||||
SUB = 2, "RRR; #0 ← #1 - #2";
|
SUB = 2, "BBB; #0 ← #1 - #2";
|
||||||
MUL = 3, "RRR; #0 ← #1 × #2";
|
MUL = 3, "BBB; #0 ← #1 × #2";
|
||||||
AND = 4, "RRR; #0 ← #1 & #2";
|
AND = 4, "BBB; #0 ← #1 & #2";
|
||||||
OR = 5, "RRR; #0 ← #1 | #2";
|
OR = 5, "BBB; #0 ← #1 | #2";
|
||||||
XOR = 6, "RRR; #0 ← #1 ^ #2";
|
XOR = 6, "BBB; #0 ← #1 ^ #2";
|
||||||
SL = 7, "RRR; #0 ← #1 « #2";
|
SL = 7, "BBB; #0 ← #1 « #2";
|
||||||
SR = 8, "RRR; #0 ← #1 » #2";
|
SR = 8, "BBB; #0 ← #1 » #2";
|
||||||
SRS = 9, "RRR; #0 ← #1 » #2 (signed)";
|
SRS = 9, "BBB; #0 ← #1 » #2 (signed)";
|
||||||
CMP = 10, "RRR; #0 ← #1 <=> #2";
|
CMP = 10, "BBB; #0 ← #1 <=> #2";
|
||||||
CMPU = 11, "RRR; #0 ← #1 <=> #2 (unsigned)";
|
CMPU = 11, "BBB; #0 ← #1 <=> #2 (unsigned)";
|
||||||
DIR = 12, "RRRR; #0 ← #2 / #3, #1 ← #2 % #3";
|
DIR = 12, "BBBB; #0 ← #2 / #3, #1 ← #2 % #3";
|
||||||
NEG = 13, "RR; #0 ← ~#1";
|
NEG = 13, "BB; #0 ← ~#1";
|
||||||
NOT = 14, "RR; #0 ← !#1";
|
NOT = 14, "BB; #0 ← !#1";
|
||||||
|
|
||||||
ADDI = 18, "RRI; #0 ← #1 + imm #2";
|
ADDI = 18, "BBD; #0 ← #1 + imm #2";
|
||||||
MULI = 19, "RRI; #0 ← #1 × imm #2";
|
MULI = 19, "BBD; #0 ← #1 × imm #2";
|
||||||
ANDI = 20, "RRI; #0 ← #1 & imm #2";
|
ANDI = 20, "BBD; #0 ← #1 & imm #2";
|
||||||
ORI = 21, "RRI; #0 ← #1 | imm #2";
|
ORI = 21, "BBD; #0 ← #1 | imm #2";
|
||||||
XORI = 22, "RRI; #0 ← #1 ^ imm #2";
|
XORI = 22, "BBD; #0 ← #1 ^ imm #2";
|
||||||
SLI = 23, "RRI; #0 ← #1 « imm #2";
|
SLI = 23, "BBD; #0 ← #1 « imm #2";
|
||||||
SRI = 24, "RRI; #0 ← #1 » imm #2";
|
SRI = 24, "BBD; #0 ← #1 » imm #2";
|
||||||
SRSI = 25, "RRI; #0 ← #1 » imm #2 (signed)";
|
SRSI = 25, "BBD; #0 ← #1 » imm #2 (signed)";
|
||||||
CMPI = 26, "RRI; #0 ← #1 <=> imm #2";
|
CMPI = 26, "BBD; #0 ← #1 <=> imm #2";
|
||||||
CMPUI = 27, "RRI; #0 ← #1 <=> imm #2 (unsigned)";
|
CMPUI = 27, "BBD; #0 ← #1 <=> imm #2 (unsigned)";
|
||||||
|
|
||||||
CP = 28, "RR; Copy #0 ← #1";
|
CP = 28, "BB; Copy #0 ← #1";
|
||||||
LI = 29, "RI; Load immediate, #0 ← imm #1";
|
SWA = 29, "BB; Swap #0 and #1";
|
||||||
LB = 30, "RRI; Load byte (8 bits), #0 ← [#1 + imm #2]";
|
LI = 30, "BD; #0 ← imm #1";
|
||||||
LD = 31, "RRI; Load doublet (16 bits)";
|
LD = 31, "BBDB; #0 ← [#1 + imm #3], imm #4 bytes, overflowing";
|
||||||
LQ = 32, "RRI; Load quadlet (32 bits)";
|
ST = 32, "BBDB; [#1 + imm #3] ← #0, imm #4 bytes, overflowing";
|
||||||
LO = 33, "RRI; Load octlet (64 bits)";
|
BMC = 33, "BBD; [#0] ← [#1], imm #2 bytes";
|
||||||
SB = 34, "RRI; Store byte, [#1 + imm #2] ← #0";
|
BRC = 34, "BBB; #0 ← #1, imm #2 registers";
|
||||||
SD = 35, "RRI; Store doublet";
|
|
||||||
SQ = 36, "RRI; Store quadlet";
|
|
||||||
SO = 37, "RRI; Store octlet";
|
|
||||||
|
|
||||||
JMP = 38, "RI; Unconditional jump [#0 + imm #1]";
|
JMP = 35, "BD; Unconditional jump [#0 + imm #1]";
|
||||||
JEQ = 39, "RRI; if #0 = #1 → jump imm #2";
|
JEQ = 36, "BBD; if #0 = #1 → jump imm #2";
|
||||||
JNE = 40, "RRI; if #0 ≠ #1 → jump imm #2";
|
JNE = 37, "BBD; if #0 ≠ #1 → jump imm #2";
|
||||||
JLT = 41, "RRI; if #0 < #1 → jump imm #2";
|
JLT = 38, "BBD; if #0 < #1 → jump imm #2";
|
||||||
JGT = 42, "RRI; if #0 > #1 → jump imm #2";
|
JGT = 39, "BBD; if #0 > #1 → jump imm #2";
|
||||||
JLTU = 43, "RRI; if #0 < #1 → jump imm #2 (unsigned)";
|
JLTU = 40, "BBD; if #0 < #1 → jump imm #2 (unsigned)";
|
||||||
JGTU = 44, "RRI; if #0 > #1 → jump imm #2 (unsigned)";
|
JGTU = 41, "BBD; if #0 > #1 → jump imm #2 (unsigned)";
|
||||||
ECALL = 45, "N; Issue system call";
|
ECALL = 42, "N; Issue system call";
|
||||||
|
|
||||||
ADDF = 46, "RRR; #0 ← #1 +. #2";
|
ADDF = 43, "BBB; #0 ← #1 +. #2";
|
||||||
MULF = 47, "RRR; #0 ← #1 +. #2";
|
MULF = 44, "BBB; #0 ← #1 +. #2";
|
||||||
DIRF = 48, "RRRR; #0 ← #2 / #3, #1 ← #2 % #3";
|
DIRF = 45, "BBBB; #0 ← #2 / #3, #1 ← #2 % #3";
|
||||||
|
|
||||||
ADDFI = 49, "RRI; #0 ← #1 +. imm #2";
|
ADDFI = 46, "BBD; #0 ← #1 +. imm #2";
|
||||||
MULFI = 50, "RRI; #0 ← #1 *. imm #2";
|
MULFI = 47, "BBD; #0 ← #1 *. imm #2";
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
/// Register-register-register-register instruction parameter
|
|
||||||
#[repr(packed)]
|
#[repr(packed)]
|
||||||
pub struct ParamRRRR(pub u8, pub u8, pub u8, pub u8);
|
pub struct ParamBBBB(pub u8, pub u8, pub u8, pub u8);
|
||||||
|
|
||||||
/// Register-register-register instruction parameter
|
|
||||||
#[repr(packed)]
|
#[repr(packed)]
|
||||||
pub struct ParamRRR(pub u8, pub u8, pub u8);
|
pub struct ParamBBB(pub u8, pub u8, pub u8);
|
||||||
|
|
||||||
/// Register-register-immediate intruction parameter
|
|
||||||
#[repr(packed)]
|
#[repr(packed)]
|
||||||
pub struct ParamRRI(pub u8, pub u8, pub u64);
|
pub struct ParamBBDH(pub u8, pub u8, pub u64, pub u16);
|
||||||
|
|
||||||
/// Register-register instruction parameter
|
|
||||||
#[repr(packed)]
|
#[repr(packed)]
|
||||||
pub struct ParamRR(pub u8, pub u8);
|
pub struct ParamBBDB(pub u8, pub u8, pub u64, pub u8);
|
||||||
|
|
||||||
/// Register-immediate instruction parameter
|
|
||||||
#[repr(packed)]
|
#[repr(packed)]
|
||||||
pub struct ParamRI(pub u8, pub u64);
|
pub struct ParamBBD(pub u8, pub u8, pub u64);
|
||||||
|
|
||||||
|
#[repr(packed)]
|
||||||
|
pub struct ParamBB(pub u8, pub u8);
|
||||||
|
|
||||||
|
#[repr(packed)]
|
||||||
|
pub struct ParamBD(pub u8, pub u64);
|
||||||
|
|
||||||
/// # Safety
|
/// # Safety
|
||||||
/// Has to be valid to be decoded from bytecode.
|
/// Has to be valid to be decoded from bytecode.
|
||||||
pub unsafe trait OpParam {}
|
pub unsafe trait OpParam {}
|
||||||
unsafe impl OpParam for ParamRRRR {}
|
unsafe impl OpParam for ParamBBBB {}
|
||||||
unsafe impl OpParam for ParamRRR {}
|
unsafe impl OpParam for ParamBBB {}
|
||||||
unsafe impl OpParam for ParamRRI {}
|
unsafe impl OpParam for ParamBBDB {}
|
||||||
unsafe impl OpParam for ParamRR {}
|
unsafe impl OpParam for ParamBBDH {}
|
||||||
unsafe impl OpParam for ParamRI {}
|
unsafe impl OpParam for ParamBBD {}
|
||||||
|
unsafe impl OpParam for ParamBB {}
|
||||||
|
unsafe impl OpParam for ParamBD {}
|
||||||
unsafe impl OpParam for u64 {}
|
unsafe impl OpParam for u64 {}
|
||||||
unsafe impl OpParam for () {}
|
unsafe impl OpParam for () {}
|
||||||
|
|
|
@ -11,4 +11,5 @@ delegate = "0.9"
|
||||||
hashbrown = "0.13"
|
hashbrown = "0.13"
|
||||||
hbbytecode.path = "../hbbytecode"
|
hbbytecode.path = "../hbbytecode"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
|
paste = "1.0"
|
||||||
static_assertions = "1.0"
|
static_assertions = "1.0"
|
||||||
|
|
|
@ -1,20 +1,8 @@
|
||||||
macro_rules! bail {
|
|
||||||
($kind:ident, $start:expr, $curr:expr, $offset:expr) => {
|
|
||||||
return Err(Error {
|
|
||||||
kind: ErrorKind::$kind,
|
|
||||||
index: ($curr.as_ptr() as usize) - ($start.as_ptr() as usize) + $offset,
|
|
||||||
})
|
|
||||||
};
|
|
||||||
($kind:ident, $start:expr, $curr:expr) => {
|
|
||||||
bail!($kind, $start, $curr, 0)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||||
pub enum ErrorKind {
|
pub enum ErrorKind {
|
||||||
InvalidInstruction,
|
InvalidInstruction,
|
||||||
InvalidRegister,
|
|
||||||
Unimplemented,
|
Unimplemented,
|
||||||
|
RegisterArrayOverflow,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
||||||
|
@ -26,57 +14,39 @@ pub struct Error {
|
||||||
pub fn validate(mut program: &[u8]) -> Result<(), Error> {
|
pub fn validate(mut program: &[u8]) -> Result<(), Error> {
|
||||||
use hbbytecode::opcode::*;
|
use hbbytecode::opcode::*;
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn reg(regs: &[u8]) -> Option<usize> {
|
|
||||||
regs.iter()
|
|
||||||
.enumerate()
|
|
||||||
.filter_map(|(n, &r)| (r > 59).then_some(n))
|
|
||||||
.next()
|
|
||||||
}
|
|
||||||
|
|
||||||
let start = program;
|
let start = program;
|
||||||
loop {
|
loop {
|
||||||
program = match program {
|
program = match program {
|
||||||
[] => return Ok(()),
|
[] => return Ok(()),
|
||||||
// N
|
[LD..=ST, reg, _, _, _, _, _, _, _, _, _, count, ..]
|
||||||
[NOP | ECALL, rest @ ..] => rest,
|
if usize::from(*reg) * 8 + usize::from(*count) > 2048 =>
|
||||||
// RRRR
|
|
||||||
[DIR | DIRF, _, _, _, _, rest @ ..] => {
|
|
||||||
if let Some(n) = reg(&program[1..=4]) {
|
|
||||||
bail!(InvalidRegister, start, program, n + 1);
|
|
||||||
}
|
|
||||||
rest
|
|
||||||
}
|
|
||||||
// RRR
|
|
||||||
[ADD..=CMPU | ADDF..=MULF, _, _, _, rest @ ..] => {
|
|
||||||
if let Some(n) = reg(&program[1..=3]) {
|
|
||||||
bail!(InvalidRegister, start, program, n + 1);
|
|
||||||
}
|
|
||||||
rest
|
|
||||||
}
|
|
||||||
// RR
|
|
||||||
[NOT | CP, _, _, rest @ ..] => {
|
|
||||||
if let Some(n) = reg(&program[1..=2]) {
|
|
||||||
bail!(InvalidRegister, start, program, n + 1);
|
|
||||||
}
|
|
||||||
rest
|
|
||||||
}
|
|
||||||
// RI
|
|
||||||
[LI | JMP, reg, _, _, _, _, _, _, _, _, rest @ ..] => {
|
|
||||||
if *reg > 59 {
|
|
||||||
bail!(InvalidRegister, start, program, 1);
|
|
||||||
}
|
|
||||||
rest
|
|
||||||
}
|
|
||||||
// RRI
|
|
||||||
[ADDI..=CMPUI | ADDFI..=MULFI | LB..=SO | JEQ..=JGTU, _, _, _, _, _, _, _, _, _, _, rest @ ..] =>
|
|
||||||
{
|
{
|
||||||
if let Some(n) = reg(&program[1..=2]) {
|
return Err(Error {
|
||||||
bail!(InvalidRegister, start, program, n + 1);
|
kind: ErrorKind::RegisterArrayOverflow,
|
||||||
}
|
index: (program.as_ptr() as usize) - (start.as_ptr() as usize),
|
||||||
rest
|
})
|
||||||
|
}
|
||||||
|
[BRC, src, dst, count, ..]
|
||||||
|
if src.checked_add(*count).is_none() || dst.checked_add(*count).is_none() =>
|
||||||
|
{
|
||||||
|
return Err(Error {
|
||||||
|
kind: ErrorKind::RegisterArrayOverflow,
|
||||||
|
index: (program.as_ptr() as usize) - (start.as_ptr() as usize),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
[NOP | ECALL, rest @ ..]
|
||||||
|
| [DIR | DIRF, _, _, _, _, rest @ ..]
|
||||||
|
| [ADD..=CMPU | BRC | ADDF..=MULF, _, _, _, rest @ ..]
|
||||||
|
| [NEG..=NOT | CP..=SWA, _, _, rest @ ..]
|
||||||
|
| [LI | JMP, _, _, _, _, _, _, _, _, _, rest @ ..]
|
||||||
|
| [ADDI..=CMPUI | BMC | JEQ..=JGTU | ADDFI..=MULFI, _, _, _, _, _, _, _, _, _, _, rest @ ..]
|
||||||
|
| [LD..=ST, _, _, _, _, _, _, _, _, _, _, _, _, rest @ ..] => rest,
|
||||||
|
_ => {
|
||||||
|
return Err(Error {
|
||||||
|
kind: ErrorKind::InvalidInstruction,
|
||||||
|
index: (program.as_ptr() as usize) - (start.as_ptr() as usize),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
_ => bail!(InvalidInstruction, start, program),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,7 @@
|
||||||
// HACK: This is temporary implementation so we can have memory instructions working
|
|
||||||
|
|
||||||
mod paging;
|
mod paging;
|
||||||
|
|
||||||
use self::paging::{PageTable, Permission, PtEntry};
|
use self::paging::{PageTable, Permission, PtEntry};
|
||||||
use alloc::boxed::Box;
|
use alloc::boxed::Box;
|
||||||
use core::mem::MaybeUninit;
|
|
||||||
use {crate::vm::value::Value, ma_size::MemAccessSize};
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Memory {
|
pub struct Memory {
|
||||||
|
@ -56,67 +52,62 @@ impl Memory {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Load value from an address
|
/// Load value from an address
|
||||||
pub fn load<S: MemAccessSize>(&self, addr: u64) -> Option<Value> {
|
pub unsafe fn load(&self, addr: u64, target: *mut u8, count: usize) -> Result<(), ()> {
|
||||||
let lookup = self.page_lookup(addr)?;
|
self.memory_access(
|
||||||
match lookup.perm {
|
addr,
|
||||||
Permission::Empty | Permission::Node => return None,
|
target,
|
||||||
Permission::Readonly | Permission::Write | Permission::Exec => (),
|
count,
|
||||||
}
|
|perm| {
|
||||||
|
matches!(
|
||||||
let mut value = MaybeUninit::<Value>::zeroed();
|
perm,
|
||||||
let overflow = (lookup.offset + S::BYTES).saturating_sub(lookup.size - 1);
|
Permission::Readonly | Permission::Write | Permission::Exec
|
||||||
let normal = S::BYTES - overflow;
|
)
|
||||||
|
},
|
||||||
unsafe {
|
|src, dst, count| core::ptr::copy_nonoverlapping(src, dst, count),
|
||||||
core::ptr::copy_nonoverlapping::<u8>(lookup.ptr, value.as_mut_ptr().cast(), normal);
|
)
|
||||||
if overflow != 0 {
|
|
||||||
let lookup = self.page_lookup(lookup.ptr as u64 + lookup.size as u64)?;
|
|
||||||
match lookup.perm {
|
|
||||||
Permission::Empty | Permission::Node => return None,
|
|
||||||
Permission::Readonly | Permission::Write | Permission::Exec => (),
|
|
||||||
}
|
|
||||||
|
|
||||||
core::ptr::copy_nonoverlapping::<u8>(
|
|
||||||
lookup.ptr,
|
|
||||||
value.as_mut_ptr().cast::<u8>().add(normal),
|
|
||||||
overflow,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Some(value.assume_init())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Store value to an address
|
/// Store value to an address
|
||||||
pub fn store<S: MemAccessSize>(&mut self, addr: u64, value: Value) -> Result<(), ()> {
|
pub unsafe fn store(&mut self, addr: u64, source: *const u8, count: usize) -> Result<(), ()> {
|
||||||
let lookup = self.page_lookup(addr).ok_or(())?;
|
self.memory_access(
|
||||||
if lookup.perm != Permission::Write {
|
addr,
|
||||||
return Err(());
|
source.cast_mut(),
|
||||||
}
|
count,
|
||||||
|
|perm| perm == Permission::Write,
|
||||||
|
|dst, src, count| core::ptr::copy_nonoverlapping(src, dst, count),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
let overflow = (lookup.offset + S::BYTES).saturating_sub(lookup.size - 1);
|
/// Copy a block of memory
|
||||||
let normal = S::BYTES - overflow;
|
pub unsafe fn block_copy(&mut self, src: u64, dst: u64, count: u64) -> Result<(), ()> {
|
||||||
unsafe {
|
let count = usize::try_from(count).expect("?conradluget a better CPU");
|
||||||
core::ptr::copy_nonoverlapping::<u8>(
|
|
||||||
(&value as *const Value).cast(),
|
|
||||||
lookup.ptr,
|
|
||||||
normal,
|
|
||||||
);
|
|
||||||
|
|
||||||
if overflow != 0 {
|
let mut srcs = PageSplitter::new(src, count, self.root_pt);
|
||||||
let lookup = self
|
let mut dsts = PageSplitter::new(dst, count, self.root_pt);
|
||||||
.page_lookup(lookup.ptr as u64 + lookup.size as u64)
|
let mut c_src = srcs.next().ok_or(())?;
|
||||||
.ok_or(())?;
|
let mut c_dst = dsts.next().ok_or(())?;
|
||||||
|
|
||||||
core::ptr::copy_nonoverlapping::<u8>(
|
loop {
|
||||||
(&value as *const Value).cast::<u8>().add(normal),
|
let min_size = c_src.size.min(c_dst.size);
|
||||||
lookup.ptr,
|
unsafe {
|
||||||
overflow,
|
core::ptr::copy(c_src.ptr, c_dst.ptr, min_size);
|
||||||
);
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
Ok(())
|
match (
|
||||||
|
match c_src.size.saturating_sub(min_size) {
|
||||||
|
0 => srcs.next(),
|
||||||
|
size => Some(PageSplitResult { size, ..c_src }),
|
||||||
|
},
|
||||||
|
match c_dst.size.saturating_sub(min_size) {
|
||||||
|
0 => dsts.next(),
|
||||||
|
size => Some(PageSplitResult { size, ..c_dst }),
|
||||||
|
},
|
||||||
|
) {
|
||||||
|
(None, None) => return Ok(()),
|
||||||
|
(Some(src), Some(dst)) => (c_src, c_dst) = (src, dst),
|
||||||
|
_ => return Err(()),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -129,71 +120,96 @@ impl Memory {
|
||||||
unsafe { &mut *self.root_pt }
|
unsafe { &mut *self.root_pt }
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Resolve page and offset from the page
|
fn memory_access(
|
||||||
fn page_lookup(&self, addr: u64) -> Option<PageLookupResult> {
|
&self,
|
||||||
let mut current_pt = self.root_pt;
|
src: u64,
|
||||||
for lvl in (0..5).rev() {
|
mut dst: *mut u8,
|
||||||
unsafe {
|
len: usize,
|
||||||
let entry = (*current_pt).get_unchecked(
|
permission_check: impl Fn(Permission) -> bool,
|
||||||
usize::try_from((addr >> (lvl * 9 + 12)) & ((1 << 9) - 1))
|
action: impl Fn(*mut u8, *mut u8, usize),
|
||||||
.expect("?conradluget a better CPU"),
|
) -> Result<(), ()> {
|
||||||
);
|
for PageSplitResult { ptr, size, perm } in PageSplitter::new(src, len, self.root_pt) {
|
||||||
|
if !permission_check(perm) {
|
||||||
let ptr = entry.ptr();
|
return Err(());
|
||||||
match entry.permission() {
|
|
||||||
Permission::Empty => return None,
|
|
||||||
Permission::Node => current_pt = ptr as _,
|
|
||||||
_ if lvl > 2 => return None,
|
|
||||||
perm => {
|
|
||||||
return Some(PageLookupResult {
|
|
||||||
perm,
|
|
||||||
ptr: ptr as _,
|
|
||||||
size: match lvl {
|
|
||||||
0 => 4096,
|
|
||||||
1 => 1024_usize.pow(2) * 2,
|
|
||||||
2 => 1024_usize.pow(3),
|
|
||||||
_ => unreachable!(),
|
|
||||||
},
|
|
||||||
offset: addr as usize & ((1 << (lvl * 9 + 12)) - 1),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
action(ptr, dst, size);
|
||||||
|
dst = unsafe { dst.add(size) };
|
||||||
}
|
}
|
||||||
None
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct PageLookupResult {
|
struct PageSplitResult {
|
||||||
perm: Permission,
|
|
||||||
ptr: *mut u8,
|
ptr: *mut u8,
|
||||||
size: usize,
|
size: usize,
|
||||||
offset: usize,
|
perm: Permission,
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! size_markers {
|
struct PageSplitter {
|
||||||
($($name:ident = $size:expr),* $(,)?) => {
|
addr: u64,
|
||||||
pub mod ma_size {
|
size: usize,
|
||||||
/// # Safety
|
pagetable: *const PageTable,
|
||||||
/// Implementor has to assure that [`MemAccessSize::BYTES`] won't be larger than
|
}
|
||||||
/// size of [`Value`]
|
|
||||||
pub unsafe trait MemAccessSize {
|
|
||||||
const BYTES: usize;
|
|
||||||
}
|
|
||||||
|
|
||||||
$(
|
impl PageSplitter {
|
||||||
pub struct $name;
|
pub const fn new(addr: u64, size: usize, pagetable: *const PageTable) -> Self {
|
||||||
unsafe impl MemAccessSize for $name {
|
Self {
|
||||||
const BYTES: usize = $size;
|
addr,
|
||||||
}
|
size,
|
||||||
)*
|
pagetable,
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
size_markers! {
|
impl Iterator for PageSplitter {
|
||||||
Byte = 1,
|
type Item = PageSplitResult;
|
||||||
Doublet = 2,
|
|
||||||
Quadlet = 4,
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
Octlet = 8,
|
if self.size == 0 {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let (base, perm, size, offset) = 'a: {
|
||||||
|
let mut current_pt = self.pagetable;
|
||||||
|
for lvl in (0..5).rev() {
|
||||||
|
unsafe {
|
||||||
|
let entry = (*current_pt).get_unchecked(
|
||||||
|
usize::try_from((self.addr >> (lvl * 9 + 12)) & ((1 << 9) - 1))
|
||||||
|
.expect("?conradluget a better CPU"),
|
||||||
|
);
|
||||||
|
|
||||||
|
let ptr = entry.ptr();
|
||||||
|
match entry.permission() {
|
||||||
|
Permission::Empty => return None,
|
||||||
|
Permission::Node => current_pt = ptr as _,
|
||||||
|
perm => {
|
||||||
|
break 'a (
|
||||||
|
ptr as *mut u8,
|
||||||
|
perm,
|
||||||
|
match lvl {
|
||||||
|
0 => 4096,
|
||||||
|
1 => 1024_usize.pow(2) * 2,
|
||||||
|
2 => 1024_usize.pow(3),
|
||||||
|
_ => return None,
|
||||||
|
},
|
||||||
|
self.addr as usize & ((1 << (lvl * 9 + 12)) - 1),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return None;
|
||||||
|
};
|
||||||
|
|
||||||
|
let avail = (size - offset).clamp(0, self.size);
|
||||||
|
self.addr += size as u64;
|
||||||
|
self.size = self.size.saturating_sub(size);
|
||||||
|
Some(PageSplitResult {
|
||||||
|
ptr: unsafe { base.add(offset) },
|
||||||
|
size: avail,
|
||||||
|
perm,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,24 +3,22 @@
|
||||||
//! All unsafe code here should be sound, if input bytecode passes validation.
|
//! All unsafe code here should be sound, if input bytecode passes validation.
|
||||||
|
|
||||||
// # General safety notice:
|
// # General safety notice:
|
||||||
// - Validation has to assure there is 60 registers (r0 - r59)
|
// - Validation has to assure there is 256 registers (r0 - r255)
|
||||||
// - Instructions have to be valid as specified (values and sizes)
|
// - Instructions have to be valid as specified (values and sizes)
|
||||||
// - Mapped pages should be at least 8 KiB
|
// - Mapped pages should be at least 4 KiB
|
||||||
// - Yes, I am aware of the UB when jumping in-mid of instruction where
|
// - Yes, I am aware of the UB when jumping in-mid of instruction where
|
||||||
// the read byte corresponds to an instruction whose lenght exceets the
|
// the read byte corresponds to an instruction whose lenght exceets the
|
||||||
// program size. If you are (rightfully) worried about the UB, for now just
|
// program size. If you are (rightfully) worried about the UB, for now just
|
||||||
// append your program with 11 zeroes.
|
// append your program with 11 zeroes.
|
||||||
|
|
||||||
use hbbytecode::ParamRRRR;
|
|
||||||
|
|
||||||
mod mem;
|
mod mem;
|
||||||
mod value;
|
mod value;
|
||||||
|
|
||||||
use {
|
use {
|
||||||
crate::validate,
|
crate::validate,
|
||||||
core::ops,
|
core::ops,
|
||||||
hbbytecode::{OpParam, ParamRI, ParamRR, ParamRRI, ParamRRR},
|
hbbytecode::{OpParam, ParamBB, ParamBBB, ParamBBBB, ParamBBD, ParamBBDH, ParamBD},
|
||||||
mem::{ma_size, Memory},
|
mem::Memory,
|
||||||
static_assertions::assert_impl_one,
|
static_assertions::assert_impl_one,
|
||||||
value::Value,
|
value::Value,
|
||||||
};
|
};
|
||||||
|
@ -41,7 +39,7 @@ macro_rules! param {
|
||||||
|
|
||||||
macro_rules! binary_op {
|
macro_rules! binary_op {
|
||||||
($self:expr, $ty:ident, $handler:expr) => {{
|
($self:expr, $ty:ident, $handler:expr) => {{
|
||||||
let ParamRRR(tg, a0, a1) = param!($self, ParamRRR);
|
let ParamBBB(tg, a0, a1) = param!($self, ParamBBB);
|
||||||
$self.write_reg(
|
$self.write_reg(
|
||||||
tg,
|
tg,
|
||||||
$handler(
|
$handler(
|
||||||
|
@ -55,7 +53,7 @@ macro_rules! binary_op {
|
||||||
|
|
||||||
macro_rules! binary_op_imm {
|
macro_rules! binary_op_imm {
|
||||||
($self:expr, $ty:ident, $handler:expr) => {{
|
($self:expr, $ty:ident, $handler:expr) => {{
|
||||||
let ParamRRI(tg, a0, imm) = param!($self, ParamRRI);
|
let ParamBBD(tg, a0, imm) = param!($self, ParamBBD);
|
||||||
$self.write_reg(
|
$self.write_reg(
|
||||||
tg,
|
tg,
|
||||||
$handler(Value::$ty(&$self.read_reg(a0)), Value::$ty(&imm.into())).into(),
|
$handler(Value::$ty(&$self.read_reg(a0)), Value::$ty(&imm.into())).into(),
|
||||||
|
@ -63,38 +61,10 @@ macro_rules! binary_op_imm {
|
||||||
}};
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! load {
|
|
||||||
($self:expr, $size:ty) => {{
|
|
||||||
let ParamRRI(tg, a0, offset) = param!($self, ParamRRI);
|
|
||||||
$self.write_reg(
|
|
||||||
tg,
|
|
||||||
match $self
|
|
||||||
.memory
|
|
||||||
.load::<$size>($self.read_reg(a0).int() + offset)
|
|
||||||
{
|
|
||||||
Some(x) => x,
|
|
||||||
None => return HaltReason::LoadAccessEx,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}};
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! store {
|
|
||||||
($self:expr, $size:ty) => {{
|
|
||||||
let ParamRRI(src, a0, offset) = param!($self, ParamRRI);
|
|
||||||
if let Err(()) = $self
|
|
||||||
.memory
|
|
||||||
.store::<$size>($self.read_reg(a0).int() + offset, $self.read_reg(src))
|
|
||||||
{
|
|
||||||
return HaltReason::StoreAccessEx;
|
|
||||||
}
|
|
||||||
}};
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! cond_jump {
|
macro_rules! cond_jump {
|
||||||
($self:expr, $ty:ident, $expected:ident) => {{
|
($self:expr, $ty:ident, $expected:ident) => {{
|
||||||
let ParamRRI(a0, a1, jt) = param!($self, ParamRRI);
|
let ParamBBD(a0, a1, jt) = param!($self, ParamBBD);
|
||||||
if core::cmp::Ord::cmp(&$self.read_reg(a0), &$self.read_reg(a1))
|
if core::cmp::Ord::cmp(&$self.read_reg(a0).as_u64(), &$self.read_reg(a1).as_u64())
|
||||||
== core::cmp::Ordering::$expected
|
== core::cmp::Ordering::$expected
|
||||||
{
|
{
|
||||||
$self.pc = jt as usize;
|
$self.pc = jt as usize;
|
||||||
|
@ -103,7 +73,7 @@ macro_rules! cond_jump {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Vm<'a> {
|
pub struct Vm<'a> {
|
||||||
pub registers: [Value; 60],
|
pub registers: [Value; 256],
|
||||||
pub memory: Memory,
|
pub memory: Memory,
|
||||||
pc: usize,
|
pc: usize,
|
||||||
program: &'a [u8],
|
program: &'a [u8],
|
||||||
|
@ -114,7 +84,7 @@ impl<'a> Vm<'a> {
|
||||||
/// Program code has to be validated
|
/// Program code has to be validated
|
||||||
pub unsafe fn new_unchecked(program: &'a [u8]) -> Self {
|
pub unsafe fn new_unchecked(program: &'a [u8]) -> Self {
|
||||||
Self {
|
Self {
|
||||||
registers: [Value::from(0_u64); 60],
|
registers: [Value::from(0_u64); 256],
|
||||||
memory: Default::default(),
|
memory: Default::default(),
|
||||||
pc: 0,
|
pc: 0,
|
||||||
program,
|
program,
|
||||||
|
@ -135,38 +105,40 @@ impl<'a> Vm<'a> {
|
||||||
unsafe {
|
unsafe {
|
||||||
match opcode {
|
match opcode {
|
||||||
NOP => param!(self, ()),
|
NOP => param!(self, ()),
|
||||||
ADD => binary_op!(self, int, u64::wrapping_add),
|
ADD => binary_op!(self, as_u64, u64::wrapping_add),
|
||||||
SUB => binary_op!(self, int, u64::wrapping_sub),
|
SUB => binary_op!(self, as_u64, u64::wrapping_sub),
|
||||||
MUL => binary_op!(self, int, u64::wrapping_mul),
|
MUL => binary_op!(self, as_u64, u64::wrapping_mul),
|
||||||
AND => binary_op!(self, int, ops::BitAnd::bitand),
|
AND => binary_op!(self, as_u64, ops::BitAnd::bitand),
|
||||||
OR => binary_op!(self, int, ops::BitOr::bitor),
|
OR => binary_op!(self, as_u64, ops::BitOr::bitor),
|
||||||
XOR => binary_op!(self, int, ops::BitXor::bitxor),
|
XOR => binary_op!(self, as_u64, ops::BitXor::bitxor),
|
||||||
SL => binary_op!(self, int, ops::Shl::shl),
|
SL => binary_op!(self, as_u64, ops::Shl::shl),
|
||||||
SR => binary_op!(self, int, ops::Shr::shr),
|
SR => binary_op!(self, as_u64, ops::Shr::shr),
|
||||||
SRS => binary_op!(self, sint, ops::Shr::shr),
|
SRS => binary_op!(self, as_i64, ops::Shr::shr),
|
||||||
CMP => {
|
CMP => {
|
||||||
let ParamRRR(tg, a0, a1) = param!(self, ParamRRR);
|
let ParamBBB(tg, a0, a1) = param!(self, ParamBBB);
|
||||||
self.write_reg(
|
self.write_reg(
|
||||||
tg,
|
tg,
|
||||||
(self.read_reg(a0).sint().cmp(&self.read_reg(a1).sint()) as i64).into(),
|
(self.read_reg(a0).as_i64().cmp(&self.read_reg(a1).as_i64()) as i64)
|
||||||
|
.into(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
CMPU => {
|
CMPU => {
|
||||||
let ParamRRR(tg, a0, a1) = param!(self, ParamRRR);
|
let ParamBBB(tg, a0, a1) = param!(self, ParamBBB);
|
||||||
self.write_reg(
|
self.write_reg(
|
||||||
tg,
|
tg,
|
||||||
(self.read_reg(a0).int().cmp(&self.read_reg(a1).int()) as i64).into(),
|
(self.read_reg(a0).as_u64().cmp(&self.read_reg(a1).as_u64()) as i64)
|
||||||
|
.into(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
NOT => {
|
NOT => {
|
||||||
let param = param!(self, ParamRR);
|
let param = param!(self, ParamBB);
|
||||||
self.write_reg(param.0, (!self.read_reg(param.1).int()).into());
|
self.write_reg(param.0, (!self.read_reg(param.1).as_u64()).into());
|
||||||
}
|
}
|
||||||
NEG => {
|
NEG => {
|
||||||
let param = param!(self, ParamRR);
|
let param = param!(self, ParamBB);
|
||||||
self.write_reg(
|
self.write_reg(
|
||||||
param.0,
|
param.0,
|
||||||
match self.read_reg(param.1).int() {
|
match self.read_reg(param.1).as_u64() {
|
||||||
0 => 1_u64,
|
0 => 1_u64,
|
||||||
_ => 0,
|
_ => 0,
|
||||||
}
|
}
|
||||||
|
@ -174,55 +146,112 @@ impl<'a> Vm<'a> {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
DIR => {
|
DIR => {
|
||||||
let ParamRRRR(dt, rt, a0, a1) = param!(self, ParamRRRR);
|
let ParamBBBB(dt, rt, a0, a1) = param!(self, ParamBBBB);
|
||||||
let a0 = self.read_reg(a0).int();
|
let a0 = self.read_reg(a0).as_u64();
|
||||||
let a1 = self.read_reg(a1).int();
|
let a1 = self.read_reg(a1).as_u64();
|
||||||
self.write_reg(dt, (a0.checked_div(a1).unwrap_or(u64::MAX)).into());
|
self.write_reg(dt, (a0.checked_div(a1).unwrap_or(u64::MAX)).into());
|
||||||
self.write_reg(rt, (a0.checked_rem(a1).unwrap_or(u64::MAX)).into());
|
self.write_reg(rt, (a0.checked_rem(a1).unwrap_or(u64::MAX)).into());
|
||||||
}
|
}
|
||||||
ADDI => binary_op_imm!(self, int, ops::Add::add),
|
ADDI => binary_op_imm!(self, as_u64, ops::Add::add),
|
||||||
MULI => binary_op_imm!(self, int, ops::Mul::mul),
|
MULI => binary_op_imm!(self, as_u64, ops::Mul::mul),
|
||||||
ANDI => binary_op_imm!(self, int, ops::BitAnd::bitand),
|
ANDI => binary_op_imm!(self, as_u64, ops::BitAnd::bitand),
|
||||||
ORI => binary_op_imm!(self, int, ops::BitOr::bitor),
|
ORI => binary_op_imm!(self, as_u64, ops::BitOr::bitor),
|
||||||
XORI => binary_op_imm!(self, int, ops::BitXor::bitxor),
|
XORI => binary_op_imm!(self, as_u64, ops::BitXor::bitxor),
|
||||||
SLI => binary_op_imm!(self, int, ops::Shl::shl),
|
SLI => binary_op_imm!(self, as_u64, ops::Shl::shl),
|
||||||
SRI => binary_op_imm!(self, int, ops::Shr::shr),
|
SRI => binary_op_imm!(self, as_u64, ops::Shr::shr),
|
||||||
SRSI => binary_op_imm!(self, sint, ops::Shr::shr),
|
SRSI => binary_op_imm!(self, as_i64, ops::Shr::shr),
|
||||||
CMPI => {
|
CMPI => {
|
||||||
let ParamRRI(tg, a0, imm) = param!(self, ParamRRI);
|
let ParamBBD(tg, a0, imm) = param!(self, ParamBBD);
|
||||||
self.write_reg(
|
self.write_reg(
|
||||||
tg,
|
tg,
|
||||||
(self.read_reg(a0).sint().cmp(&Value::from(imm).sint()) as i64).into(),
|
(self.read_reg(a0).as_i64().cmp(&Value::from(imm).as_i64()) as i64)
|
||||||
|
.into(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
CMPUI => {
|
CMPUI => {
|
||||||
let ParamRRI(tg, a0, imm) = param!(self, ParamRRI);
|
let ParamBBD(tg, a0, imm) = param!(self, ParamBBD);
|
||||||
self.write_reg(tg, (self.read_reg(a0).int().cmp(&imm) as i64).into());
|
self.write_reg(tg, (self.read_reg(a0).as_u64().cmp(&imm) as i64).into());
|
||||||
}
|
}
|
||||||
CP => {
|
CP => {
|
||||||
let param = param!(self, ParamRR);
|
let param = param!(self, ParamBB);
|
||||||
self.write_reg(param.0, self.read_reg(param.1));
|
self.write_reg(param.0, self.read_reg(param.1));
|
||||||
}
|
}
|
||||||
|
SWA => {
|
||||||
|
let ParamBB(src, dst) = param!(self, ParamBB);
|
||||||
|
if src + dst != 0 {
|
||||||
|
core::ptr::swap(
|
||||||
|
self.registers.get_unchecked_mut(usize::from(src)),
|
||||||
|
self.registers.get_unchecked_mut(usize::from(dst)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
LI => {
|
LI => {
|
||||||
let param = param!(self, ParamRI);
|
let param = param!(self, ParamBD);
|
||||||
self.write_reg(param.0, param.1.into());
|
self.write_reg(param.0, param.1.into());
|
||||||
}
|
}
|
||||||
LB => load!(self, ma_size::Byte),
|
LD => {
|
||||||
LD => load!(self, ma_size::Doublet),
|
let ParamBBDH(dst, base, off, count) = param!(self, ParamBBDH);
|
||||||
LQ => load!(self, ma_size::Quadlet),
|
let n: usize = match dst {
|
||||||
LO => load!(self, ma_size::Octlet),
|
0 => 1,
|
||||||
SB => store!(self, ma_size::Byte),
|
_ => 0,
|
||||||
SD => store!(self, ma_size::Doublet),
|
};
|
||||||
SQ => store!(self, ma_size::Quadlet),
|
|
||||||
SO => store!(self, ma_size::Octlet),
|
if self
|
||||||
|
.memory
|
||||||
|
.load(
|
||||||
|
self.read_reg(base).as_u64() + off + n as u64,
|
||||||
|
self.registers.as_mut_ptr().add(usize::from(dst) + n).cast(),
|
||||||
|
usize::from(count).saturating_sub(n),
|
||||||
|
)
|
||||||
|
.is_err()
|
||||||
|
{
|
||||||
|
return HaltReason::LoadAccessEx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ST => {
|
||||||
|
let ParamBBDH(dst, base, off, count) = param!(self, ParamBBDH);
|
||||||
|
if self
|
||||||
|
.memory
|
||||||
|
.store(
|
||||||
|
self.read_reg(base).as_u64() + off,
|
||||||
|
self.registers.as_ptr().add(usize::from(dst)).cast(),
|
||||||
|
count.into(),
|
||||||
|
)
|
||||||
|
.is_err()
|
||||||
|
{
|
||||||
|
return HaltReason::LoadAccessEx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BMC => {
|
||||||
|
let ParamBBD(src, dst, count) = param!(self, ParamBBD);
|
||||||
|
if self
|
||||||
|
.memory
|
||||||
|
.block_copy(
|
||||||
|
self.read_reg(src).as_u64(),
|
||||||
|
self.read_reg(dst).as_u64(),
|
||||||
|
count,
|
||||||
|
)
|
||||||
|
.is_err()
|
||||||
|
{
|
||||||
|
return HaltReason::LoadAccessEx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BRC => {
|
||||||
|
let ParamBBB(src, dst, count) = param!(self, ParamBBB);
|
||||||
|
core::ptr::copy(
|
||||||
|
self.registers.get_unchecked(usize::from(src)),
|
||||||
|
self.registers.get_unchecked_mut(usize::from(dst)),
|
||||||
|
usize::from(count * 8),
|
||||||
|
);
|
||||||
|
}
|
||||||
JMP => {
|
JMP => {
|
||||||
let ParamRI(reg, offset) = param!(self, ParamRI);
|
let ParamBD(reg, offset) = param!(self, ParamBD);
|
||||||
self.pc = (self.read_reg(reg).int() + offset) as usize;
|
self.pc = (self.read_reg(reg).as_u64() + offset) as usize;
|
||||||
}
|
}
|
||||||
JEQ => cond_jump!(self, int, Equal),
|
JEQ => cond_jump!(self, int, Equal),
|
||||||
JNE => {
|
JNE => {
|
||||||
let ParamRRI(a0, a1, jt) = param!(self, ParamRRI);
|
let ParamBBD(a0, a1, jt) = param!(self, ParamBBD);
|
||||||
if self.read_reg(a0) != self.read_reg(a1) {
|
if self.read_reg(a0).as_u64() != self.read_reg(a1).as_u64() {
|
||||||
self.pc = jt as usize;
|
self.pc = jt as usize;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -234,18 +263,18 @@ impl<'a> Vm<'a> {
|
||||||
param!(self, ());
|
param!(self, ());
|
||||||
return HaltReason::Ecall;
|
return HaltReason::Ecall;
|
||||||
}
|
}
|
||||||
ADDF => binary_op!(self, float, ops::Add::add),
|
ADDF => binary_op!(self, as_f64, ops::Add::add),
|
||||||
MULF => binary_op!(self, float, ops::Mul::mul),
|
MULF => binary_op!(self, as_f64, ops::Mul::mul),
|
||||||
DIRF => {
|
DIRF => {
|
||||||
let ParamRRRR(dt, rt, a0, a1) = param!(self, ParamRRRR);
|
let ParamBBBB(dt, rt, a0, a1) = param!(self, ParamBBBB);
|
||||||
let a0 = self.read_reg(a0).float();
|
let a0 = self.read_reg(a0).as_f64();
|
||||||
let a1 = self.read_reg(a1).float();
|
let a1 = self.read_reg(a1).as_f64();
|
||||||
self.write_reg(dt, (a0 / a1).into());
|
self.write_reg(dt, (a0 / a1).into());
|
||||||
self.write_reg(rt, (a0 % a1).into());
|
self.write_reg(rt, (a0 % a1).into());
|
||||||
}
|
}
|
||||||
ADDFI => binary_op_imm!(self, float, ops::Add::add),
|
ADDFI => binary_op_imm!(self, as_f64, ops::Add::add),
|
||||||
MULFI => binary_op_imm!(self, float, ops::Mul::mul),
|
MULFI => binary_op_imm!(self, as_f64, ops::Mul::mul),
|
||||||
_ => core::hint::unreachable_unchecked(),
|
_ => return HaltReason::InvalidOpcode,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -273,6 +302,7 @@ impl<'a> Vm<'a> {
|
||||||
pub enum HaltReason {
|
pub enum HaltReason {
|
||||||
ProgramEnd,
|
ProgramEnd,
|
||||||
Ecall,
|
Ecall,
|
||||||
|
InvalidOpcode,
|
||||||
LoadAccessEx,
|
LoadAccessEx,
|
||||||
StoreAccessEx,
|
StoreAccessEx,
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,59 +1,37 @@
|
||||||
use core::fmt::Debug;
|
use core::fmt::Debug;
|
||||||
|
|
||||||
/// # Safety
|
|
||||||
/// The macro invoker shall make sure that byte reinterpret-cast
|
|
||||||
/// or zero-init won't cause undefined behaviour.
|
|
||||||
macro_rules! value_def {
|
macro_rules! value_def {
|
||||||
($($fname:ident : $fty:ident, $getter:ident);* $(;)?) => {
|
($($ty:ident),* $(,)?) => {
|
||||||
#[derive(Clone, Copy)]
|
#[derive(Copy, Clone)]
|
||||||
|
#[repr(packed)]
|
||||||
pub union Value {
|
pub union Value {
|
||||||
$($fname: $fty),*
|
$(pub $ty: $ty),*
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Value {$(
|
paste::paste! {
|
||||||
#[inline]
|
impl Value {$(
|
||||||
pub fn $getter(&self) -> $fty {
|
#[inline]
|
||||||
unsafe { self.$fname }
|
pub fn [<as_ $ty>](&self) -> $ty {
|
||||||
}
|
unsafe { self.$ty }
|
||||||
)*}
|
}
|
||||||
|
)*}
|
||||||
|
}
|
||||||
|
|
||||||
$(impl From<$fty> for Value {
|
$(
|
||||||
#[inline]
|
impl From<$ty> for Value {
|
||||||
fn from($fname: $fty) -> Self {
|
#[inline]
|
||||||
Self { $fname }
|
fn from(value: $ty) -> Self {
|
||||||
|
Self { $ty: value }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})*
|
)*
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
value_def! {
|
value_def!(u64, i64, f64);
|
||||||
i: u64, int;
|
|
||||||
s: i64, sint;
|
|
||||||
f: f64, float;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Debug for Value {
|
impl Debug for Value {
|
||||||
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
|
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
|
||||||
self.int().fmt(f)
|
self.as_u64().fmt(f)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl PartialEq for Value {
|
|
||||||
fn eq(&self, other: &Self) -> bool {
|
|
||||||
self.int().eq(&other.int())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Eq for Value {}
|
|
||||||
|
|
||||||
impl PartialOrd for Value {
|
|
||||||
fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
|
|
||||||
self.int().partial_cmp(&other.int())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Ord for Value {
|
|
||||||
fn cmp(&self, other: &Self) -> core::cmp::Ordering {
|
|
||||||
self.int().cmp(&other.int())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
120
spec.md
120
spec.md
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
# Bytecode format
|
# Bytecode format
|
||||||
- All numbers are encoded little-endian
|
- All numbers are encoded little-endian
|
||||||
- There is 60 registers (0 – 59), they are represented by a byte
|
- There is 256 registers, they are represented by a byte
|
||||||
- Immediate values are 64 bit
|
- Immediate values are 64 bit
|
||||||
|
|
||||||
### Instruction encoding
|
### Instruction encoding
|
||||||
|
@ -10,17 +10,20 @@
|
||||||
- [opcode, …parameters…]
|
- [opcode, …parameters…]
|
||||||
|
|
||||||
### Instruction parameter types
|
### Instruction parameter types
|
||||||
- R = Register
|
- B = Byte
|
||||||
- I = Immediate
|
- D = Doubleword (64 bits)
|
||||||
|
- H = Halfword (16 bits)
|
||||||
|
|
||||||
| Name | Size |
|
| Name | Size |
|
||||||
|:----:|:--------|
|
|:----:|:--------|
|
||||||
| RRRR | 32 bits |
|
| BBBB | 32 bits |
|
||||||
| RRR | 24 bits |
|
| BBB | 24 bits |
|
||||||
| RRI | 80 bits |
|
| BBDH | 96 bits |
|
||||||
| RR | 16 bits |
|
| BBDB | 88 bits |
|
||||||
| RI | 72 bits |
|
| BBD | 80 bits |
|
||||||
| I | 64 bits |
|
| BB | 16 bits |
|
||||||
|
| BD | 72 bits |
|
||||||
|
| D | 64 bits |
|
||||||
| N | 0 bits |
|
| N | 0 bits |
|
||||||
|
|
||||||
# Instructions
|
# Instructions
|
||||||
|
@ -37,7 +40,7 @@
|
||||||
| 0 | NOP | Do nothing |
|
| 0 | NOP | Do nothing |
|
||||||
|
|
||||||
## Integer binary ops.
|
## Integer binary ops.
|
||||||
- RRR type
|
- BBB type
|
||||||
- `#0 ← #1 <op> #2`
|
- `#0 ← #1 <op> #2`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|
@ -66,7 +69,7 @@
|
||||||
| > | 1 |
|
| > | 1 |
|
||||||
|
|
||||||
### Division-remainder
|
### Division-remainder
|
||||||
- Type RRRR
|
- Type BBBB
|
||||||
- In case of `#3` is zero, the resulting value is all-ones
|
- In case of `#3` is zero, the resulting value is all-ones
|
||||||
- `#0 ← #2 ÷ #3`
|
- `#0 ← #2 ÷ #3`
|
||||||
- `#1 ← #2 % #3`
|
- `#1 ← #2 % #3`
|
||||||
|
@ -76,7 +79,7 @@
|
||||||
| 12 | DIR | Divide and remainder combinated |
|
| 12 | DIR | Divide and remainder combinated |
|
||||||
|
|
||||||
### Negations
|
### Negations
|
||||||
- Type RR
|
- Type BB
|
||||||
- `#0 ← #1 <op> #2`
|
- `#0 ← #1 <op> #2`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|
@ -85,7 +88,7 @@
|
||||||
| 14 | NOT | Logical negation |
|
| 14 | NOT | Logical negation |
|
||||||
|
|
||||||
## Integer immediate binary ops.
|
## Integer immediate binary ops.
|
||||||
- Type RRI
|
- Type BBD
|
||||||
- `#0 ← #1 <op> imm #2`
|
- `#0 ← #1 <op> imm #2`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|
@ -110,100 +113,113 @@
|
||||||
## Register value set / copy
|
## Register value set / copy
|
||||||
|
|
||||||
### Copy
|
### Copy
|
||||||
- Type RR
|
- Type BB
|
||||||
- `#0 ← #1`
|
- `#0 ← #1`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:----:|:------:|
|
|:------:|:----:|:------:|
|
||||||
| 28 | CP | Copy |
|
| 28 | CP | Copy |
|
||||||
|
|
||||||
|
### Swap
|
||||||
|
- Type BB
|
||||||
|
- Swap #0 and #1
|
||||||
|
|
||||||
|
| Opcode | Name | Action |
|
||||||
|
|:------:|:----:|:------:|
|
||||||
|
| 29 | SWA | Swap |
|
||||||
|
|
||||||
### Load immediate
|
### Load immediate
|
||||||
- Type RI
|
- Type BD
|
||||||
- `#0 ← #1`
|
- `#0 ← #1`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:----:|:--------------:|
|
|:------:|:----:|:--------------:|
|
||||||
| 29 | LI | Load immediate |
|
| 30 | LI | Load immediate |
|
||||||
|
|
||||||
## Memory operations
|
## Memory operations
|
||||||
- Type RRI
|
- Type BBDH
|
||||||
|
- If loaded/store value exceeds one register size, continue accessing following registers
|
||||||
|
|
||||||
### Load
|
### Load / Store
|
||||||
- `#0 ← [#1 + imm #2]`
|
| Opcode | Name | Action |
|
||||||
|
|:------:|:----:|:---------------------------------------:|
|
||||||
|
| 31 | LD | `#0 ← [#1 + imm #3], copy imm #4 bytes` |
|
||||||
|
| 32 | ST | `[#1 + imm #3] ← #0, copy imm #4 bytes` |
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
## Block copy
|
||||||
|:------:|:----:|:----------------------:|
|
- Block copy source and target can overlap
|
||||||
| 30 | LB | Load byte (8 bits) |
|
|
||||||
| 31 | LD | Load doublet (16 bits) |
|
|
||||||
| 32 | LQ | Load quadlet (32 bits) |
|
|
||||||
| 33 | LO | Load octlet (64 bits) |
|
|
||||||
|
|
||||||
### Store
|
### Memory copy
|
||||||
- `[#1 + imm #2] ← #0`
|
- Type BBD
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:----:|:-----------------------:|
|
|:------:|:----:|:--------------------------------:|
|
||||||
| 34 | SB | Store byte (8 bits) |
|
| 33 | BMC | `[#0] ← [#1], copy imm #2 bytes` |
|
||||||
| 35 | SD | Store doublet (16 bits) |
|
|
||||||
| 36 | SQ | Store quadlet (32 bits) |
|
### Register copy
|
||||||
| 37 | SO | Store octlet (64 bits) |
|
- Type BBB
|
||||||
|
- Copy a block a register to another location (again, overflowing to following registers)
|
||||||
|
|
||||||
|
| Opcode | Name | Action |
|
||||||
|
|:------:|:----:|:--------------------------------:|
|
||||||
|
| 34 | BRC | `#0 ← #1, copy imm #2 registers` |
|
||||||
|
|
||||||
## Control flow
|
## Control flow
|
||||||
|
|
||||||
### Unconditional jump
|
### Unconditional jump
|
||||||
- Type RI
|
- Type BD
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:----:|:---------------------:|
|
|:------:|:----:|:---------------------:|
|
||||||
| 38 | JMP | Jump at `#0 + imm #1` |
|
| 35 | JMP | Jump at `#0 + imm #1` |
|
||||||
|
|
||||||
### Conditional jumps
|
### Conditional jumps
|
||||||
- Type RRI
|
- Type BBD
|
||||||
- Jump at `imm #2` if `#0 <op> #1`
|
- Jump at `imm #2` if `#0 <op> #1`
|
||||||
|
|
||||||
| Opcode | Name | Comparsion |
|
| Opcode | Name | Comparsion |
|
||||||
|:------:|:----:|:------------:|
|
|:------:|:----:|:------------:|
|
||||||
| 39 | JEQ | = |
|
| 36 | JEQ | = |
|
||||||
| 40 | JNE | ≠ |
|
| 37 | JNE | ≠ |
|
||||||
| 41 | JLT | < (signed) |
|
| 38 | JLT | < (signed) |
|
||||||
| 42 | JGT | > (signed) |
|
| 39 | JGT | > (signed) |
|
||||||
| 43 | JLTU | < (unsigned) |
|
| 40 | JLTU | < (unsigned) |
|
||||||
| 44 | JGTU | > (unsigned) |
|
| 41 | JGTU | > (unsigned) |
|
||||||
|
|
||||||
### Environment call
|
### Environment call
|
||||||
- Type N
|
- Type N
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:-----:|:-------------------------------------:|
|
|:------:|:-----:|:-------------------------------------:|
|
||||||
| 45 | ECALL | Cause an trap to the host environment |
|
| 42 | ECALL | Cause an trap to the host environment |
|
||||||
|
|
||||||
## Floating point operations
|
## Floating point operations
|
||||||
- Type RRR
|
- Type BBB
|
||||||
- `#0 ← #1 <op> #2`
|
- `#0 ← #1 <op> #2`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:----:|:--------------:|
|
|:------:|:----:|:--------------:|
|
||||||
| 46 | ADDF | Addition |
|
| 43 | ADDF | Addition |
|
||||||
| 47 | MULF | Multiplication |
|
| 44 | MULF | Multiplication |
|
||||||
|
|
||||||
### Division-remainder
|
### Division-remainder
|
||||||
- Type RRRR
|
- Type BBBB
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:----:|:--------------------------------------:|
|
|:------:|:----:|:--------------------------------------:|
|
||||||
| 48 | DIRF | Same flow applies as for integer `DIR` |
|
| 45 | DIRF | Same flow applies as for integer `DIR` |
|
||||||
|
|
||||||
## Floating point immediate operations
|
## Floating point immediate operations
|
||||||
- Type RRI
|
- Type BBD
|
||||||
- `#0 ← #1 <op> imm #2`
|
- `#0 ← #1 <op> imm #2`
|
||||||
|
|
||||||
| Opcode | Name | Action |
|
| Opcode | Name | Action |
|
||||||
|:------:|:-----:|:--------------:|
|
|:------:|:-----:|:--------------:|
|
||||||
| 49 | ADDFI | Addition |
|
| 46 | ADDFI | Addition |
|
||||||
| 50 | MULFI | Multiplication |
|
| 47 | MULFI | Multiplication |
|
||||||
|
|
||||||
# Registers
|
# Registers
|
||||||
- There is 59 registers + one zero register (with index 0)
|
- There is 255 registers + one zero register (with index 0)
|
||||||
- Reading from zero register yields zero
|
- Reading from zero register yields zero
|
||||||
- Writing to zero register is a no-op
|
- Writing to zero register is a no-op
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue