

# Microprocessor Interfacing & Programming

LECTURE 13 &14

# I/O Port Programming in PIC18

- ▶ PIC18F458 is a 40-pin chip.
- ▶ Total 33 pins are set aside for the five ports. PORTA, PORTB, PORTC, PORTD and PORTE.
- ▶ Several dedicated pins for power, clock, reset, and special features.
- ▶ To use any of the above ports as an input or output port, it must be programmed.
- ▶ In addition to being used for simple I/O, each port has some other functions such as ADC, timers, interrupts, and serial communication pins.



| Pins   | 18-pin     | 28-pin     | 40-pin    | 64-pin     | 80-pin     |
|--------|------------|------------|-----------|------------|------------|
| Chip   | PIC18F1220 | PIC18F2220 | PIC18F458 | PIC18F6525 | PIC18F8525 |
| Port A | X          | X          | X         | X          | X          |
| Port B | X          | X          | X         | X          | X          |
| Port C |            | X          | X         | X          | X          |
| Port D |            |            | X         | X          | X          |
| Port E |            |            | X         | X          | X          |
| Port F |            |            |           | X          | X          |
| Port G |            |            |           | X          | X          |
| Port H |            |            |           | X          | X          |
| Port J |            |            |           | X          | X          |
| Port K |            |            |           |            | X          |
| Port L |            |            |           |            | X          |

Note: X indicates that the port is available.

- ▶ Each port has **three** SFRs associated with it.

PORT<sub>x</sub> , LAT<sub>x</sub> and TRIS<sub>x</sub>

- ▶ Like for Port B we have, PORTB, TRISB and LATB.

TRIS ----- TRIState  
LAT ----- LATch

**Table 4-2: Ports' SFR Addresses for PIC18F458**

| <b>Port</b>  | <b>Address</b> |
|--------------|----------------|
| <b>PORTA</b> | <b>F80H</b>    |
| <b>PORTB</b> | <b>F81H</b>    |
| <b>PORTC</b> | <b>F82H</b>    |
| <b>PORTD</b> | <b>F83H</b>    |
| <b>PORTE</b> | <b>F84H</b>    |
| <b>LATA</b>  | <b>F89H</b>    |
| <b>LATB</b>  | <b>F8AH</b>    |
| <b>LATC</b>  | <b>F8BH</b>    |
| <b>LATD</b>  | <b>F8CH</b>    |
| <b>LATE</b>  | <b>F8DH</b>    |
| <b>TRISA</b> | <b>F92H</b>    |
| <b>TRISB</b> | <b>F93H</b>    |
| <b>TRISC</b> | <b>F94H</b>    |
| <b>TRISD</b> | <b>F95H</b>    |
| <b>TRISE</b> | <b>F96H</b>    |

|      |       |
|------|-------|
| 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  |

|      |         |
|------|---------|
| FA0h | PIE2    |
| FA1h | PIR2    |
| FA2h | IPR2    |
| FA3h | ---     |
| FA4h | ---     |
| FA5h | ---     |
| FA6h | ---     |
| FA7h | ---     |
| FA8h | ---     |
| FA9h | ---     |
| FAAh | ---     |
| FABh | RCSTA   |
| FACh | TXSTA   |
| FADh | TXREG   |
| FAEh | RCREG   |
| FAFh | SPBRG   |
| FB0h | ---     |
| FB1h | T3CON   |
| FB2h | TMR3L   |
| FB3h | TMR3H   |
| FB4h | ---     |
| FB5h | ---     |
| FB6h | ---     |
| FB7h | ---     |
| FB8h | ---     |
| FB9h | ---     |
| FBAh | CCP2CON |
| FBBh | CCPR2L  |
| FBCh | CCPR2H  |
| FBDh | CCP1CON |
| FBEh | CCPR1L  |
| FBFh | CCPR1H  |

