Architecture specification

This specification describes the architecture required to run Bedrock bytecode.

Terminology

Byte

A byte is an 8-bit value, and is the base unit of data used by Bedrock.

Double

The term double is short for ‘double-width value’, and is a 16-bit value represented as a pair of bytes in big-endian order.

Undefined behaviour

Any operation with undefined behaviour must be avoided by a program.

Program memory

The program memory is a 65536 byte block of read-write memory that holds the current program. The initial value of each byte of memory is zero.

Systems with fewer than 65536 bytes of program memory are partially supported. The behaviour when an unimplemented memory address is read from or written to is undefined.

Reading from memory

Reading a byte from program memory will return the byte at the given memory address.

Reading a double from program memory will read the high byte from the given memory address, and the low byte from the following memory address. The behaviour when a double is read from the highest memory address is undefined.

Writing to memory

Writing a byte to program memory will write that byte to the given memory address.

Writing a double to program memory will write the high byte to the given memory address, and the low byte to the following memory address. The behaviour when a double is written to the highest memory address is undefined.

Loading a program

To load a program into program memory, the program bytecode is copied into program memory starting at address zero. If the program exceeds the length of program memory, the remaining bytes are discarded.

Stacks

The stacks are two identical 256 byte stacks, the working stack and the return stack. The initial value of each byte of stack memory is undefined.

Systems with fewer than 256 bytes of memory per stack are partially supported.

Stack pointer

Each stack has a stack pointer, an 8-bit register that points to the next free byte of stack memory. The initial value of each stack pointer is zero. The behaviour on overflow or underflow is undefined.

Pushing to a stack

Pushing a byte to a stack will write the byte to the stack memory address referenced by the stack pointer, and then will increment the stack pointer by 1.

Pushing a double to a stack will push the high byte of the double, followed by the low byte of the double.

Popping from a stack

Popping a byte from a stack will decrement the stack pointer by 1, and then will read the byte from the stack memory address referenced by the stack pointer.

Popping a double from a stack will pop the low byte of the double first, followed by the high byte of the double.

Device bus

The device bus is a 256 port communications bus for interfacing with up to sixteen different devices. Each device is allocated sixteen ports on the device bus.

Each port is identified by an 8-bit port number. The upper four bits of the port number identify a device slot, and the lower four bits identify a port within that device.

Reading from a device

Reading a byte from the device bus will send a read request to the given device port, and then the device connected to that port will process the read request and return a byte. There is no limit to the time a device can take while processing a read request. If there is no device connected to the given device port, the request will return immediately with a value of zero.

Reading a double from the device bus will read the high byte of the double from the given device port, and the low byte of the double from the following device port. The behaviour when a double is read from the highest port number is undefined.

Writing to a device

Writing a byte to the device bus will send a write request with that byte to the given device port, and then the device connected to that port will process the write request and return. There is no limit to the time a device can take while processing a write request. If there is no device connected to the given device port, the request will return immediately.

Writing a double to the device bus will write the high byte of the double to the given device port, and the low byte of the double to the following device port. The behaviour when a double is written to the highest port number is undefined.

Processor

The processor sequentially executes instructions from program memory.

Program counter

The program counter is a 16-bit register that points to the memory address of the next instruction to be executed. The initial value of the program counter is zero. The behaviour when the program counter overflows is undefined.

After an instruction is read from program memory, the program counter is incremented by 1. After a value is read from program memory due to the effect of immediate mode, the program counter is incremented by 1 for each byte read.

Instructions

Instructions are stored as bytes in program memory. The lower five bits of an instruction byte determine the operation to be performed, and the upper three bits are mode flags that affect how the operation will be performed.

Return mode

Bit 0x80 of the instruction byte is the return mode flag. When set, the operation will use the return stack in place of the working stack, and the working stack in place of the return stack.

Immediate mode

Bit 0x40 of the instruction byte is the immediate mode flag. When set, the first time that the operation would pop a value from a stack, a value of the same size is read instead from program memory, from the address referenced by the program counter.

Wide mode

Bit 0x20 of the instruction byte is the wide mode flag. When set, the instruction will operate on double-size values, otherwise the instruction will operate on byte-size values. Values specified as fixed-size by an instruction are not affected by this mode.

Operation summary

