Harddisk Controller RLV12/Hack

Disk Controllers

DEC made a lot of different disk controllers for the UNIBUS system but only a few for the Q-Bus system and of them only a few support 22-bit addressing, which with the exception of the RLV12 are all MSCP controllers.

The MSCP controllers make use of a very sophisticated software protocol and virtually support any disk size. The RLV12 on the other hand uses as simple device register interface with only 5 valid registers but supports only RL01 or RL02 drives with a very limited capacity of 5Mbyte or 10Mbyte respectively.

RLV12 Emulation

Looking at the users guide it was clear that building a RLV12 controller would be much less effort than a MSCP controller. And after I found that Mentec again added support for RL02 in their latest release for RSX-11MPlus I decided to go for the RLV12 emulation. This is a very simple controller that has only 5 device registers and only implements 8 disk control functions.

Later I have adopted the design to the real Q-Bus, so I have now a RLV12 Emulator for the PDP-11/Hack and the real Q-Bus PDP-11 systems.

A description of the RLV12 emulators can be found here.

Schematic

Note the following are the schematics and boards as they are used in the prototype currently used in the PDP-11/Hack system. However there are some modifications required. First I forgot to add decoupling capacitors with 10µF to buffer the high current peak when inserting an SD-Card. This can cause a reset of the whole system. I added them on the fly to the board.

Also the Hack version connected a different pin of the microcontroller to the CLK1 input of the CPLD than the Q-Bus version. This has been ECO’d and now the same pin connects to CLK1 input on both RLV12 versions to have the same microcontroller program in both versions.

The change swaps pins 83 and 84 on the CPLD (cut the two traces and install two new wires)

The PDP-11 Hack

The PDP-11 Hack

The PDP-11 Hack

CPLD Design

Name     Qbridge11-Mini ;
PartNo   01 ;
Date     05.12.2018 ;
Revision 01 ;
Designer cbscpe ;
Company  PRIVAT ;
Assembly None ;
Location ATHOME ;
Device   f1508ispplcc84 ;

/*
    2018-12-05  Based on the working RVL12 Emulator (See Qbridge11, v1) now
                with DMA implemented as state machine. The interface towards
                the MCU consists now only of 2 pins. A DMR (DMA Request) from
                the MCU and the DMG (DMA Grant) to the MCU to signal the
                progress and allows the MCU to read and write the DMA data
                using the normal data register. The DMA data register is gone.
                The Base Control Register CS1 is now implemented as an internal
                register. That is when reading CS1 then the internal register
                is immediately returned to the PDP-11 and the MCU is not interrupted.

    2018-12-12  Correct BDAL select signals, use positive logic for UB and LB, adapt
                the comments in the section to the new design, remove uneccessary PINNODE
                
    2018-12-19  New Statemachine concept

    2018-12-21  Additional cycles between SACK and SYNC and SYNC and data phase
                Bidirectional open collector signals must not be registered else WinCUPL
                will use the output of the flip-flop and not the PIN as input, sometimes
                you must even force equations

    2018-12-22  New PDP-11 interrupt logic. Now the MCU just sets IRQST when an interrupt
                is requested. This asserts BIRQ and on the next BDIN cycle the status is
                latched. In case we are the highest priority device when the interrupt
                is acknowledged by the PDP-11 the MCU is alerted via INT2. The MCU has
                then to take care of the various signals related to the interrupt. I.e.
                the MCU must de-assert IRQST and must clear the INT2 interrupt. If required
                the MCU can then also update the CS1 register. E.g. the RH70 always clears
                IE if an interrupt is acknowledged.
 */

$INCLUDE Z:\OneDrive\Dokumente\WinCupl\QBridge11-v2\Qbridge-11-v2-0-0.txt 

PROPERTY ATMEL {SOFT_BUFFER=BUSEN};
PROPERTY ATMEL {preassign KEEP};
PROPERTY ATMEL {open_collector=BRPLY, BDMR, BSACK, BDOUT, BDIN, BSYNC, BRPLY, BWTBT};

