# **Chapter 3 Assembly language programming**

- \* PIC18 assembly language programming
- \* branching, looping, time delay
- \* I/O port
- \* arithmetic and logic instructions
- \* addressing modes
- \* look-up table, stack, subroutine

# 3.1 PIC18 assembly language programming

- revision of PIC18 architecture
- PIC18 assembly language
- structure of PIC18 assembly language program
- assemble and run PIC18 program
- ROM space
- flags and status register
- register banks

### 3.1.1 revision of PIC18 architecture

- structure of PIC18 components and their relation
- function of PIC18 necessary for writing assembly language program

#### Harvard architecture



### **Programming model**





• Microprocessor unit contains Arithmetic Logic Unit (ALU), registers, and control unit

#### • ALU contains:

16-bit instruction decoder

8-bit Working Register (WREG) – involved in the execution of many instructions

status register (5 flag bits) – contains arithmetic status of ALU

# Flags in Status Register

(check the state of WREG)



- N (Negative Flag)
  - Set when bit 7 is 1 as the result of an arithmetic/logic operation
- OV (Overflow Flag)
  - Set when there is overflow from high-order bit into sign bit
- Z (Zero Flag)
  - Set when result of an operation is zero
- DC (Digit Carry Flag, Half Carry)
  - Set when there is carry from bit 3 to bit 4
- C (Carry Flag)
  - Set when there is a carry out from bit 7

### • Registers:

21-bit Program Counter (PC) – pointer to program memory during program execution

21-bit Table Pointer – memory pointer to copy bytes between program memory and data registers