In the notation used in the signature column of the following table, the letters to the left of the -- token represent the values on the stack before the instruction is executed, and the letters to the right represent the values on the stack after the instruction is executed. A * suffix represents a fixed-size double, a ! suffix represents a fixed-size byte, and a plain letter represents a value with a size determined by the state of the double mode flag.

Base Name Description Signature
0x00 HLT Halt the processor. ( -- )
0x01 JMP Jump to an address. ( a* -- )
0x02 JCN Conditionally jump to an address. ( t! a* -- )
0x03 JCK Conditionally jump to an address, keeping the value. ( t a* -- t )
0x04 LDA Read a value from program memory. ( a* -- v )
0x05 STA Write a value to program memory. ( v a* -- )
0x06 LDD Read a value from the device bus. ( p! -- v )
0x07 STD Write a value to the device bus. ( v p! -- )
0x08 PSH Move the top value from one stack to the other. ( -- x ) ( x -- )
0x09 POP Remove the top value of a stack. ( x -- )
0x0A CPY Copy the top value from one stack to the other. ( -- x ) ( x -- x )
0x0B SPL Split packed values into multiple bytes. ( x -- y z )
0x0C DUP Duplicate the top value of a stack. ( x -- x x )
0x0D OVR Duplicate the second value of a stack. ( x y -- x y x )
0x0E SWP Swap the top two values on a stack. ( x y -- y x )
0x0F ROT Rotate the top three values on a stack. ( x y z -- y z x )
0x10 ADD Find the sum of two values. ( x y -- r )
0x11 SUB Find the difference of two values. ( x y -- r )
0x12 INC Increment a value by 1. ( x -- r )
0x13 DEC Decrement a value by 1. ( x -- r )
0x14 LTH Test if one value is less than another. ( x y -- t! )
0x15 GTH Test if one value is greater than another. ( x y -- t! )
0x16 EQU Test if two values are equal. ( x y -- t! )
0x17 NQK Test if two values are inequal, keeping the values. ( x y -- x y t! )
0x18 IOR Find the bitwise disjunction of two values. ( x y -- r )
0x19 XOR Find the bitwise non-equivalence of two values. ( x y -- r )
0x1A AND Find the bitwise conjugation of two values. ( x y -- r )
0x1B NOT Find the bitwise compliment of a value. ( x -- r )
0x1C SHF Shift the bits of a value. ( x y! -- r )
0x1D SHC Rotate the bits of a value. ( x y! -- r )
0x1E TAL Count the number of set bits in a value. ( x -- r! )
0x1F REV Reverse the order of bits in a value. ( x -- r )

Operations

The terms primary stack and return stack are used in the following descriptions to refer to the stacks once the return mode flag has been applied. The primary stack is normally the working stack, but will be the return stack while the return mode flag is set. The secondary stack is normally the return stack, but will be the working stack while the return mode flag is set.

In the instruction table for each operation, the * column contains a character when the wide mode flag is set, the : column contains a character when the immediate mode flag is set, and the r column contains a character when the return mode flag is set.

Category
Flow control HLT JMP JCN JCK
Access LDA STA LDD STD
Stack PSH POP CPY SPL
Movement DUP OVR SWP ROT
Arithmetic ADD SUB INC DEC
Relational LTH GTH EQU NQK
Logical IOR XOR AND NOT
Binary SHF SHC TAL REV

HLT

Halt the processor, terminating the current program. If any mode flag is set, do nothing.

The instruction variants DB1, DB2, DB3, DB4, DB5, and DB6 can be used as hooks for implementation defined debug functionality.

Signature
( -- )
Instr. Mnm. * : r Working stack Return stack Notes
0x00 HLT ( -- ) ( -- ) Halt the processor.
0x20 NOP * ( -- ) ( -- ) No effect.
0x40 DB1 : ( -- ) ( -- ) No effect.
0x60 DB2 * : ( -- ) ( -- ) No effect.
0x80 DB3 r ( -- ) ( -- ) No effect.
0xA0 DB4 * r ( -- ) ( -- ) No effect.
0xC0 DB5 : r ( -- ) ( -- ) No effect.
0xE0 DB6 * : r ( -- ) ( -- ) No effect.

JMP

Jump to an address.