PINNODE         =  [D15..0]      ; /* Data Output Register                                  */
PINNODE         =  [DD15..0]     ; /* DMA Data and Interrupt Vector Register                */
PINNODE         =  [A21..1]      ; /* DMA Address                                           */
PINNODE         =  [C5..0]       ; /* IO  Address                                           */
PINNODE         =  CSRADDR       ; /* CSR Address                                           */
PINNODE         =  CSR           ; /* CSR Address                                           */
PINNODE         =  CS1ADDR       ; /* CS1 Address                                           */
PINNODE         =  CS1           ; /* CS1 Address                                           */
PINNODE         =  [DMA4..0]     ; /*                                                       */
PINNODE         =  SYNC          ; /*                                                       */
PINNODE         =  DMGI          ; /*                                                       */
PINNODE         =  SACK          ; /*                                                       */
PINNODE         =  DATA          ; /*                                                       */
PINNODE         =  RPLY          ; /*                                                       */
PINNODE         =  DMARW         ; /*  Control bit: DMA read(1) or DMA write(0)             */
PINNODE         =  BUSEN         ; /*                                                       */
PINNODE         =  DIRQ          ; /* Device Interrupt Request                              */
PINNODE         =  DACK          ; /* Device Interrupt Acknowledge                          */
PINNODE         =  TDOUT;
PINNODE         =  TSYNC;
PINNODE         =  TDIN;


/*
    Define one of the possible emulations: RLV12, RH70
 */
$DEFINE RLV12

/*
    Field definitions
 */
FIELD  ADDR     = [C5..0];      /* Latched address used for register address                */
FIELD  RS       = [RS2..0];     /* Register Select Address for Atmega 162 reads/writes      */
FIELD  IOADDR   = [BDAL12..0];  /* Access to IO Page only requires address lines 0..12      */
FIELD  DMA      = [DMA4..0];    /* DMA State Machine                                        */
/*
    Address bits BDAL0..5 are latched for every bus cycle and made available to the MCU
    to identify the selected device register
 */
ADDR.d        =  [BDAL5..0].io;
ADDR.ck       =  BSYNC.io;

/*
    BINIT is propagated to the MCU via the INIT signal. This is a normal MCU input so
    we can use the pin change interrupt to take actions for both edges of BINIT.
 */
INIT          =  BINIT.io;

/*
    CSRADDR and CS1ADDR are asserted whenever the bus addresses a CSR or the first CSR
    respectively. These signals are used to latch the selected states into CSR and CS1.

    RLV12 addresses
 */
$IFDEF RLV12
CSRADDR       =  IOADDR.io:['o'174400..174417];
CS1ADDR       =  IOADDR.io:['o'174400..174401];
$ENDIF

/*
    RH70 addresses
 */
$IFDEF RH70
CSRADDR       =  IOADDR.io:['o'176702..176757];
CS1ADDR       =  IOADDR.io:['o'176700..176701];
$ENDIF

CSR.ck        =  BSYNC.io;
CSR.d         =               BBS7.io &  CSRADDR;

CS1.ck        =  BSYNC.io;
CS1.d         =               BBS7.io &  CS1ADDR;
/*
    DATI to a device register, except for CS1, triggers the INT1 interrupt. The first
    device register is implemented as local register and bus reads to this register 
    are handeled in the CPLD. This allows us to always let the PDP-11 interrupt the MCU
    to support overlapped processing of register IO and data transfer/seek functions.
    CS1 is implemented as local register in order to avoid the MCU beeing permanently
    interrupted when the PDP-11 polls the controller instead of using interrupts.
    This is mainly the case during the boot phase.

    When the MCU has finished the processing the device register access it must create
    a pulse to IACK to acknowledge the interrupt and the CPLD then asserts BRPLY.
 */
