Skip to content

(ASM Programming) Interrupts and interrupt handling

Stephen Horn edited this page Oct 28, 2019 · 9 revisions

Interrupts and interrupt handling

Address Description
$FFFE (ROM) Universal interrupt vector
$F583 (ROM) Kernal interrupt selector
$0316 (RAM) BRK vector
$0314 (RAM) IRQ vector

What is an interrupt?

An interrupt is a signal to the CPU to stop whatever it's doing and execute the "interrupt handler" at some address in memory. This is usually caused by a device connected to the CPU, such as the video card signalling the end of one displayed frame and the start of another, but can also represent something more direct, like a "reset" button that the user has pressed on the computer's case. They can also be triggered by software, through the brk instruction.

When something triggers an interrupt on the CPU, the CPU is allowed to finish its current instruction before beginning the interrupt handler. Once done, it pushes the program counter to the stack followed by the processor's current status bits. If the interrupt was triggered by a "break" instruction (BRK), or else was caused by a non-maskable interrupt signal, it leaves these status bits alone. For all other interrupts, the processor clears the "Break" bit from the set of status bits that are pushed onto the stack.

The processor then sets the program counter to the 16-bit address stored in $FFFE. As of r33, this address is $F583, which is in ROM.

Interrupts on the X16

Starting from $F583, the kernal pushes the contents of the A, X, and Y registers, in that order. It then reads the status bits that were pushed to the stack and checks for the "Break" flag. If that flag exists, it jumps to the 16-bit address stored at $0316 (the BRK handler), otherwise it jumps to the 16-bit address stored at $0314 (the IRQ handler).

Programs can set their own IRQ handlers by overwriting $0314 with the low byte of their handler's address, and $0315 with the high byte of their handler's address.

When done, programs have two choices: They can perform a jmp instruction to whatever address the kernal had previously written to $0314, or it can exit the interrupt process manually.

If the program wishes to jump back to the kernal's interrupt handling, it must store the values of $0314 and $0315 before overwriting them:

Default_irq_handler: !le16 $0000

preserve_default_irq:
    lda $0314
    sta Default_irq_handler
    lda $0315
    sta Default_irq_handler+1
    rts

At this point, the program can end its IRQ handler by jumping to the kernal's default handler:

custom_irq_handler:
    ; Whatever code your program
    ; wanted to execute...
    
    ; Return to kernal handling:
    jmp (Default_irq_handler)

The kernal's default handler will ensure that it properly restores all processor values before returning control to the interrupted program code.

If, on the other hand, the program would rather end the IRQ immediately and return to whatever code was interrupted, it must first restore the registers that kernal code had pushed onto the stack, followed by an rti instruction:

custom_irq_handler:
    ; Whatever code your program
    ; wanted to execute...
    
    ; Return to whatever had been interrupted:
    pla
    tay
    pla
    tax
    pla
    rti

The most common interrupt: VSYNC

By far, the most common interrupt you can expect to handle right now is the VSYNC interrupt generated by the Vera chip. This potentially adds two wrinkles to a custom IRQ handler: First, verifying whether the interrupt was generated by the VERA. Second, if the custom IRQ handler does not jump to the kernal's default IRQ handler at the end, it must clear the Vera's interrupt signal on behalf of the kernal.

Detecting the Vera's interrupt requires checking $9F27 for the flag $01, whereas clearing the Vera's interrupt requires writing $01 to $9F27.

Consider the following example where the custom IRQ handles VSYNC:

custom_irq_handler:
    lda $9F27
    and #$01
    beq irq_done

    ; Whatever code your program
    ; wanted to execute...

    lda #$01
    sta $9F27
    
    ; Return to whatever had been interrupted:
irq_done:
    pla
    tay
    pla
    tax
    pla
    rti