The double a is popped from the primary stack. If the wide mode flag is set, the double b is then read from the program counter and pushed to the secondary stack. The double a is then written to the program counter.

Signature
( a* -- : -- [b*] )
Instr. Mnm. * : r Working stack Return stack
0x01 JMP ( a* -- ) ( -- )
0x21 JMS * ( a* -- ) ( -- b* )
0x41 JMP: : ( -- ) ( -- )
0x61 JMS: * : ( -- ) ( -- b* )
0x81 JMPr r ( -- ) ( a* -- )
0xA1 JMSr * r ( -- b* ) ( a* -- )
0xC1 JMPr: : r ( -- ) ( -- )
0xE1 JMSr: * : r ( -- b* ) ( -- )

JCN

Conditionally jump to an address.

The double a is popped from the primary stack, then the byte t is popped from the primary stack. If the wide mode flag is set and t is not zero, the double b is then read from the program counter and pushed to the secondary stack. If t is not zero, the double a is then written to the program counter.

Signature
( t! a* -- : -- [b*] )
Instr. Mnm. * : r Working stack Return stack
0x02 JCN ( t a* -- ) ( -- )
0x22 JCS * ( t a* -- ) ( -- b* )
0x42 JCN: : ( t -- ) ( -- )
0x62 JCS: * : ( t -- ) ( -- b* )
0x82 JCNr r ( -- ) ( t a* -- )
0xA2 JCSr * r ( -- b* ) ( t a* -- )
0xC2 JCNr: : r ( -- ) ( t -- )
0xE2 JCSr: * : r ( -- b* ) ( t -- )

JCK

Conditionally jump to an address, keeping the test value.

The double a is popped from the primary stack, then the value t is popped from the primary stack, then the value t is pushed to the primary stack. The double a is then written to the program counter if t is not zero.

Signature
( t a* -- t )
Instr. Mnm. * : r Working stack Return stack
0x03 JCK ( t a* -- t ) ( -- )
0x23 JCK* * ( t* a* -- t* ) ( -- )
0x43 JCK: : ( t -- t ) ( -- )
0x63 JCK*: * : ( t* -- t* ) ( -- )
0x83 JCKr r ( -- ) ( t a* -- t )
0xA3 JCKr* * r ( -- ) ( t* a* -- t* )
0xC3 JCKr: : r ( -- ) ( t -- t )
0xE3 JCKr*: * : r ( -- ) ( t* -- t* )

LDA

Read a value from program memory.

The double a is popped from the primary stack, and then the value v is read from program memory starting at the memory address referenced by a and is pushed to the primary stack.

Signature
( a* -- v )
Instr. Mnm. * : r Working stack Return stack
0x04 LDA ( a* -- v ) ( -- )
0x24 LDA* * ( a* -- v* ) ( -- )
0x44 LDA: : ( -- v ) ( -- )
0x64 LDA*: * : ( -- v* ) ( -- )
0x84 LDAr r ( -- ) ( a* -- v )
0xA4 LDAr* * r ( -- ) ( a* -- v* )
0xC4 LDAr: : r ( -- ) ( -- v )
0xE4 LDAr*: * : r ( -- ) ( -- v* )

STA

Write a value to program memory.

The double a is popped from the primary stack, then the value v is popped from the primary stack, and then the value v is written to program memory starting at the memory address referenced by a.

Signature
( v a* -- )
Instr. Mnm. * : r Working stack Return stack
0x05 STA ( v a* -- ) ( -- )
0x25 STA* * ( v* a* -- ) ( -- )
0x45 STA: : ( v -- ) ( -- )
0x65 STA*: * : ( v* -- ) ( -- )
0x85 STAr r ( -- ) ( v a* -- )
0xA5 STAr* * r ( -- ) ( v* a* -- )
0xC5 STAr: : r ( -- ) ( v -- )
0xE5 STAr*: * : r ( -- ) ( v* -- )

LDD

Read a value from the device bus.

The byte p is popped from the primary stack, then the value v is read from the device bus starting at the device port referenced by p and is pushed to the primary stack.

