UART test code

The next stage, after adding ROM, was to add a way to communicate with my Z80 system - I went with a UART, which will be a fairly simple programming exercise and has quite
a lot of flexibility in communications. I chose a recent UART, an NXP 16C550B, which is capable of 3Mbps at full tilt... which, of course, I'd almost certainly never use,
but it sounds good.

My initial board has the UART as the only I/O device, so it's directly connected to the Z80 CPU - meaning its I/O address starts at $00. This is important when it comes to
addressing the UART's registers - obviously if you do this with I/O decoding your UART(s) will be in different locations, so change your addressing accordingly.

Configuring the UART
The first step in working with a UART is to configure the device so it knows what kind of line speed to use (baud rate) and the byte size/parity/stop bits configuration that
you're going to use. You must match both ends of the serial link in these settings, and in my case I was going to use a dumb terminal (initially a DEC VT240, then later a
Linux laptop running minicom), which I planned on connecting to at 9600bps, 8N1 (8-bit byte, No parity, 1 stop bit).
The UART takes a clock source which you then must convert down to your line speed with a series of divisions - firstly, the UART runs a communication cycle every sixteen
clock ticks, so right there you divide by 16. You then must divide this again by your divisor number to get to your targeted baud rate.
In my case, I was use a 16mHz oscillator.
16mHz / 16 = 1mHz
Now I just needed to get 1mHz down to ~9600bps - it doesn't have to be exact because this is supposed to be an asynchronous protocol, but it does need to be vaguely close
1mHz / 104 = 9615.4bps
And, lo, we have our divisor - 104.

The UART's line settings are configured using three registers - the DLL (low byte of the divisor), DLM (high byte of the divisor) and LCR (line control register). However there's
a catch - the DLL and DLM are passed through the first two registers in the UART, the Transmit/Receive Register and the Interrupt Enable Register. To actually talk to the DLL/DLM
you need to latch it in - using the Divisor Latch Enable, or DLE. So first we have to set the DLE bit (bit 7) of the IER to a binary 1, then load our DLL and DLM. When we're
done with setting the baud rate, we flip the DLE back off (bit 7 of the IER to binary 0).
But before we do, we need to set the DLL and DLM to 104. 104 decimal is $0068 in hexadecimal, so the DLL needs to be set to $68 and the DLM to $00.
Finally, the DLE is set to 8N1 - this involves a bitmask, which is defined by the manufacturer, although I imagine most UART DLE registers follow the same mask, in this case
00000011

This code chunk looks like this:

  LD A,10000000b	; Bitmask to set the DLE to 1
  OUT ($03),A		; Write the mask to the LCR (register $03)
 
  LD A,$68		; 104
  OUT ($00),A		; Write 104 to the DLL
  LD A,$00		; 00
  OUT ($01),A		; Write 00 to the DLM - thus giving us a final divisor of $0068, or 104

  LD A,00000011b	; Bitmask to set DLE back to 0, and configure the LCR for 8, N, 1
  OUT ($03),A		; Write to the LCR
  

GPIO pins
Now we're getting somewhere. The UART also has a pair of general-purpose I/O pins. These are +5v pins that can be turned on and off by flipping a bit on the UART's Modem
Control Register, or MCR. For my first board I used the second GPIO pin, /OUT2, to tell me the system was up, and the other, /OUT1, toggled each time it attempted to write
a byte to the transmit buffer - like a heartbeat light, telling me it was trying to do something. I used an OR to ensure the /OUT2 pin was on, and an XOR to toggle
the /OUT1 bit in the MCR, like so:

  IN A,($04)		; Read the MCR into the Accumulator
  OR 00001000b		; Make sure the /OUT2 GPIO pin in the bitmask is on
  OUT ($04),A		; Write the bitmask back out to the MCR, enabling our change

  ; Or, for /OUT1, toggle the bit in the mask:
  IN A,($04)
  XOR 00000100b
  OUT ($04),A
  

Writing to the terminal
So we're making good progress. The next part to work out is how to write a byte out through the UART to the terminal on the other side of the serial cable. We do this by sending
a byte to the Transmit Holding Register and then polling the Line Status Register to verify the byte has cleared, that way we're not pummelling the UART with a stream of bytes
before it has the ability to transmit them - the THR is only a single byte in size, so it is possible to overwrite it if we were sending things to it fast enough.
Now, this being said, the "right" way to do this is to use an interrupt - that way we can continue processing other things and pump bytes to the UART when it's ready. In my
case, there's nothing else to do, so polling it suits me just fine.

  LD C,$00		; Load the THR address into the B register, $00
  LD B,"A"		; Put the ASCII character 'A' in the B register
  OUT (C),B		; Push the contents of the B register ("A") out to the address in the C register ($00)
Loop:
  IN A,($05)		; Read the LSR
  BIT 6,A		; Check bit 6 on the byte read from the LSR
  JP Z,Loop		; If zero, loop
  JP SentByte		; Elsewise, go somewhere else
  

Now you may be asking why I used C to store the THR address and then had to do an OUT (C),B - this is because you have to use an index register to talk to an I/O
device unless you're using the Accumulator. So why didn't I use the Accumulator? Because the only register you can operate compares on is the A register, and I wanted
to keep state on what character I'd sent because in my final programme, I run through the alphabet - so I need to store the sent letter somewhere else.

Putting it all together
So, finally, here is the full chunk of code.

  IN A,($04)		; Read the MCR into the Accumulator
  OR 000010000b		; Make sure the /OUT2 GPIO pin in the bitmask is on
  OUT ($04),A		; Write the bitmask back out to the MCR, enabling our change

  LD A,10000000b	; Bitmask to set the DLE to 1
  OUT ($03),A		; Write the mask to the LCR (register $03)
 
  LD A,$68		; 104
  OUT ($00),A		; Write 104 to the DLL
  LD A,$00		; 00
  OUT ($01),A		; Write 00 to the DLM - thus giving us a final divisor of $0068, or 104

  LD A,00000011b	; Bitmask to set DLE back to 0, and configure the LCR for 8, N, 1
  OUT ($03),A		; Write to the LCR

  LD C,$00		; Put the THR address in the C register
  LD B,'A'		; Load the ASCII character 'A' in the B register

Output:
  IN A,($04)		; Read the MCR
  XOR 00000100b		; Toggle the /OUT1 pin
  OUT ($04),A		; Write back to the MCR

  OUT (C),B		; Push the contents of the B register ("A") out to the address in the C register ($00)
  INC B			; Increment the letter in the B register
  LD A,'['		; The character after Z in the ASCII character map is the left square bracket, load this into A
  CP B			; Compare to B
  JP Z,ResetByte	; If they are the same (B is also '['), jump to ResetByte

Loop:
  IN A,($05)		; Read the LSR
  BIT 6,A		; Check bit 6 on the byte read from the LSR
  JP Z,Loop		; If zero, loop
  JP Output		; Elsewise, go back to outputting the next character

ResetByte:
  LD B,'A'		; Put an ASCII 'A' back into B
  JP Loop		; Jump back to the LSR polling loop  
  

One final thing to note - I'm using jumps here a lot and no call/return subroutines. To do a call/return you need to pop the return address onto the stack and in my ROM-only
system there is no stack. The programme is simple enough that this isn't a big deal.