#![cfg_attr(docsrs, feature(doc_auto_cfg))]
#![cfg_attr(not(feature = "std"), no_std)]
#![cfg_attr(feature = "std", doc = include_str!("../README.md"))]
#![deny(
clippy::arithmetic_side_effects,
clippy::cast_sign_loss,
clippy::cast_possible_truncation,
clippy::cast_possible_wrap,
clippy::string_slice
)]
#![deny(missing_docs)]
#![deny(unsafe_code)]
#![deny(unused_crate_dependencies)]
#[cfg(feature = "alloc")]
extern crate alloc;
mod args;
mod panic_instruction;
#[macro_use]
pub mod macros;
pub mod op;
mod pack;
mod panic_reason;
mod unpack;
#[cfg(test)]
mod encoding_tests;
#[doc(no_inline)]
pub use args::{
wideint,
GMArgs,
GTFArgs,
};
pub type RegisterId = usize;
pub type Word = u64;
pub use panic_instruction::PanicInstruction;
pub use panic_reason::PanicReason;
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[cfg_attr(feature = "typescript", wasm_bindgen::prelude::wasm_bindgen)]
pub struct RegId(u8);
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[cfg_attr(feature = "typescript", wasm_bindgen::prelude::wasm_bindgen)]
pub struct Imm06(u8);
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[cfg_attr(feature = "typescript", wasm_bindgen::prelude::wasm_bindgen)]
pub struct Imm12(u16);
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[cfg_attr(feature = "typescript", wasm_bindgen::prelude::wasm_bindgen)]
pub struct Imm18(u32);
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[cfg_attr(feature = "typescript", wasm_bindgen::prelude::wasm_bindgen)]
pub struct Imm24(u32);
pub type RawInstruction = u32;
#[derive(Debug, Eq, PartialEq)]
pub struct InvalidOpcode;
bitflags::bitflags! {
pub struct Flags: Word {
const UNSAFEMATH = 0x01;
const WRAPPING = 0x02;
}
}
pub trait CheckRegId {
fn check(self) -> RegId;
}
impl CheckRegId for RegId {
fn check(self) -> RegId {
self
}
}
impl CheckRegId for u8 {
fn check(self) -> RegId {
RegId::new_checked(self).expect("CheckRegId was given invalid RegId")
}
}
impl_instructions! {
"Adds two registers."
0x10 ADD add [dst: RegId lhs: RegId rhs: RegId]
"Bitwise ANDs two registers."
0x11 AND and [dst: RegId lhs: RegId rhs: RegId]
"Divides two registers."
0x12 DIV div [dst: RegId lhs: RegId rhs: RegId]
"Compares two registers for equality."
0x13 EQ eq [dst: RegId lhs: RegId rhs: RegId]
"Raises one register to the power of another."
0x14 EXP exp [dst: RegId lhs: RegId rhs: RegId]
"Compares two registers for greater-than."
0x15 GT gt [dst: RegId lhs: RegId rhs: RegId]
"Compares two registers for less-than."
0x16 LT lt [dst: RegId lhs: RegId rhs: RegId]
"The integer logarithm of a register."
0x17 MLOG mlog [dst: RegId lhs: RegId rhs: RegId]
"The integer root of a register."
0x18 MROO mroo [dst: RegId lhs: RegId rhs: RegId]
"Modulo remainder of two registers."
0x19 MOD mod_ [dst: RegId lhs: RegId rhs: RegId]
"Copy from one register to another."
0x1A MOVE move_ [dst: RegId src: RegId]
"Multiplies two registers."
0x1B MUL mul [dst: RegId lhs: RegId rhs: RegId]
"Bitwise NOT a register."
0x1C NOT not [dst: RegId arg: RegId]
"Bitwise ORs two registers."
0x1D OR or [dst: RegId lhs: RegId rhs: RegId]
"Left shifts a register by a register."
0x1E SLL sll [dst: RegId lhs: RegId rhs: RegId]
"Right shifts a register by a register."
0x1F SRL srl [dst: RegId lhs: RegId rhs: RegId]
"Subtracts two registers."
0x20 SUB sub [dst: RegId lhs: RegId rhs: RegId]
"Bitwise XORs two registers."
0x21 XOR xor [dst: RegId lhs: RegId rhs: RegId]
"Fused multiply-divide with arbitrary precision intermediate step."
0x22 MLDV mldv [dst: RegId mul_lhs: RegId mul_rhs: RegId divisor: RegId]
"Return from context."
0x24 RET ret [value: RegId]
"Return from context with data."
0x25 RETD retd [addr: RegId len: RegId]
"Allocate a number of bytes from the heap."
0x26 ALOC aloc [bytes: RegId]
"Clear a variable number of bytes in memory."
0x27 MCL mcl [dst_addr: RegId len: RegId]
"Copy a variable number of bytes in memory."
0x28 MCP mcp [dst_addr: RegId src_addr: RegId len: RegId]
"Compare bytes in memory."
0x29 MEQ meq [result: RegId lhs_addr: RegId rhs_addr: RegId len: RegId]
"Get block header hash for height."
0x2A BHSH bhsh [dst: RegId heigth: RegId]
"Get current block height."
0x2B BHEI bhei [dst: RegId]
"Burns `amount` coins of the asset ID created from `sub_id` for the current contract."
0x2C BURN burn [amount: RegId sub_id_addr: RegId]
"Call a contract."
0x2D CALL call [target_struct: RegId fwd_coins: RegId asset_id_addr: RegId fwd_gas: RegId]
"Copy contract code for a contract."
0x2E CCP ccp [dst_addr: RegId contract_id_addr: RegId offset: RegId len: RegId]
"Get code root of a contract."
0x2F CROO croo [dst_addr: RegId contract_id_addr: RegId]
"Get code size of a contract."
0x30 CSIZ csiz [dst: RegId contract_id_addr: RegId]
"Get current block proposer's address."
0x31 CB cb [dst: RegId]
"Load a contract's code as executable."
0x32 LDC ldc [contract_id_addr: RegId offset: RegId len: RegId]
"Log an event."
0x33 LOG log [a: RegId b: RegId c: RegId d: RegId]
"Log data."
0x34 LOGD logd [a: RegId b: RegId addr: RegId len: RegId]
"Mints `amount` coins of the asset ID created from `sub_id` for the current contract."
0x35 MINT mint [amount: RegId sub_id_addr: RegId]
"Halt execution, reverting state changes and returning a value."
0x36 RVRT rvrt [value: RegId]
"Clear a series of slots from contract storage."
0x37 SCWQ scwq [key_addr: RegId status: RegId lenq: RegId]
"Load a word from contract storage."
0x38 SRW srw [dst: RegId status: RegId key_addr: RegId]
"Load a series of 32 byte slots from contract storage."
0x39 SRWQ srwq [dst_addr: RegId status: RegId key_addr:RegId lenq: RegId]
"Store a word in contract storage."
0x3A SWW sww [key_addr: RegId status: RegId value: RegId]
"Store a series of 32 byte slots in contract storage."
0x3B SWWQ swwq [key_addr: RegId status: RegId src_addr: RegId lenq: RegId]
"Transfer coins to a contract unconditionally."
0x3C TR tr [contract_id_addr: RegId amount: RegId asset_id_addr: RegId]
"Transfer coins to a variable output."
0x3D TRO tro [contract_id_addr: RegId output_index: RegId amount: RegId asset_id_addr: RegId]
"The 64-byte public key (x, y) recovered from 64-byte signature on 32-byte message."
0x3E ECK1 eck1 [dst_addr: RegId sig_addr: RegId msg_hash_addr: RegId]
"The 64-byte Secp256r1 public key (x, y) recovered from 64-byte signature on 32-byte message."
0x3F ECR1 ecr1 [dst_addr: RegId sig_addr: RegId msg_hash_addr: RegId]
"Verify ED25519 public key and signature match a 32-byte message."
0x40 ED19 ed19 [pub_key_addr: RegId sig_addr: RegId msg_hash_addr: RegId]
"The keccak-256 hash of a slice."
0x41 K256 k256 [dst_addr: RegId src_addr: RegId len: RegId]
"The SHA-2-256 hash of a slice."
0x42 S256 s256 [dst_addr: RegId src_addr: RegId len: RegId]
"Get timestamp of block at given height."
0x43 TIME time [dst: RegId heigth: RegId]
"Performs no operation."
0x47 NOOP noop []
"Set flag register to a register."
0x48 FLAG flag [value: RegId]
"Get the balance of contract of an asset ID."
0x49 BAL bal [dst: RegId asset_id_addr: RegId contract_id_addr: RegId]
"Dynamic jump."
0x4A JMP jmp [abs_target: RegId]
"Conditional dynamic jump."
0x4B JNE jne [abs_target: RegId lhs: RegId rhs: RegId]
"Send a message to recipient address with call abi, coins, and output."
0x4C SMO smo [recipient_addr: RegId data_addr: RegId data_len: RegId coins: RegId]
"Adds a register and an immediate value."
0x50 ADDI addi [dst: RegId lhs: RegId rhs: Imm12]
"Bitwise ANDs a register and an immediate value."
0x51 ANDI andi [dst: RegId lhs: RegId rhs: Imm12]
"Divides a register and an immediate value."
0x52 DIVI divi [dst: RegId lhs: RegId rhs: Imm12]
"Raises one register to the power of an immediate value."
0x53 EXPI expi [dst: RegId lhs: RegId rhs: Imm12]
"Modulo remainder of a register and an immediate value."
0x54 MODI modi [dst: RegId lhs: RegId rhs: Imm12]
"Multiplies a register and an immediate value."
0x55 MULI muli [dst: RegId lhs: RegId rhs: Imm12]
"Bitwise ORs a register and an immediate value."
0x56 ORI ori [dst: RegId lhs: RegId rhs: Imm12]
"Left shifts a register by an immediate value."
0x57 SLLI slli [dst: RegId lhs: RegId rhs: Imm12]
"Right shifts a register by an immediate value."
0x58 SRLI srli [dst: RegId lhs: RegId rhs: Imm12]
"Subtracts a register and an immediate value."
0x59 SUBI subi [dst: RegId lhs: RegId rhs: Imm12]
"Bitwise XORs a register and an immediate value."
0x5A XORI xori [dst: RegId lhs: RegId rhs: Imm12]
"Conditional jump."
0x5B JNEI jnei [cond_lhs: RegId cond_rhs: RegId abs_target: Imm12]
"A byte is loaded from the specified address offset by an immediate value."
0x5C LB lb [dst: RegId addr: RegId offset: Imm12]
"A word is loaded from the specified address offset by an immediate value."
0x5D LW lw [dst: RegId addr: RegId offset: Imm12]
"Write the least significant byte of a register to memory."
0x5E SB sb [addr: RegId value: RegId offset: Imm12]
"Write a register to memory."
0x5F SW sw [addr: RegId value: RegId offset: Imm12]
"Copy an immediate number of bytes in memory."
0x60 MCPI mcpi [dst_addr: RegId src_addr: RegId len: Imm12]
"Get transaction fields."
0x61 GTF gtf [dst: RegId arg: RegId selector: Imm12]
"Clear an immediate number of bytes in memory."
0x70 MCLI mcli [addr: RegId count: Imm18]
"Get metadata from memory."
0x71 GM gm [dst: RegId selector: Imm18]
"Copy immediate value into a register"
0x72 MOVI movi [dst: RegId val: Imm18]
"Conditional jump against zero."
0x73 JNZI jnzi [cond_nz: RegId abs_target: Imm18]
"Unconditional dynamic relative jump forwards, with a constant offset."
0x74 JMPF jmpf [dynamic: RegId fixed: Imm18]
"Unconditional dynamic relative jump backwards, with a constant offset."
0x75 JMPB jmpb [dynamic: RegId fixed: Imm18]
"Dynamic relative jump forwards, conditional against zero, with a constant offset."
0x76 JNZF jnzf [cond_nz: RegId dynamic: RegId fixed: Imm12]
"Dynamic relative jump backwards, conditional against zero, with a constant offset."
0x77 JNZB jnzb [cond_nz: RegId dynamic: RegId fixed: Imm12]
"Dynamic relative jump forwards, conditional on comparsion, with a constant offset."
0x78 JNEF jnef [cond_lhs: RegId cond_rhs: RegId dynamic: RegId fixed: Imm06]
"Dynamic relative jump backwards, conditional on comparsion, with a constant offset."
0x79 JNEB jneb [cond_lhs: RegId cond_rhs: RegId dynamic: RegId fixed: Imm06]
"Jump."
0x90 JI ji [abs_target: Imm24]
"Extend the current call frame's stack by an immediate value."
0x91 CFEI cfei [amount: Imm24]
"Shrink the current call frame's stack by an immediate value."
0x92 CFSI cfsi [amount: Imm24]
"Extend the current call frame's stack"
0x93 CFE cfe [amount: RegId]
"Shrink the current call frame's stack"
0x94 CFS cfs [amount: RegId]
"Push a bitmask-selected set of registers in range 16..40 to the stack."
0x95 PSHL pshl [bitmask: Imm24]
"Push a bitmask-selected set of registers in range 40..64 to the stack."
0x96 PSHH pshh [bitmask: Imm24]
"Pop a bitmask-selected set of registers in range 16..40 to the stack."
0x97 POPL popl [bitmask: Imm24]
"Pop a bitmask-selected set of registers in range 40..64 to the stack."
0x98 POPH poph [bitmask: Imm24]
"Compare 128bit integers"
0xa0 WDCM wdcm [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Compare 256bit integers"
0xa1 WQCM wqcm [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Simple 128bit operations"
0xa2 WDOP wdop [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Simple 256bit operations"
0xa3 WQOP wqop [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Multiply 128bit"
0xa4 WDML wdml [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Multiply 256bit"
0xa5 WQML wqml [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Divide 128bit"
0xa6 WDDV wddv [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Divide 256bit"
0xa7 WQDV wqdv [dst: RegId lhs: RegId rhs: RegId flags: Imm06]
"Fused multiply-divide 128bit"
0xa8 WDMD wdmd [dst: RegId mul_lhs: RegId mul_rhs: RegId divisor: RegId]
"Fused multiply-divide 256bit"
0xa9 WQMD wqmd [dst: RegId mul_lhs: RegId mul_rhs: RegId divisor: RegId]
"AddMod 128bit"
0xaa WDAM wdam [dst: RegId add_lhs: RegId add_rhs: RegId modulo: RegId]
"AddMod 256bit"
0xab WQAM wqam [dst: RegId add_lhs: RegId add_rhs: RegId modulo: RegId]
"MulMod 128bit"
0xac WDMM wdmm [dst: RegId mul_lhs: RegId mul_rhs: RegId modulo: RegId]
"MulMod 256bit"
0xad WQMM wqmm [dst: RegId mul_lhs: RegId mul_rhs: RegId modulo: RegId]
"Call external function"
0xb0 ECAL ecal [a: RegId b: RegId c: RegId d: RegId]
}
impl Instruction {
pub const SIZE: usize = core::mem::size_of::<Instruction>();
pub fn to_bytes(self) -> [u8; 4] {
self.into()
}
}
#[cfg(feature = "typescript")]
mod typescript {
#[derive(Clone, Eq, Hash, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
#[wasm_bindgen::prelude::wasm_bindgen]
pub struct Instruction(Box<crate::Instruction>);
impl Instruction {
pub fn new(instruction: crate::Instruction) -> Self {
Self(Box::new(instruction))
}
}
#[wasm_bindgen::prelude::wasm_bindgen]
impl Instruction {
pub fn to_bytes(&self) -> Vec<u8> {
use core::ops::Deref;
self.deref().to_bytes().to_vec()
}
pub fn size() -> usize {
crate::Instruction::SIZE
}
}
impl core::ops::Deref for Instruction {
type Target = crate::Instruction;
fn deref(&self) -> &crate::Instruction {
self.0.as_ref()
}
}
impl core::ops::DerefMut for Instruction {
fn deref_mut(&mut self) -> &mut crate::Instruction {
self.0.as_mut()
}
}
impl core::borrow::Borrow<crate::Instruction> for Instruction {
fn borrow(&self) -> &crate::Instruction {
self.0.as_ref()
}
}
impl core::borrow::BorrowMut<crate::Instruction> for Instruction {
fn borrow_mut(&mut self) -> &mut crate::Instruction {
self.0.as_mut()
}
}
}
impl RegId {
pub const BAL: Self = Self(0x0B);
pub const CGAS: Self = Self(0x0A);
pub const ERR: Self = Self(0x08);
pub const FLAG: Self = Self(0x0F);
pub const FP: Self = Self(0x06);
pub const GGAS: Self = Self(0x09);
pub const HP: Self = Self(0x07);
pub const IS: Self = Self(0x0C);
pub const OF: Self = Self(0x02);
pub const ONE: Self = Self(0x01);
pub const PC: Self = Self(0x03);
pub const RET: Self = Self(0x0D);
pub const RETL: Self = Self(0x0E);
pub const SP: Self = Self(0x05);
pub const SSP: Self = Self(0x04);
pub const WRITABLE: Self = Self(0x10);
pub const ZERO: Self = Self(0x00);
pub const fn new(u: u8) -> Self {
Self(u & 0b_0011_1111)
}
pub const fn to_u8(self) -> u8 {
self.0
}
}
#[cfg_attr(feature = "typescript", wasm_bindgen::prelude::wasm_bindgen)]
impl RegId {
pub fn new_checked(u: u8) -> Option<RegId> {
let r = Self::new(u);
(r.0 == u).then_some(r)
}
}
#[cfg(feature = "typescript")]
#[wasm_bindgen::prelude::wasm_bindgen]
impl RegId {
pub fn bal() -> Self {
Self::BAL
}
pub fn cgas() -> Self {
Self::CGAS
}
pub fn err() -> Self {
Self::ERR
}
pub fn flag() -> Self {
Self::FLAG
}
pub fn fp() -> Self {
Self::FP
}
pub fn ggas() -> Self {
Self::GGAS
}
pub fn hp() -> Self {
Self::HP
}
pub fn is() -> Self {
Self::IS
}
pub fn of() -> Self {
Self::OF
}
pub fn one() -> Self {
Self::ONE
}
pub fn pc() -> Self {
Self::PC
}
pub fn ret() -> Self {
Self::RET
}
pub fn retl() -> Self {
Self::RETL
}
pub fn sp() -> Self {
Self::SP
}
pub fn spp() -> Self {
Self::SSP
}
pub fn writable() -> Self {
Self::WRITABLE
}
pub fn zero() -> Self {
Self::ZERO
}
#[wasm_bindgen(constructor)]
pub fn new_typescript(u: u8) -> Self {
Self::new(u)
}
#[wasm_bindgen(js_name = to_u8)]
pub fn to_u8_typescript(self) -> u8 {
self.to_u8()
}
}
impl Imm06 {
pub const MAX: Self = Self(0b_0011_1111);
pub const fn new(u: u8) -> Self {
Self(u & Self::MAX.0)
}
pub fn new_checked(u: u8) -> Option<Self> {
let imm = Self::new(u);
(imm.0 == u).then_some(imm)
}
pub const fn to_u8(self) -> u8 {
self.0
}
}
impl Imm12 {
pub const MAX: Self = Self(0b_0000_1111_1111_1111);
pub const fn new(u: u16) -> Self {
Self(u & Self::MAX.0)
}
pub fn new_checked(u: u16) -> Option<Self> {
let imm = Self::new(u);
(imm.0 == u).then_some(imm)
}
pub const fn to_u16(self) -> u16 {
self.0
}
}
impl Imm18 {
pub const MAX: Self = Self(0b_0000_0000_0000_0011_1111_1111_1111_1111);
pub const fn new(u: u32) -> Self {
Self(u & Self::MAX.0)
}
pub fn new_checked(u: u32) -> Option<Self> {
let imm = Self::new(u);
(imm.0 == u).then_some(imm)
}
pub const fn to_u32(self) -> u32 {
self.0
}
}
impl Imm24 {
pub const MAX: Self = Self(0b_0000_0000_1111_1111_1111_1111_1111_1111);
pub const fn new(u: u32) -> Self {
Self(u & Self::MAX.0)
}
pub fn new_checked(u: u32) -> Option<Self> {
let imm = Self::new(u);
(imm.0 == u).then_some(imm)
}
pub const fn to_u32(self) -> u32 {
self.0
}
}
impl Opcode {
#[allow(clippy::match_like_matches_macro)]
pub fn is_predicate_allowed(&self) -> bool {
use Opcode::*;
match self {
ADD | AND | DIV | EQ | EXP | GT | LT | MLOG | MROO | MOD | MOVE | MUL
| NOT | OR | SLL | SRL | SUB | XOR | WDCM | WQCM | WDOP | WQOP | WDML
| WQML | WDDV | WQDV | WDMD | WQMD | WDAM | WQAM | WDMM | WQMM | PSHH
| PSHL | POPH | POPL | RET | ALOC | MCL | MCP | MEQ | ECK1 | ECR1 | ED19
| K256 | S256 | NOOP | FLAG | ADDI | ANDI | DIVI | EXPI | MODI | MULI
| MLDV | ORI | SLLI | SRLI | SUBI | XORI | JNEI | LB | LW | SB | SW
| MCPI | MCLI | GM | MOVI | JNZI | JI | JMP | JNE | JMPF | JMPB | JNZF
| JNZB | JNEF | JNEB | CFEI | CFSI | CFE | CFS | GTF => true,
_ => false,
}
}
}
impl From<u8> for RegId {
fn from(u: u8) -> Self {
RegId::new(u)
}
}
impl From<u8> for Imm06 {
fn from(u: u8) -> Self {
Imm06::new(u)
}
}
impl From<u16> for Imm12 {
fn from(u: u16) -> Self {
Imm12::new(u)
}
}
impl From<u32> for Imm18 {
fn from(u: u32) -> Self {
Imm18::new(u)
}
}
impl From<u32> for Imm24 {
fn from(u: u32) -> Self {
Imm24::new(u)
}
}
impl From<RegId> for u8 {
fn from(RegId(u): RegId) -> Self {
u
}
}
impl From<Imm06> for u8 {
fn from(Imm06(u): Imm06) -> Self {
u
}
}
impl From<Imm12> for u16 {
fn from(Imm12(u): Imm12) -> Self {
u
}
}
impl From<Imm18> for u32 {
fn from(Imm18(u): Imm18) -> Self {
u
}
}
impl From<Imm24> for u32 {
fn from(Imm24(u): Imm24) -> Self {
u
}
}
impl From<RegId> for usize {
fn from(r: RegId) -> usize {
u8::from(r).into()
}
}
impl From<Imm06> for u16 {
fn from(imm: Imm06) -> Self {
u8::from(imm).into()
}
}
impl From<Imm06> for u32 {
fn from(imm: Imm06) -> Self {
u8::from(imm).into()
}
}
impl From<Imm06> for u64 {
fn from(imm: Imm06) -> Self {
u8::from(imm).into()
}
}
impl From<Imm06> for u128 {
fn from(imm: Imm06) -> Self {
u8::from(imm).into()
}
}
impl From<Imm12> for u32 {
fn from(imm: Imm12) -> Self {
u16::from(imm).into()
}
}
impl From<Imm12> for u64 {
fn from(imm: Imm12) -> Self {
u16::from(imm).into()
}
}
impl From<Imm12> for u128 {
fn from(imm: Imm12) -> Self {
u16::from(imm).into()
}
}
impl From<Imm18> for u64 {
fn from(imm: Imm18) -> Self {
u32::from(imm).into()
}
}
impl From<Imm18> for u128 {
fn from(imm: Imm18) -> Self {
u32::from(imm).into()
}
}
impl From<Imm24> for u64 {
fn from(imm: Imm24) -> Self {
u32::from(imm).into()
}
}
impl From<Imm24> for u128 {
fn from(imm: Imm24) -> Self {
u32::from(imm).into()
}
}
impl From<Opcode> for u8 {
fn from(op: Opcode) -> Self {
op as u8
}
}
impl From<Instruction> for RawInstruction {
fn from(inst: Instruction) -> Self {
RawInstruction::from_be_bytes(inst.into())
}
}
impl core::convert::TryFrom<RawInstruction> for Instruction {
type Error = InvalidOpcode;
fn try_from(u: RawInstruction) -> Result<Self, Self::Error> {
Self::try_from(u.to_be_bytes())
}
}
impl<T> core::ops::Index<RegId> for [T]
where
[T]: core::ops::Index<usize, Output = T>,
{
type Output = T;
fn index(&self, ix: RegId) -> &Self::Output {
&self[usize::from(ix)]
}
}
impl<T> core::ops::IndexMut<RegId> for [T]
where
[T]: core::ops::IndexMut<usize, Output = T>,
{
fn index_mut(&mut self, ix: RegId) -> &mut Self::Output {
&mut self[usize::from(ix)]
}
}
#[cfg(feature = "alloc")]
impl core::iter::FromIterator<Instruction> for alloc::vec::Vec<u8> {
fn from_iter<I: IntoIterator<Item = Instruction>>(iter: I) -> Self {
iter.into_iter().flat_map(Instruction::to_bytes).collect()
}
}
#[cfg(feature = "alloc")]
impl core::iter::FromIterator<Instruction> for alloc::vec::Vec<u32> {
fn from_iter<I: IntoIterator<Item = Instruction>>(iter: I) -> Self {
iter.into_iter().map(u32::from).collect()
}
}
pub fn from_bytes<I>(bs: I) -> impl Iterator<Item = Result<Instruction, InvalidOpcode>>
where
I: IntoIterator<Item = u8>,
{
let mut iter = bs.into_iter();
core::iter::from_fn(move || {
let a = iter.next()?;
let b = iter.next()?;
let c = iter.next()?;
let d = iter.next()?;
Some(Instruction::try_from([a, b, c, d]))
})
}
pub fn from_u32s<I>(us: I) -> impl Iterator<Item = Result<Instruction, InvalidOpcode>>
where
I: IntoIterator<Item = u32>,
{
us.into_iter().map(Instruction::try_from)
}
fn check_imm06(u: u8) -> Imm06 {
Imm06::new_checked(u)
.unwrap_or_else(|| panic!("Value `{u}` out of range for 6-bit immediate"))
}
fn check_imm12(u: u16) -> Imm12 {
Imm12::new_checked(u)
.unwrap_or_else(|| panic!("Value `{u}` out of range for 12-bit immediate"))
}
fn check_imm18(u: u32) -> Imm18 {
Imm18::new_checked(u)
.unwrap_or_else(|| panic!("Value `{u}` out of range for 18-bit immediate"))
}
fn check_imm24(u: u32) -> Imm24 {
Imm24::new_checked(u)
.unwrap_or_else(|| panic!("Value `{u}` out of range for 24-bit immediate"))
}
#[test]
fn test_instruction_size() {
assert_eq!(
core::mem::size_of::<Instruction>(),
core::mem::size_of::<RawInstruction>()
);
assert_eq!(core::mem::size_of::<Instruction>(), Instruction::SIZE);
}
#[test]
fn test_opcode_size() {
assert_eq!(core::mem::size_of::<Opcode>(), 1);
}
#[test]
#[allow(clippy::match_like_matches_macro)]
fn check_predicate_allowed() {
use Opcode::*;
for byte in 0..u8::MAX {
if let Ok(repr) = Opcode::try_from(byte) {
let should_allow = match repr {
BAL | BHEI | BHSH | BURN | CALL | CB | CCP | CROO | CSIZ | LDC | LOG
| LOGD | MINT | RETD | RVRT | SMO | SCWQ | SRW | SRWQ | SWW | SWWQ
| TIME | TR | TRO | ECAL => false,
_ => true,
};
assert_eq!(should_allow, repr.is_predicate_allowed());
}
}
}
#[test]
fn test_opcode_u8_conv() {
for u in 0..=u8::MAX {
if let Ok(op) = Opcode::try_from(u) {
assert_eq!(op as u8, u);
}
}
}