Signature
( p! -- v )
Instr. Mnm. * : r Working stack Return stack
0x06 LDD ( p -- v ) ( -- )
0x26 LDD* * ( p -- v* ) ( -- )
0x46 LDD: : ( -- v ) ( -- )
0x66 LDD*: * : ( -- v* ) ( -- )
0x86 LDDr r ( -- ) ( p -- v )
0xA6 LDDr* * r ( -- ) ( p -- v* )
0xC6 LDDr: : r ( -- ) ( -- v )
0xE6 LDDr*: * : r ( -- ) ( -- v* )

STD

Write a value to the device bus.

The byte p is popped from the primary stack, then the value v is popped from the primary stack and is written to the device bus starting at the device port referenced by p.

Signature
( v p! -- )
Instr. Mnm. * : r Working stack Return stack
0x07 STD ( v p -- ) ( -- )
0x27 STD* * ( v* p -- ) ( -- )
0x47 STD: : ( v -- ) ( -- )
0x67 STD*: * : ( v* -- ) ( -- )
0x87 STDr r ( -- ) ( v p -- )
0xA7 STDr* * r ( -- ) ( v* p -- )
0xC7 STDr: : r ( -- ) ( v -- )
0xE7 STDr*: * : r ( -- ) ( v* -- )

PSH

Move the top value from one stack to the other.

The value x is popped from the secondary stack and is pushed to the primary stack.

Signature
( -- x ) ( x -- )
Instr. Mnm. * : r Working stack Return stack
0x08 PSH ( -- x ) ( x -- )
0x28 PSH* * ( -- x* ) ( x* -- )
0x48 PSH: : ( -- l ) ( -- )
0x68 PSH*: * : ( -- l* ) ( -- )
0x88 PSHr r ( x -- ) ( -- x )
0xA8 PSHr* * r ( x* -- ) ( -- x* )
0xC8 PSHr: : r ( -- ) ( -- l )
0xE8 PSHr*: * : r ( -- ) ( -- l* )

POP

Remove the top value of a stack.

The value x is popped from the primary stack.

Signature
( x -- )
Instr. Mnm. * : r Working stack Return stack
0x09 POP ( x -- ) ( -- )
0x29 POP* * ( x* -- ) ( -- )
0x49 POP: : ( -- ) ( -- )
0x69 POP*: * : ( -- ) ( -- )
0x89 POPr r ( -- ) ( x -- )
0xA9 POPr* * r ( -- ) ( x* -- )
0xC9 POPr: : r ( -- ) ( -- )
0xE9 POPr*: * : r ( -- ) ( -- )

CPY

Copy the top value from one stack to the other.

The value x is popped from the secondary stack, and then is pushed to the secondary stack, and then is pushed to the primary stack.

Signature
( -- x ) ( x -- x )
Instr. Mnm. * : r Working stack Return stack
0x0A CPY ( -- x ) ( x -- x )
0x2A CPY* * ( -- x* ) ( x* -- x* )
0x4A CPY: : ( -- l ) ( -- l )
0x6A CPY*: * : ( -- l* ) ( -- l* )
0x8A CPYr r ( x -- x ) ( -- x )
0xAA CPYr* * r ( x* -- x* ) ( -- x* )
0xCA CPYr: : r ( -- l ) ( -- l )
0xEA CPYr*: * : r ( -- l* ) ( -- l* )

SPL

Split packed bytes into two bytes each.

The value x is popped from the primary stack. For each byte b of x, starting with the high byte, create a byte y where the high four bits of b are the low four bits of y and the remainder are unset, and push y to the primary stack, then create a byte z where the low four bits of b are the low four bits of z and the remainder are unset, and push z to the primary stack.

Signature
( x -- y z )
Instr. Mnm. * : r Working stack Return stack
0x0B SPL ( x -- y z ) ( -- )
0x2B SPL* * ( x* -- y z y z ) ( -- )
0x4B SPL: : ( -- y z ) ( -- )
0x6B SPL*: * : ( -- y z y z) ( -- )
0x8B SPLr r ( -- ) ( x -- y z )
0xAB SPLr* * r ( -- ) ( x* -- y z y z )
0xCB SPLr: : r ( -- ) ( -- y z )
0xEB SPLr*: * : r ( -- ) ( -- y z y z)

DUP

Duplicate the top value of a stack.

The value x is popped from the primary stack, then is pushed to the primary stack, and then is pushed once more to the primary stack.

