



Université d'Ottawa • University of Ottawa

SCHOOL OF INFORMATION TECHNOLOGY AND ENGINEERING

**COURSE:** CEG3131

**SEMESTER:** Fall 2009

**PROFESSOR:** Gilbert Arbez

**DATE:** October 24, 2009

**TIME:** 13h00 – 14h30

**MIDTERM  
EXAMINATION**  
**SOLUTION**

**NAME and STUDENT NUMBER:** \_\_\_\_\_ / \_\_\_\_\_

**Instructions:**

- Answer ALL questions on the questionnaire.
- This is a close-book examination.
- Use the provided space to answer the following questions. If more space is needed, use the back of the page.
- Show all your calculations to obtain full marks.
- Calculators are allowed.
- Read all the questions carefully before you start.
- You may detach pages 8 to 12.

1. There are three (3) parts in this examination.

|               |                     |                 |  |
|---------------|---------------------|-----------------|--|
| <b>Part 1</b> | <b>Short Answer</b> | <b>18 marks</b> |  |
| <b>Part 2</b> | <b>Theory</b>       | <b>10 marks</b> |  |
| <b>Part 3</b> | <b>Application</b>  | <b>22 marks</b> |  |
| <b>Total</b>  |                     | <b>50 marks</b> |  |

**Part 1a – Short Answer Questions (2 points per question – total 10 points)**

Answer questions 1 to 5 given that the state of the memory and CPU contents shown on page 8. The diagram represents the initial conditions for each question.

- 1) Give the address of the memory location affected by the following instructions. **\$2509**

Give the contents of the location after the instructions have been executed. **\$9A**.

|                 |                                                      |
|-----------------|------------------------------------------------------|
| LDAA 1,X+       | loads from M(\$2508) into A \$5D, %0101 1101         |
| ANDA 0,X        | AND \$47 from M(\$2509) with \$5D = \$45, %0100 0101 |
| ORAA #%01100000 | changes contents of A to %0110 0101                  |
| COMA            | complements contents of A to produce %1001 1010      |
| STAA 0,X        | Stores contents of A, \$9A into address \$2509       |

- 2) Give the contents of D register after the execution of the following 2 instructions. **8547**

|           |                                                                                         |
|-----------|-----------------------------------------------------------------------------------------|
| LDAA 2,-X | pre-decrements X by 2 to \$2506, loads \$85 into A (from address \$2506 in register X), |
| LDAB 3, X | computes effective address as \$2506+3=\$2509, and loads \$47 into B using this EA.     |

- 3) Consider the following instructions:

|              |                                                          |
|--------------|----------------------------------------------------------|
| LDAB \$2500  | loads into B the value \$F5 (from address \$2500)        |
| CMPB 0,Y     | subtracts \$B3 (found at address \$2510 in Y) from \$F5. |
| B = \$F5     | 1111 0101 OR: 1111 0101 1111 0101 - 11                   |
| M(Y) = \$B3  | 1011 0011 +2's compl(1011 0011) +0100 1101 (-77)         |
| B - M(Y) = 7 | 0100 0010 + 66                                           |

BGT NEXT

..

NEXT LDAA \$CC

Will the instruction BLT branch to NEXT? Yes, since \$F5 (-11) is larger (i.e., less negative) than \$B3 (-77). BLT treats numbers as signed. What will be the contents of the register B after the CMPB instruction? **\$F5 (unchanged!)**

- 4) Given that 16-bit unsigned integers are stored at addresses NUMERATOR and DENOMINATOR, what sequence of instructions would you use to store NUMERATOR%DENOMINATOR at address MODULO, where % is the modulo operator.

**LDD NUMERATOR**  
**LDX DENOMINATOR**  
**IDIV**  
**STD MODULO**

- 5) Consider the following instructions

OFFSET 0

LCL\_VAR1 DS.B 6

LCL\_VAR2 DS.B 1

LEAS 2,X      changes the value of the SP to 2+\$2508 (contents of S) to \$250A

