#### Notes on the Altos 586 System Board

Work in progress, May 2022 by Lubomir Rintel

I've recently obtained the system board from the Altos 586 computer. Unfortunately, it came without documentation, the hard drive or the controller board.

With the goal of putting the system to some use I've decided to examine the hardware and firmware, learn how the system is put together and perhaps eventually substitute the missing parts.

This document contains some notes from my investigation, provided in hope of being useful to other owners of Altos 586 systems.

#### 1. Altos 586 vs. 586T

The Altos 586 system seems to be largely similar to Altos 586T. The "T" in the latter system's designation seems to stand for "Tape", as, unlike the 586, 586T includes a tape drive.

There seems to be some documentation available for the Altos 586T system. While there's no service manuals or schematics, there's a reference manual that describes many aspects of the computer architecture and firmware itnerfaces.

## 2. Controller Board

I know little about the board, since I don't own one. The board used in 586 seems to be vastly different from the controller board used in the Altos 586T.

The 586T board includes a Floppy and Tape controllers and Z80 processor to handle the I/O. On the other hand, the controller board for the 586 seems to be based around the 8089 I/O controller and lack a tape or floppy interface (the latter is on the 586 system board).

 $<sup>^{\</sup>rm 1}$ 586T/986T System Reference Manual, P/N 690-15813-002, April 1985

## 3. Z80 Peripheral Controller

As on 586T, the peripheral controller is a Z80 controlling a bunch of serial interfaces, timers for baud rate generators and a RTC. It's equipped with a PIO for generating utility signal lines.

On a 586 it also has an extra duty of controlling the floppy drives, described below.

#### 3.1. Peripherals

The peripherals are mostly the same as in the 586T machine aside from the addition of the Floppy Controller and the related circuitry.

Consult the 586T/986T System Reference Manual for description of the peripherals common to 586 and 586T.

#### 3.2. I/O Port Map

| Address   | Name  | Part                        | Description                      |
|-----------|-------|-----------------------------|----------------------------------|
| 00h       |       |                             | Bus Address Top Bits Latch       |
| 20h – 23h | PIT 0 | Intel 8254                  | Programmable Interval Timer      |
| 24h – 24h | PIT 1 | Intel 8254                  | Programmable Interval Timer      |
| 28h – 2Bh | SIO 0 | Zilog Z8440APS (Z80A SIO/0) | Serial I/O Controller            |
| 2Ch – 2Fh | SIO 1 | Zilog Z8440APS (Z80A SIO/0) | Serial I/O Controller            |
| 30h – 33h | SIO 2 | Zilog Z8440APS (Z80A SIO/0) | Serial I/O Controller            |
| 34h – 37h | PIO   | Zilog Z8420APS (Z80A PIO)   | Parallel I/O Controller          |
| 38h – 3Bh | FDC   | Western Digital FDC1797     | Floppy Disk Controller/Formatter |
| 3Ch       | DMA   | Zilog Z8410APS (Z80A DMA)   | Direct Memory Access Controller  |
| 40h       |       |                             | DMA Carrier/Parity               |
| 80h – 9Fh | RTC   | MM58167AN                   | Microprocessor Real Time Clock   |

#### 3.3. Floppy Disk Controller/Formatter

The floppy disk controller is the FDC1797 part hooked on in a straightforward manner. The output control signals are interfaced to a standard Shuggart floppy connector via open-collector drivers, the input signals go through a line drivers with external pull up register array. The read data signal is passed through external data separator circuitry to generate clock and data from MFM encoded signal.

Some of the FDC lines involved in the Floppy Drive access are connected to the PIO. In particular, the #DDEN pin needs to be set low in order to read double-density (MFM) disks, which is almost certainly the case. The INTRQ pin is connected to PIO's PA7 input pin. No idea if it's actually used.

To actually access the floppy drive, the appropriate *Drive Select* signal needs to be pulled low. For reading double-sided floppies, the drive head/side is selected by the *Side Select* signal. The *Drive Select* and *Side Select* signals are also both controlled by the PIO.

For the details about the afforementioned control lines consult the paragraph on the Parallel I/O Controller

below.