INT1.d        = !(!BWTBT.io &  BBS7.io &  CSRADDR & !CS1ADDR);
INT1.ck       =  BSYNC.io;
INT1.ap       =  IACK;

/*
    DATO to a device register triggers INT0 interrupt.
 */
INT0.d        =  !(BWTBT.io &  BBS7.io &  CSRADDR);
INT0.ck       =  BSYNC.io;
INT0.ap       =  IACK;

BRPLY         =  CSR &  BSYNC.io &  INT1 &  BDIN.io       /* Assert RPLY when MCU is done   */
              #  CSR &  BSYNC.io &  INT0 &  BDOUT.io;     /* Assert RPLY when MCU is done   */

/*
    The interrupt logic is now fully handled in software so we can also update CS1 if needed.
    The CPU sets IRQST whenever an interrupt is required. This asserts !BIRQ on the QBUS. When
    the PDP-11 acknowledges the interrupt first DACK will be set which causes INT2 to be reset
    when !BIAKI is asserted. In the INT2_isr of the MCU the first action will be to deassert
    IRQST and then eventually update CS1. Then it will have to set INT2 in the CPLD with a 
    short pulse on IACK which then asserts BRPLY. 

           +----------------------------+
IRQST      |                            |
        ---+                            +------------------------------------------------

        ---+                            +------------------------------------------------
!BIRQ      |                            |
           +----------------------------+

        ----------+                                           +--------------------------
!BDIN             |                                           |
                  +-------------------------------------------+

    DACK This reflects always IRQST latched with BDIN as on all devices and therefore is
    only de-asserted on next falling edge of !BDIN
                   +---------------------------------------------------------------------
DACK               | 
        -----------+

        -----------------+                                    +--------------------------
!BIAKI                   |                                    |
                         +------------------------------------+
                         
                                             +---+
IACK                                         |   |
        -------------------------------------+   +---------------------------------------

        -----------------+                    +------------------------------------------
INT2                     |                    |
                         +--------------------+

        ---------------------------------------+               +-------------------------   
!BRPLY                                         |               |
                                               +---------------+

 */
DACK.d        =  IRQST;
DACK.ck       = !DMA3     &  BDIN.io;
DACK.ar       =  BINIT.io;

BIAKO         =  BIAKI    & !DACK;
BIRQ          =  IRQST;

INT2.d        = !DACK;
INT2.ck       =  BIAKI;
INT2.ap       =  IACK;

APPEND
BRPLY         =  BIAKI & !IRQST &  INT2 & BDIN.io;

/*
    Q-Bus Data Register

 */
[D7..0].d     =  [AD7..0].io;
[D15..8].d    =  [AD7..0].io;
[D15..0].ck   =  WR;
[D7..0].ce    =  RS:['d'0];
[D15..8].ce   =  RS:['d'1];


/*
    Control and Status Register
 */
[DD7..0].d    =  [AD7..0].io;
[DD15..8].d   =  [AD7..0].io;
[DD15..0].ck  =  WR;
[DD7..0].ce   =  RS:['d'2];
[DD15..8].ce  =  RS:['d'3];


/*
    DMA Address
 */
DMARW.ar      =  WR  &  RS:['d'4] & !AD0.io;
DMARW.ap      =  WR  &  RS:['d'4] &  AD0.io;
DMARW.ck      = 'b'0;
DMARW.d       = 'b'0;

[A7..1].ar    =  WR  &  RS:['d'4] & ![AD7..1].io;
[A7..1].ap    =  WR  &  RS:['d'4] &  [AD7..1].io;

[A15..8].ar   =  WR  &  RS:['d'5] & ![AD7..0].io;
[A15..8].ap   =  WR  &  RS:['d'5] &  [AD7..0].io;

[A21..16].ar  =  WR  &  RS:['d'7] & ![AD5..0].io;
[A21..16].ap  =  WR  &  RS:['d'7] &  [AD5..0].io;