LDI 0,X      loads register D with contents at address \$2508, which is \$5D47

STD LCL\_VAR2,SP    stores the contents of D at address \$250A+6 = \$2510

The last instruction stores the contents of D into a location in memory. Give the address of this location **\$2510** and the value that gets stored there **\$5D47**

## Part 1b – Short Answer Questions (total 8 points)

- 1) (5 points) Translate the following short C program into assembler.
- Use the stack to exchange ALL parameters and the result (assume *int*'s take 2 bytes of storage, and *byte*'s take 1 byte of storage).
  - Ensure that the registers used by the subroutine are not changed after the subroutine has executed.
  - Define the stack usage using the OFFSET directive and labels as offsets into the stack.

```
/*
Function: addInts
Description: Adds two 8-bit integers.
*/
int addInts(byte val1, byte val2)
{
    int sum;
    sum = val1 + val2;
    return(sum);
}

-----Assembler Code-----
; Subroutine - addInts
; Parameters - val1 - on stack
;               val2 - on stack
; Results sum - on stack
; Description: Adds two integers.
; Stack usage:
    OFFSET 0
        DS.W 1 ; preserve Register D
        DS.W 1 ; return address
ADI_SUM  DS.W 1 ; sum and return value
ADI_VAL1 DS.B 1 ; byte val1
ADI_VAL2 DS.B 1 ; byte val2

addInts: PSHD ; preserve acc D
        CLRA
        LDAB ADI_VAL1,SP
        ADDB ADI_VAL2,SP
        ADCA #00
        STD ADI_SUM,SP
        PULD
        RTS
```

2) (3 points) Complete the translation of the C function to Assembler code.

```
-----  
Function: addByteArray  
Parameters: arrPt - pointer to array  
           num - number of elements  
           to sum  
Description: Adds the contents of  
           an integer array.  
Assumption: array contains at  
           least one element.  
-----*/  
int addByteArray(byte *arrPt, byte num)  
{  
    int sum;  
    sum=0;  
    do  
    {  
        sum=sum+*arrPt++;  
        num--;  
    }while(num != 0);  
    return(sum);  
}
```

```
;-----Assembler Code-----  
; Subroutine - addArr  
; Parameters - arrPt - register X  
;               num - register Y  
; Results:     sum - register D  
; Description: Adds contents of an integer  
;               array.  
  
addByteArray: PSHX  
              PSHY ; preserve Registers  
  
              LDD #0 ; sum=0;  
  
loop      ADDB 1,X+ ; sum=sum+*arrPt++;  
              ADCA #0  
              DEY      ; num--  
              BNE loop ; or dbne Y,loop  
              PULY ; restore registers  
              PULX  
              RTS
```

## Part 2 Theory (total 10 points)

(10 points) Modular design provides the means of separating the tasks of any project into manageable pieces. Each of the modules in a project is typically stored in a separate assembler file. Although this does help facilitate the development of a software project, it produces a challenge – how to collect the executable code, constant global data and variable global data from each module into separate sections of software where code and constant data are stored in Read Only Memory (these sections follow one another) and the variable global data is stored in RAM.

Describe how assembler directives, such as ORG, SECTION, SWITCH available in MiniIDE, can be used to perform this organization of software sections.



**Directive SECTION – defines the start of a section.**

**Directive ORG – sets the location counter of a section.**

The above two directives can be used to define sections and their locations in memory.

**Directive SWITCH – changes section, such that any subsequent assembly instructions (including storage instruction such as DS) are placed in the section.**

**Note:** Absolute addresses are defined only on the second pass and thus ENDCODE which defines the location of the global data is placed after all code assembly and consequently places the global constant data section after the code section.

- ENDCODE value can change when code changes when source code is re-assembled.

**Part 3 – Application Question (total 22 points)**

The C standard library provides a function to concatenate two strings:

```
char *strcat(char *str1, char *str2)
```