Stack Pointer (SP) – point to stack (31 registers used for temporary storage of memory addresses during program execution

Product – 16-bit product of 8-bit by 8-bit multiply

4-bit Bank Select Register (BSR) – upper 4-bit of 12-bit address of data memory

File Select Registers (FSRs) – FSR0, FSR1, FSR2, each composed of 8-bit H and 8-bit L

### Program Memory

- 21-bit address bus
- address up to  $2^{21}$ =2M bytes of memory
- not all memory locations are implemented
- 16-bit data bus

### • Data Memory

- also called file register
- 12-bit address bus
- address up to  $2^{12}$ =4K bytes of memory
- 8-bit data bus

• data memory is divided into 2 sections – GPR and SFR

• General-purpose registers (GPRs) provide storage for variables used in a program.

• Special-function registers (SFRs) are used to control the operation of the CPU and peripherals.

I/O Ports (A to E)

Interrupts

**EEPROM** 

Serial I/O

**Timers** 

Capture/Compare/PWM (CCP)

Analog-to-Digital (A/D) Converter

#### Access (Default) Bank Mode:

Top 128 bytes
of Data Memory
(GPR)

Bottom 128 bytes
of Data Memory
(SFR)

Bank specified by Bank Select Register (BSR)



128 bytes SFRs
How many bytes GPRs?



When a = 1, the BSR is used to specify the RAM location that the instruction uses.

### **Banks in Data Memory**

- Divided into 16 banks.
- 256 bytes per bank
- Access (Default) Bank is a 256-byte bank consisting of:
  - 128 bytes of GPRs located at 00H to 7FH in the access bank,
     mapped from 000H to 07FH of the data memory
  - 128 bytes of SFRs located at 80H to FFH in the access bank,
     mapped from F80H to FFFH of the data memory
- A program that requires more than the amount of RAM provided in the access bank necessitates *bank switching*.

PIC18 uses the bank concept because in many instructions there are 8 bits to indicate the RAM address (12 bits address)

# 3.1.2 PIC18 assembly language

some data types

• some PIC18 instructions

• access data memory (file register)

# **Data Format Representation**

- Data can only be represented as 8-bit number in PIC18
- Four ways to represent a byte:
  - Hexadecimal (Default)
  - Binary
  - Decimal
  - ASCII

#### **Hexadecimal Numbers**

- Four ways to show that hex representation is used:
  - 1. Put nothing in front or back of the number, e.g. movlw 99 (Hex is the default representation)
  - 2. Use h (or H) right after the number, e.g. movlw 99H
  - 3. Put 0x (or 0X) before the number, e.g. movlw 0x99
  - 4. Put h in front of the number, with single quotes around the number, e.g. movlw h'99'
- If the starting hex digit is A-F, the number must be preceded by a 0.
  - e.g. movlw C6 is invalid. Must be movlw 0C6

### Binary, Decimal, ASCII

- The only way to represent a binary number is to put a B (or b) in front, with single quotes around the binary digits, e.g. movlw B'10011001'
- Two ways to present a decimal number:
  - 1. Put a D (or d) in front, with single quotes around the decimal digits, e.g. movlw D'12'
  - 2. Use the ".value" format, e.g. movlw .12
- The only way to represent an ASCII character is to put a A (or a) in front, with single quotes around the ASCII character, e.g. movlw A'2'.
- The ASCII code 0x32 is used to represent the character '2'. 0x32 is stored in WREG.

#### **PIC18 Instruction Set**

- Includes 77 instructions
  - 73 one word (16-bit) long
  - 4 two words (32-bit) long
- Divided into seven groups
  - Move (Data Copy) and Load
  - Arithmetic
  - Logic
  - Program Redirection (Branch/Jump)
  - Bit Manipulation
  - Table Read/Write
  - Machine Control

#### **MOVLW**

Moves 8-bit data into WREG

• MOVLW k; move literal value k into WREG

Example

MOVLW 25H

MOVLW 0A5H

Is the following code correct? MOVLW 9H

MOVLW A23H

### **ADDLW**



#### **ADDLW**

ADDLW k; add literal value k to WREG (k +WREG)

Example: WREG

ADDLW 16H; 0 0 1 0 0 0

ADDLW 43H; 0 1 1 1 1 0 0

### The PIC File register (data memory)

• Used for data storage, scratch pad and registers for internal use and function

• 8-bit width





### **Special-Function Registers**

- dedicated to specific functions such as ALU status, timers, serial communication, I/O ports, ADC,...
- They are used for control of the microcontroller or peripheral
- 8-bit registers
- Their numbers varies from one chip to another

### **General-Purpose Registers**

- Group of RAM locations
- 8-bit registers
- Larger than SFR

Difficult to manage them by using Assembly language

Easier to handle them by C Compiler.

#### **GPRAM vs. EEPROM**

EEPROM: an add-on memory (for holding data after power off), can be zero size

# File Register Size

|            | File Register | = | SFR     | + | GPR     |
|------------|---------------|---|---------|---|---------|
|            | (Bytes)       | • | (Bytes) |   | (Bytes) |
| PIC12F508  | 32            |   | 7       |   | 25      |
| PIC16F84   | 80            |   | 12      |   | 68      |
| PIC18F1220 | 512           |   | 256     |   | 256     |
| PIC18F452  | 1792          |   | 256     |   | 1536    |
| PIC18F2220 | 768           |   | 256     |   | 512     |
| PIC18F458  | 1792          |   | 256     |   | 1536    |
| PIC18F8722 | 4096          |   | 158     |   | 3938    |
| PIC18F4550 | 2048          |   | 160     |   | 1888    |

### File Register and access bank in PIC18

- The PIC18 Family can have a maximum of 4096 bytes.
- The File Register
  - has addresses of 000- FFFH divided into 256-byte banks maximum 16 banks (Why?)
- At least there is one bank
- ☐ Known as default access bank.
- Bank switching is a method used to access all the banks



#### Access bank in PIC18

- It is 256-Byte bank
- Divided into two discontinuous sections
   (each 128 Bytes)
   GP RAM, from 0 to 7FH
   SFR, from F80H to FFFH

# SFRs of PIC18

| F80h | PORTA |   |
|------|-------|---|
| F81h | PORTB |   |
| F82h | PORTC |   |
| F83h | PORTD |   |
| F84h | PORTE |   |
| F85h |       |   |
| F86h |       |   |
| F87h |       |   |
| F88h |       |   |
| F89h | LATA  |   |
| F8Ah | LATB  |   |
| F8Bh | LATC  |   |
| F8Ch | LATD  |   |
| F8Dh | LATE  |   |
| F8Eh |       |   |
| F8Fh |       |   |
| F90h |       |   |
| F91h |       |   |
| F92h | TRISA |   |
| F93h | TRISB |   |
| F94h | TRISC |   |
| F95h | TRISD |   |
| F96h | TRISE |   |
| F97h |       |   |
| F98h |       |   |
| F99h |       |   |
| F9Ah |       |   |
| F9Bh |       |   |
| F9Ch |       |   |
| F9Dh | PIE1  |   |
| F9Eh | PIR1  |   |
| F9Fh | IPR1  |   |
|      |       | - |

| F AO h | PIE2    | 1 |
|--------|---------|---|
| FA1h   | PIR2    | ı |
| FA2h   | IPR2    | ı |
| FA3h   |         | ı |
| FA4h   |         | ı |
| F Æ h  |         | ı |
| FA6h   |         | ı |
| F A7 h |         | ı |
| FA8h   |         | ı |
| FA9h   |         | ı |
| FAAh   |         | ı |
| FABh   | RCSTA   | ı |
| FACh   | TXSTA   | ı |
| FADh   | TXREG   | ı |
| FAEh   | RCREG   | ı |
| FAFh   | SPBRG   | ı |
| FBOh   |         | ı |
| FB1h   | T3C ON  | ı |
| FB2h   | TMR3L   | ı |
| FB3h   | TMR3H   | ı |
| FB4h   |         | ı |
| FB5h   |         | ı |
| FB6h   |         | ı |
| F 87 h |         | ı |
| FB8h   |         | ı |
| FB9h   |         | ı |
| FBAh   | CCP2CON | ı |
| FBBh   | CCPR2L  | ı |
| FBCh   | CCPR2H  |   |
| FBDh   | CCP1CON |   |
| FBEh   | CCPR1L  |   |
| FBFh   | CCPR1H  | ı |

| FCOh |          | l |
|------|----------|---|
| FC1h | ADC ON1  |   |
| FC2h | ADC ON0  |   |
| FСЗh | ADRESL   | l |
| FC4h | ADRESH   |   |
| FC5h | SSPC ON2 |   |
| FC6h | SSPC ON1 |   |
| FC7h | SSPSTAT  |   |
| FC8h | SSPADD   |   |
| FC9h | SSPBUF   |   |
| FCAh | T2CON    | l |
| FCBh | PR2      | l |
| FCCh | TMR2     |   |
| FCDh | T1CON    |   |
| FCEh | TMR 1L   |   |
| FCFh | TMR1H    |   |
| FDOh | RCON     |   |
| FD1h | WDTCON   |   |
| FD2h | LVDCON   |   |
| FDЗh | OSCCON   |   |
| FD4h |          |   |
| FD5h | TOCON    |   |
| FD6h | TMROL    |   |
| FD7h | TMROH    |   |
| FD8h | STATUS   |   |
| FD9h | FSR2L    |   |
| FDAh | FSR2H    |   |
| FDBh | PLUSW2   | * |
| FDCh | PREINC2  | * |
| FDDh | POSTDEC2 | * |
| FDEh | POSTINC2 | × |
| FDFh | INDF2    | * |

| FEOh  | BSR      |
|-------|----------|
| FE1h  | FSR1L    |
| FE2h  | FSR1H    |
| FE3h  | PLUSW1   |
| FE4h  | PREINC1  |
| FE5h  | POSTDEC1 |
| FE6h  | POSTINC1 |
| FE7h  | INDF1    |
| FE8h  | WREG     |
| FE9h  | FSROL    |
| FEAh  | FSROH    |
| FEBh  | PLUSWO   |
| FECh  | PREINCO  |
| FEDh  | POSTDECO |
| FEEh  | POSTINCO |
| FEFh  | INDFO    |
| F FOh | INTCONS  |
| FF1h  | INTCON2  |
| FF2h  | INTCON   |
| F F3h | PRODL    |
| FF4h  | PRODH    |
| FF5h  | TABLAT   |
| F F6h | TBLPTRL  |
| FF7h  | TBLPTRH  |
| F F8h | TBLPTRU  |
| F F9h | PCL      |
| FF.Ah | PCLATH   |
| FFBh  | PCLATU   |
| FFCh  | STKPTR   |
| FFDh  | TOSL     |
| FFEh  | TOSH     |
| FFFh  | TOSU     |
|       | 1        |

# Using instructions with the default access bank

Instructions to access other locations in the file register for ALU and other operations.

- •□ MOVWF
- •□ COMF
- •□ DECF
- •□ MOVF
- •□ MOVFF

### **MOVWF** instruction

• F indicates a file register

#### **MOVWF** Address

- It tells the CPU to copy the source, WREG, to a destination in the file register.
- ☐ a location in the SFR
- ☐ a location in GP RAM

### **MOVWF** instruction

|       |     | WREG |
|-------|-----|------|
| MOVLW | 99H | 99   |
| MOVWF | 12H |      |
| MOVLW | 85H | 85   |
| MOVWF | 13H |      |
| MOVLW | 3FH | 3F   |
| MOVWF | 14H |      |
| MOVLW | 63H | 63   |
| MOVWF | 15H |      |
| MOVLW | 12H | 12   |
| MOVWF | 16H |      |

Note: We cannot move literal values directly into the GP RAM location in the PIC18.

They must be moved there via WREG.

#### **Data Memory**

| Address | Data |
|---------|------|
| 012H    |      |
| 013H    |      |
| 014H    |      |
| 015H    |      |
| 016H    |      |
|         |      |
| Address | Data |

| Address | Data |  |
|---------|------|--|
| 012H    | 99   |  |
| 013H    | 85   |  |
| 014H    | 3F   |  |
| 015H    | 63   |  |
| 016H    | 12   |  |
|         |      |  |

### **ADDWF** instruction

- Add together the contents of WREG and a file register location ADDWF File Reg. Address, D
- The result will be placed in either the WREG or in the File Reg. location

D indicates the destination bit

- If D=0 or (D=w)

  The result will be placed in WREG
- If D=1 or (D=f)
   The result will be placed in File Reg.

| MOVLW | 22H   | ;WREG=22H                            |
|-------|-------|--------------------------------------|
| MOVWF | 5H    | ; copy WREG contents to location 5H  |
| MOVWF | 6H    | ; copy WREG contents to location 6H  |
| MOVWF | 7 H   | ; copy WREG contents to location 7H  |
| ADDWF | 5H, 0 | ;add W and loc 5, put result in WREG |
|       |       | ; WREG=44H                           |
| ADDWF | 6H, 0 | ;add W and loc 6, put result in WREG |
|       |       | ; WREG=66H                           |
| ADDWF | 7H, 0 | ;add W and loc 7, put result in WREG |
|       |       | ; WREG=88H                           |

| Data |
|------|
|      |
| 22   |
| 22   |
| 22   |
|      |

GPR after the execution up to "ADDWF 7H, 0" WREG = 88H

| MOVLW | 22H   | ;WREG=22H                            |
|-------|-------|--------------------------------------|
| MOVWF | 5H    | ; copy WREG contents to location 5H  |
| MOVWF | 6H    | ; copy WREG contents to location 6H  |
| MOVWF | 7 H   | ; copy WREG contents to location 7H  |
| ADDWF | 5H, 0 | ;add W and loc 5, put result in WREG |
|       |       | ; WREG=44H                           |
| ADDWF | 6H, 0 | ;add W and loc 6, put result in WREG |
|       |       | ; WREG=66H                           |
| ADDWF | 7H, 1 | ;add W and loc 7, put result in      |
|       |       | ; loc 7, content of loc $07 = 88H$ , |
|       |       | ;WREG=66.                            |

| ata |
|-----|
|     |
| ,   |
|     |
|     |
|     |

GP RAM after the execution up to "ADDWF 7H, 1"
WREG = 66H

### **COMF** instruction

### COMF File Reg. Address, D

• It tells the CPU to complement the contents of File Reg. and place the result in WREG or in File Reg.

D indicates the destination bit

- If D=0 or (D=w)

  The result will be placed in WREG
- If D=1 or (D=f)

  The result will be placed in File Reg.

Write a simple program to toggle the SFR of Port C continuously forever.

Solution:

MOVLW 00

MOVWF TRISC, 0

MOVLW 55H

MOVWF PORTC

B1: COMF PORTC, F

GOTO B1

What are the values (in hexadecimal) in Port C

What do you see if Port C is connected to 8 LEDs?



## **DECF (INCF) instruction**

## DECF File Reg. Address, D

• It tells the CPU to decrement the content of File Reg. and place the result in WREG or in File Reg.

```
MOVLW 3 ; WREG=3

MOVWF 20H ; 20H=(3)

DECF 20H, F ; WREG=3, 20H=(2)

DECF 20H, F ; WREG=3, 20H=(1)

DECF 20H, F ; WREG=3, 20H=(0)
```

```
MOVLW 3 ; WREG=3

MOVWF 20H ; 20H=(3)

DECF 20H, W ; WREG=2, 20H=(3)

DECF 20H, W ; WREG=2, 20H=(3)

DECF 20H, W ; WREG=2, 20H=(3)
```

## **MOVF** instruction

MOVF File Reg. Address, D

It is intended to copy contents of File Reg. to WREG

- If D=0 copy contents of File Reg. (from I/O pin) to WREG
- If D=1

  contents of File Reg. are copied to itself

  (Why do this?)

Write a simple program to get data from SFRs of Port B and send it to SFRs of PORT C continuously.

Solution:

Again: MOVF PORTB, W

MOVWF PORTC

GOTO Again

#### **MOVFF** instruction

Copy contents of one location in File Reg. to another location in File Reg.

MOVFF source File Reg, destination File Reg



Write a simple program to get data from SFRs of Port B and send it to SFRs of PORT C continuously.

Solution:

Again: MOVFF PORTB, PORTC

GOTO Again

## 3.1.3 structure of PIC18 assembly language program

- programming language
- directives

- Machine language:
  - a program that consists of 0 and 1.
  - CPU can work on machine language directly.
  - Example: 7D25
- Low-level language:
  - It deals directly with the internal structure of the CPU.
  - Programmers must know all details of the CPU.
  - Example: MOVFF 20H, 21H
- High-level language:
  - Machine independent
  - Example: a=37; (C++)

- Assembly languages were developed which provided mnemonics for the machine code instructions, plus other features.
  - Mnemonic: the instruction
    - Example: MOVFF, MOVLW
  - Provide decimal number, named registers, label, command
  - programming faster and less prone to error
- Assembly language programs must be translated into machine code by a program called an assembler.

#### • Assembler:

- a software program can translate an assembly language program into machine code
- source program
- object program, object code (opcode)



## **Structure of Assembly Language**

• An assembly language program is a series of statements.

```
[label:] mnemonic [operands] [;comment]
```

- Brackets indicate that a field is optional.
- Label is the name to refer to a line of program code. A label referring to an instruction must be followed by a ":".

```
Here: GOTO Here
```

- Mnemonic and operand(s) perform the real work of the program.
- The comment field begins with ";".

- Two types of assembly instructions:
  - Mnemonic: tell the CPU what to do
    - Example: MOVFF, ADDLW (opcodes)
  - pseudo-instruction: give directions to the assembler
    - Example: ORG 0H, END
    - Pseudo-instruction is called directives

## ORG & END

• ORG tells the assembler to place the opcode at ROM with a chosen start address.

#### **ORG** start-address

```
ORG 0200H ;put the following codes ;start at location 200H
```

• END indicates to the assembler the end of the source code.

#### **END**

```
END ; end of asm source file
```

## **EQU** and **SET**

- EQU associates a constant number with an address label.
- e.g., COUNT equ 0x25

• • • • • • • • •

movlw COUNT; WREG = 0x25

- SET identical to EQU, but value assigned by SET can be reassigned later.
- e.g., COUNT set 0x00

• • • • • • • • •

COUNT set 0x25

• • • • • • • •

movlw COUNT; WREG = 0x25

# e.g., Move 22H into two file registers with addresses 0x05 and 0x06, add the contents of all three registers and put the result in WREG:

| Without | EQU   |   | With EQU |           |   |  |  |
|---------|-------|---|----------|-----------|---|--|--|
| movlw   | 0x22  |   | FirstReg | EQU 0x05  |   |  |  |
| movwf   | 0×05  |   | SecReg   | EQU 0x06  |   |  |  |
|         |       |   | movlw    | 0x22      |   |  |  |
| movwf   | UXU6  |   | movwf    | FirstReg  |   |  |  |
| addwf   | 0x05, | M | movwf    | SecReg    |   |  |  |
| addwf   | 0x06, | M | addwf    | FirstReg, | W |  |  |
|         |       |   | addwf    | SecReg, W |   |  |  |

Which method is easier to change the file registers?

### **CBLOCK**

- Defines a list of named constants.
- Format: cblock <num><constant label> [:<inc>]endc
- e.g. 1:

  cblock 0x50

  test1, test2, test3, test4
  endc
- Values Assigned:

test1 = 
$$0x50$$
, test2 =  $0x51$ ,  
test3 =  $0x52$ , test4 =  $0x53$ .

```
• e.g. 2:

cblock 0x30

twoBytesValue: 0, twoByteHi, twoByteLo
queue: d'40'
queuehead, queuetail
double1: 2, double2: 2
endc
```

## • Value Assigned: twoBytesValue = 0x30, twoByteHi = 0x30

twoBytes value = 0x30, twoBytes i = 0x30 twoByteLo = 0x31, queue = 0x32 queuehead = 0x5A, queuetail = 0x5B double1 = 0x5C, double2 = 0x5E

## Sample of an Assembly Language Program

```
LIST P=18F4520
                      ; directive to define processor
#include <P18F4520.INC> ;CPU specific variable
                      ; definitions
     SUM EQU 10H
                      ; RAM loc 10H for SUM
     ORG OH
                      ; start at address 0
                      ; WREG = 25
     MOVIW 25H
     ADDLW 0x34; add 34H to WREG = 59H
                      ; add 11H to WREG = 6AH
     ADDLW 11H
     ADDLW d'18'
                      W = W + 12H = 7CH
     ADDLW 1CH
                      W = W + 1CH = 98H
     ADDLW b'00000110'; W = W + 6H = 9EH
     MOVWF SUM
                      ; save the result in SUM location
HERE: GOTO HERE
                      ; stay here forever
                      : end of asm source file
     END
```

## **Summary**

- ♦ review PIC18 architecture major components, programming model
- ◆ some data types and instructions
- ♦ access bank and file registers
- **♦** directives
- ♦ assembly language program

## 3.1.4 assemble and run PIC18 program

- enter assembly language program, create project, build and run the program on IDE (refer to section 2.5 and Tutorial 2 "MPLAB IDE")
- next, you can choose Debugger, e.g. PICKit 3, to run the program on the development kit (refer to Tutorial 3 "upload program")

```
LIST P=18F4520 ; directive to define processor
#include <P18F4520.INC> ;CPU specific variable
                      ;definitions
     SUM EQU 10H
                      ; RAM loc 10H for SUM
                      ; start at address 0
     ORG OH
     MOVLW 25H ; WREG = 25
     ADDLW 0x34 ; WREG = WREG + 34H = 59H
     ADDLW 11H \Rightarrow WREG = WREG + 11H = 6AH
     ADDLW d'18'; WREG = WREG + 12H = 7CH
     ADDLW 1CH ; WREG = WREG + 1CH = 98H
     ADDLW b'00000110'; WREG = WREG + 6H = 9EH
                      ; save the result in SUM location
     MOVWF SUM
HERE: GOTO HERE
                      ; stay here forever
     END
                      ; end of asm source file
```

## observe Program Counter, machine code

| Progra   | ■ Program Memory |             |        |                    |  |  |  |
|----------|------------------|-------------|--------|--------------------|--|--|--|
|          | Line             | Address     | Opcode | Disassembl ^       |  |  |  |
|          | 1                | 0000        | 0E25   | MOVLW 0x25         |  |  |  |
|          | 2                | 0002        | 0F34   | ADDLW 0x34         |  |  |  |
|          | 3                | 0004        | 0F11   | ADDLW 0x11         |  |  |  |
|          | 4                | 0006        | 0F12   | ADDLW 0x12         |  |  |  |
|          | 5                | 8000        | 0F1C   | ADDLW 0x1c         |  |  |  |
|          | 6                | A000        | 0F06   | ADDLW 0x6          |  |  |  |
|          | 7                | 000C        | 6E10   | MOVWF 0x10, ACCESS |  |  |  |
|          | 8                | 000E        | EF07   | GOTO 0xe           |  |  |  |
|          | 9                | 0010        | F000   | NOP +              |  |  |  |
| 4        |                  | 111         |        | ► F                |  |  |  |
| Opcode H | lex Machi        | ne Symbolic |        |                    |  |  |  |



```
LOC
     OBJECT CODE
                 LINE
                        SOURCE TEXT
                                                   VALUE
                 00001
                        LIST
                              P=18F4520 ; directive to define processor
                 00002
                       #include <P18F4520.INC>;
0000010
                                                   ; RAM loc 10H fro SUM
                 00004 SUM:
                                EQU 10H
000000
                                ORG OH
                 00005
                                                   ; start at address 0
000000 0E25
                 00006
                                MOVLW 25H
                                                   ; WREG = 25
000002 OF34
                 00007
                                ADDLW 0x34
                                                   ; add 34H to WREG=59H
000004 OF11
                 80000
                                ADDLW 11H
                                                   ; add 11H to WREG=6AH
000006 OF12
                 00009
                                ADDLW d'18'
                                                   W = W + 12H = 7CH
000008 OF1C
                 00010
                                                   W = W + 1CH = 98H
                                ADDLW 1CH
00000A 0F06
                 00011
                                ADDLW b'00000110'; W = W + 6H = 9EH
                                                   ; save the result in SUM location
00000C 6E10
                 00012
                                MOVWF SUM
00000E EF07 F000 00013 HERE:
                                GOTO HERE
                                                   ; stay here forever
                 00014
                                END
                                                   ; end of asm source file
```

## 3.1.5 ROM space

- Program Counter (PC) is used by the CPU to point to the address of the next instruction to be executed (*increase automatically*)
- The wider the program counter, more the memory locations can be accessed

PIC16 has 14 bits (16 KB)

PIC18 has 21 bits (2 MB)

8051 has 16 bits (64 KB)

## PIC18 PC



#### PIC18 Microcontroller Family

|            |         |          | Data  | Memory |       |        |                      |       |               |      |          |                      |       |
|------------|---------|----------|-------|--------|-------|--------|----------------------|-------|---------------|------|----------|----------------------|-------|
|            | Program | n Memory | RAM   | EEPROM | 1/0   | ADC    |                      |       |               | CCP/ | Timers   |                      |       |
| Product    | Туре    | Bytes    | Bytes | Bytes  | Ports | 10-bit | MSSP                 | USART | <b>O</b> ther | PWM  | 8/16-bit | Packages             | Pins  |
| PIC18F1220 | FLASH   | 4K       | 256   | 256    | 16    | 7      | _                    | 1     | 6x PMM        | 1    | 1/3      | DIP, SOIC, SSOP, QFN | 18    |
| PIC18F1320 | FLASH   | 8K       | 256   | 256    | 16    | 7      | _                    | 1     | 6x PMM        | 1    | 1/3      | DIP, SOIC, SSOP, QFN | 18    |
| PIC18F2220 | FLASH   | 4K       | 512   | 256    | 23    | 10     | I <sup>2</sup> C/SPI | 1     | 6x PMM        | 2    | 1/3      | DIP, SOIC            | 28    |
| PIC18F2320 | FLASH   | 8K       | 512   | 256    | 23    | 10     | I2C/SPI              | 1     | 6x PMM        | 2    | 1/3      | DIP, SOIC            | 28    |
| PIC18C242  | OTP     | 16K      | 512   | _      | 23    | 5      | I <sup>2</sup> C/SPI | 1     | _             | 2    | 1/3      | DIP, SOIC            | 28    |
| PIC18C252  | OTP     | 32K      | 1536  | _      | 23    | 5      | I <sup>2</sup> C/SPI | 1     | _             | 2    | 1/3      | DIP, SOIC            | 28    |
| PIC18F242  | FLASH   | 16K      | 512   | 256    | 23    | 5      | I <sup>2</sup> C/SPI | 1     | _             | 2    | 1/3      | DIP, SOIC, SSOP      | 28    |
| PIC18F252  | FLASH   | 32K      | 1536  | 256    | 23    | 5      | I <sup>2</sup> C/SPI | 1     | _             | 2    | 1/3      | DIP, SOIC, SSOP      | 28    |
| PIC18F258  | FLASH   | 32K      | 1536  | 256    | 22    | 5      | I2C/SPI              | 1     | CAN 2.0B      | 1    | 1/3      | DIP, SOIC            | 28    |
| PIC18F4220 | FLASH   | 4K       | 512   | 256    | 34    | 13     | I <sup>2</sup> C/SPI | 1     | 6x PMM        | 2    | 1/3      | DIP, TQFP, QFN       | 40/44 |
| PIC18F4320 | FLASH   | 8K       | 512   | 256    | 34    | 13     | I2C/SPI              | 1     | 6x PMM        | 2    | 1/3      | DIP, TQFP, QFN       | 40/44 |
| PIC18C442  | OTP     | 16K      | 512   | _      | 34    | 8      | I <sup>2</sup> C/SPI | 1     | _             | 2    | 1/3      | DIP, PLCC, TQFP      | 40/44 |
| PIC18C452  | OTP     | 32K      | 1536  | _      | 34    | 8      | I2C/SPI              | 1     | _             | 2    | 1/3      | DIP, PLCC, TQFP      | 40/44 |
| PIC18F442  | FLASH   | 16K      | 512   | 256    | 34    | 8      | I <sup>2</sup> C/SPI | 1     | _             | 2    | 1/3      | DIP, PLCC, TQFP      | 40/44 |
| PIC18F452  | FLASH   | 32K      | 1536  | 256    | 34    | 8      | I2C/SPI              | 1     | _             | 2    | 1/3      | DIP, PLCC, TQFP      | 40/44 |
| PIC18F458  | FLASH   | 32K      | 1536  | 256    | 33    | 5      | I <sup>2</sup> C/SPI | 1     | CAN 2.0B      | 1    | 1/3      | DIP, PLCC, TQFP      | 40/44 |
| PIC18C601  | _       | ROMless  | 1536  | _      | 31    | 8      | I2C/SPI              | 1     | _             | 2    | 1/3      | PLCC, TQFP           | 64/68 |
| PIC18C658  | OTP     | 32K      | 1536  | _      | 52    | 12     | I <sup>2</sup> C/SPI | 1     | CAN 2.0B      | 2    | 1/3      | PLCC, TQFP           | 64/68 |
| PIC18F6520 | FLASH   | 32K      | 2048  | 1024   | 52    | 12     | I <sup>2</sup> C/SPI | 2     | _             | 5    | 2/3      | TQFP                 | 64    |
| PIC18F6620 | FLASH   | 64K      | 3840  | 1024   | 52    | 12     | I <sup>2</sup> C/SPI | 2     | _             | 5    | 2/3      | TQFP                 | 64    |
| PIC18F6720 | FLASH   | 128K     | 3840  | 1024   | 52    | 12     | I <sup>2</sup> C/SPI | 2     | _             | 5    | 2/3      | TQFP                 | 64    |
| PIC18C801  | _       | ROMless  | 1536  | _      | 42    | 12     | I2C/SPI              | 1     | _             | 2    | 1/3      | PLCC, TQFP           | 80/84 |
| PIC18C858  | OTP     | 32K      | 1536  | _      | 68    | 16     | I <sup>2</sup> C/SPI | 1     | CAN 2.0B      | 2    | 1/3      | PLCC, TQFP           | 80/84 |
| PIC18F8520 | FLASH   | 32K      | 2048  | 1024   | 68    | 16     | I2C/SPI              | 2     | EMA           | 5    | 2/3      | TQFP                 | 80    |
| PIC18F8620 | FLASH   | 64K      | 3840  | 1024   | 68    | 16     | I <sup>2</sup> C/SPI | 2     | EMA           | 5    | 2/3      | TQFP                 | 80    |
| PIC18F8720 | FLASH   | 128K     | 3840  | 1024   | 68    | 16     | I2C/SPI              | 2     | EMA           | 5    | 2/3      | TQFP                 | 80    |
|            |         |          |       |        |       |        |                      |       |               |      |          |                      |       |

Abbreviation: ADC = Analog-to-Digital Converter
PWM = Pulse Width Modulation

CCP = Capture/Compare/PWM SPI = Serial Peripheral Interface PC = Inter-Integrated Circuit Bus

PMM = Power Managed Mode

Find the ROM size for each of the following PIC chips:

- a) PIC18F2220
- b) PIC18F2410
- c) PIC18F458