/*
    The MCU can read the 16-bit Q-Bus data using the two register address 0 and 1 for
    the even and odd byte respectively. Register address 4 contains the latched CSR
    offset together with two bits that indicate whether the upper or lower byte are
    written. 

 */
UB            = !BWTBT.io
              #  BWTBT.io  &  C0;
LB            = !BWTBT.io
              #  BWTBT.io  & !C0;

[AD7..0]      =  RS:['d'0] &  [BDAL7..0].io 
              #  RS:['d'1] &  [BDAL15..8].io
              #  RS:['d'4] &  [UB, LB, C0, C5, C4, C3, C2, C1];
/*
              #  RS:['d'4] &  [A7..1, DMARW] 
              #  RS:['d'5] &  [A15..8]
              #  RS:['d'7] &  ['b'0, 'b'0, A21..16];
 */
[AD7..0].oe   =  RD;


/*
    Synchronous counter for DMA address which is updated after every DMA cycle. The address
    is incremented whenever a DMA cycle has finished.
 */ 
[A21..1].ck   = !DMA3;

A1.t          = 'b'1;
A2.t          = A1;
A3.t          = A1 & A2;
A4.t          = A1 & A2 & A3;
A5.t          = A1 & A2 & A3 & A4;
A6.t          = A1 & A2 & A3 & A4 & A5;
A7.t          = A1 & A2 & A3 & A4 & A5 & A6;
A8.t          = A1 & A2 & A3 & A4 & A5 & A6 & A7;
A9.t          = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8;
A10.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9;
A11.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10;
A12.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11;
A13.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12;
A14.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13;
A15.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14;
A16.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14 & A15;
A17.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14 & A15 & A16;
A18.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14 & A15 & A16 & A17;
A19.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14 & A15 & A16 & A17 & A18;
A20.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14 & A15 & A16 & A17 & A18 & A19;
A21.t         = A1 & A2 & A3 & A4 & A5 & A6 & A7 & A8 & A9 & A10 & A11 & A12 & A13 & A14 & A15 & A16 & A17 & A18 & A19 & A20;


/*
    Q-Bus Output Select and Output Enable
    
    DMA4    DMA Read is active
    DMA3    DMA is active
    DMA2    DMA data transfer
 */
[BDAL15..0]   = !BIAKI & !DMA3  &  CS1   &  [DD15..0]
              # !BIAKI & !DMA3  & !CS1   &  [D15..0]
              # !BIAKI &  DMA3  & !DMA2  &  [A15..1, 'b'0]
              # !BIAKI &  DMA3  &  DMA2  &  [D15..0]
$IFDEF RLV12
              #  BIAKI                   &  [D15..0];
$ENDIF
$IFDEF RH70
              #  BIAKI                   &  'o'160;
              #  BIAKI                   &  'o'254;
$ENDIF

BUSEN         = !BIAKI          & !DMA3  &  CSR   &  BDIN.io &  BSYNC.io
              #  BIAKI          & !DMA3  &  DACK  &  BDIN.io
              # !BIAKI & !DMA4  &  DMA3                      /* 0x08..0x0F DMA Write   */
              # !BIAKI &  DMA4  &  DMA3  & !DMA2;            /* 0x18..0x1B DMA Read    */

/*
    The upper address bits are just a copy of the DMA address.
 */
[BDAL21..16]  =  [A21..16];
[BDAL21..0].oe=  BUSEN;
/*
    Partially CPLD controlled DMA state machine
 */

 /*
    The rising edge of DMR from the MCU triggers BDMR of the Q-Bus when
    the DMA is granted BDMR is cleared but DMR will be kept asserted by
    the MCU until the MCU has finished the DMA. Used to control the state
    machine.
  */
BDMR.d        = 'b'0;
BDMR.ck       =  DMR;
BDMR.ap       =  DMA3 # BINIT # ABORT;
 
TSYNC.ck      =  CLK;
TDOUT.ck      =  CLK;
TDIN.ck       =  CLK;
/*
    Synchronous Signals
 */