|      |          |
|------|----------|
| FC0h | ---      |
| FC1h | ADCON1   |
| FC2h | ADCON0   |
| FC3h | ADRESL   |
| FC4h | ADRESH   |
| FC5h | SSPCON2  |
| FC6h | SSPCON1  |
| FC7h | SSPSTAT  |
| FC8h | SSPADD   |
| FC9h | SSPBUF   |
| FCAh | T2CON    |
| FCBh | PR2      |
| FCCh | TMR2     |
| FCDh | T1CON    |
| FCEh | TMR1L    |
| FCFh | TMR1H    |
| FD0h | RCON     |
| FD1h | WDTCON   |
| FD2h | LVDCON   |
| FD3h | OSCCON   |
| FD4h | ---      |
| FD5h | T0CON    |
| FD6h | TMR0L    |
| FD7h | TMR0H    |
| FD8h | STATUS   |
| FD9h | FSR2L    |
| FDAh | FSR2H    |
| FDBh | PLUSW2   |
| FDCh | PREINC2  |
| FDDh | POSTDEC2 |
| FDEh | POSTINC2 |
| FDFh | INDF2    |

|      |          |
|------|----------|
| FE0h | BSR      |
| FE1h | FSR1L    |
| FE2h | FSR1H    |
| FE3h | PLUSW1   |
| FE4h | PREINC1  |
| FE5h | POSTDEC1 |
| FE6h | POSTINC1 |
| FE7h | INDF1    |
| FE8h | WREG     |
| FE9h | FSR0L    |
| FEAh | FSR0H    |
| FEBh | PLUSW0   |
| FECh | PREINC0  |
| FEDh | POSTDEC0 |
| FEEh | POSTINC0 |
| FEFh | INDF0    |
| FF0h | INTCON3  |
| FF1h | INTCON2  |
| FF2h | INTCON   |
| FF3h | PRODL    |
| FF4h | PRODH    |
| FF5h | TABLAT   |
| FF6h | TBLPTRL  |
| FF7h | TBLPTRH  |
| FF8h | TBLPTRU  |
| FF9h | PCL      |
| FFAh | PCLATH   |
| FFBh | PCLATU   |
| FFCh | STKPTR   |
| FFDh | TOSL     |
| FFEh | TOSH     |
| FFFh | TOSU     |

# TRISx register role

- ▶ TRISx SFR dictates direction of pin ----- Input or Output.
- ▶ To make a port an output, we write 0s to the TRISx register.
- ▶ To output data to any of the pins of the Port B, we first put 0s into the TRISB register to make it an output port, and then send the data to the Port B SFR itself.

# Example

```
    MOVLW 0x0          ;WREG = 00
    MOVWF TRISB        ;make Port B an output port 0000 0000
L1    MOVLW 0x55       ;WREG = 55h
    MOVWF PORTB        ;put 55h on port B pins
    CALL  DELAY
    MOVLW 0xAA         ;WREG = AAh
    MOVWF PORTB        ;put AAh on port B pins
    CALL  DELAY
    GOTO  L1
```

# TRIS register for inputting data

Get the data at the pins of Port C and send it infinitely to Port B after adding the value 5 in it.

```
        MOVLW B'00000000' ;WREG = 0000000 (binary)
        MOVWF TRISB      ;Port B an output port(0 for 0)
        MOVLW B'11111111' ;WREG = 11111111 (binary)
        MOVWF TRISC      ;Port C an input port (1 for I)
L2    MOVF  PORTC,W   ;move data from Port C to WREG
        ADDLW 5          ;add some value to it
        MOVWF PORTB      ;send it to Port B
        GOTO L2          ;continue forever
```

# Efficient approach

```
CLRF TRISB      ;clear TRISB (Port B an output port)
SETF TRISC      ;set TRISC (Port C an input port)
L2  MOVF PORTC,W    ;get data from port C
          ADDLW 5      ;add some value
          MOVWF PORTB    ;send it to port B
          BRA  L2
```

# Port A