# 3.4. Parallel I/O Controller

For the most part, the pins I've traced out are related to the floppy circuitry. Others are unknown.

| #  | Pin | Direction | Description                                          |
|----|-----|-----------|------------------------------------------------------|
| 15 | PA0 |           |                                                      |
| 14 | PA1 |           |                                                      |
| 13 | PA2 |           |                                                      |
| 12 | PA3 |           |                                                      |
| 10 | PA4 |           |                                                      |
| 9  | PA5 |           |                                                      |
| 8  | PA6 |           |                                                      |
| 7  | PA7 | Input     | FDC INTRQ (Pin 39)                                   |
| 27 | PB0 | Output    | "586 sync" sub-board (Pin 7)                         |
| 28 | PB1 | Output    | "586 sync" sub-board (Pin 8)                         |
| 29 | PB2 | Input     | Goes into unpopulated connector next to FDD (Pin 10) |
| 30 | PB3 | Output    | Floppy Drive Select 0                                |
| 31 | PB4 | Output    | Floppy Drive Select 1                                |
| 32 | PB5 | Output    | Floppy Side Select                                   |
| 33 | PB6 | Output    | FDC #DDEN (Pin 37)                                   |
| 34 | PB7 | Input     | PIT ? OUT2 (Pin 17)                                  |

#### 4. Firmware Interface

The firmware interface is unsurprisingly very similar to 586T. A notable difference is addition of the Floppy Channel, described below.

The 586T/986T System Reference Manual is not too clear about the offset of various register blocks though they can be for the most part inferred from their respective lengths, since they immediately follow each other.

The layout of register blocks on the 586 is somewhat different due to the inclusion of the *Floppy Drive Interface Registers* block before the *Time of Day Registers*. offsets of each block. It is incidentally also the offset of each command register the firmware looks at, since each block starts with a command register.

| Offset | Description                              |
|--------|------------------------------------------|
| 0Ah    | Communication Channel Registers (Port 1) |
| 20h    | Communication Channel Registers (Port 2) |
| 36h    | Communication Channel Registers (Port 3) |
| 4Ch    | Communication Channel Registers (Port 4) |
| 62h    | Communication Channel Registers (Port 5) |
| 78h    | Communication Channel Registers (Port 6) |
| 8Ah    | Floppy Drive Interface Registers         |
| D4h    | Time of Day Registers                    |

The Communication Channel Registers and Time of Day Registers are described in the 586T/986T System Reference Manual, respectively. The rest of this section deals with the Floppy interface exclusively.

## 4.1. Floppy Drive Interface Registers

This structure is present at offset 8Ah from the command control block (CCB). Not all members are used by all commands.

| Offset      | Size     | Description                  | Typical Value                   |
|-------------|----------|------------------------------|---------------------------------|
| +00h        | BYTE     | Command                      | 87h, 88h,                       |
| +01h        | BYTE     | Status                       | 00h, 48h, 40h, C0h,             |
| +02h - +05h | DWORD    | Floppy Command Queue Address | Pointer to Floppy Command Queue |
| +06h - +06h | WORD     | Floppy Command Count         |                                 |
| +08h        | BYTE     | Unknown/Reserved             | 00h                             |
| +09h        | BYTE     | Unknown/Reserved             | 00h                             |
| +0Ah - +B3h | BYTE[32] | Floppy 1 Parameters          | Floppy Parameters               |
| +B4h - +D3h | BYTE[32] | Floppy 2 Parameters          | Floppy Parameters               |

## **4.2. Floppy Drive Interface Commands**

There are two known commands:

| Command | Description                 |
|---------|-----------------------------|
| 87h     | Set Floppy Drive Parameters |
| 88h     | Submit Floppy Command Queue |

As with the other command blocks, the most significant bit indicates to the peripheral controller that there's a new command to process. Once the processing is finished, the peripheral controller sets the bit to zero.

The Set Floppy Drive Parameters command reads the Floppy Parameters structure (see below) and saves it in the controller's SRAM. The other registers in the block are ignored. This command needs to be called to set the sector size before any Floppy I/O. Failure to set the sector size up will cause the read commands to overflow a data buffer, corrupting the controller SRAM. The main processor firmware calls this command on startup.