DMGI.ck       =  CLK;
DMA.ck        =  CLK;
RPLY.ck       =  CLK;
DMG.ck        =  CLK;
 
DMA.ar        =  BINIT.io # ABORT;
/*
    Latched BDMGI and BRPLY as we need a stable DMGI and RPLY for the state machine
 */
DMGI.d        =  BDMGI;
RPLY.d        =  BRPLY.io;


/*
    New DMA state machine.
    
    0. If signals are simultanuously configured as open collector, register and
    are used bidirectional WinCUPL unfortunately does not take the .PIN when we
    refer to .io extension. Therefore we cannot use the statemachine signals as
    open collector IO pins. Therefore we need additional signals TSYNC, TDOUT
    and TDIN.
    1. This state machine interacts directly with the MCU for the data transfer.
    First the MCU requests a DMA with a rising edge on DMR which asserts BDMR.
    When the CPU acknoweldges a DMA request it asserts BDMGI. This daisy chain
    signal must be propagated by all devices until it reaches the first that
    requested the DMA. In case we did not request the DMA (DMR is de-asserted)
    the statemachine will propagate BDMGI to BDMGO and wait until BDMGI is
    de-asserted and returns to the idle state. When we have requested the DMA
    the statemachine will assert BSACK, place the DMA address onto the data
    bus and then asserts BSYNC. 
    2. For DMA writes the state machine then asserts DMG which is polled by the
    MCU which then loads the bus data register with the DMA data. Once the data
    has been written the MCU de-asserts DMR and the statemachine will finish
    the DMA write request.
    3. For DMA reads the the state machine waits for BRPLY to be asserted and
    will then assert DMG to signal the MCU that the DMA data can be read from
    the QBUS. Once the MCU has done this it will de-assert DMR and the state
    machine will finish the DMA cycle.
    4. DMG will be deasserted when the state machine has finished the DMA cycle.
    So before the MCU starts another DMA cycle it must check DMG to be de-asserted.

    Transfer rate is approximatively 1Mbyte/second and a DMA cycle takes about
    1usec. DMA on a real PDP-11 has about twice the performance, i.e. the transfer
    rate is about 2Mbyte/second and a DMA cycle only takes about 500ns. However
    the SD-Card has no seek and rotational latency. This more then compensates it.
 */

sequence DMA {
 
   PRESENT 'h'00 IF !DMGI                NEXT 'h'00;
                 IF  DMGI                NEXT 'h'01;
    
   PRESENT 'h'01 IF !DMR                 NEXT 'h'02;
                 IF  DMR                 NEXT 'h'03;
 
   PRESENT 'h'02 IF !DMGI                NEXT 'h'00;
                 IF  DMGI OUT BDMGO;     NEXT 'h'02;
 
   PRESENT 'h'03 IF !DMARW               NEXT 'h'08;
                 IF  DMARW               NEXT 'h'18;
/*
     Write
*/
   PRESENT 'h'08                         NEXT 'h'09           OUT DMG;
  
   PRESENT 'h'09                         NEXT 'h'0A           OUT DMG;
 
   PRESENT 'h'0A                         NEXT 'h'0B OUT TSYNC OUT DMG;
 
   PRESENT 'h'0B                         NEXT 'h'0C OUT TSYNC OUT DMG;
 
   PRESENT 'h'0C IF  DMR                 NEXT 'h'0C OUT TSYNC OUT DMG;
                 IF !DMR                 NEXT 'h'0D OUT TSYNC OUT DMG OUT TDOUT;
 
   PRESENT 'h'0D IF !RPLY                NEXT 'h'0D OUT TSYNC OUT DMG OUT TDOUT;
                 IF  RPLY                NEXT 'h'0E OUT TSYNC OUT DMG OUT TDOUT;
 
   PRESENT 'h'0E IF  RPLY                NEXT 'h'0E OUT TSYNC OUT DMG;
                 IF !RPLY                NEXT 'h'0F           OUT DMG;
 
   PRESENT 'h'0F                         NEXT 'h'00;
/*
     Read
*/
   PRESENT 'h'18                         NEXT 'h'19;
  
   PRESENT 'h'19                         NEXT 'h'1A;
 
   PRESENT 'h'1A                         NEXT 'h'1B OUT TSYNC;
 
   PRESENT 'h'1B                         NEXT 'h'1C OUT TSYNC;
 
   PRESENT 'h'1C IF !RPLY                NEXT 'h'1C OUT TSYNC         OUT TDIN;
                 IF  RPLY                NEXT 'h'1D OUT TSYNC OUT DMG OUT TDIN;
 
   PRESENT 'h'1D IF  DMR                 NEXT 'h'1D OUT TSYNC OUT DMG OUT TDIN;
                 IF !DMR                 NEXT 'h'1E OUT TSYNC OUT DMG;
 
   PRESENT 'h'1E IF  RPLY                NEXT 'h'1E OUT TSYNC OUT DMG;
                 IF !RPLY                NEXT 'h'1F           OUT DMG;
 
   PRESENT 'h'1F                         NEXT 'h'00;
}