```
;toggle all bits of PORTA

    MOVLW B'00000000' ;WREG = 00000000 (binary)
    MOVWF TRISA        ;make Port A an output port (0 for Out)
L1     MOVLW 0x55      ;WREG = 55h
    MOVWF PORTA        ;put 55h on Port A pins
    CALL  DELAY
    MOVLW 0xAA          ;WREG = AAh
    MOVWF PORTA        ;put AAh on Port A pins
    CALL  DELAY
    GOTO  L1
```

It must be noted that 55H (01010101) when complemented becomes AAH (10101010). Although by sending 55H and AAH to Port A continuously, we toggle all 8 bits of the Port A register, only 6 pins (RA0–RA5) will show the toggling data.

# Port A as input

```
MYREG EQU 0X20 ;save it here

MOVLW B'11111111' ;WREG = 11111111 (binary)
MOVWF TRISA        ;make Port A an input port (1 for In)
MOVF PORTA,W       ;move from fileReg of Port A to WREG
MOVWF MYREG        ;save it in fileReg of MYREG
```

# Dual role of Ports A and B

**Table 4-3: Port A Alternate Functions**

| <b>Bit</b> | <b>Function</b> |
|------------|-----------------|
| RA0        | AN0/CVREF       |
| RA1        | AN1             |
| RA2        | AN2/VREF-       |
| RA3        | AN3/VREF+       |
| RA4        | T0CKI           |
| RA5        | AN4/SS/LVDIN    |
| RA6        | OSC2/CLKO       |

**Table 4-4: Port B Alternate Functions**

| <b>Bit</b> | <b>Function</b> |
|------------|-----------------|
| RB0        | INT0            |
| RB1        | INT1            |
| RB2        | INT2/CANTX      |
| RB3        | CANRX           |
| RB4        |                 |
| RB5        | PGM             |
| RB6        | PGC             |
| RB7        | PGD             |

# Dual role of Ports C and D

**Table 4-5: Port C Alternate Functions**

| <b>Bit</b> | <b>Function</b> |
|------------|-----------------|
| RC0        | T1OSO/T1CKI     |
| RC1        | T1OSI           |
| RC2        | CCP1            |
| RC3        | SCK/SCL         |
| RC4        | SDI/SDA         |
| RC5        | SDO             |
| RC6        | TX/CK           |
| RC7        | RX/DT           |

**Table 4-6: Port D Alternate Functions**

| <b>Bit</b> | <b>Function</b> |
|------------|-----------------|
| RD0        | PSP0/C1IN+      |
| RD1        | PSP1/C1IN-      |
| RD2        | PSP2/C2IN+      |
| RD3        | PSP3/C2IN-      |
| RD4        | PSP4/ECCP1/P1A  |
| RD5        | PSP5/P1B        |
| RD6        | PSP6/P1C        |
| RD7        | PSP7/P1D        |

# Different ways of accessing the entire 8 bits

```
;toggle all bits of PORTB  
  
L1    MOVLW 0x0          ;WREG = 00  
      MOVWF TRISB        ;make Port B an output port  
      MOVLW 0x55         ;WREG = 55h  
      MOVWF PORTB        ;put 55h on Port B pins  
      CALL DELAY  
      MOVLW 0xAA         ;WREG = AAh  
      MOVWF PORTB        ;put AAh on Port B pins  
      CALL DELAY  
      GOTO L1
```

read-modify-write technique →

```
L1    CLRF  TRISB       ;make Port B an output port  
      MOVLW 0x55         ;WREG = 55h  
      MOVWF PORTB        ;put 55h on Port B pins  
      CALL  DELAY  
      MOVLW 0xAA         ;WREG = AAh  
      MOVWF PORTB        ;put AAh on Port B pins  
      CALL  DELAY  
      GOTO  L1
```

```
L2    CLRF  TRISB       ;make Port B an output port  
      MOVLW 0x55         ;WREG = 55h  
      MOVWF PORTB        ;put 55h on Port B pins  
      COMF  PORTB, F     ;toggle bits of Port B  
      CALL  DELAY  
      BRA   L2
```

# Data Dependency in CPU Design

- ▶ Due to the timing issue, we must be careful not to have two I/O operations one right after the other.