#### Place program code in ROM

At what address does the CPU wake up when power is applied?

- PIC18 wakes up at memory address 000000H
- PC has the value 000000H
- ORG directive put the address of the first opcode at the memory location 000000H



#### LOC OBJECT CODE LINE SOURCE TEXT VALUE

00001 LIST P=18F4520 ;directive to define processor 00002 #include <P18F4520.INC> ;

| 0000010<br>000000 | 00004 SUM:<br>00005 | EQU 10H ; RAM loc 10H for SUM ORG 0H; start at address 0 |
|-------------------|---------------------|----------------------------------------------------------|
| 000000 0E25       | 00006               | MOVLW 25H ; $WREG = 25$                                  |
| 000002 OF34       | 00007               | ADDLW 0x34 ;add 34H to WREG=59H                          |
| 000004 OF11       | 00008               | ADDLW 11H ;add 11H to WREG=6AH                           |
| 000006 OF12       | 00009               | ADDLW d'18'; $W = W+12H=7CH$                             |
| 000008 OF1C       | 00010               | ADDLW 1CH; $W = W+1CH=98H$                               |
| 00000A 0F06       | 00011               | ADDLW b'00000110'; $W = W+6H=9EH$                        |
| 00000C 6E10       | 00012               | MOVWF SUM ; save the result in SUM location              |
| 00000E EF07 F000  | 00013 HERE:         | GOTO HERE ;stay here forever                             |
|                   | 00014               | END ; end of asm source file                             |



