Tny

Tutorial

Tny is a tiny virtual console. It can execute sequences of bytes that have a special meaning. These sequences are called "ROMs" and the bytes "machine code". To make writing ROMs easier for Humans, Tny also has an assembly language. The language exposes special words that translate directly to machine instructions. These are called mnemonics. In this book, we will learn about the mnemonics to build simple games with Tny.

Getting started

All Tny ROMs must start with `0 0` or other numbers.

0 0 LIT 1 INC

There is a lot going on in the above. Let's work through it.

First, LIT is an instruction that tells Tny to push the next number into the parameter stack (pstack). The parameter stack is a special part of memory in which we can push or pop values. It's used to pass parameters, thus its name.

Then, INC is another instruction that increases by one the top of the pstack.

If you haven't done so, execute the above program. You should see that the pstack now has 02 stored in it.

Manipulating the stack

Now that we've seen the stack, let's see how we can manipulate it.

LIT lets you push the next number to the stack.

You can read about the other instructions in the glossary below. Let's look at DUP. It says that the purpose is the "Duplicate the top of the stack". There's another column called "Effect" which says:

n -- n n

This is a way to indicate the effect of DUP on the stack. DUP requires a number to be on the stack. This number is the "n" represented on the left of "--". After its execution, the stack now has "n" twice.

Learning how to read stack effects is a great way to quickly see what parameters an instruction needs on the stack, and how it will modify the stack.

Arithmetic

All numbers are expressed in hexadecimal.

LIT f \ push the decimal value 15 to the stack

Let's add two numbers together:

LIT 1 LIT 1 ADD

Now, let's multiply the result by two:

LIT 2 MUL

How about we divide it by 4?

LIT 4 DIV

If you ever need a random number, you can use the RND instruction which will place on the stack a random number in the 0 255 range.

Labels and addresses

When your program gets assembled, every mnemonic gets mapped to a corresponding machine code value, which fits in a byte. Since every instruction fits in a single byte, we can refer to a particular place in your ROM by its offset. In Tny, this offset is called an address.

Imagine the following program:

0 0 LIT 5 LIT 5 ADD

The address of the first LIT instruction is 2. The address of the ADD instruction is 6. Let's now look at the following program:

0 0 start: LIT 5 LIT 5 ADD

In the above code, we added a label called start. It could have been called anything else. When Tny's preprocessor finds a label, it remembers its name and location, so that we can use this location later on.

Jumping

Let's look at the following program:

0 0 LIT @start JMP
start: LIT 5 LIT 5 ADD BRK
LIT 2

Let's try to understand what happens. When executed, the Instruction Pointer (IP) starts at address 2. It finds LIT, so it pushes the number that follows it into the stack. In this case, @start represents the memory address of the start label, which is 5. Our stack now has a 5 in it. Then it finds a JMP instruction. The JMP instruction tells the IP to take the value of the top of the stack, so 5, and to carry on. Our program then executes the LIT instructions, then the ADD one, and finally BRK. BRK is a special instruction that stops the execution of the program. Meaning that the remaining LIT 2 instruction won't be executed.

There are different ways to jump. For example, JMR will store the position of the next instruction in a special stack called the return stack. This allows to resume the execution with the RET instruction.

Drawing on the screen

The screen tny is 16x16, meaning that a screen position can be encoded in one byte. The address of the top left pixel is 00, and the address of the bottom right if ff. The high nibble represents the rows, and the low nibble the columns. For example, the pixel at the 4th row and 10th column is 4a.

A pixel can be either on or off.

0 0 LIT 11 LIT 1 SET

Here, we set the pixel at offset 11.

Controller

Like most consoles, 45M has a controller. A very basic one! It has your usual up left down right keys, as well as A and B which are mapped the the X and C key of the keyboard, respectively.

You can push to the stack the value of the current key being pressed with the KEY instruction. The value has the following format:

0 0 0 B A R L D U

B: the B button
A: the A button R: the right arrow L: the left arrow D: the down arrow U: the up arrow