Signature
( x -- x x )
Instr. Mnm. * : r Working stack Return stack
0x0C DUP ( x -- x x ) ( -- )
0x2C DUP* * ( x* -- x* x* ) ( -- )
0x4C DUP: : ( -- l l ) ( -- )
0x6C DUP*: * : ( -- l* l* ) ( -- )
0x8C DUPr r ( -- ) ( x -- x x )
0xAC DUPr* * r ( -- ) ( x* -- x* x* )
0xCC DUPr: : r ( -- ) ( -- l l )
0xEC DUPr*: * : r ( -- ) ( -- l* l* )

OVR

Duplicate the second value of a stack.

The value y is popped from the primary stack, then the value x is popped from the primary stack, then the value x is pushed to the primary stack, then the value y is pushed to the primary stack, and then the value x is pushed to the primary stack.

Signature
( x y -- x y x )
Instr. Mnm. * : r Working stack Return stack
0x0D OVR ( x y -- x y x ) ( -- )
0x2D OVR* * ( x* y* -- x* y* x* ) ( -- )
0x4D OVR: : ( x -- x l x ) ( -- )
0x6D OVR*: * : ( x* -- x* l* x* ) ( -- )
0x8D OVRr r ( -- ) ( x y -- x y x )
0xAD OVRr* * r ( -- ) ( x* y* -- x* y* x* )
0xCD OVRr: : r ( -- ) ( x -- x l x )
0xED OVRr*: * : r ( -- ) ( x* -- x* l* x* )

SWP

Swap the top two values on a stack.

The value y is popped from the primary stack, then the value x is popped from the primary stack, then the value y is pushed to the primary stack, and then the value x is pushed to the primary stack.

Signature
( x y -- y x )
Instr. Mnm. * : r Working stack Return stack
0x0E SWP ( x y -- y x ) ( -- )
0x2E SWP* * ( x* y* -- y* x* ) ( -- )
0x4E SWP: : ( x -- l x ) ( -- )
0x6E SWP*: * : ( x* -- l* x* ) ( -- )
0x8E SWPr r ( -- ) ( x y -- y x )
0xAE SWPr* * r ( -- ) ( x* y* -- y* x* )
0xCE SWPr: : r ( -- ) ( x -- l x )
0xEE SWPr*: * : r ( -- ) ( x* -- l* x* )

ROT

Rotate the top three values on a stack.

The value z is popped from the primary stack, then the value y is popped from the primary stack, then the value x is popped from the primary stack, then the value y is pushed to the primary stack, then the value z is pushed to the primary stack, and then the value x is pushed to the primary stack.

Signature
( x y z -- y z x )
Instr. Mnm. * : r Working stack Return stack
0x0F ROT ( x y z -- y z x ) ( -- )
0x2F ROT* * ( x* y* z* -- y* z* x* ) ( -- )
0x4F ROT: : ( x y -- y l x ) ( -- )
0x6F ROT*: * : ( x* y* -- y* l* x* ) ( -- )
0x8F ROTr r ( -- ) ( x y z -- y z x )
0xAF ROTr* * r ( -- ) ( x* y* z* -- y* z* x* )
0xCF ROTr: : r ( -- ) ( x y -- y l x )
0xEF ROTr*: * : r ( -- ) ( x* y* -- y* l* x* )

ADD

Find the sum of two values.

The value y is popped from the primary stack, then the value x is popped from the primary stack, and then the value r is calculated as the wrapped addition of y to x and is pushed to the primary stack.

Signature
( x y -- r )
Instr. Mnm. * : r Working stack Return stack
0x10 ADD ( x y -- r ) ( -- )
0x30 ADD* * ( x* y* -- r* ) ( -- )
0x50 ADD: : ( x -- r ) ( -- )
0x70 ADD*: * : ( x* -- r* ) ( -- )
0x90 ADDr r ( -- ) ( x y -- r )
0xB0 ADDr* * r ( -- ) ( x* y* -- r* )
0xD0 ADDr: : r ( -- ) ( x -- r )
0xF0 ADDr*: * : r ( -- ) ( x* -- r* )

SUB

Find the difference of two values.

The value y is popped from the primary stack, then the value x is popped from the primary stack, and then the value r is calculated as the wrapped subtraction of y from x and is pushed to the primary stack.