## If we change ORG 0H to ORG 20H, we have

| LOC OBJECT CODE  | LINE SOURCE TE | EXT VALUE                                   |
|------------------|----------------|---------------------------------------------|
| 00000010         | 00004 SUM:     | EQU 10H ; RAM loc 10H for SUM               |
| 000000           | 00005          | ORG 00H; start at address 0                 |
| 000000 EF10 F000 | 00006          | GOTO Main                                   |
| 000020           | 00007 Main:    | ORG 20H                                     |
| 000020 0E25      | 80000          | MOVLW 25H ; WREG = 25                       |
| 000022 0F34      | 00009          | ADDLW 0x34 ;add 34H to WREG=59H             |
| 000024 0F11      | 00010          | ADDLW 11H ;add 11H to WREG=6AH              |
| 000026 0F12      | 00011          | ADDLW $d'18'$ ; $W = W+12H=7CH$             |
| 000028 0F1C      | 00012          | ADDLW 1CH; $W = W+1CH=98H$                  |
| 00002A 0F06      | 00013          | ADDLW b'00000110'; $W = W+6H=9EH$           |
| 00002C 6E10      | 00014          | MOVWF SUM ; save the result in SUM location |
| 00002E EF17 F000 | 00015 HERE:    | GOTO HERE ;stay here forever                |
|                  | 00016          | END ; end of asm source file                |