BSACK         =          DMA3;
BWTBT         = !DMA4 &  DMA3 & !DMA2 & !DMA1;
BSYNC         =  DMA3 &  TSYNC;
BDOUT         =  DMA3 &  TDOUT;
BDIN          =  DMA3 &  TDIN;

The pin allocation definitions already reflects the ECO (swap of IACK and WR).

/* This is the workfile to change pin allocations

    2018-12-05  Based on the working RVL12 Emulator (See Qbridge11, v1) now
                with DMA implemented as state machine. The interface towards
                the MCU consists now only of 2 pins. A DMR (DMA Request) from
                the MCU and the DMG (DMA Grant) to the MCU to signal the
                progress.
    2018-12-28  Swap RD and IACK to keep changes to the real Q-Bus version minimal,
                i.e. only need to swap IACK and WR in the Atmega1284P definitions
    2018-07-03  Swap differently, just swap WR and IACK and keep RD on a sub-optimal
                input, this results in the same pin allocation of the MCU on both
                versions. RD should in fact go to a OE input pin to save product
                terms and fan-in, however the AD multiplexor is not at it's limit
                so it does still fit the CPLD.

*/

Device   f1508ispplcc84 ;

/*************************** Control Bus *******************************************************/
PIN    79  = !BSYNC          ; /* Address Latch Enable Transmit                           QBUS */
PIN     1  = !BINIT          ; /* Master reset                                            QBUS */
PIN    16  = !BDOUT          ; /* IO Bus write                                            QBUS */
PIN    34  = !BIAKI          ; /* Interrupt acknowledge daisy chain in                    QBUS */
PIN    11  = !BIRQ           ; /* Interrupt request                                       QBUS */
PIN    25  = !BIAKO          ; /* Interrupt acknowledge daisy chain out                   QBUS */

PIN    31  =  BDMR           ; /*                                                         QBUS */
PIN    29  = !BDMGI          ; /*                                                         QBUS */
PIN    22  = !BDMGO          ; /*                                                         QBUS */
PIN    80  = !BSACK          ; /*                                                         QBUS */

PIN    12  = !BDIN           ; /* IO Bus read                                             QBUS */
PIN    27  = !BWTBT          ; /* Multiplexed Write/Byte                                  QBUS */
PIN    10  = !BBS7           ; /* Bank Select 7 i.e. registers in IO page                 QBUS */
PIN     8  = !BRPLY          ; /* Continue daisy chain in                                 QBUS */