Vectors

Remember, each ROM starts with two values: 0 0. These values are actually memory addresses. Let's take the following example:

@frame 0 BRK
frame:  LIT 0 LIT 1 SET

Let's start with memory address 0. It's the screen vector. The value at this address will be called 60 times per second. In the example above, we have set it to the frame label.

Now let's continue with memory address 1. The controller vector:

@frame @key BRK
frame: LIT 0 LIT 1 SET
key: KEY

The key vector will be executed anytime a key from the controller has been pressed.

Looping

It's possible to use use labels, jumps, and the return stack in order to create a loop. Here is a full example:

0 0 LIT 0 LIT 10 PSH PSH do:
  RSI LIT 1 SET
  PUL INC PSH
  RSI RSJ LTH LIT @do JCN
  PUL PUL POP POP

Let's unpack!

We start by setting the screen and controller vectors to 0. Then we push 0 and 10 to the stack. They are the index and the limit respectively. The next two instructions, PSH and PSH, move the index and limit to the return stack. We use the return stack as a way to keep track of the current iteration. After that, we create a label called "do". That's the start of the loop.

The first line of the loop turns on a pixel on the first line, based on the current index. It uses the RSI instruction, which copies the top of the return stack to the parameter stack.

The next line increments the index by one by. To do that it pulls it from the return stack, increments it, and pushes it back.

The next line is the test. We compare the top two values of the return stack by using RSI and RSJ with LTH. We then jump back to our "do" label when the comparaison is true.

Finally, we pop the index and limit from the return stack.

Now that we've seen an example, here is a template:

LIT index LIT limit PSH PSH do:
  # instructions go here
  PUL INC PSH
  RSI RSJ LTH LIT @do JCN
  PUL PUL POP POP

Use FRM to control the screen vector's FPS

The screen vector is called 60 times per second, which might be too much for some programs. To change that, we can leverage the FRM and MOD instructions. Let's take an example:

2 0 CLS
FRM LIT 1e MOD LIT 0 NEQ LIT @end JCN
LIT 0 LIT 1 SET
end: BRK

This program starts by setting the screen vector to the address 2. 60 times per second, Tny's Instruction Pointer will be set to the address 2 and will start to execute each instruction sequentially. The first one, CLS, clears the screen. The following line will jump to the "end" label if the current frame should be skipped. To know how many frames should be skipped, we need to compute the following:

frame_skip = 60 / desired_fps

In our example, we want 2 frames per second, so the number of frames to skip is:

LIT 3c LIT 2 DIV

Which is equal to 30 (1e). That's why the 3rd line pushes the current frame as well as 1e to the stack and applies a modulo. We then push 0 and NEQ to make sure to jump our when the current frame isn't a multiple of 1e.

Here's a snippet you can use to controle the FPS:

FRM LIT frame_skip MOD LIT 0 NEQ LIT @end JCN
# Perform computation
end: BRK

Move a pixel around with the keyboard

Let's see how we can move a pixel around. For that, we will use the KEY instruction together with JCN. Here's an example:

@screen @ctrl
screen:
  CLS
  LIT pos: 0 LIT 1 SET
BRK
ctrl:
  KEY LIT 1 EQU LIT @up    JCN
  KEY LIT 2 EQU LIT @down  JCN
  KEY LIT 4 EQU LIT @left  JCN
  KEY LIT 8 EQU LIT @right JCN
BRK

up:    LIT @pos LDA LIT 10 SUB LIT @pos STA BRK
down:  LIT @pos LDA LIT 10 ADD LIT @pos STA BRK
left:  LIT @pos LDA DEC LIT @pos STA BRK
right: LIT @pos LDA INC LIT @pos STA BRK
BRK

As always, we start by setting the screen and controller vector. Any time an arrow key is pressed, the controller vector will get called. In there, we use KEY, which pushes the value of the controller to the stack. We then test its value and jump to the correct label based on it. Notice that we modify directly the value at the pos address.