Signature
( x y -- r )
Instr. Mnm. * : r Working stack Return stack
0x11 SUB ( x y -- r ) ( -- )
0x31 SUB* * ( x* y* -- r* ) ( -- )
0x51 SUB: : ( x -- r ) ( -- )
0x71 SUB*: * : ( x* -- r* ) ( -- )
0x91 SUBr r ( -- ) ( x y -- r )
0xB1 SUBr* * r ( -- ) ( x* y* -- r* )
0xD1 SUBr: : r ( -- ) ( x -- r )
0xF1 SUBr*: * : r ( -- ) ( x* -- r* )

INC

Increment a value by 1.

The value x is popped from the primary stack, and then the value r is calculated as the wrapped addition of 1 to x and is pushed to the primary stack.

Signature
( x -- r )
Instr. Mnm. * : r Working stack Return stack
0x12 INC ( x -- r ) ( -- )
0x32 INC* * ( x* -- r* ) ( -- )
0x52 INC: : ( -- r ) ( -- )
0x72 INC*: * : ( -- r* ) ( -- )
0x92 INCr r ( -- ) ( x -- r )
0xB2 INCr* * r ( -- ) ( x* -- r* )
0xD2 INCr: : r ( -- ) ( -- r )
0xF2 INCr*: * : r ( -- ) ( -- r* )

DEC

Decrement a value by 1.

The value x is popped from the primary stack, and then the value r is calculated as the wrapped subtraction of 1 from x and is pushed to the primary stack.

Signature
( x -- r )
Instr. Mnm. * : r Working stack Return stack
0x13 DEC ( x -- r ) ( -- )
0x33 DEC* * ( x* -- r* ) ( -- )
0x53 DEC: : ( -- r ) ( -- )
0x73 DEC*: * : ( -- r* ) ( -- )
0x93 DECr r ( -- ) ( x -- r )
0xB3 DECr* * r ( -- ) ( x* -- r* )
0xD3 DECr: : r ( -- ) ( -- r )
0xF3 DECr*: * : r ( -- ) ( -- r* )

LTH

Test if one value is less than another.

The value y is popped from the primary stack, and then the value x is popped from the primary stack. If x is less than y, the byte 0xff is pushed to the primary stack, otherwise the byte 0x00 is pushed to the primary stack.

Signature
( x y -- t! )
Instr. Mnm. * : r Working stack Return stack
0x14 LTH ( x y -- t ) ( -- )
0x34 LTH* * ( x* y* -- t ) ( -- )
0x54 LTH: : ( x -- t ) ( -- )
0x74 LTH*: * : ( x* -- t ) ( -- )
0x94 LTHr r ( -- ) ( x y -- t )
0xB4 LTHr* * r ( -- ) ( x* y* -- t )
0xD4 LTHr: : r ( -- ) ( x -- t )
0xF4 LTHr*: * : r ( -- ) ( x* -- t )

GTH

Test if one value is greater than another.

The value y is popped from the primary stack, and then the value x is popped from the primary stack. If x is greater than y, the byte 0xff is pushed to the primary stack, otherwise the byte 0x00 is pushed to the primary stack.

Signature
( x y -- t! )
Instr. Mnm. * : r Working stack Return stack
0x15 GTH ( x y -- t ) ( -- )
0x35 GTH* * ( x* y* -- t ) ( -- )
0x55 GTH: : ( x -- t ) ( -- )
0x75 GTH*: * : ( x* -- t ) ( -- )
0x95 GTHr r ( -- ) ( x y -- t )
0xB5 GTHr* * r ( -- ) ( x* y* -- t )
0xD5 GTHr: : r ( -- ) ( x -- t )
0xF5 GTHr*: * : r ( -- ) ( x* -- t )

EQU

Test if two values are equal.

The value y is popped from the primary stack, and then the value x is popped from the primary stack. If x is equal to y, the byte 0xff is pushed to the primary stack, otherwise the byte 0x00 is pushed to the primary stack.

Signature
( x y -- t! )
Instr. Mnm. * : r Working stack Return stack
0x16 EQU ( x y -- t ) ( -- )
0x36 EQU* * ( x* y* -- t ) ( -- )
0x56 EQU: : ( x -- t ) ( -- )
0x76 EQU*: * : ( x* -- t ) ( -- )
0x96 EQUr r ( -- ) ( x y -- t )
0xB6 EQUr* * r ( -- ) ( x* y* -- t )
0xD6 EQUr: : r ( -- ) ( x -- t )
0xF6 EQUr*: * : r ( -- ) ( x* -- t )