```
    CLRF  TRISB ;clear TRISB to make PORTB an output port
    SETF  TRISC      ;set TRISC all 1s (Port C as Input)
L4    MOVF  PORTC,W   ;get data from Port C into WREG
    NOP           ;NEED some NOP to ensure data is in WREG
    MOVWF PORTB     ;before it is sent to Port B
    BRA   L4        ;keep doing it
```

**NOTE:** We need a NOP (or some other instruction) to make sure that the data is written into WREG before it is read for outputting to Port B.

# Solution of the problem

- ▶ This type of data dependency is referred as RAW (Read-After-Write).
- ▶ The NOP will introduce a bubble into the pipeline to remove data dependency due to RAW.
- ▶ One way to avoid this problem is to use the MOVFF instruction.

```
CLRF TRISB      ;make Port B an output port
SETF TRISC      ;TRISC = FFh (Port C Input)
L5    MOVFF PORTC,PORTB ;get from Port C and send to PORTB
      BRA L5       ;keep doing it
```

### INSTRUCTION



MOVF PORTC,W ;Read PORTC into WREG

MOVWF PORTB ;Write WREG to PORTB

The RAW (Read – After – Write) for two consecutive instructions .

### INSTRUCTION



MOVF PORTC,W

NOP ;Bubble in Pipeline  
MOVWF PORTB

N = No Operation

D = Decode the instruction

R = Read the operand

P = Process

W = Write the result to destination register

# Ports Status upon Reset

| <b>Register</b> | <b>Reset Value (Binary)</b> |
|-----------------|-----------------------------|
| TRISA           | 11111111                    |
| TRISB           | 11111111                    |
| TRISC           | 11111111                    |
| TRISD           | 11111111                    |

*Note:* All ports are input ports upon reset.

Write a test program for the PIC18 chip to toggle all the bits of PORTB, PORTC, and PORTD every 1/4 of a second. Assume a crystal frequency of 4 MHz.

**Solution:**

```
; tested with MPLAB for the PIC18F458 and XTAL = 4 MHz

list P=PIC18F458
#include P18F458.INC

R1 equ 0x07
R2 equ 0x08

ORG 0
    CLRF TRISB      ;make Port B an output port
    CLRF TRISC      ;make Port C an output port
    CLRF TRISD      ;make Port D an output port
    MOVLW 0x55      ;WREG = 55h
    MOVWF PORTB     ;put 55h on Port B pins
    MOVWF PORTC     ;put 55h on Port C pins
    MOVWF PORTD     ;put 55h on Port D pins
L3    COMF PORTB,F  ;toggle bits of Port B
    COMF PORTC,F  ;toggle bits of Port C
    COMF PORTD,F  ;toggle bits of Port D
    CALL QDELAY     ;quarter of a second delay
    BRA L3

;-----1/4 SECOND DELAY
QDELAY
    MOVLW D'200'
    MOVWF R1
D1    MOVLW D'250'
    MOVWF R2
D2    NOP
    NOP
    DECF R2, F
    BNZ D2
    DECF R1, F
    BNZ D1
    RETURN
END
```

# I/O Bit Manipulation programming

**Table 4-8: Single-Bit (Bit-Oriented) Instructions for PIC18**

| <b>Instruction</b> | <b>Function</b>                                                    |
|--------------------|--------------------------------------------------------------------|
| BSF fileReg,bit    | Bit Set fileReg (set the bit: bit = 1)                             |
| BCF fileReg,bit    | Bit Clear fileReg (clear the bit: bit = 0)                         |
| BTG fileReg,bit    | Bit Toggle fileReg (complement the bit)                            |
| BTFS fileReg,bit   | Bit test fileReg, skip if clear (skip next instruction if bit = 0) |
| BTFS fileReg,bit   | Bit test fileReg, skip if set (skip next instruction if bit = 1)   |

**Table 4-9: Single-Bit Addressability of Ports for PIC18F458/4580**