A *string* of characters terminated with a null character is stored in the memory starting at address found in the pointer variable *str1* and a second string at address found in the pointer variable *str2*. Develop a structured assembly subroutine that concatenates the contents pointed to by *str2* to the end of the contents in string *str1*. The function returns the address of string *str1* (that is the address received by the *str1* variable). Assume single byte ASCII characters.

1. First provide a C function that illustrates the design of the subroutine (or functions/subroutines – you may use additional functions/subroutines to provide a solution).
2. Then translate the C functions to assembler code subroutine(s). Do not forget to comment your code.

**Design (C program and description):**

```
char *strcat(char *str1, char *str2)
{
    char *pt; // working pointer

    pt = findEnd(str1); // find end of string (i.e. nul character) => pt
    while(*str2 != 0)
    {
        *pt++ = *str2++;
    }
    *pt = *str2;
    return(str1);
}

char *findEnd(char *str)
{
    while(*str != 0) str++;
    return(str);
}
```

**Assembler Source Code:**

```

; Subroutine: char* strcat(char *str1, char *str2)
; Parameters
;     str1 - address of first string - on stack
;     str2 - address of second string to append to first string
;             on stack and reg Y
; Returns
;     str1 - pointer to concatenated string, str1 - on stack
; Local Variables
;     pt - register X
; Description: appends the second string to the first string.
;

; Stack Usage:
    OFFSET 0
SCAT PRY  DS.W 1 // preserve Y
SCAT PRX  DS.W 1 // preserve X
SCAT RA   DS.W 1 // return address
SCAT STR1 DS.W 1 // address of first string = also pointer to concatenated
SCAT STR2 DS.W 1 // address of second string
SCAT RET  DS.W 1 // address of return value

strcat: pshx ; preserve registers
        pshy
        ldx SCAT STR1,SP ; pass the address of the 1st string char through X
        jsr findEnd ; when returns, X (i.e. pt) points to end of string 1
        ldy SCAT_STR2,SP ; Y points to str2
scat loop:
        tst 0,Y ; (*str2 == 0)
        beq scat_endwhile ; exit loop if true
        movb 1,Y+,1,X+ ; *pt++ = *str2++ (add str2 after str1; pt=X & str2=Y);
        bra scat_loop
scat endwhile:
        movb 0,Y,0,X ; *pt = *str2
                    ; // add null char at the end of the concatenated string
        puly ; restore registers
        pulx
        rts

; Subroutine: char *findEnd(str)
; Parameters
;     str - address to a string - x
; Returns
;     adr - points to end of string in x
; Description: Finds the end of the string (nul char) and returns its address.
;
; Stack usage:
FEND_RA  DS.W 1 ; 0 return address

findEnd:
fend loop:
        tst 0,x ; (*pt == 0)
        beq fend_while ; yes exit loop

        inx ; pt++
        bra fend_loop
fend endwhile:
        rts

```

## CPU and Memory for Part 1



All values shown are hexadecimal.

**68HC12 INSTRUCTION LIST (reduced)****Loads, Stores, and Transfers**