#### Program ROM width

- Byte addressable: each location holds only one byte
   8-bit CPU will fetch one byte a time
   Increasing the data bus width will bring more information
- Solution: data bus between CPU and ROM is similar to traffic lanes on the highway
- 16-bit data bus
   Increase the processing power
   PIC18 can fetch instruction in 1 cycle



16-bit

ROM:

#### Instruction size of PIC18

- PIC instructions are 2-Byte or 4-Byte
- Most of PIC18 instructions are 2-Byte
- The first seven or eight bits represents the opcode
- The following eight bits is the operand

```
MOVLW 0000 1110 kkkk kkkk (0E XX)
ADDLW 0000 1111 kkkk kkkk (0F XX)
MOVWF 0110 111a ffff ffff (6E XX)
or (6F XX)
```

- a = 0 specifies the default access bank
- a = 1 specifies other bank

• 4-Byte instructions include

#### **GOTO**

```
1110 1111 k<sub>7</sub>kkk kkkk<sub>0</sub>
1111 k<sub>19</sub>kkk kkkk kkkk<sub>8</sub>
```

12-bit opcode, 20-bit target address  $(0 \le k \le FFFFF)$ But address is 21-bit!

MOVFF (move data within RAM) 1100 ssss ssss ssss  $(0 \le s \le FFF)$ 