The Submit Floppy Command Queue command enqueues I/O commands for processing. The commands are described by the Floppy Disk Interface Parameter Block structure (described below). They are are pointed to from a Floppy Command Queue (also described below). This command doesn't use the Floppy Parameters in the register block, though it requires that the parameters are set by the Set Floppy Drive Parameters previously.

## 4.3. Floppy Parameters

Used to issue I/O commands, not used by command.

| Offset      | Size   | Description          | Typical Value |
|-------------|--------|----------------------|---------------|
| +00h        | WORD   | Unknown              | 0200h         |
| +02h        | WORD   | Sector size          | 0200h         |
| +04h        | BYTE   | Unknown              | 06h           |
| +05h - +1fh | BYTE[] | Unknown/Filler (00h) | 00h           |

## 4.4. Floppy Command Queue

| Offset                                                                  | Size  | Description                     | Typical Value                   |
|-------------------------------------------------------------------------|-------|---------------------------------|---------------------------------|
| +00h DWORD Floppy Command Block Address Pointer to Floppy Command Block |       | Pointer to Floppy Command Block |                                 |
| +04h                                                                    | DWORD | Floppy Command Block Address    | Pointer to Floppy Command Block |
|                                                                         |       |                                 |                                 |
| +04h * n                                                                | DWORD | Floppy Command Block Address    | Pointer to Floppy Command Block |

## 4.5. Floppy Disk Interface Parameter Block

| Offset      | Size  | Description | Typical Value |
|-------------|-------|-------------|---------------|
| +00h        | WORD  | Command     | 10h, 20h,     |
| +01h        | WORD  | Status      |               |
| +02h        | WORD  | Unknown     | 00h           |
| +03h        | WORD  | Track       | 00h - 28h     |
| +04h        | WORD  | Head        | 00h, 01h      |
| +05h        | WORD  | Sector      | 01h - 09h     |
| +06h - +09h | DWORD | Data Buffer |               |

## 5. Reading a Floppy from the Firmware ROM Monitor

This section provides a concrete example of using the Floppy Interface of the peripheral controller.

```
FAILED POWER-UP TEST B Controller board error. I got no board.

Monitor Version a2.2

Enter [1] to boot from Hard Disk

Enter [2] to boot from Floppy Disk

Enter [3] to enter Monitor

Enter option: 3
```

First let's find the Channel Control Block address (CCB). Read it from 1FFFCh:

It's 416h. The channel register offsets are calculated from this address. The new command register is at offset 05h from the beginning of the CCB, therefore at address 41Bh.

Now let's prepare a small program that just bumps the command counter so that the controller firmware will know it needs to check for new commands. It then invokes to return to the monitor. We'll place it at address 2000h, which is an arbitrarily chosen address in the DRAM.

It's not strictly necessary here, since each key press results in a serial command and the command registers are checked anyway, but would be necessary for floppy communication outside the monitor.

We can proceed setting the floppy parameters. This is also not strictly necessary if we're using the ROM monitor, because the firmware must have initialized the parameters for us.

Let's do it anyway though, for demonstration purposes. It might also be useful if the floppy format uses a sector of size other than 512 bytes.

The floppy command register is at offset 8Ah from the beginning of the CCB, therefore at address 4A0h.

For reasons explained below, we start setting the bytes from 4A1h and set the command byte (4A0h) last.

```
0000:04A9 00-00
  0000:04AA 00-00
                              Floppy 1 parameters
  0000:04AB 02-02
  0000:04AC 00-00
  0000:04AD 02-02
  0000:04AE 06-06
  0000:04AF 00-00
  0000:04C9 00-00
  0000:04CA 00-00
                              Floppy 2 parameters
  0000:04CB 02-02
  0000:04CC 00-00
  0000:04CD 02-02
  0000:04CE 06-06
  0000:04CF 00-00
  0000:04E9 00-00
  0000:04EA 11-,
< A, B, D, G, I, K, L, M, O, R, S, X > A 0:04A0
  0000:04A0 00-87
                              The "Floppy Parameters" command byte
  0000:04A1 00-,
```