| Function                                | Mnemonic   | IMM | DIR | EXT | IDX | [IDX] | INH | Operation                        |
|-----------------------------------------|------------|-----|-----|-----|-----|-------|-----|----------------------------------|
| Clear Memory Byte                       | CLR        |     |     | X   | X   | X     |     | $m(ea) \leq 0$                   |
| Clear Accumulator A (B)                 | CLRA (B)   |     |     |     |     |       | X   | $A \leq 0$                       |
| Load Accumulator A (B)                  | LDAA (B)   | X   | X   | X   | X   | X     |     | $A \leq [m(ea)]$                 |
| Load Double Accumulator D               | LDL (B)    | X   | X   | X   | X   | X     |     | $D \leq [m(ea, ea+1)]$           |
| Load Effective Address into SP (X or Y) | LEAS (A,B) |     |     |     |     |       |     | $SP \leq ea$                     |
| Store Accumulator A (B)                 | STAA (B)   | X   | X   | X   | X   | X     |     | $m(ea) \leq (A)$                 |
| Store Double Accumulator D              | STD (B)    | X   | X   | X   | X   | X     |     | $m(ea, ea+1) \leq D$             |
| Transfer A to B                         | TAB        |     |     |     |     |       | X   | $B \leq (A)$                     |
| Transfer A to CCR                       | TAP        |     |     |     |     |       | X   | $CCR \leq (A)$                   |
| Transfer B to A                         | TBA        |     |     |     |     |       | X   | $A \leq B$                       |
| Transfer CCR to A                       | TPA        |     |     |     |     |       | X   | $A \leq (CCR)$                   |
| Exchange D with X (Y)                   | XGDX       |     |     |     |     |       | X   | $D \leftrightarrow (X)$          |
| Pull A (B) from Stack                   | PULA(B)    |     |     |     |     |       | X   | $A \leq [m(SP)], SP \leq (SP)+1$ |
| Push A (B) onto Stack                   | PSHA(B)    |     |     |     |     |       | X   | $SP \leq (SP)-1, m(SP) \leq A$   |

**Arithmetic Operations**

| Function                          | Mnemonic | IMM | DIR | EXT | IDX | [IDX] | INH | Operation                                                             |
|-----------------------------------|----------|-----|-----|-----|-----|-------|-----|-----------------------------------------------------------------------|
| Add Accumulators                  | ABA      |     |     |     |     |       | X   | $A \leq (A) + (B)$                                                    |
| Add with Carry to A (B)           | ADCA (B) | X   | X   | X   | X   | X     |     | $A \leq (A) + [m(ea)] + (C)$                                          |
| Add Memory to A (B)               | ADDA (B) | X   | X   | X   | X   | X     |     | $A \leq (A) + [m(ea)]$                                                |
| Add Memory to D (16 Bit)          | ADDI (B) | X   | X   | X   | X   | X     |     | $D \leq (D) + [m(ea, ea+1)]$                                          |
| Decrement Memory Byte             | DEC      |     |     | X   | X   | X     |     | $m(ea) \leq [m(ea)] - 1$                                              |
| Decrement Accumulator A (B)       | DECA (B) |     |     |     |     |       | X   | $A \leq (A) - 1$                                                      |
| Increment Memory Byte             | INC      |     |     | X   | X   | X     |     | $m(ea) \leq [m(ea)] + 1$                                              |
| Increment Accumulator A (B)       | INCA (B) |     |     |     |     |       | X   | $A \leq (A) + 1$                                                      |
| Subtract with Carry from A (B)    | SBCA (B) | X   | X   | X   | X   | X     |     | $A \leq (A) - [m(ea)] - C$                                            |
| Subtract Memory from A (B)        | SUBA (B) | X   | X   | X   | X   | X     |     | $A \leq (A) - [m(ea)]$                                                |
| Subtract Memory from D (16 Bit)   | SUBD (B) | X   | X   | X   | X   | X     |     | $D \leq (D) - [m(ea, ea+1)]$                                          |
| Multiply (byte, unsigned)         | MUL      |     |     |     |     |       | X   | $D \leq (A) \times (B)$                                               |
| Multiply word, unsigned (signed)  | EMUL(S)  |     |     |     |     |       | X   | $Y:D \leq (D) \times (Y)$                                             |
| Unsigned (signed) 32 by 16 divide | EDIV(S)  |     |     |     |     |       | X   | $X \leq (Y:D) /.(X), Y \leq \text{quotient}, D \leq \text{remainder}$ |
| Fractional Divide (D < X)         | FDIV     |     |     |     |     |       | X   | $X \leq (D) /.(X), D \leq \text{remainder}$                           |
| Integer Divide (unsigned)         | IDIV     |     |     |     |     |       | X   | $X \leq (D) /.(X), D \leq \text{remainder}$                           |

**Logical Operations**