1111 dddd dddd dddd ( $0 \le d \le FFF$ )

First 16-bit for opcode and address of source file register Second 16-bit for opcode and address of destination file register

## 3.1.6 flags and status register

- when CPU performs operations, sometimes an exception may occur, e.g. overflow
- How does the CPU tells control unit that an exception occurs?
- Answer flags
  - C Carry Flag
  - DC Digital Carry Flag
  - Zero Flag
  - OV Overflow Flag
  - N Negative Flag

## Effect of addlw on the status register

### Example 1:

```
MOVLW 0x38 38h
ADDLW 0x2F + 2Fh
67h
```

```
N = 0; bit7=0
```

OV=0; +ve + +ve = + ve => no overflow (in signed sense)

Z=0; NOT all zeros

DC=1; A carry from the first to second nibble

C=0; No carry

## Example 2:

N = 0; bit7=0

OV=0 ; -ve + +ve = + ve => no overflow (in signed sense)

Z=1; All zeros

DC=1; A carry from the first to second nibble

C=1; A carry is generated (in unsigned sense)

#### Example 3:

N = 0; bit7=0

OV=1; -ve + -ve = + ve => overflow (in signed sense)

Z=0; NOT all zeros

DC=0; No carry from the first to second nibble

C=1; A carry is generated (in unsigned sense)