NQK

Test if two values are inequal, keeping the values.

The value y is popped from the primary stack, then the value x is popped from the primary stack, then the value x is pushed to the primary stack, and then the value y is pushed to the primary stack. If x is not equal to y, the byte 0xff is pushed to the primary stack, otherwise the byte 0x00 is pushed to the primary stack.

Signature
( x y -- x y t! )
Instr. Mnm. * : r Working stack Return stack
0x17 NQK ( x y -- x y t ) ( -- )
0x37 NQK* * ( x* y* -- x* y* t ) ( -- )
0x57 NQK: : ( x -- x l t ) ( -- )
0x77 NQK*: * : ( x* -- x* l* t ) ( -- )
0x97 NQKr r ( -- ) ( x y -- x y t )
0xB7 NQKr* * r ( -- ) ( x* y* -- x* y* t )
0xD7 NQKr: : r ( -- ) ( x -- x l t )
0xF7 NQKr*: * : r ( -- ) ( x* -- x* l* t )

IOR

Find the bitwise disjunction of two values. This operation is also known as ‘inclusive-or’.

The value y is popped from the primary stack, and then the value x is popped from the primary stack. The value r is then calculated bitwise, where each bit of r is set only if at least one of the corresponding bits of x and y is set. The value r is then pushed to the primary stack.

Signature
( x y -- r )
Instr. Mnm. * : r Working stack Return stack
0x18 IOR ( x y -- r ) ( -- )
0x38 IOR* * ( x* y* -- r* ) ( -- )
0x58 IOR: : ( x -- r ) ( -- )
0x78 IOR*: * : ( x* -- r* ) ( -- )
0x98 IORr r ( -- ) ( x y -- r )
0xB8 IORr* * r ( -- ) ( x* y* -- r* )
0xD8 IORr: : r ( -- ) ( x -- r )
0xF8 IORr*: * : r ( -- ) ( x* -- r* )

XOR

Find the bitwise non-equivalence of two values. This operation is also known as ‘exclusive-or’.

The value y is popped from the primary stack, and then the value x is popped from the primary stack. The value r is then calculated bitwise, where each bit of r is set only if exactly one of the corresponding bits of x and y is set. The value r is then pushed to the primary stack.

Signature
( x y -- r )
Instr. Mnm. * : r Working stack Return stack
0x19 XOR ( x y -- r ) ( -- )
0x39 XOR* * ( x* y* -- r* ) ( -- )
0x59 XOR: : ( x -- r ) ( -- )
0x79 XOR*: * : ( x* -- r* ) ( -- )
0x99 XORr r ( -- ) ( x y -- r )
0xB9 XORr* * r ( -- ) ( x* y* -- r* )
0xD9 XORr: : r ( -- ) ( x -- r )
0xF9 XORr*: * : r ( -- ) ( x* -- r* )

AND

Find the bitwise conjunction of two values. This operation is also known as ‘and’.

The value y is popped from the primary stack, and then the value x is popped from the primary stack. The value r is then calculated bitwise, where each bit of r is set only if both of the corresponding bits of x and y are set. The value r is then pushed to the primary stack.

Signature
( x y -- r )
Instr. Mnm. * : r Working stack Return stack
0x1A AND ( x y -- r ) ( -- )
0x3A AND* * ( x* y* -- r* ) ( -- )
0x5A AND: : ( x -- r ) ( -- )
0x7A AND*: * : ( x* -- r* ) ( -- )
0x9A ANDr r ( -- ) ( x y -- r )
0xBA ANDr* * r ( -- ) ( x* y* -- r* )
0xDA ANDr: : r ( -- ) ( x -- r )
0xFA ANDr*: * : r ( -- ) ( x* -- r* )

NOT

Find the bitwise compliment of a value. This operation is also known as ‘not’.

The value x is popped from the primary stack. The value r is then calculated bitwise, where each bit of r is set only if the corresponding bit of x is not set. The value r is then pushed to the primary stack.