| Function                         | Mnemonic | IMM | DIR | EXT | IDX | [IDX] | INH | Operation                  |
|----------------------------------|----------|-----|-----|-----|-----|-------|-----|----------------------------|
| AND A (B) with Memory            | ANDA (B) | X   | X   | X   | X   | X     |     | $A \leq A \bullet [m(ea)]$ |
| Bit(s) Test A (B) with Memory    | BITA (B) | X   | X   | X   | X   | X     |     | $A \bullet [m(ea)]$        |
| One's Complement Memory Byte     | COM      |     |     | X   | X   | X     |     | $m(ea) \leq [m(ea)]$       |
| One's Complement A (B)           | COMA (B) |     |     |     |     |       | X   | $A \leq /A$                |
| OR A (B) with Memory (Exclusive) | EORA (B) | X   | X   | X   | X   | X     |     | $A \leq A \oplus [m(ea)]$  |
| OR A (B) with Memory (Inclusive) | ORAA (B) | X   | X   | X   | X   | X     |     | $A \leq A + [m(ea)]$       |

**Shift and Rotate**

| Function                             | Mnemonic  | IMM | DIR | EXT | IDX | [IDX] | INH | Operation |
|--------------------------------------|-----------|-----|-----|-----|-----|-------|-----|-----------|
| Arithmetic/Logical Shift Left Memory | ASL/LSL   |     |     | X   | X   | X     |     |           |
| Arithmetic/Logical Shift Left A (B)  | ASLA(B)   |     |     |     |     |       | X   |           |
| Arithmetic/Logical Shift Left Double | ASLD/LSLD |     |     |     |     |       | X   |           |
| Arithmetic Shift Right Memory        | ASR       |     |     | X   | X   | X     |     |           |
| Arithmetic Shift Right A (B)         | ASRA(B)   |     |     |     |     |       | X   |           |
| Logical Shift Right A (B)            | LSRA(B)   |     |     |     |     |       | X   |           |
| Logical Shift Right Memory           | LSR       |     |     | X   | X   | X     |     |           |
| Logical Shift Right D                | LSRD      |     |     |     |     |       | X   |           |
| Rotate Left Memory                   | ROL       |     |     | X   | X   | X     |     |           |
| Rotate Left A (B)                    | ROLA(B)   |     |     |     |     |       | X   |           |
| Rotate Right A (B)                   | RORA(B)   |     |     |     |     |       | X   |           |
| Rotate Right Memory                  | ROR       |     |     | X   | X   | X     |     |           |

**Compare & Test**

| Function                         | Mnemonic | IMM | DIR | EXT | IDX | [IDX] | INH | Operation           |
|----------------------------------|----------|-----|-----|-----|-----|-------|-----|---------------------|
| Compare A to B                   | CBA      |     |     |     |     |       | X   | (A)-(B)             |
| Compare A (B) to Memory          | CMPA (B) | X   | X   | X   | X   | X     |     | (A) - [m(ea)]       |
| Compare D to Memory (16 Bit)     | CPD      | X   | X   | X   | X   | X     |     | (D) - [m(ea,ea+1)]  |
| Compare SP to Memory (16 Bit)    | CPS      | X   | X   | X   | X   | X     |     | (SP) - [m(ea,ea+1)] |
| Compare X (Y) to Memory (16 Bit) | CPX      | X   | X   | X   | X   | X     |     | (X) - [m(ea,ea+1)]  |
| Test memory for 0 or minus       | TST      |     |     | X   | X   | X     |     | m(ea) - 0           |
| Test A (B) for 0 or minus        | TSTA (B) |     |     |     |     |       | X   | (A)-0               |

**Short Branches**