We set the command byte last to make sure the controller firmware doesn't see a valid command until the parameters are all set.

Now we can bump the command counter with the program we entered in at the beginning (though, as stated above, this is not really necessary):

Let's enter some actual floppy I/O requests now.

Start with a seek:

Seek to a different track, so that we're sure we see the head move:

| 0000:2203 | 00- <b>25</b> | Track  |
|-----------|---------------|--------|
| 0000:2204 | 00-01         | Head   |
| 0000:2205 | 00-05         | Sector |
| 0000:2206 | 00-,          |        |

And then read a sector. The controller will seek back to the appropriate track:

```
< A, B, D, G, I, K, L, M, O, R, S, X > A 0:2300
 0000:2300 00-10
                               Read command
 0000:2301 00-00
                               Status
 0000:2302 00-00
                               Unknown, 00h
 0000:2303 00-05
                               Track
 0000:2304 00-01
                               Head
 0000:2305 00-05
                               Sector
 0000:2306 00-00
                               Destination buffer address: 00002500h
 0000:2307 00-25
 0000:2308 00-00
 0000:2309 00-00
 0000:230A 00-,
```

Create a queue with the three commands we created above:

```
< A, B, D, G, I, K, L, M, O, R, S, X > A 0:2400
  0000:2400 00-00
                                First command at 00002100h
  0000:2401 00-21
                                (Seek to track 5)
  0000:2402 00-00
  0000:2403 00-00
  0000:2404 00-00
                                First command at 00002200h
  0000:2405 00-22
                                (Seek to track 25h)
  0000:2406 00-00
  0000:2407 00-00
  0000:2408 00-00
                                First command at 00002300h
  0000:2409 00-23
                                (Read from CHS 5/1/5 to 2500h)
  0000:240A 00-00
  0000:240B 00-00
  0000:240C 00-,
```

Now invoke the queue the same way as we set the parameters:

```
< A, B, D, G, I, K, L, M, O, R, S, X > A 0:04A1
 0000:04A1 00-ff
                               Status
 0000:04A2 60-00
                               The queue is at 00002400h
 0000:04A3 09-24
 0000:04A4 00-00
 0000:04A5 02-00
 0000:04A6 00-03
                               There are 0003h commands in the queue
 0000:04A7
            00-00
 0000:04A8
            00-00
 0000:04A9 00-00
 0000:04AA 00-,
< A, B, D, G, I, K, L, M, O, R, S, X > A 0:04A0
 0000:04A0 00-c8
                             Now set the command byte
 0000:04A1 48-,
                              to process the queue
```

< A, B, D, G, I, K, L, M, O, R, S, X >

```
< A, B, D, G, I, K, L, M, O, R, S, X > G 0:2000
Break ....
CS:IP FC00:0008 Flags - - - S - A - -
      BX CX DX SI DI DS
                                         ES
                                               SS
 The command is in. Let's keep checking the status byte until it indicates success:
< A, B, D, G, I, K, L, M, O, R, S, X > D 0:04A1 1
 0000:04A1
             48
                                                *.H.$.....*
The status is now 48h. We need it to be 40h. Try some more:
< A, B, D, G, I, K, L, M, O, R, S, X > D 0:04A1 1
                                                *.@.$....*
 0000:04A1
Now the status is 40h, which indicates no error.
The sector data is now at 2500h, let's check it out:
< A, B, D, G, I, K, L, M, O, R, S, X > D 0:2500 200
 0000:2500 EB080000 30000000 0002CCEB FD000000 *k...0....lk}...*
 0000:26F0 00000000 00000000 00000000 *.....*
```

## 5.1. Z80 ROM Monitor

The factory firmware seems to include a ROM Monitor software. Having a firmware monitor is very helpful for investigation of the system. It would allow controlling the Z80 peripherals directly and operating them without the main 8086 processor.

Unfortunately, I didn't discover a way to enter the factory ROM monitor. I've ended up temporarily replacing the ROM with a different one, allowing me to poke around the controller. That way I've lost the way of utilizing the factory firmware on the main processor, which would just end up being stuck in a tight loop.