/*************************** Address/Data Bus **************************************************/
PIN    45  =  BDAL0           ; /*                                                         QBUS */
PIN    39  =  BDAL1           ; /*                                                         QBUS */
PIN    46  =  BDAL2           ; /*                                                         QBUS */
PIN    44  =  BDAL3           ; /*                                                         QBUS */
PIN    41  =  BDAL4           ; /*                                                         QBUS */
PIN    40  =  BDAL5           ; /*                                                         QBUS */
PIN    36  =  BDAL6           ; /*                                                         QBUS */
PIN    35  =  BDAL7           ; /*                                                         QBUS */
PIN    37  =  BDAL8           ; /*                                                         QBUS */
PIN     4  =  BDAL9           ; /*                                                         QBUS */
PIN    20  =  BDAL10          ; /*                                                         QBUS */
PIN    21  =  BDAL11          ; /*                                                         QBUS */
PIN     9  =  BDAL12          ; /*                                                         QBUS */
PIN     5  =  BDAL13          ; /*                                                         QBUS */
PIN    18  =  BDAL14          ; /*                                                         QBUS */
PIN     6  =  BDAL15          ; /*                                                         QBUS */
PIN    33  =  BDAL16          ; /*                                                         QBUS */
PIN    30  =  BDAL17          ; /*                                                         QBUS */
PIN    28  =  BDAL18          ; /*                                                         QBUS */
PIN    24  =  BDAL19          ; /*                                                         QBUS */
PIN    15  =  BDAL20          ; /*                                                         QBUS */
PIN    17  =  BDAL21          ; /*                                                         QBUS */

/*************************** AVR Interface *****************************************************/
PIN    81  =  CLK            ; /* Master Clock                                     Atmega1248P */

PIN    57  =  AD0            ; /* MCU <-> CPLD Data                                Atmega1248P */
PIN    58  =  AD1            ; /* "                                                Atmega1248P */
PIN    63  =  AD2            ; /* "                                                Atmega1248P */
PIN    64  =  AD3            ; /* "                                                Atmega1248P */
PIN    65  =  AD4            ; /* "                                                Atmega1248P */
PIN    68  =  AD5            ; /* "                                                Atmega1248P */
PIN    70  =  AD6            ; /* "                                                Atmega1248P */
PIN    73  =  AD7            ; /* "                                                Atmega1248P */
PIN    54  =  INT0           ; /* DATO Interrupt                                   Atmega1248P */
PIN    74  =  INT1           ; /* DATI Interrupt                                   Atmega1248P */
PIN    50  =  RS0            ; /* Register Select 0                                Atmega1248P */
PIN    49  =  RS1            ; /* Register Select 1                                Atmega1248P */
PIN    48  =  RS2            ; /* Register Select 2                                Atmega1248P */
PIN    84  =  IACK           ; /* Acknowledge CPDL Interrupts                      Atmega1248P */
PIN    83  =  WR             ; /* Write Register                                   Atmega1248P */
PIN    77  =  RD             ; /* AD Output Enable                                 Atmega1248P */
PIN    60  =  INT2           ; /* Interrupt Acknowledge                            Atmega1248P */
PIN    55  =  IRQST          ; /* Start Interrupt Request Cycle pulse              Atmega1284P */
PIN    69  =  INIT           ; /* Propagate BINIT (Pin Change Interrupt)           Atmega1248P */
PIN    75  =  ABORT          ; /* Abort DMA cycle (bus timeout)                    Atmega1248P */ 

/*************************** DMA Signals *******************************************************/
PIN     2  =  DMR            ; /* DMA Request                                      Atmega1248P */
PIN    61  =  DMG            ; /* DMA Grant                                        Atmega1248P */

/*************************** Previous interface Pins *******************************************/
/*     69  =  RPLY           ; /* BUS Reply                                        Atmega1248P */
/*     61  =  DMGI           ; /* DMA Grant Input                                  Atmega1248P */
/*      2  =  SACK           ; /* Service Acknowledge                              Atmega1248P */
/*     83  =  DMR            ; /* DMA Request                                      Atmega1248P */

MCU Software

The PDP-11/Hack version of the RLV12 emulator uses the same code as the Q-Bus version. You can find the software in the download section and some description in the RLV12 Emulator section