| Function                               | Mnemonic | REL | DIR | IDX | [IDX] | PC <= ea if       |
|----------------------------------------|----------|-----|-----|-----|-------|-------------------|
| Branch ALWAYS                          | BRA      | X   |     |     |       |                   |
| Branch if Carry Clear                  | BCC      | X   |     |     |       | C = 0 ?           |
| Branch if Carry Set                    | BCS      | X   |     |     |       | C = 1 ?           |
| Branch if Equal Zero                   | BEQ      | X   |     |     |       | Z = 1 ?           |
| Branch if Not Equal                    | BNE      | X   |     |     |       | Z = 0 ?           |
| Branch if Minus                        | BMI      | X   |     |     |       | N = 1 ?           |
| Branch if Plus                         | BPL      | X   |     |     |       | N = 0 ?           |
| Branch if Bit(s) Clear in Memory Byte  | BRCLR    |     | X   | X   |       | [m(ea)] • mask=0  |
| Branch if Bit(s) Set in Memory Byte    | BRSET    |     | X   | X   |       | [/m(ea)] • mask=0 |
| Branch if Overflow Clear               | BVC      | X   |     |     |       | V = 0 ?           |
| Branch if Overflow Set                 | BVS      | X   |     |     |       | V = 1 ?           |
| Branch if Greater Than                 | BGT      | X   |     |     |       | Signed >          |
| Branch if Greater Than or Equal        | BGE      | X   |     |     |       | Signed ≥          |
| Branch if Less Than or Equal           | BLE      | X   |     |     |       | Signed ≤          |
| Branch if Less Than                    | BLT      | X   |     |     |       | Signed <          |
| Branch if Higher                       | BHI      | X   |     |     |       | Unsigned >        |
| Branch if Higher or Same (same as BCC) | BHS      | X   |     |     |       | Unsigned ≥        |
| Branch if Lower or Same                | BLS      | X   |     |     |       | Unsigned ≤        |
| Branch if Lower (same as BCS)          | BLO      | X   |     |     |       | Unsigned <        |
| Branch Never                           | BRN      | X   |     |     |       | 3-cycle NOP       |

**Long branch mnemonic = L + Short branch mnemonic, e.g.: BRA → LBRA**

**Loop Primitive Instructions** (counter ctr = A, B, or D)

| Function                         | Mnemonic | REL | DIR | EXT | IDX | [IDX] | INH | Operation                               |
|----------------------------------|----------|-----|-----|-----|-----|-------|-----|-----------------------------------------|
| Decrement counter & branch if =0 | DBEQ     | X   |     |     |     |       |     | ctr <= (ctr)-1, if (ctr)=0 => PC <= ea  |
| Decrement counter & branch if ≠0 | DBNE     | X   |     |     |     |       |     | ctr <= (ctr)-1, if (ctr) ≠0 => PC <= ea |
| Increment counter & branch if =0 | IBEQ     | X   |     |     |     |       |     | ctr <= (ctr)+1, if (ctr)=0 => PC <= ea  |
| Increment counter & branch if ≠0 | IBNE     | X   |     |     |     |       |     | ctr <= (ctr)+1, if (ctr) ≠0 => PC <= ea |
| Test counter & branch if =0      | DBEQ     | X   |     |     |     |       |     | if (ctr)=0 => PC <= ea                  |

**Subroutine Calls and Returns**

| Function                               | Mnemonic | REL | DIR | EXT | IDX | [IDX] | INH | Operation                                                                       |
|----------------------------------------|----------|-----|-----|-----|-----|-------|-----|---------------------------------------------------------------------------------|
| Branch to Subroutine                   | BSR      | X   |     |     |     |       |     | SP <= (SP)-2, m(SP) <= (PC), PC <= ea                                           |
| Jump to Subroutine                     | JSR      |     | X   | X   | X   | X     |     | SP <= (SP)-2, m(SP) <= (PC), PC <= ea                                           |
| CALL a Subroutine<br>(expanded memory) | CALL     |     | X   | X   | X   | X     |     | SP <= (SP)-2, m(SP) <= (PC), PC <= ea<br>SP <= (SP)-1, m(SP) <= (PPG), PC <= pg |
| Return from Subroutine                 | RTS      |     |     |     |     |       | X   | PC <= [m(SP)], SP <= (SP)+2                                                     |
| Return from call                       | RTC      |     |     |     |     |       | X   | PPG <= [m(SP)], SP <= (SP)+1,<br>PC <= [m(SP)], SP <= (SP)+2                    |