#### Example 4:

```
N = 1 ; bit7=1
OV=1 ; +ve + +ve = -ve => overflow (in signed sense)
Z=0 ; NOT all zeros
DC=1; A carry from the first to second nibble
C=0 ; No carry
```

## Instructions That Affect Flag Bits

| Instruction | С | DC | Z | ov | N   |
|-------------|---|----|---|----|-----|
| ADDLW       | X | X  | X | X  | X   |
| ADDWF       | X | X  | X | X  | X   |
| ADDWFC      | X | X  | X | X  | X   |
| ANDLW       |   |    | X |    | X   |
| ANDWF       |   |    | X |    | X   |
| CLRF        |   |    | X |    |     |
| COMF        |   |    | X |    | X   |
| DAW         | X |    |   |    |     |
| DECF        | X | X  | X | X  | . X |
| INCF        | X | X  | X | X  | X   |
| IORLW       |   |    | X |    | X   |
| IORWF       |   |    | X |    | X   |
| MOVF        |   |    | X |    |     |
| NEGF        | X | X  | X | X  | X   |
| RLCF        | X |    | X |    | X   |
| RLNCF       |   |    | X |    | X   |
| RRCF        | X |    | X |    | X   |
| RRNCF       |   |    | X |    | X   |
| SUBFWB      | X | X  | X | X  | X   |
| SUBLW       | X | X  | X | X  | Χ.  |
| SUBWF       | X | X  | X | X  | X   |
| SUBWFB      | X | X  | X | X  | X   |
| XORLW       |   |    | X |    | X   |
| XORWF       |   |    | X |    | X   |