Signature
( x -- r )
Instr. Mnm. * : r Working stack Return stack
0x1B NOT ( x -- r ) ( -- )
0x3B NOT* * ( x* -- r* ) ( -- )
0x5B NOT: : ( -- r ) ( -- )
0x7B NOT*: * : ( -- r* ) ( -- )
0x9B NOTr r ( -- ) ( x -- r )
0xBB NOTr* * r ( -- ) ( x* -- r* )
0xDB NOTr: : r ( -- ) ( -- r )
0xFB NOTr*: * : r ( -- ) ( -- r* )

SHF

Shift the bits of a value.

The byte y is popped from the primary stack, then the value x is popped from the primary stack. The value r is calculated from x by shifting the bits of x leftwards by a distance represented by the upper four bits of y, and then by shifting the resultant bits rightwards by a distance represented by the lower four bits of y. The value r is then pushed to the primary stack.

When a value is shifted n bits in one direction, each bit of the value is set to the state of the bit lying n bits in the opposite direction. If this would reach past the end of the value, the bit is unset.

Signature
( x y! -- r )
Instr. Mnm. * : r Working stack Return stack
0x1C SHF ( x y -- r ) ( -- )
0x3C SHF* * ( x* y -- r ) ( -- )
0x5C SHF: : ( x -- r ) ( -- )
0x7C SHF*: * : ( x* -- r ) ( -- )
0x9C SHFr r ( -- ) ( x y -- r )
0xBC SHFr* * r ( -- ) ( x* y -- r )
0xDC SHFr: : r ( -- ) ( x -- r )
0xFC SHFr*: * : r ( -- ) ( x* -- r )

SHC

Rotate the bits of a value. This operation is also known as a ‘circular shift’.

The byte y is popped from the primary stack, then the value x is popped from the primary stack. The value r is calculated from x by rotating the bits of x leftwards by a distance represented by the upper four bits of y, and then by rotating the resultant bits rightwards by a distance represented by the lower four bits of y. The value r is then pushed to the primary stack.

When a value is rotated n bits in one direction, each bit of the value is set to the state of the bit lying n bits in the opposite direction. If this would reach past the end of the value, wrap around to the other end of the value and continue.

Signature
( x y! -- r )
Instr. Mnm. * : r Working stack Return stack
0x1D SHC ( x y -- r ) ( -- )
0x3D SHC* * ( x* y -- r ) ( -- )
0x5D SHC: : ( x -- r ) ( -- )
0x7D SHC*: * : ( x* -- r ) ( -- )
0x9D SHCr r ( -- ) ( x y -- r )
0xBD SHCr* * r ( -- ) ( x* y -- r )
0xDD SHCr: : r ( -- ) ( x -- r )
0xFD SHCr*: * : r ( -- ) ( x* -- r )

TAL

Count the number of set bits in a value. This operation is also known as ‘population count’.

The value x is popped from the primary stack, and then the value r is calculated as the number of bits set in x and is pushed to the primary stack.

Signature
( x -- r! )
Instr. Mnm. * : r Working stack Return stack
0x1E TAL ( x -- r ) ( -- )
0x3E TAL* * ( x* -- r ) ( -- )
0x5E TAL: : ( -- r ) ( -- )
0x7E TAL*: * : ( -- r ) ( -- )
0x9E TALr r ( -- ) ( x -- r )
0xBE TALr* * r ( -- ) ( x* -- r )
0xDE TALr: : r ( -- ) ( -- r )
0xFE TALr*: * : r ( -- ) ( -- r )

REV

Reverse the order of bits in a value.

The value x is popped from the primary stack. The value r is then calculated bitwise, where each bit of r starting from one end of the value is set only if the corresponding bit of x starting from the opposite end of the value is set. The value r is then pushed to the primary stack.

Signature
( x -- r )
Instr. Mnm. * : r Working stack Return stack
0x1F REV ( x -- r ) ( -- )
0x3F REV* * ( x* -- r* ) ( -- )
0x5F REV: : ( -- r ) ( -- )
0x7F REV*: * : ( -- r* ) ( -- )
0x9F REVr r ( -- ) ( x -- r )
0xBF REVr* * r ( -- ) ( x* -- r* )
0xDF REVr: : r ( -- ) ( -- r )
0xFF REVr*: * : r ( -- ) ( -- r* )