| Function | Mnemonic | DIR | EXT | IDX | [IDX] | INH | Operation |
|----------|----------|-----|-----|-----|-------|-----|-----------|
| Jump     | JMP      | X   | X   | X   | X     |     | PC <= ea  |

The **jump** instruction allows control to be passed to any address in the 64-Kbyte memory map.

**Stack and Index Register Instructions**

| Function                       | Mnemonic | IMM | DIR | EXT | IDX | [IDX] | INH | Operation                           |
|--------------------------------|----------|-----|-----|-----|-----|-------|-----|-------------------------------------|
| Decrement Index Register X (Y) | DEX (Y)  |     |     |     |     |       | X   | X <= (X) - 1                        |
| Increment Index Register X (Y) | INX (Y)  |     |     |     |     |       | X   | X <= (X) + 1                        |
| Load Index Register X (Y)      | LDX(Y)   | X   | X   | X   | X   | X     |     | X <= [m(ea,ea+1)]                   |
| Pull X (Y) from Stack          | PULX     |     |     |     |     |       | X   | X <= [m(SP,SP+1)]<br>SP <= (SP) + 2 |
| Push X (Y) onto Stack          | PSHX (Y) |     |     |     |     |       | X   | m(SP,SP+1) <= (X)<br>SP <= (SP) - 2 |
| Store Index Register X (Y)     | STX (X)  | X   | X   | X   | X   | X     |     | m(ea,ea+1) <= X                     |
| Add Accumulator B to X (Y)     | ABX (Y)  |     |     |     |     |       | X   | X <= (X) + (B)                      |
| Decrement Stack Pointer        | DES      |     |     |     |     |       | X   | SP <= (SP) - 1                      |
| Increment Stack Pointer        | INS      |     |     |     |     |       | X   | SP <= (SP) + 1                      |
| Load Stack Pointer             | LDS      | X   | X   | X   | X   | X     |     | SP <= [m(ea,ea+1)]                  |
| Store Stack Pointer            | STS      | X   | X   | X   | X   | X     |     | m(ea,ea+1) <= (SP)                  |
| Transfer SP to X (Y)           | TSX (Y)  |     |     |     |     |       | X   | X <= (SP)                           |
| Transfer X (Y) to SP           | TXS (Y)  |     |     |     |     |       | X   | SP <= (X)                           |
| Exchange D with X (Y)          | XGDX (Y) |     |     |     |     |       | X   | (D) <=> (X)                         |

| Function              | Mnemonic | INH | Operation                                                                                                                                                                                                                                                                                                                                                                                |
|-----------------------|----------|-----|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Return from Interrupt | RTI      | X   | $(M_{(SP)}) \Rightarrow CCR; (SP) + \$0001 \Rightarrow SP$<br>$(M_{(SP)} : M_{(SP+1)}) \Rightarrow B : A; (SP) + \$0002 \Rightarrow S$<br>$(M_{(SP)} : M_{(SP+1)}) \Rightarrow X_H : X_L; (SP) + \$0004 \Rightarrow S$<br>$(M_{(SP)} : M_{(SP+1)}) \Rightarrow PC_H : PC_L; (SP) + \$0002 \Rightarrow S$<br>$(M_{(SP)} : M_{(SP+1)}) \Rightarrow Y_H : Y_L; (SP) + \$0004 \Rightarrow S$ |
| Software Interrupt    | SWI      | X   |                                                                                                                                                                                                                                                                                                                                                                                          |
| Wait for Interrupt    | WAI      | X   |                                                                                                                                                                                                                                                                                                                                                                                          |

**Interrupt Handling**

The software interrupt (SWI) instruction is similar to a JSR instruction, except the contents of all working CPU registers are saved on the stack rather than just the return address. SWI is unusual in that it is requested by the software program as opposed to other interrupts that are requested asynchronously to the executing program.