| PORT | PORTE | PORTE | PORTE | PORTE | Port Bit |
|------|-------|-------|-------|-------|----------|
| RA0  | RB0   | RC0   | RD0   | RE0   | D0       |
| RA1  | RB1   | RC1   | RD1   | RE1   | D1       |
| RA2  | RB2   | RC2   | RD2   | RE2   | D2       |
| RA3  | RB3   | RC3   | RD3   |       | D3       |
| RA4  | RB4   | RC4   | RD4   |       | D4       |
| RA5  | RB5   | RC5   | RD5   |       | D5       |
|      | RB6   | RC6   | RD6   |       | D6       |
|      | RB7   | RC7   | RD7   |       | D7       |

# BSF (bit set fileReg) & BCF (bit clear fileReg)

- ▶ Syntax: BSF fileReg, bit\_num
- ▶ Syntax: BCF fileReg, bit\_num

where, fileReg can be any location in file register and bit\_num is the desired bit number from 0 to 7.

**Remember:** For I/O ports, we must activate the appropriate bit in the TRISx register if we want the pin to reflect the changes.

An LED is connected to each pin of Port D. Write a program to turn on each LED from pin D0 to pin D7. Call a delay module before turning on the next LED.

**Solution:**

```
CLRF TRISD      ;make PORTD an output port
BSF  PORTD, 0   ;bit set turns on RD0
CALL DELAY
BSF  PORTD, 1   ;delay before next one
CALL DELAY
BSF  PORTD, 2
CALL DELAY
BSF  PORTD, 3
CALL DELAY
BSF  PORTD, 4
CALL DELAY
BSF  PORTD, 5
CALL DELAY
BSF  PORTD, 6
CALL DELAY
BSF  PORTD, 7
CALL DELAY
```

Ex: 4-2



# BCF (bit clear fileReg)

```
BCF TRISB, 2           ;bit = 0, make RB2 an output pin
AGAIN BSF PORTB, 2      ;bit set (RB2 = high)
    CALL DELAY
    BCF PORTB, 2          ;bit clear(RB2 = low)
    CALL DELAY
    BRA AGAIN
```

**Write the following programs:**

- (a) Create a square wave of 50% duty cycle on bit 0 of Port C.
- (b) Create a square wave of 66% duty cycle on bit 3 of Port C.

**Solution:**

- (a) The 50% duty cycle means that the “on” and “off” states (or the high and low portions of the pulse) have the same length. Therefore, we toggle RC0 with a time delay between each state.

```
HERE    BCF      TRISC, 0          ;clear TRIS bit for RC0 = out
        BSF      PORTC, 0          ;set to HIGH RC0 (RC0 = 1)
        CALL    DELAY             ;call the delay subroutine
        BCF      PORTC, 0          ;RC0 = 0
        CALL    DELAY             ;twice for 66%
        BRA     HERE              ;keep doing it
```

Another way to write the above program is:

```
HERE    BCF      TRISC, 0          ;make RC0 = out
        BTG      PORTC, 0          ;complement bit 0 of PORTC
        CALL    DELAY             ;call the delay subroutine
        BRA     HERE              ;keep doing it
```

PIC18F458

RC0



- (b) A 66% duty cycle means that the “on” state is twice the “off” state.

```
BACK   BCF      TRISC, 3          ;clear TRIS3 bit for output
        BSF      PORTC, 3          ;RC3 = 1
        CALL    DELAY             ;call the delay subroutine
        CALL    DELAY             ;twice for 66%
        BCF      PORTC, 3          ;RC3 = 0
        CALL    DELAY             ;call delay once for 33%
        BRA     BACK              ;keep doing it
```

PIC18F458

RC3



# BTG (bit toggle fileReg)

- ▶ Syntax: BTG fileReg , bit\_number

```
        BCF    TRISB, 2           ;make RB2 an output pin
BACK  BTG    PORTB, 2         ;toggle pin RB2 only
        CALL   DELAY
        BRA    BACK
```

**Notice:** RB2 is the third bit of Port B (the first bit is RB0, the second bit is RB1)