



# Ch.3 Input and Output



## Topics

- Introduction: input and output (I/O)
  - Gaming industry devices
- 3.1.1: I/O device – system interface
- 3.1.2: Program-controlled I/O transfer
- • 3.2: Interrupt-based I/O
  - 3.2.6: Exceptions



# 3.2 Interrupt I/O Fig. 3.6



Interrupt is external or internal?

Timer is an IO device ?

## Program 1

- Extensive computation
- Periodic display of current result

Steps: Instructions

1. An I/O device raises 1  
an interrupt while CPU  
is executing  
instruction  $i$

Interrupt  
occurs  
here

2

$i$

$i + 1$

M

## Set a Timer; raise an interrupt when timeout

Program 1

COMPUTE routine

Program 2

DISPLAY routine

## Program 2: An Interrupt Service Routine ISR

4. Executes DISPLAY ISR

2. Completes instruction  $i$

3. Saves  $PC_i+4$  & Status on Stack

6. Restores  $PC=PC_i+4$  and Status

5. Return-from-interrupt inst.

Steps 3 & 6 done by processor  
or user in assembly



# Interrupts Handling/Handshaking-1



Processor

I/O Device

1. Processor is executing a program

2. An I/O device  $T$  raises an interrupt

How?

Send a hardware-based ***Interrupt-request*** signal

3. Processor suspends current program execution

When?

After current instruction is processed

4. Processor identifies the specific interrupt  $T$

How?

By a) polling or b) from the interrupt vector

5. Processor acknowledges the interrupt has been recognized

How?

Send a hardware-based ***Interrupt-acknowledge INTA*** signal



# Interrupts Handling/Handshaking-2



Processor

I/O Device

6. I/O device  $T$  de-activates *Interrupt-request*

7. Processor saves PC and status

Why?

To allow for its next IRQ

Where? On Stack

8. Processor branches to and executes *ISR T*

How? Get ISR address from Interrupt Table

9. Processor executes ISR  $T$  ...till Return

10. Processor returns from ISR  $T$  and restores PC and status

Where? From Stack

11. Processor continues current program execution or *process next IRQ from some device*



# Disable and Enable Interrupts



- Some tasks cannot tolerate interrupt latency
  - In both main program and interrupt service routine
  - Processor must complete a task without interruption
- How?
  - Need to disable interrupts temporarily
  - Then enable interrupt again when the task is done
- Normal case:
  - **DI**: disable all interrupts
  - **EI**: enable all interrupts

Interrupt Latency ?



# DI & EI Interrupt





# Device Level: EI in I/O Control Registers



Device Level:  
EI or DI to that device only

(a) Keyboard interface

Move #2, 0x4008

Why #2 ?

#2=0000 0010



(b) Display interface

OR 0x4018, #4

Why #4 ?

#4=0000 0100



# Processor Level: EI/DI in Control Registers





# Single Interrupt: Event Sequence



1. Program sets IE (e.g., in PS to 1) to enable interrupts (assuming interrupt enabled at the other two levels: **Device** and **IENABLE**)



PS (Processor Status) register: IE bit setting affects all devices

2. When an interrupt request  $IR_x$  is recognized, processor saves program counter and status register
3. IE bit is cleared to 0 so that there is no further interruptions
4. After servicing interrupt  $IR_x$ , restores saved state, and sets IE to 1 again



## 3.2.5 Processor-Keyboard-Display



Example application:



- Keyboard Input: interrupt to read entered characters
- Echo each keyboard character to Display
- Display Output: polling within Key-ISR
- Done when carriage return (CR) is entered and detected



# Fig 3.8 Main Program



| Main program |                         |                                                               |
|--------------|-------------------------|---------------------------------------------------------------|
| START:       | Move R2, #LINE          |                                                               |
|              | Store R2, PNTR          | Initialize buffer pointer.                                    |
|              | Clear R2                |                                                               |
|              | Store R2, EOL           | Clear end-of-line indicator.                                  |
|              | Move R2, #2             |                                                               |
|              | Move R2, KBD_CONT       | Enable interrupts in the keyboard interface.                  |
|              | MoveControl R2, IENABLE |                                                               |
|              | Or R2, R2, #2           | Enable keyboard interrupts in the processor control register. |
|              | MoveControl IENABLE, R2 |                                                               |
|              | MoveControl R2, PS      |                                                               |
|              | Or R2, R2, #1           |                                                               |
|              | MoveControl PS, R2      | Set interrupt-enable bit in PS.                               |
|              | next instruction        |                                                               |

How many levels of interrupt control?

Why the OR ?

0x4008



KBD\_CONT

LINE: 1<sup>st</sup> Char



Initialize buffer pointer.

Clear end-of-line indicator.

Enable interrupts in the keyboard interface.

Enable keyboard interrupts in the processor control register.

Set interrupt-enable bit in PS.

Address Memory

PNTR  
LINE  
0



IENABLE



PS



# Fig. 3.8 Keyboard ISR