## Flag Bits and Decision Making

Status flags are also called conditions, there are instructions that will make a conditional Jump (branch) based on the status of the flag.

| Instruction | Action           |
|-------------|------------------|
| ВС          | Branch if C = 1  |
| BNC         | Branch if C = 0  |
| BZ          | Branch if Z = 1  |
| BNZ         | Branch if Z = 0  |
| BN          | Branch if N = 1  |
| BNN         | Branch if N = 0  |
| BOV         | Branch if OV = 1 |
| BNOV        | Branch if OV = 0 |

## 3.1.7 register banks

So far, we only consider the access bank in data memory. Actually, we can choose other banks

INCF MYREG, D, A
MOVWF MYREG, A

A=0, the access bank

A=1, other bank

To use this feature,

Load BSR with the desired bank number

Make A=1 in the instruction



#### **Example:**

```
MYREG EQU 0x40; define a location

MOVLB 0x2; load 2 into BSR (use bank 2)

MOVLW 0; WREG=0

MOVWF MYREG, 1; loc (240)=0, WREG=0, A=1

INCF MYREG,F,1; loc (240)=1, WREG=0, A=1

INCF MYREG,F,1; loc (240)=2, WREG=0, A=1

INCF MYREG,F,1; loc (240)=3, WREG=0, A=1
```

#### **Example:**

```
MYREG EQU 0x40; define a location

MOVLB 0x2; load 2 into BSR (use bank 2)

MOVLW 0; WREG=0

MOVWF MYREG; loc (40)=0, WREG=0, A=0

INCF MYREG,F; loc (40)=1, WREG=0, A=0

INCF MYREG,F; loc (40)=2, WREG=0, A=0

INCF MYREG,F; loc (40)=3, WREG=0, A=0
```

## Summary

- ◆ procedure to assemble and run PIC18 program
- ◆ program code in ROM
- ♦ status of flag bits in instruction execution
- ♦ selection of register bank