| Interrupt-service routine    |       |                     |                                                      |
|------------------------------|-------|---------------------|------------------------------------------------------|
| Initialization<br>Save Reg's | ILOC: | Subtract            | SP, SP, #8<br>Save registers.                        |
|                              |       | Store               | R2, 4(SP)                                            |
|                              |       | Store               | R3, (SP)                                             |
|                              |       | Load                | R2, PNTR<br>Load address pointer.                    |
| Input Key-Data               |       | LoadByte            | R3, KBD_DATA<br>Read character from keyboard.        |
|                              |       | StoreByte           | R3, (R2)<br>Write the character into memory          |
|                              |       | Add                 | and increment the pointer.                           |
|                              |       | Store               | R2, PNTR<br>Update the pointer in memory.            |
| Output when<br>Display ready | ECHO: | LoadByte            | R2, DISP_STATUS<br>Wait for display to become ready. |
|                              |       | And                 | R2, R2, #4                                           |
|                              |       | Branch_if_[R2]=0    | ECHO                                                 |
|                              |       | StoreByte           | R3, DISP DATA<br>Display the character just read.    |
| CR ?                         |       | Move                | R2, #CR<br>ASCII code for Carriage Return.           |
|                              |       | Branch_if_[R3]≠[R2] | RTRN<br>Return if not CR.                            |
| If CR, done                  |       | Move                | R2, #1                                               |
|                              |       | Store               | R2, EOL<br>Indicate end of line.                     |
|                              |       | Clear               | R2<br>Disable interrupts in                          |
|                              |       | StoreByte           | R2, KBD_CONT<br>the keyboard interface.              |
| Restore<br>Registers Used    | RTRN: | Load                | R3, (SP)                                             |
|                              |       | Load                | R2, 4(SP)                                            |
|                              |       | Add                 | SP, SP, #8<br>Return-from-interrupt                  |

**Note: no parameter passed?**



# Fig 3.8 Main (Assignment)



## Main program

START: Move R2, #LINE  
Store R2, PNTR  
Clear R2  
Store R2, EOL  
Move R2, #2  
StoreByte R2, KBD\_CONT  
MoveControl R2, IENABLE  
Or R2, R2, #2  
MoveControl IENABLE, R2  
MoveControl R2, PS  
Or R2, R2, #1  
MoveControl PS, R2  
next instruction

3. Why these two Writes to Control Registers use different instructions?

R2, #LINE  
R2, PNTR  
R2  
R2, EOL  
R2, #2  
R2, KBD\_CONT  
R2, IENABLE  
R2, R2, #2  
IENABLE, R2  
R2, PS  
R2, R2, #1  
PS, R2

Initialize buffer pointer.  
Clear end-of-line indicator.  
Enable interrupts in the keyboard interface.  
2. Why these EI's use different instructions?  
Enable keyboard interrupts in the processor control register.  
Set interrupt-enable bit in PS.

0x4008



KBD\_CONT

LINE: 1<sup>st</sup> Char



IENABLE



PS



# Fig. 3.8 ISR (Assignment)



## Interrupt-service routine

|       |                                                                                 |                                                                                                       |                                                                                                                                                                                               |                                                                                             |
|-------|---------------------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------|
| ILOC: | Subtract<br>Store<br>Store<br>Load<br>LoadByte<br>StoreByte<br>Add<br>Store     | SP, SP, #8<br>R2, 4(SP)<br>R3, (SP)<br>R2, PNTR<br>R3, KBD_DATA<br>R3, (R2)<br>R2, R2, #1<br>R2, PNTR | Save registers.                                                                                                                                                                               | 4. Where is the Frame Pointer?                                                              |
| ECHO: | LoadByte<br>And<br>Branch if [R2]=0<br>StoreByte<br>Move<br>Branch if [R3]≠[R2] | R2, DISP_STATUS<br>R2, R2, #4<br>ECHO<br>R3, DISP_DATA<br>R2, #CR<br>RTRN                             | Load address pointer.<br>Read character from keyboard.<br>Write the character into memory<br>and increment the pointer.<br>Update the pointer in memory.<br>Wait for display to become ready. | 5. Why the PTR is incremented<br>by only one address location?                              |
|       | Move<br>Store<br>Clear<br>StoreByte                                             | R2, #1<br>R2, EOL<br>R2<br>R2, KBD_CONT                                                               | Display the character just read.<br>ASCII code for Carriage Return.<br>Return if not CR.                                                                                                      |                                                                                             |
| RTRN: | Load<br>Load<br>Add                                                             | R3, (SP)<br>R2, 4(SP)<br>SP, SP, #8                                                                   | Indicate end of line.<br>Disable interrupts in<br>the keyboard interface.<br>Restore registers.                                                                                               | 7. Why DI at keyboard<br>interface, and not in PS or<br>IENABLE registers?                  |
|       | Return-from-interrupt                                                           |                                                                                                       |                                                                                                                                                                                               | 8. Comment on the difference<br>in saving/restoring registers in<br>ISR vs Call Subroutine? |