# PIC16F87XA ## 28/40/44-Pin Enhanced Flash Microcontrollers #### Devices Included in this Data Sheet: - PIC16F873A PIC16F876A - PIC16F874A PIC16F877A ### High-Performance RISC CPU: - · Only 35 single-word instructions to learn - All single-cycle instructions except for program branches, which are two-cycle - Operating speed: DC 20 MHz clock input DC – 200 ns instruction cycle - Up to 8K x 14 words of Flash Program Memory, Up to 368 x 8 bytes of Data Memory (RAM), Up to 256 x 8 bytes of EEPROM Data Memory - Pinout compatible to other 28-pin or 40/44-pin PIC16CXXX and PIC16FXXX microcontrollers #### Peripheral Features: - Timer0: 8-bit timer/counter with 8-bit prescaler - Timer1: 16-bit timer/counter with prescaler, can be incremented during Sleep via external crystal/clock - Timer2: 8-bit timer/counter with 8-bit period register, prescaler and postscaler - Two Capture, Compare, PWM modules - Capture is 16-bit, max. resolution is 12.5 ns - Compare is 16-bit, max. resolution is 200 ns - PWM max. resolution is 10-bit - Synchronous Serial Port (SSP) with SPI (Master mode) and I<sup>2</sup>C™ (Master/Slave) - Universal Synchronous Asynchronous Receiver Transmitter (USART/SCI) with 9-bit address detection - Parallel Slave Port (PSP) 8 bits wide with external RD, WR and CS controls (40/44-pin only) - Brown-out detection circuitry for Brown-out Reset (BOR) #### Analog Features: - 10-bit, up to 8-channel Analog-to-Digital Converter (A/D) - · Brown-out Reset (BOR) - · Analog Comparator module with: - Two analog comparators - Programmable on-chip voltage reference (VREF) module - Programmable input multiplexing from device inputs and internal voltage reference - Comparator outputs are externally accessible #### **Special Microcontroller Features:** - 100,000 erase/write cycle Enhanced Flash program memory typical - 1,000,000 erase/write cycle Data EEPROM memory typical - Data EEPROM Retention > 40 years - · Self-reprogrammable under software control - In-Circuit Serial Programming™ (ICSP™) via two pins - Single-supply 5V In-Circuit Serial Programming - Watchdog Timer (WDT) with its own on-chip RC oscillator for reliable operation - · Programmable code protection - · Power saving Sleep mode - Selectable oscillator options - In-Circuit Debug (ICD) via two pins #### CMOS Technology: - Low-power, high-speed Flash/EEPROM technology - · Fully static design - Wide operating voltage range (2.0V to 5.5V) - · Commercial and Industrial temperature ranges - · Low-power consumption | | Prog | ram Memory | Data | EEPROM | | 10-bit | ССР | MSSP | | | Timers | | |------------|-------|-------------------------------|-----------------|---------|-----|----------|-----|------|----------------------------|-------|----------|-------------| | Device | Bytes | # Single Word<br>Instructions | SRAM<br>(Bytes) | (Bytes) | I/O | A/D (ch) | | SPI | Master<br>I <sup>2</sup> C | USART | 8/16-bit | Comparators | | PIC16F873A | 7.2K | 4096 | 192 | 128 | 22 | 5 | 2 | Yes | Yes | Yes | 2/1 | 2 | | PIC16F874A | 7.2K | 4096 | 192 | 128 | 33 | 8 | 2 | Yes | Yes | Yes | 2/1 | 2 | | PIC16F876A | 14.3K | 8192 | 368 | 256 | 22 | 5 | 2 | Yes | Yes | Yes | 2/1 | 2 | | PIC16F877A | 14.3K | 8192 | 368 | 256 | 33 | 8 | 2 | Yes | Yes | Yes | 2/1 | 2 | #### 40-Pin PDIP #### 1.0 DEVICE OVERVIEW This document contains device specific information about the following devices: - PIC16F873A - PIC16F874A - PIC16F876A - PIC16F877A PIC16F873A/876A devices are available only in 28-pin packages, while PIC16F874A/877A devices are available in 40-pin and 44-pin packages. All devices in the PIC16F87XA family share common architecture with the following differences: - The PIC16F873A and PIC16F874A have one-half of the total on-chip memory of the PIC16F876A and PIC16F877A - The 28-pin devices have three I/O ports, while the 40/44-pin devices have five - The 28-pin devices have fourteen interrupts, while the 40/44-pin devices have fifteen - The 28-pin devices have five A/D input channels, while the 40/44-pin devices have eight - The Parallel Slave Port is implemented only on the 40/44-pin devices The available features are summarized in Table 1-1. Block diagrams of the PIC16F873A/876A and PIC16F874A/877A devices are provided in Figure 1-1 and Figure 1-2, respectively. The pinouts for these device families are listed in Table 1-2 and Table 1-3. Additional information may be found in the PIC® Mid-Range Reference Manual (DS33023), which may be obtained from your local Microchip Sales Representative or downloaded from the Microchip web site. The Reference Manual should be considered a complementary document to this data sheet and is highly recommended reading for a better understanding of the device architecture and operation of the peripheral modules. TABLE 1-1: PIC16F87XA DEVICE FEATURES | Key Features | PIC16F873A | PIC16F874A | PIC16F876A | PIC16F877A | |----------------------------------------|---------------------------------------------------------|---------------------------------------------------------|---------------------------------------------------------|---------------------------------------------------------| | Operating Frequency | DC - 20 MHz | DC - 20 MHz | DC - 20 MHz | DC - 20 MHz | | Resets (and Delays) | POR, BOR<br>(PWRT, OST) | POR, BOR<br>(PWRT, OST) | POR, BOR<br>(PWRT, OST) | POR, BOR<br>(PWRT, OST) | | Flash Program Memory<br>(14-bit words) | 4K | 4K | 8K | 8K | | Data Memory (bytes) | 192 | 192 | 368 | 368 | | EEPROM Data Memory (bytes) | 128 | 128 | 256 | 256 | | Interrupts | 14 | 15 | 14 | 15 | | I/O Ports | Ports A, B, C | Ports A, B, C, D, E | Ports A, B, C | Ports A, B, C, D, E | | Timers | 3 | 3 | 3 | 3 | | Capture/Compare/PWM modules | 2 | 2 | 2 | 2 | | Serial Communications | MSSP, USART | MSSP, USART | MSSP, USART | MSSP, USART | | Parallel Communications | _ | PSP | _ | PSP | | 10-bit Analog-to-Digital Module | 5 input channels | 8 input channels | 5 input channels | 8 input channels | | Analog Comparators | 2 | 2 | 2 | 2 | | Instruction Set | 35 Instructions | 35 Instructions | 35 Instructions | 35 Instructions | | Packages | 28-pin PDIP<br>28-pin SOIC<br>28-pin SSOP<br>28-pin QFN | 40-pin PDIP<br>44-pin PLCC<br>44-pin TQFP<br>44-pin QFN | 28-pin PDIP<br>28-pin SOIC<br>28-pin SSOP<br>28-pin QFN | 40-pin PDIP<br>44-pin PLCC<br>44-pin TQFP<br>44-pin QFN | #### FIGURE 1-1: PIC16F873A/876A BLOCK DIAGRAM Note 1: Higher order bits are from the Status register. #### FIGURE 1-2: PIC16F874A/877A BLOCK DIAGRAM Note 1: Higher order bits are from the Status register. TABLE 1-2: PIC16F873A/876A PINOUT DESCRIPTION | Pin Name | PDIP, SOIC,<br>SSOP Pin# | QFN<br>Pin# | I/O/P<br>Type | Buffer<br>Type | Description | |---------------------------------------------------------|--------------------------|-------------|----------------------------------|------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | OSC1/CLKI<br>OSC1<br>CLKI | 9 | 6 | I<br>I | ST/CMOS <sup>(3)</sup> | Oscillator crystal or external clock input. Oscillator crystal input or external clock source input. ST buffer when configured in RC mode; otherwise CMOS. External clock source input. Always associated with pin function OSC1 (see OSC1/CLKI, OSC2/CLKO pins). | | OSC2/CLKO<br>OSC2<br>CLKO | 10 | 7 | 0 | _ | Oscillator crystal or clock output. Oscillator crystal output. Connects to crystal or resonator in Crystal Oscillator mode. In RC mode, OSC2 pin outputs CLKO, which has 1/4 the frequency of OSC1 and denotes the instruction cycle rate. | | MCLR/VPP<br>MCLR<br>VPP | 1 | 26 | I<br>P | ST | Master Clear (input) or programming voltage (output). Master Clear (Reset) input. This pin is an active low Reset to the device. Programming voltage input. | | RA0/AN0<br>RA0<br>AN0 | 2 | 27 | I/O<br>I | TTL | PORTA is a bidirectional I/O port. Digital I/O. Analog input 0. | | RA1/AN1<br>RA1<br>AN1 | 3 | 28 | I/O<br>I | TTL | Digital I/O.<br>Analog input 1. | | RA2/AN2/VREF-/<br>CVREF<br>RA2<br>AN2<br>VREF-<br>CVREF | 4 | 1 | I/O<br> <br> <br> <br> <br> <br> | TTL | Digital I/O. Analog input 2. A/D reference voltage (Low) input. Comparator VREF output. | | RA3/AN3/VREF+<br>RA3<br>AN3<br>VREF+ | 5 | 2 | I/O<br>I<br>I | TTL | Digital I/O. Analog input 3. A/D reference voltage (High) input. | | RA4/T0CKI/C1OUT<br>RA4<br>T0CKI<br>C1OUT | 6 | 3 | I/O<br>I<br>O | ST | Digital I/O – Open-drain when configured as output. Timer0 external clock input. Comparator 1 output. | | RA5/AN4/SS/C2OUT<br>RA5<br>AN4<br>SS<br>C2OUT | 7 | 4 | I/O<br> <br> <br> <br> <br> | TTL | Digital I/O. Analog input 4. SPI slave select input. Comparator 2 output. | Legend: I = input O = output I/O = input/output - = Not used TTL = TTL input ST = Schmitt Trigger input Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt. 2: This buffer is a Schmitt Trigger input when used in Serial Programming mode. 3: This buffer is a Schmitt Trigger input when configured in RC Oscillator mode and a CMOS input otherwise. TABLE 1-2: PIC16F873A/876A PINOUT DESCRIPTION (CONTINUED) | IABLE 1-2. | FICTOR OF S | MOTOM | FINOU | DESCRIP | CKIPTION (CONTINUED) | | | | |------------|-----------------------|-------|---------------|-----------------------|---------------------------------------------------------------------------------------------------------------|--|--|--| | Pin Name | PDIP, SOI<br>SSOP Pir | | I/O/P<br>Type | Buffer<br>Type | Description | | | | | | | | | | PORTB is a bidirectional I/O port. PORTB can be software programmed for internal weak pull-ups on all inputs. | | | | | RB0/INT | 21 | 18 | | TTL/ST <sup>(1)</sup> | | | | | | RB0 | | | I/O | | Digital I/O. | | | | | INT | | | - 1 | | External interrupt. | | | | | RB1 | 22 | 19 | I/O | TTL | Digital I/O. | | | | | RB2 | 23 | 20 | I/O | TTL | Digital I/O. | | | | | RB3/PGM | 24 | 21 | | TTL | | | | | | RB3 | | | I/O | | Digital I/O. | | | | | PGM | | | - 1 | | Low-voltage (single-supply) ICSP programming enable pin. | | | | | RB4 | 25 | 22 | I/O | TTL | Digital I/O. | | | | | RB5 | 26 | 23 | I/O | TTL | Digital I/O. | | | | | RB6/PGC | 27 | 24 | | TTL/ST <sup>(2)</sup> | | | | | | RB6 | | | I/O | | Digital I/O. | | | | | PGC | | | - 1 | | In-circuit debugger and ICSP programming clock. | | | | | RB7/PGD | 28 | 25 | | TTL/ST <sup>(2)</sup> | | | | | | RB7 | | | I/O | | Digital I/O. | | | | | PGD | | | I/O | | In-circuit debugger and ICSP programming data. | | | | | | | | | - | | | | | | | | | | PORTC is a bidirectional I/O port. | |-------|----------------------------------------|------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------| | 11 | 8 | | ST | | | | | I/O | | Digital I/O. | | | | 0 | | Timer1 oscillator output. | | | | ı | | Timer1 external clock input. | | 12 | 9 | | ST | | | | | I/O | | Digital I/O. | | | | ı | | Timer1 oscillator input. | | | | I/O | | Capture2 input, Compare2 output, PWM2 output. | | 13 | 10 | | ST | | | | | I/O | | Digital I/O. | | | | I/O | | Capture1 input, Compare1 output, PWM1 output. | | 14 | 11 | | ST | | | | | I/O | | Digital I/O. | | | | I/O | | Synchronous serial clock input/output for SPI mode. | | | | I/O | | Synchronous serial clock input/output for I <sup>2</sup> C mode. | | 15 | 12 | | ST | | | | | I/O | | Digital I/O. | | | | I | | SPI data in. | | | | I/O | | I <sup>2</sup> C data I/O. | | 16 | 13 | | ST | | | | | | | Digital I/O. | | | | 0 | | SPI data out. | | 17 | 14 | | ST | | | | | I/O | | Digital I/O. | | | | _ | | USART asynchronous transmit. | | | | I/O | | USART1 synchronous clock. | | 18 | 15 | | ST | | | | | I/O | | Digital I/O. | | | | - 1 | | USART asynchronous receive. | | | | | | USART synchronous data. | | 8, 19 | 5, 6 | P | _ | Ground reference for logic and I/O pins. | | 20 | 17 | P | _ | Positive supply for logic and I/O pins. | | | 12<br>13<br>14<br>15<br>16<br>17<br>18 | 12 9 13 10 14 11 15 12 16 13 17 14 18 15 8, 19 5, 6 | 12 9 1/O O I I 1/O 1/O I | 12 9 ST | Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt. 2: This buffer is a Schmitt Trigger input when used in Serial Programming mode. 3: This buffer is a Schmitt Trigger input when configured in RC Oscillator mode and a CMOS input otherwise. TABLE 1-3: PIC16F874A/877A PINOUT DESCRIPTION | TABLE 1-3: PIC | PDIP | PLCC | | QFN | I/O/P | Buffer | | |-----------------------------------------------------|------|------|------|------|-----------------------------|------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | Pin Name | Pin# | Pin# | Pin# | Pin# | Туре | Туре | Description | | OSC1/CLKI<br>OSC1<br>CLKI | 13 | 14 | 30 | 32 | 1 | ST/CMOS <sup>(4)</sup> | Oscillator crystal or external clock input. Oscillator crystal input or external clock source input. ST buffer when configured in RC mode; otherwise CMOS. External clock source input. Always associated with pin function OSC1 (see OSC1/CLKI, OSC2/CLKO pins). | | OSC2/CLKO<br>OSC2 | 14 | 15 | 31 | 33 | 0 | _ | Oscillator crystal or clock output. Oscillator crystal output. Connects to crystal or resonator in Crystal | | CLKO | | | | | 0 | | Oscillator mode.<br>In RC mode, OSC2 pin outputs CLKO, which<br>has 1/4 the frequency of OSC1 and denotes the<br>instruction cycle rate. | | MCLR/VPP<br>MCLR<br>VPP | 1 | 2 | 18 | 18 | I<br>P | ST | Master Clear (input) or programming voltage (output). Master Clear (Reset) input. This pin is an active low Reset to the device. Programming voltage input. | | VFF | | | | | | | PORTA is a bidirectional I/O port. | | RA0/AN0<br>RA0<br>AN0 | 2 | 3 | 19 | 19 | I/O<br>I | TTL | Digital I/O. Analog input 0. | | RA1/AN1<br>RA1<br>AN1 | 3 | 4 | 20 | 20 | I/O<br>I | ΠL | Digital I/O.<br>Analog input 1. | | RA2/AN2/VREF-/CVREF<br>RA2<br>AN2<br>VREF-<br>CVREF | 4 | 5 | 21 | 21 | I/O<br> <br> <br> <br> | ΠL | Digital I/O. Analog input 2. A/D reference voltage (Low) input. Comparator VREF output. | | RA3/AN3/VREF+<br>RA3<br>AN3<br>VREF+ | 5 | 6 | 22 | 22 | I/O<br>I<br>I | TTL | Digital I/O. Analog input 3. A/D reference voltage (High) input. | | RA4/T0CKI/C1OUT<br>RA4 | 6 | 7 | 23 | 23 | I/O | ST | Digital I/O – Open-drain when configured as output. | | T0CKI<br>C1OUT | | | | | 0 | | Timer0 external clock input.<br>Comparator 1 output. | | RA5/AN4/SS/C2OUT<br>RA5<br>AN4<br>SS<br>C2OUT | 7 | 8 | 24 | 24 | I/O<br> <br> <br> <br> <br> | TTL | Digital I/O. Analog input 4. SPI slave select input. Comparator 2 output. | Legend: I = input O = output I/O = input/output P = power - = Not used TTL = TTL input ST = Schmitt Trigger input Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt. 2: This buffer is a Schmitt Trigger input when used in Serial Programming mode. 3: This buffer is a Schmitt Trigger input when configured in RC Oscillator mode and a CMOS input otherwise. TABLE 1-3: PIC16F874A/877A PINOUT DESCRIPTION (CONTINUED) | | THE THE THE TAKE THE THE TENED TO | | | | | | | | | |------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------|--------------|-------------|---------------|-----------------------|--------------------------------------------------------------|--|--| | Pin Name | PDIP<br>Pin# | PLCC<br>Pin# | TQFP<br>Pin# | QFN<br>Pin# | I/O/P<br>Type | Buffer<br>Type | Description | | | | | | | | | | | PORTB is a bidirectional I/O port. PORTB can be | | | | | | | | | | | software programmed for internal weak pull-up on all inputs. | | | | RB0/INT | 33 | 36 | 8 | 9 | | TTL/ST(1) | | | | | RB0<br>INT | | | | | I/O<br>I | | Digital I/O.<br>External interrupt. | | | | RB1 | 34 | 37 | 9 | 10 | I/O | TTL | Digital I/O. | | | | RB2 | 35 | 38 | 10 | 11 | I/O | TTL | Digital I/O. | | | | RB3/PGM | 36 | 39 | 11 | 12 | | TTL | | | | | RB3 | | | | | I/O | | Digital I/O. | | | | PGM | | | | | ı | | Low-voltage ICSP programming enable pin. | | | | RB4 | 37 | 41 | 14 | 14 | I/O | TTL | Digital I/O. | | | | RB5 | 38 | 42 | 15 | 15 | I/O | TTL | Digital I/O. | | | | RB6/PGC | 39 | 43 | 16 | 16 | | TTL/ST <sup>(2)</sup> | | | | | RB6 | | | | | I/O | | Digital I/O. | | | | PGC | | | | | I | | In-circuit debugger and ICSP programming clock. | | | | RB7/PGD | 40 | 44 | 17 | 17 | | TTL/ST <sup>(2)</sup> | | | | | RB7 | | | | | I/O | | Digital I/O. | | | | PGD | | | | | I/O | | In-circuit debugger and ICSP programming data. | | | Legend: I I = input - = Not used O = output TTL = TTL input nput ST = Schmitt Trigger input I/O = input/output P = power Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt. 2: This buffer is a Schmitt Trigger input when used in Serial Programming mode. 3: This buffer is a Schmitt Trigger input when configured in RC Oscillator mode and a CMOS input otherwise. TABLE 1-3: PIC16F874A/877A PINOUT DESCRIPTION (CONTINUED) | Pin Name | PDIP<br>Pin# | PLCC<br>Pin# | TQFP<br>Pin# | QFN<br>Pin# | I/O/P<br>Type | Buffer<br>Type | Description | |------------------------------------------|--------------|--------------|--------------|-------------|-----------------|----------------|-------------------------------------------------------------------------------------| | | | | | | | | PORTC is a bidirectional I/O port. | | RC0/T1OSO/T1CKI<br>RC0<br>T1OSO<br>T1CKI | 15 | 16 | 32 | 34 | I/O<br>O<br>I | ST | Digital I/O. Timer1 oscillator output. Timer1 external clock input. | | RC1/T1OSI/CCP2<br>RC1<br>T1OSI<br>CCP2 | 16 | 18 | 35 | 35 | I/O<br>I<br>I/O | ST | Digital I/O. Timer1 oscillator input. Capture2 input, Compare2 output, PWM2 output. | | RC2/CCP1<br>RC2<br>CCP1 | 17 | 19 | 36 | 36 | 1/0 | ST | Digital I/O. Capture1 input, Compare1 output, PWM1 output. | | RC3/SCK/SCL<br>RC3<br>SCK | 18 | 20 | 37 | 37 | I/O<br>I/O | ST | Digital I/O. Synchronous serial clock input/output for SPI mode. | | SCL | | | | | I/O | | Synchronous serial clock input/output for I <sup>2</sup> C mode. | | RC4/SDI/SDA<br>RC4<br>SDI<br>SDA | 23 | 25 | 42 | 42 | I/O<br>I<br>I/O | ST | Digital I/O.<br>SPI data in.<br>I <sup>2</sup> C data I/O. | | RC5/SDO<br>RC5<br>SDO | 24 | 26 | 43 | 43 | I/O<br>O | ST | Digital I/O.<br>SPI data out. | | RC6/TX/CK<br>RC6<br>TX<br>CK | 25 | 27 | 44 | 44 | I/O<br>O<br>I/O | ST | Digital I/O. USART asynchronous transmit. USART1 synchronous clock. | | RC7/RX/DT<br>RC7<br>RX<br>DT | 26 | 29 | 1 | 1 | I/O<br>I/O | ST | Digital I/O. USART asynchronous receive. USART synchronous data. | Legend: I = input t O = outpr I/O = input/output P = power - = Not used TTL = TTL input ST = Schmitt Trigger input Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt. 2: This buffer is a Schmitt Trigger input when used in Serial Programming mode. 3: This buffer is a Schmitt Trigger input when configured in RC Oscillator mode and a CMOS input otherwise. TABLE 1-3: PIC16F874A/877A PINOUT DESCRIPTION (CONTINUED) | Pin Name | PDIP<br>Pin# | PLCC<br>Pin# | TQFP<br>Pin# | QFN<br>Pin# | I/O/P<br>Type | Buffer<br>Type | Description | |-------------------|--------------|------------------|------------------|--------------|---------------|-----------------------|---------------------------------------------------------------------------------| | | | | | | | | PORTD is a bidirectional I/O port or Parallel Slave | | | | | | | | (2) | Port when interfacing to a microprocessor bus. | | RD0/PSP0 | 19 | 21 | 38 | 38 | I/O | ST/TTL <sup>(3)</sup> | Digital I/O | | RD0<br>PSP0 | | | | | 1/0 | | Digital I/O. Parallel Slave Port data. | | RD1/PSP1 | 20 | 22 | 39 | 39 | " | ST/TTL <sup>(3)</sup> | Talaliel Glave Folt data. | | RD1 | 20 | 22 | 39 | 39 | Ι/O | SITTLE | Digital I/O. | | PSP1 | | | | | I/O | | Parallel Slave Port data. | | RD2/PSP2 | 21 | 23 | 40 | 40 | | ST/TTL <sup>(3)</sup> | | | RD2 | | | | | I/O | | Digital I/O. | | PSP2 | | | | | I/O | | Parallel Slave Port data. | | RD3/PSP3 | 22 | 24 | 41 | 41 | | ST/TTL <sup>(3)</sup> | | | RD3 | | | | | I/O | | Digital I/O. Parallel Slave Port data. | | PSP3 | | | | _ | I/O | ST/TTL <sup>(3)</sup> | Parallel Slave Port data. | | RD4/PSP4<br>RD4 | 27 | 30 | 2 | 2 | I/O | SI/IIL | Digital I/O. | | PSP4 | | | | | I/O | | Parallel Slave Port data. | | RD5/PSP5 | 28 | 31 | 3 | 3 | | ST/TTL <sup>(3)</sup> | | | RD5 | | | | | I/O | | Digital I/O. | | PSP5 | | | | | I/O | | Parallel Slave Port data. | | RD6/PSP6 | 29 | 32 | 4 | 4 | | ST/TTL <sup>(3)</sup> | | | RD6 | | | | | I/O | | Digital I/O. | | PSP6 | | | | | I/O | (2) | Parallel Slave Port data. | | RD7/PSP7 | 30 | 33 | 5 | 5 | | ST/TTL <sup>(3)</sup> | Distriction | | RD7<br>PSP7 | | | | | I/O | | Digital I/O. Parallel Slave Port data. | | 1017 | | | | | | | PORTE is a bidirectional I/O port. | | RE0/RD/AN5 | 8 | 9 | 25 | 25 | | ST/TTL <sup>(3)</sup> | PORTE IS a bidirectional tro port. | | RE0 | • | 9 | 25 | 25 | VO | SITTLE | Digital I/O. | | RD | | | | | ı | | Read control for Parallel Slave Port. | | AN5 | | | | | - 1 | | Analog input 5. | | RE1/WR/AN6 | 9 | 10 | 26 | 26 | | ST/TTL <sup>(3)</sup> | | | RE1 | | | | | I/O | | Digital I/O. | | WR | | | | | | | Write control for Parallel Slave Port. | | AN6 | 40 | | | | ' | ST/TTL <sup>(3)</sup> | Analog input 6. | | RE2/CS/AN7<br>RE2 | 10 | 11 | 27 | 27 | I/O | SI/ITL | Digital I/O. | | CS | | | | | 1 | | Chip select control for Parallel Slave Port. | | AN7 | | | | | i | | Analog input 7. | | Vss | 12, 31 | 13, 34 | 6, 29 | 6, 30,<br>31 | Р | _ | Ground reference for logic and I/O pins. | | VDD | 11, 32 | 12, 35 | 7, 28 | 7, 8, | Р | _ | Positive supply for logic and I/O pins. | | - | | , | , == | 28, 29 | | | | | NC | _ | 1, 17,<br>28, 40 | 12,13,<br>33, 34 | 13 | _ | _ | These pins are not internally connected. These pins should be left unconnected. | Legend: I = input O = output I/O = input/output P = power - = Not used TTL = TTL input ST = Schmitt Trigger input Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt. 2: This buffer is a Schmitt Trigger input when used in Serial Programming mode. 3: This buffer is a Schmitt Trigger input when configured in RC Oscillator mode and a CMOS input otherwise. #### 2.0 MEMORY ORGANIZATION There are three memory blocks in each of the PIC16F87XA devices. The program memory and data memory have separate buses so that concurrent access can occur and is detailed in this section. The EEPROM data memory block is detailed in **Section 3.0** "Data EEPROM and Flash Program Memory". Additional information on device memory may be found in the PIC® Mid-Range MCU Family Reference Manual (DS33023). ### 2.1 Program Memory Organization The PIC16F87XA devices have a 13-bit program counter capable of addressing an 8K word x 14 bit program memory space. The PIC16F876A/877A devices have 8K words x 14 bits of Flash program memory, while PIC16F873A/874A devices have 4K words x 14 bits. Accessing a location above the physically implemented address will cause a wraparound. The Reset vector is at 0000h and the interrupt vector is at 0004h. FIGURE 2-1: PIC16F876A/877A PROGRAM MEMORY MAP AND STACK FIGURE 2-2: PIC16F873A/874A PROGRAM MEMORY MAP AND STACK #### 2.2 Data Memory Organization The data memory is partitioned into multiple banks which contain the General Purpose Registers and the Special Function Registers. Bits RP1 (Status<6>) and RP0 (Status<5>) are the bank select bits. | RP1:RP0 | Bank | |---------|------| | 00 | 0 | | 01 | 1 | | 10 | 2 | | 11 | 3 | Each bank extends up to 7Fh (128 bytes). The lower locations of each bank are reserved for the Special Function Registers. Above the Special Function Registers are General Purpose Registers, implemented as static RAM. All implemented banks contain Special Function Registers. Some frequently used Special Function Registers from one bank may be mirrored in another bank for code reduction and quicker access. | Note: | The EEPROM data memory description can | |-------|----------------------------------------| | | be found in Section 3.0 "Data EEPROM | | | and Flash Program Memory" of this data | | | sheet. | ## 2.2.1 GENERAL PURPOSE REGISTER FILE The register file can be accessed either directly, or indirectly, through the File Select Register (FSR). FIGURE 2-3: PIC16F876A/877A REGISTER FILE MAP | | File<br>Address | A | File<br>Address | | File<br>Address | , | File<br>Address | |----------------------|-----------------|--------------------------------------------|-----------------|--------------------------------|-----------------|--------------------------------------------|-----------------| | Indirect addr.(*) | 00h | Indirect addr.(*) | 80h | Indirect addr.(*) | 100h | Indirect addr.(*) | 180h | | TMR0 | 01h | OPTION_REG | 81h | TMR0 | 101h | OPTION_REG | 181h | | PCL | 02h | PCL | 82h | PCL | 102h | PCL | 182h | | STATUS | 03h | STATUS | 83h | STATUS | 103h | STATUS | 183h | | FSR | 04h | FSR | 84h | FSR | 104h | FSR | 184h | | PORTA | 05h | TRISA | 85h | | 105h | | 185h | | PORTB | 06h | TRISB | 86h | PORTB | 106h | TRISB | 186h | | PORTC | 07h | TRISC | 87h | | 107h | | 187h | | PORTD <sup>(1)</sup> | 08h | TRISD(1) | 88h | | 108h | | 188h | | PORTE <sup>(1)</sup> | 09h | TRISE(1) | 89h | | 109h | | 189h | | PCLATH | 0Ah | PCLATH | 8Ah | PCLATH | 10Ah | PCLATH | 18Ah | | INTCON | 0Bh | INTCON | 8Bh | INTCON | 10Bh | INTCON | 18Bh | | PIR1 | 0Ch | PIE1 | 8Ch | EEDATA | 10Ch | EECON1 | 18Ch | | PIR2 | 0Dh | PIE2 | 8Dh | EEADR | 10Dh | EECON2 | 18Dh | | TMR1L | 0Eh | PCON | 8Eh | EEDATH | 10Eh | Reserved <sup>(2)</sup> | 18Eh | | TMR1H | 0Fh | | 8Fh | EEADRH | 10Fh | Reserved <sup>(2)</sup> | 18Fh | | T1CON | 10h | | 90h | | 110h | | 190h | | TMR2 | 11h | SSPCON2 | 91h | | 111h | | 191h | | T2CON | 12h | PR2 | 92h | | 112h | | 192h | | SSPBUF | 13h | SSPADD | 93h | | 113h | | 193h | | SSPCON | 14h | SSPSTAT | 94h | | 114h | | 194h | | CCPR1L | 15h | | 95h | | 115h | | 195h | | CCPR1H | 16h | | 96h | | 116h | | 196h | | CCP1CON | 17h | | 97h | General<br>Purpose | 117h | General<br>Purpose | 197h | | RCSTA | 18h | TXSTA | 98h | Register | 118h | Register | 198h | | TXREG | 19h | SPBRG | 99h | 16 Bytes | 119h | 16 Bytes | 199h | | RCREG | 1Ah | | 9Ah | | 11Ah | | 19Ah | | CCPR2L | 1Bh | | 9Bh | | 11Bh | | 19Bh | | CCPR2H | 1Ch | CMCON | 9Ch | | 11Ch | | 19Ch | | CCP2CON | 1Dh | CVRCON | 9Dh | | 11Dh | | 19Dh | | ADRESH | 1Eh | ADRESL | 9Eh | | 11Eh | | 19Eh | | ADCON0 | 1Fh | ADCON1 | 9Fh | | 11Fh | | 19Fh | | | 20h | | A0h | | 120h | | 1A0h | | General<br>Purpose | | General<br>Purpose<br>Register<br>80 Bytes | | General<br>Purpose<br>Register | | General<br>Purpose<br>Register<br>80 Bytes | | | Register | | oo bytes | | 80 Bytes | | oo bytes | | | 96 Bytes | | | EFh | | 16Fh | | 1EFh | | | 7Fh | accesses<br>70h-7Fh | F0h<br>FFh | accesses<br>70h-7Fh | 170h<br>17Fh | accesses<br>70h - 7Fh | 1F0h<br>1FFh | | Bank 0 | | Bank 1 | | Bank 2 | | Bank 3 | | Unimplemented data memory locations, read as '0'. Note 1: These registers are not implemented on the PIC16F876A. Not a physical register. <sup>2:</sup> These registers are reserved; maintain these registers clear. FIGURE 2-4: PIC16F873A/874A REGISTER FILE MAP | , | File<br>Address | | File<br>Address | | File<br>Address | | File<br>Addr | |--------------------------------|-----------------|--------------------------------------------------|-----------------|-------------------|-----------------|-------------------------|--------------| | Indirect addr.(*) | 00h | Indirect addr.(*) | 80h | Indirect addr.(*) | 100h | Indirect addr.(*) | 180 | | TMR0 | 01h | OPTION_REG | 81h | TMR0 | 101h | OPTION_REG | 181 | | PCL | 02h | PCL | 82h | PCL | 102h | PCL | 182 | | STATUS | 03h | STATUS | 83h | STATUS | 103h | STATUS | 183 | | FSR | 04h | FSR | 84h | FSR | 104h | FSR | 184 | | PORTA | 05h | TRISA | 85h | | 105h | | 185 | | PORTB | 06h | TRISB | 86h | PORTB | 106h | TRISB | 186 | | PORTC | 07h | TRISC | 87h | | 107h | | 187 | | PORTD <sup>(1)</sup> | 08h | TRISD <sup>(1)</sup> | 88h | | 108h | | 188 | | PORTE <sup>(1)</sup> | 09h | TRISE(1) | 89h | | 109h | | 189 | | PCLATH | 0Ah | PCLATH | 8Ah | PCLATH | 10Ah | PCLATH | 18/ | | INTCON | 0Bh | INTCON | 8Bh | INTCON | 10Bh | INTCON | 186 | | PIR1 | 0Ch | PIE1 | 8Ch | EEDATA | 10Ch | EECON1 | 180 | | PIR2 | 0Dh | PIE2 | 8Dh | EEADR | 10Dh | EECON2 | 180 | | TMR1L | 0Eh | PCON | 8Eh | EEDATH | 10Eh | Reserved <sup>(2)</sup> | 181 | | TMR1H | 0Fh | | 8Fh | EEADRH | 10Fh | Reserved <sup>(2)</sup> | 181 | | T1CON | 10h | | 90h | | 110h | | 190 | | TMR2 | 11h | SSPCON2 | 91h | | | | | | T2CON | 12h | PR2 | 92h | | | | | | SSPBUF | 13h | SSPADD | 93h | | | | | | SSPCON | 14h | SSPSTAT | 94h | | | | | | CCPR1L | 15h | | 95h | | | | | | CCPR1H | 16h | | 96h | | | | | | CCP1CON | 17h | | 97h | | | | | | RCSTA | 18h | TXSTA | 98h | | | | | | TXREG | 19h | SPBRG | 99h | | | | | | RCREG | 1Ah | | 9Ah | | | | | | CCPR2L | 1Bh | | 9Bh | | | | | | CCPR2H | 1Ch | CMCON | 9Ch | | | | | | CCP2CON | 1Dh | CVRCON | 9Dh | | | | | | ADRESH | 1Eh | ADRESL | 9Eh | | | | | | ADCON0 | 1Fh | ADCON1 | 9Fh | | | | | | 7.500.10 | 20h | | A0h | | 120h | | 1A( | | General<br>Purpose<br>Register | | General<br>Purpose<br>Register | | accesses | | accesses | | | 96 Bytes | | | | 20h-7Fh | 1055 | A0h - FFh | 1EF | | 30 Dyles | | 96 Bytes | | | 16Fh<br>170h | | 1F0 | | David 0 | 7Fh | Dort d | FFh | Donk 0 | 17Fh | Dork 2 | 1FF | | Bank 0 | | Bank 1 | | Bank 2 | | Bank 3 | | | * Not a | physical re | data memory loca<br>gister.<br>are not implement | | | | | | #### 2.2.2 SPECIAL FUNCTION REGISTERS The Special Function Registers are registers used by the CPU and peripheral modules for controlling the desired operation of the device. These registers are implemented as static RAM. A list of these registers is given in Table 2-1. The Special Function Registers can be classified into two sets: core (CPU) and peripheral. Those registers associated with the core functions are described in detail in this section. Those related to the operation of the peripheral features are described in detail in the peripheral features section. TABLE 2-1: SPECIAL FUNCTION REGISTER SUMMARY | Address | Name | Bit 7 | Bit 6 | Bit 5 | Bit 4 | Bit 3 | Bit 2 | Bit 1 | Bit 0 | | e on:<br>BOR | Details on page: | |----------------------|---------|-------------|----------------|---------------|----------------|-----------------|-----------------|--------------|---------------|------|--------------|------------------| | Bank 0 | | | | | | | | | | | | | | 00h <sup>(3)</sup> | INDF | Addressing | g this locatio | n uses cont | ents of FSR t | o address da | ata memory ( | not a physic | cal register) | 0000 | 0000 | 31, 150 | | 01h | TMR0 | Timer0 Mo | dule Registe | er | | | | | | xxxx | xxxx | 55, 150 | | 02h <sup>(3)</sup> | PCL | Program C | Counter (PC) | Least Signi | ficant Byte | | | | | 0000 | 0000 | 30, 150 | | 03h <sup>(3)</sup> | STATUS | IRP | RP1 | RP0 | TO | PD | Z | DC | С | 0001 | 1xxx | 22, 150 | | 04h <sup>(3)</sup> | FSR | Indirect Da | ta Memory | Address Poi | inter | | | | | xxxx | xxxx | 31, 150 | | 05h | PORTA | _ | _ | PORTA Da | ta Latch whe | en written: PO | ORTA pins w | hen read | | 0x | 0000 | 43, 150 | | 06h | PORTB | PORTB Da | ata Latch wh | en written: | PORTB pins | when read | | | | xxxx | xxxx | 45, 150 | | 07h | PORTC | PORTC Da | ata Latch wh | en written: | PORTC pins | when read | | | | xxxx | xxxx | 47, 150 | | 08h <sup>(4)</sup> | PORTD | PORTD Da | ata Latch wh | en written: | PORTD pins | when read | | | | xxxx | xxxx | 48, 150 | | 09h <sup>(4)</sup> | PORTE | _ | _ | _ | _ | _ | RE2 | RE1 | RE0 | | -xxx | 49, 150 | | 0Ah <sup>(1,3)</sup> | PCLATH | _ | _ | _ | Write Buffer | for the uppe | r 5 bits of the | Program C | Counter | 0 | 0000 | 30, 150 | | 0Bh <sup>(3)</sup> | INTCON | GIE | PEIE | TMR0IE | INTE | RBIE | TMR0IF | INTF | RBIF | 0000 | 000x | 24, 150 | | 0Ch | PIR1 | PSPIF(3) | ADIF | RCIF | TXIF | SSPIF | CCP1IF | TMR2IF | TMR1IF | 0000 | 0000 | 26, 150 | | 0Dh | PIR2 | _ | CMIF | _ | EEIF | BCLIF | _ | _ | CCP2IF | -0-0 | 00 | 28, 150 | | 0Eh | TMR1L | Holding Re | egister for th | e Least Sigr | nificant Byte | of the 16-bit | TMR1 Regis | ter | | xxxx | xxxx | 60, 150 | | 0Fh | TMR1H | Holding Re | egister for th | e Most Sign | ificant Byte o | of the 16-bit 1 | TMR1 Regist | er | | xxxx | xxxx | 60, 150 | | 10h | T1CON | _ | _ | T1CKPS1 | T1CKPS0 | T10SCEN | T1SYNC | TMR1CS | TMR10N | 00 | 0000 | 57, 150 | | 11h | TMR2 | Timer2 Mo | dule Registe | er | | | | | | 0000 | 0000 | 62, 150 | | 12h | T2CON | _ | TOUTPS3 | TOUTPS2 | TOUTPS1 | TOUTPS0 | TMR2ON | T2CKPS1 | T2CKPS0 | -000 | 0000 | 61, 150 | | 13h | SSPBUF | Synchrono | us Serial Po | ort Receive I | Buffer/Transr | mit Register | | | | xxxx | xxxx | 79, 150 | | 14h | SSPCON | WCOL | SSPOV | SSPEN | CKP | SSPM3 | SSPM2 | SSPM1 | SSPM0 | 0000 | 0000 | 82, 82,<br>150 | | 15h | CCPR1L | Capture/Co | ompare/PW | M Register | 1 (LSB) | | | | | xxxx | xxxx | 63, 150 | | 16h | CCPR1H | Capture/C | ompare/PW | M Register | 1 (MSB) | | | | | xxxx | xxxx | 63, 150 | | 17h | CCP1CON | _ | _ | CCP1X | CCP1Y | CCP1M3 | CCP1M2 | CCP1M1 | CCP1M0 | 00 | 0000 | 64, 150 | | 18h | RCSTA | SPEN | RX9 | SREN | CREN | ADDEN | FERR | OERR | RX9D | 0000 | 000x | 112, 150 | | 19h | TXREG | USART Tra | ansmit Data | Register | | | | | | 0000 | 0000 | 118, 150 | | 1Ah | RCREG | USART RE | eceive Data | Register | | | | | | 0000 | 0000 | 118, 150 | | 1Bh | CCPR2L | Capture/C | ompare/PW | M Register: | 2 (LSB) | | | | | xxxx | xxxx | 63, 150 | | 1Ch | CCPR2H | Capture/C | ompare/PW | M Register | 2 (MSB) | | | | | xxxx | xxxx | 63, 150 | | 1Dh | CCP2CON | _ | _ | CCP2X | CCP2Y | CCP2M3 | CCP2M2 | CCP2M1 | CCP2M0 | 00 | 0000 | 64, 150 | | 1Eh | ADRESH | A/D Result | Register Hi | gh Byte | | | | | | xxxx | xxxx | 133, 150 | | 1Fh | ADCON0 | ADCS1 | ADCS0 | CHS2 | CHS1 | CHS0 | GO/DONE | _ | ADON | 0000 | 00-0 | 127, 150 | Legend: x = unknown, u = unchanged, q = value depends on condition, - = unimplemented, read as '0', r = reserved. Shaded locations are unimplemented, read as '0'. Note 1: The upper byte of the program counter is not directly accessible. PCLATH is a holding register for the PC<12:8>, whose contents are transferred to the upper byte of the program counter. - 2: Bits PSPIE and PSPIF are reserved on PIC16F873A/876A devices; always maintain these bits clear. - These registers can be addressed from any bank. PORTD, PORTE, TRISD and TRISE are not implemented on PIC16F873A/876A devices, read as '0'. - 5: Bit 4 of EEADRH implemented only on the PIC16F876A/877A devices. TABLE 2-1: SPECIAL FUNCTION REGISTER SUMMARY (CONTINUED) | IABLE | 2-1. OF | LUIAL I | UNCTIC | IN KEGI | SIER SU | INIMIAI | (CONTIN | NOLD) | | | | |----------------------|------------|----------------------|----------------|---------------------------|----------------|--------------|------------------|--------------|---------------|-----------------------|---------------------| | Address | Name | Bit 7 | Bit 6 | Bit 5 | Bit 4 | Bit 3 | Bit 2 | Bit 1 | Bit 0 | Value on:<br>POR, BOR | Details<br>on page: | | Bank 1 | | | | | | | | | | | | | 80h <sup>(3)</sup> | INDF | Addressing | g this locatio | n uses cont | ents of FSR t | o address d | ata memory ( | not a physic | cal register) | 0000 0000 | 31, 150 | | 81h | OPTION_REG | RBPU | INTEDG | TOCS | TOSE | PSA | PS2 | PS1 | PS0 | 1111 1111 | 23, 150 | | 82h <sup>(3)</sup> | PCL | Program C | counter (PC) | Least Sign | ificant Byte | | | | | 0000 0000 | 30, 150 | | 83h <sup>(3)</sup> | STATUS | IRP | RP1 | RP0 | TO | PD | Z | DC | С | 0001 1xxx | 22, 150 | | 84h <sup>(3)</sup> | FSR | Indirect Da | ata Memory | Address Po | inter | | | | | xxxx xxxx | 31, 150 | | 85h | TRISA | _ | _ | PORTA Da | ta Direction F | Register | | | | 11 1111 | 43, 150 | | 86h | TRISB | PORTB D | ata Direction | Register | | | | | | 1111 1111 | 45, 150 | | 87h | TRISC | PORTC D | ata Direction | Register | | | | | | 1111 1111 | 47, 150 | | 88h <sup>(4)</sup> | TRISD | PORTD D | ata Direction | Register | | | | | | 1111 1111 | 48, 151 | | 89h <sup>(4)</sup> | TRISE | IBF | OBF | IBOV | PSPMODE | _ | PORTE Date | a Direction | bits | 0000 -111 | 50, 151 | | 8Ah <sup>(1,3)</sup> | PCLATH | _ | _ | _ | Write Buffer | for the uppe | er 5 bits of the | Program C | ounter | 0 0000 | 30, 150 | | 8Bh <sup>(3)</sup> | INTCON | GIE | PEIE | TMR0IE | INTE | RBIE | TMR0IF | INTF | RBIF | 0000 000x | 24, 150 | | 8Ch | PIE1 | PSPIE <sup>(2)</sup> | ADIE | RCIE | TXIE | SSPIE | CCP1IE | TMR2IE | TMR1IE | 0000 0000 | 25, 151 | | 8Dh | PIE2 | _ | CMIE | _ | EEIE | BCLIE | _ | _ | CCP2IE | -0-0 00 | 27, 151 | | 8Eh | PCON | _ | _ | _ | _ | _ | _ | POR | BOR | qq | 29, 151 | | 8Fh | _ | Unimplem | ented | | • | | | | | _ | _ | | 90h | _ | Unimplem | ented | | | | | | | _ | _ | | 91h | SSPCON2 | GCEN | ACKSTAT | ACKDT | ACKEN | RCEN | PEN | RSEN | SEN | 0000 0000 | 83, 151 | | 92h | PR2 | Timer2 Pe | riod Registe | r | | | | | | 1111 1111 | 62, 151 | | 93h | SSPADD | Synchrono | ous Serial Po | ort (I <sup>2</sup> C mod | e) Address R | tegister | | | | 0000 0000 | 79, 151 | | 94h | SSPSTAT | SMP | CKE | D/A | Р | S | R/W | UA | BF | 0000 0000 | 79, 151 | | 95h | _ | Unimplem | ented | | | | | | | _ | _ | | 96h | _ | Unimplem | ented | | | | | | | _ | _ | | 97h | _ | Unimplem | ented | | | | | | | _ | _ | | 98h | TXSTA | CSRC | TX9 | TXEN | SYNC | _ | BRGH | TRMT | TX9D | 0000 -010 | 111, 151 | | 99h | SPBRG | Baud Rate | Generator I | Register | | | | | | 0000 0000 | 113, 151 | | 9Ah | _ | Unimplem | ented | | | | | | | _ | _ | | 9Bh | _ | Unimplem | ented | | | | | | | _ | _ | | 9Ch | CMCON | C2OUT | C10UT | C2INV | C1INV | CIS | CM2 | CM1 | CM0 | 0000 0111 | 135, 151 | | 9Dh | CVRCON | CVREN | CVROE | CVRR | _ | CVR3 | CVR2 | CVR1 | CVR0 | 000- 0000 | 141, 151 | | 9Eh | ADRESL | A/D Result | t Register Lo | w Byte | | | | | | xxxx xxxx | 133, 151 | | 9Fh | ADCON1 | ADFM | ADCS2 | _ | _ | PCFG3 | PCFG2 | PCFG1 | PCFG0 | 00 0000 | 128, 151 | | | | | | | | | | | | | | Legend: x = unknown, u = unchanged, q = value depends on condition, - = unimplemented, read as '0', r = reserved. Shaded locations are unimplemented, read as '0'. Note 1: The upper byte of the program counter is not directly accessible. PCLATH is a holding register for the PC<12:8>, whose contents are transferred to the upper byte of the program counter. - 2: Bits PSPIE and PSPIF are reserved on PIC16F873A/876A devices; always maintain these bits clear. - 3: These registers can be addressed from any bank. - 4: PORTD, PORTE, TRISD and TRISE are not implemented on PIC16F873A/876A devices, read as '0'. - 5: Bit 4 of EEADRH implemented only on the PIC16F876A/877A devices. TABLE 2-1: SPECIAL FUNCTION REGISTER SUMMARY (CONTINUED) | Address | Name | Bit 7 | Bit 6 | Bit 5 | Bit 4 | Bit 3 | Bit 2 | Bit 1 | Bit 0 | Value on:<br>POR, BOR | Details on page: | |-----------------------|------------|-------------|----------------------------------------------------------|--------------|---------------|--------------|------------------|--------------|---------------|-----------------------|------------------| | Bank 2 | | | | | | | | | | | | | 100h <sup>(3)</sup> | INDF | Addressing | this locatio | n uses cont | ents of FSR t | o address d | ata memory ( | not a physic | cal register) | 0000 0000 | 31, 150 | | 101h | TMR0 | Timer0 Mo | r0 Module Register | | | | | | | | 55, 150 | | 102h <sup>(3)</sup> | PCL | Program C | ounter's (Po | C) Least Sig | nificant Byte | | | | | 0000 0000 | 30, 150 | | 103h <sup>(3)</sup> | STATUS | IRP | RP RP1 RP0 TO PD Z DC C | | | | | | | 0001 1xxx | 22, 150 | | 104h <sup>(3)</sup> | FSR | Indirect Da | ect Data Memory Address Pointer | | | | | | | xxxx xxxx | 31, 150 | | 105h | _ | Unimpleme | mplemented | | | | | | | | _ | | 106h | PORTB | PORTB Da | RTB Data Latch when written: PORTB pins when read | | | | | | | xxxx xxxx | 45, 150 | | 107h | _ | Unimpleme | nplemented | | | | | | | | _ | | 108h | _ | Unimpleme | ented | | | | | | | _ | _ | | 109h | _ | Unimpleme | nplemented | | | | | | | _ | _ | | 10Ah <sup>(1,3)</sup> | PCLATH | _ | Write Buffer for the upper 5 bits of the Program Counter | | | | | | | 0 0000 | 30, 150 | | 10Bh <sup>(3)</sup> | INTCON | GIE | PEIE | TMR0IE | INTE | RBIE | TMR0IF | INTF | RBIF | 0000 000x | 24, 150 | | 10Ch | EEDATA | EEPROM | PROM Data Register Low Byte | | | | | | xxxx xxxx | 39, 151 | | | 10Dh | EEADR | EEPROM A | PROM Address Register Low Byte | | | | | | | xxxx xxxx | 39, 151 | | 10Eh | EEDATH | _ | _ | EEPROM I | Data Registe | r High Byte | | | | xx xxxx | 39, 151 | | 10Fh | EEADRH | _ | 1 | _ | (5) | EEPROM A | Address Regi | ster High By | yte | xxxx | 39, 151 | | Bank 3 | | | | | | | | | | | | | 180h <sup>(3)</sup> | INDF | Addressing | this locatio | n uses cont | ents of FSR t | o address d | ata memory ( | not a physic | cal register) | 0000 0000 | 31, 150 | | 181h | OPTION_REG | RBPU | INTEDG | TOCS | TOSE | PSA | PS2 | PS1 | PS0 | 1111 1111 | 23, 150 | | 182h <sup>(3)</sup> | PCL | Program C | ounter (PC) | Least Sign | ficant Byte | | | | | 0000 0000 | 30, 150 | | 183h <sup>(3)</sup> | STATUS | IRP | RP1 | RP0 | TO | PD | Z | DC | С | 0001 1xxx | 22, 150 | | 184h <sup>(3)</sup> | FSR | Indirect Da | ta Memory | Address Po | inter | | | | | xxxx xxxx | 31, 150 | | 185h | _ | Unimpleme | ented | | | | | | | _ | _ | | 186h | TRISB | PORTB Da | ata Direction | Register | | | | | | 1111 1111 | 45, 150 | | 187h | _ | Unimpleme | ented | | | | | | | _ | _ | | 188h | _ | Unimpleme | ented | | | | | | | _ | _ | | 189h | _ | Unimpleme | ented | | | | | | | _ | _ | | 18Ah <sup>(1,3)</sup> | PCLATH | _ | _ | _ | Write Buffer | for the uppe | er 5 bits of the | Program C | ounter | 0 0000 | 30, 150 | | 18Bh <sup>(3)</sup> | INTCON | GIE | GIE PEIE TMROIE INTE RBIE TMROIF INTF RBIF | | | | | | RBIF | 0000 000x | 24, 150 | | 18Ch | EECON1 | EEPGD | EPGD WRERR WREN WR RD | | | | | | RD | x x000 | 34, 151 | | 18Dh | EECON2 | EEPROM | EPROM Control Register 2 (not a physical register) | | | | | | | | 39, 151 | | 18Eh | _ | Reserved; | teserved; maintain clear | | | | | | | 0000 0000 | _ | | 18Fh | _ | Reserved; | served; maintain clear | | | | | | | 0000 0000 | _ | Legend: x = unknown, u = unchanged, q = value depends on condition, - = unimplemented, read as '0', r = reserved. Shaded locations are unimplemented, read as '0'. - Note 1: The upper byte of the program counter is not directly accessible. PCLATH is a holding register for the PC<12:8>, whose contents are transferred to the upper byte of the program counter. - 2: Bits PSPIE and PSPIF are reserved on PIC16F873A/876A devices; always maintain these bits clear. - 3: These registers can be addressed from any bank. - 4: PORTD, PORTE, TRISD and TRISE are not implemented on PIC16F873A/876A devices, read as '0'. #### 2.2.2.1 Status Register The Status register contains the arithmetic status of the ALU, the Reset status and the bank select bits for data memory. The Status register can be the destination for any instruction, as with any other register. If the Status register is the destination for an instruction that affects the Z, DC or C bits, then the write to these three bits is disabled. These bits are set or cleared according to the device logic. Furthermore, the $\overline{\text{TO}}$ and $\overline{\text{PD}}$ bits are not writable, therefore, the result of an instruction with the Status register as destination may be different than intended. For example, CLRF STATUS, will clear the upper three bits and set the Z bit. This leaves the Status register as 000u uluu (where u = unchanged). It is recommended, therefore, that only BCF, BSF, SWAPF and MOVWF instructions are used to alter the Status register because these instructions do not affect the Z, C or DC bits from the Status register. For other instructions not affecting any status bits, see Section 15.0 "Instruction Set Summary". Note: The C and DC bits operate as a borrow and digit borrow bit, respectively, in subtraction. See the SUBLW and SUBWF instructions for examples. #### REGISTER 2-1: STATUS REGISTER (ADDRESS 03h, 83h, 103h, 183h) | R/W-0 | R/W-0 | R/W-0 | R-1 | R-1 | R/W-x | R/W-x | R/W-x | |-------|-------|-------|-----|-----|-------|-------|-------| | IRP | RP1 | RP0 | TO | PD | Z | DC | С | bit 7 bit 7 IRP: Register Bank Select bit (used for indirect addressing) 1 = Bank 2, 3 (100h-1FFh) 0 = Bank 0, 1 (00h-FFh) bit 6-5 RP1:RP0: Register Bank Select bits (used for direct addressing) 11 = Bank 3 (180h-1FFh) 10 = Bank 2 (100h-17Fh) 01 = Bank 1 (80h-FFh) 00 = Bank 0 (00h-7Fh) Each bank is 128 bytes. bit 4 TO: Time-out bit 1 = After power-up, CLRWDT instruction or SLEEP instruction o = A WDT time-out occurred bit 3 PD: Power-down bit 1 = After power-up or by the CLRWDT instruction 0 = By execution of the SLEEP instruction bit 2 Z: Zero bit 1 = The result of an arithmetic or logic operation is zero 0 = The result of an arithmetic or logic operation is not zero bit 1 DC: Digit carry/borrow bit (ADDWF, ADDLW, SUBLW, SUBWF instructions) (for borrow, the polarity is reversed) 1 = A carry-out from the 4th low order bit of the result occurred 0 = No carry-out from the 4th low order bit of the result bit 0 C: Carry/borrow bit (ADDWF, ADDLW, SUBLW, SUBWF instructions) 1 = A carry-out from the Most Significant bit of the result occurred o = No carry-out from the Most Significant bit of the result occurred Note: For borrow, the polarity is reversed. A subtraction is executed by adding the two's complement of the second operand. For rotate (RRF, RLF) instructions, this bit is loaded with either the high, or low order bit of the source register. Legend: R = Readable bit W = Writable bit U = Unimplemented bit, read as '0' n = Value at POR '1' = Bit is set '0' = Bit is cleared x = Bit is unknown #### 2.2.2.2 OPTION\_REG Register The OPTION\_REG Register is a readable and writable register, which contains various control bits to configure the TMR0 prescaler/WDT postscaler (single assignable register known also as the prescaler), the external INT interrupt, TMR0 and the weak pull-ups on PORTB. To achieve a 1:1 prescaler assignment for the TMR0 register, assign the prescaler to the Watchdog Timer. #### REGISTER 2-2: OPTION\_REG REGISTER (ADDRESS 81h, 181h) | R/W-1 |-------|--------|-------|-------|-------|-------|-------|-------| | RBPU | INTEDG | T0CS | T0SE | PSA | PS2 | PS1 | PS0 | | bit 7 | | | | | • | | bit 0 | Note: bit 7 RBPU: PORTB Pull-up Enable bit 1 = PORTB pull-ups are disabled 2 = PORTB pull-ups are enabled by 0 = PORTB pull-ups are enabled by individual port latch values bit 6 INTEDG: Interrupt Edge Select bit 1 = Interrupt on rising edge of RB0/INT pin 0 = Interrupt on falling edge of RB0/INT pin bit 5 TOCS: TMR0 Clock Source Select bit 1 = Transition on RA4/T0CKI pin o = Internal instruction cycle clock (CLKO) bit 4 T0SE: TMR0 Source Edge Select bit 1 = Increment on high-to-low transition on RA4/T0CKI pin o = Increment on low-to-high transition on RA4/T0CKI pin bit 3 PSA: Prescaler Assignment bit 1 = Prescaler is assigned to the WDT o = Prescaler is assigned to the Timer0 module bit 2-0 PS2:PS0: Prescaler Rate Select bits | Bit Value | TMR0 Rate | WDT Rate | |-----------|-----------|----------| | 000 | 1:2 | 1:1 | | 001 | 1:4 | 1:2 | | 010 | 1:8 | 1:4 | | 011 | 1:16 | 1:8 | | 100 | 1:32 | 1:16 | | 101 | 1:64 | 1:32 | | 110 | 1:128 | 1:64 | | 111 | 1:256 | 1:128 | Legend: R = Readable bit W = Writable bit U = Unimplemented bit, read as '0' - n = Value at POR '1' = Bit is set '0' = Bit is cleared x = Bit is unknown Note: When using Low-Voltage ICSP Programming (LVP) and the pull-ups on PORTB are enabled, bit 3 in the TRISB register must be cleared to disable the pull-up on RB3 and ensure the proper operation of the device #### 2.2.2.3 INTCON Register The INTCON register is a readable and writable register, which contains various enable and flag bits for the TMR0 register overflow, RB port change and external RB0/INT pin interrupts. Interrupt flag bits are set when an interrupt condition occurs regardless of the state of its corresponding enable bit or the global enable bit, GIE (INTCON<7>). User software should ensure the appropriate interrupt flag bits are clear prior to enabling an interrupt. #### REGISTER 2-3: INTCON REGISTER (ADDRESS 0Bh, 8Bh, 10Bh, 18Bh) | R/W-0 R/W-x | |-------|-------|--------|-------|-------|--------|-------|-------| | GIE | PEIE | TMR0IE | INTE | RBIE | TMR0IF | INTF | RBIF | | bit 7 | | | | | | | bit 0 | Note: bit 7 GIE: Global Interrupt Enable bit 1 = Enables all unmasked interrupts o = Disables all interrupts bit 6 PEIE: Peripheral Interrupt Enable bit 1 = Enables all unmasked peripheral interrupts o = Disables all peripheral interrupts bit 5 TMR0IE: TMR0 Overflow Interrupt Enable bit 1 = Enables the TMR0 interrupt o = Disables the TMR0 interrupt bit 4 INTE: RB0/INT External Interrupt Enable bit 1 = Enables the RB0/INT external interrupt o = Disables the RB0/INT external interrupt bit 3 RBIE: RB Port Change Interrupt Enable bit 1 = Enables the RB port change interrupt o = Disables the RB port change interrupt bit 2 TMR0IF: TMR0 Overflow Interrupt Flag bit 1 = TMR0 register has overflowed (must be cleared in software) o = TMR0 register did not overflow bit 1 INTF: RB0/INT External Interrupt Flag bit 1 = The RB0/INT external interrupt occurred (must be cleared in software) o = The RB0/INT external interrupt did not occur bit 0 RBIF: RB Port Change Interrupt Flag bit 1 = At least one of the RB7:RB4 pins changed state; a mismatch condition will continue to set the bit. Reading PORTB will end the mismatch condition and allow the bit to be cleared (must be cleared in software). 0 = None of the RB7:RB4 pins have changed state #### Legend: | R = Readable bit | W = Writable bit | U = Unimplemented | bit, read as '0' | |--------------------|------------------|----------------------|--------------------| | - n = Value at POR | '1' = Bit is set | '0' = Bit is cleared | x = Bit is unknown | #### 2.2.2.4 PIE1 Register The PIE1 register contains the individual enable bits for the peripheral interrupts. Note: Bit PEIE (INTCON<6>) must be set to enable any peripheral interrupt. #### REGISTER 2-4: PIE1 REGISTER (ADDRESS 8Ch) | R/W-0 |----------|-------|-------|-------|-------|--------|--------|--------| | PSPIE(1) | ADIE | RCIE | TXIE | SSPIE | CCP1IE | TMR2IE | TMR1IE | | bit 7 | | | | | | | bit 0 | bit 7 PSPIE: Parallel Slave Port Read/Write Interrupt Enable bit(1) 1 = Enables the PSP read/write interrupt o = Disables the PSP read/write interrupt Note 1: PSPIE is reserved on PIC16F873A/876A devices; always maintain this bit clear. bit 6 ADIE: A/D Converter Interrupt Enable bit 1 = Enables the A/D converter interrupt o = Disables the A/D converter interrupt bit 5 RCIE: USART Receive Interrupt Enable bit > 1 = Enables the USART receive interrupt o = Disables the USART receive interrupt bit 4 TXIE: USART Transmit Interrupt Enable bit 1 = Enables the USART transmit interrupt o = Disables the USART transmit interrupt bit 3 SSPIE: Synchronous Serial Port Interrupt Enable bit 1 = Enables the SSP interrupt 0 = Disables the SSP interrupt bit 2 CCP1IE: CCP1 Interrupt Enable bit 1 = Enables the CCP1 interrupt o = Disables the CCP1 interrupt bit 1 TMR2IE: TMR2 to PR2 Match Interrupt Enable bit 1 = Enables the TMR2 to PR2 match interrupt o = Disables the TMR2 to PR2 match interrupt bit 0 TMR1IE: TMR1 Overflow Interrupt Enable bit 1 = Enables the TMR1 overflow interrupt o = Disables the TMR1 overflow interrupt Legend: R = Readable bit W = Writable bit U = Unimplemented bit, read as '0' '0' = Bit is cleared - n = Value at POR '1' = Bit is set x = Bit is unknown #### 2.2.2.5 PIR1 Register The PIR1 register contains the individual flag bits for the peripheral interrupts. Note: Interrupt flag bits are set when an interrupt condition occurs regardless of the state of its corresponding enable bit or the global enable bit, GIE (INTCON<7>). User software should ensure the appropriate interrupt bits are clear prior to enabling an interrupt. #### REGISTER 2-5: PIR1 REGISTER (ADDRESS 0Ch) | R/W-0 | R/W-0 | R-0 | R-0 | R/W-0 | R/W-0 | R/W-0 | R/W-0 | |----------------------|-------|------|------|-------|--------|--------|--------| | PSPIF <sup>(1)</sup> | ADIF | RCIF | TXIF | SSPIF | CCP1IF | TMR2IF | TMR1IF | | bit 7 | | | | | | | bit 0 | - bit 7 PSPIF: Parallel Slave Port Read/Write Interrupt Flag bit (1) - 1 = A read or a write operation has taken place (must be cleared in software) - 0 = No read or write has occurred Note 1: PSPIF is reserved on PIC16F873A/876A devices; always maintain this bit clear. - bit 6 ADIF: A/D Converter Interrupt Flag bit - 1 = An A/D conversion completed - 0 = The A/D conversion is not complete - bit 5 RCIF: USART Receive Interrupt Flag bit - 1 = The USART receive buffer is full - o = The USART receive buffer is empty - bit 4 TXIF: USART Transmit Interrupt Flag bit - 1 = The USART transmit buffer is empty - o = The USART transmit buffer is full - bit 3 SSPIF: Synchronous Serial Port (SSP) Interrupt Flag bit - 1 = The SSP interrupt condition has occurred and must be cleared in software before returning from the Interrupt Service Routine. The conditions that will set this bit are: - SPI A transmission/reception has taken place. - I<sup>2</sup>C Slave A transmission/reception has taken place. - I<sup>2</sup>C Master - A transmission/reception has taken place. - The initiated Start condition was completed by the SSP module. - The initiated Stop condition was completed by the SSP module. - The initiated Restart condition was completed by the SSP module. - The initiated Acknowledge condition was completed by the SSP module. A Start condition occurred while the SSP module was Idle (multi-master system). - A Stop condition occurred while the SSP module was Idle (multi-master system). - 0 = No SSP interrupt condition has occurred #### bit 2 CCP1IF: CCP1 Interrupt Flag bit #### Capture mode: - 1 = A TMR1 register capture occurred (must be cleared in software) - o = No TMR1 register capture occurred #### Compare mode: - 1 = A TMR1 register compare match occurred (must be cleared in software) - o = No TMR1 register compare match occurred #### PWM mode: Unused in this mode. - bit 1 TMR2IF: TMR2 to PR2 Match Interrupt Flag bit - 1 = TMR2 to PR2 match occurred (must be cleared in software) - o = No TMR2 to PR2 match occurred - bit 0 TMR1IF: TMR1 Overflow Interrupt Flag bit - 1 = TMR1 register overflowed (must be cleared in software) - o = TMR1 register did not overflow | Legend: | | | | |--------------------|------------------|----------------------|--------------------| | R = Readable bit | W = Writable bit | U = Unimplemented | bit, read as '0' | | - n = Value at POR | '1' = Bit is set | '0' = Bit is cleared | x = Bit is unknown | #### 2.2.2.6 PIE2 Register The PIE2 register contains the individual enable bits for the CCP2 peripheral interrupt, the SSP bus collision interrupt, EEPROM write operation interrupt and the comparator interrupt. Note: Bit PEIE (INTCON<6>) must be set to enable any peripheral interrupt. #### REGISTER 2-6: PIE2 REGISTER (ADDRESS 8Dh) | U-0 | R/W-0 | U-0 | R/W-0 | R/W-0 | U-0 | U-0 | R/W-0 | |-------|-------|-----|-------|-------|-----|-----|--------| | _ | CMIE | _ | EEIE | BCLIE | - | _ | CCP2IE | | bit 7 | | | | | | | bit 0 | bit 7 Unimplemented: Read as '0' bit 6 CMIE: Comparator Interrupt Enable bit 1 = Enables the comparator interrupt0 = Disable the comparator interrupt bit 5 Unimplemented: Read as '0' bit 4 **EEIE**: EEPROM Write Operation Interrupt Enable bit 1 = Enable EEPROM write interrupt0 = Disable EEPROM write interrupt bit 3 BCLIE: Bus Collision Interrupt Enable bit 1 = Enable bus collision interrupt 0 = Disable bus collision interrupt bit 2-1 Unimplemented: Read as '0' bit 0 CCP2IE: CCP2 Interrupt Enable bit 1 = Enables the CCP2 interrupt 0 = Disables the CCP2 interrupt Legend: R = Readable bit W = Writable bit U = Unimplemented bit, read as '0' - n = Value at POR '1' = Bit is set '0' = Bit is cleared x = Bit is unknown ### 2.2.2.8 PCON Register The Power Control (PCON) register contains flag bits to allow differentiation between a Power-on Reset (POR), a Brown-out Reset (BOR), a Watchdog Reset (WDT) and an external MCLR Reset. Note: BOR is unknown on Power-on Reset. It must be set by the user and checked on subsequent Resets to see if BOR is clear, indicating a brown-out has occurred. The BOR status bit is a "don't care" and is not predictable if the brown-out circuit is disabled (by clearing the BODEN bit in the configuration word). #### REGISTER 2-8: PCON REGISTER (ADDRESS 8Eh) | U-0 | U-0 | U <b>-</b> 0 | U-0 | U-0 | U-0 | R/W-0 | R/W-1 | | |-------|-----|--------------|-----|-----|-----|-------|-------|--| | _ | _ | _ | _ | _ | _ | POR | BOR | | | bit 7 | | • | • | • | • | | bit 0 | | bit 7-2 Unimplemented: Read as '0' bit 1 POR: Power-on Reset Status bit 1 = No Power-on Reset occurred 0 = A Power-on Reset occurred (must be set in software after a Power-on Reset occurs) bit 0 BOR: Brown-out Reset Status bit 1 = No Brown-out Reset occurred 0 = A Brown-out Reset occurred (must be set in software after a Brown-out Reset occurs) Legend: R = Readable bit W = Writable bit U = Unimplemented bit, read as '0' - n = Value at POR '1' = Bit is set '0' = Bit is cleared x = Bit is unknown #### 2.3 PCL and PCLATH The Program Counter (PC) is 13 bits wide. The low byte comes from the PCL register which is a readable and writable register. The upper bits (PC<12:8>) are not readable, but are indirectly writable through the PCLATH register. On any Reset, the upper bits of the PC will be cleared. Figure 2-5 shows the two situations for the loading of the PC. The upper example in the figure shows how the PC is loaded on a write to PCL (PCLATH<4:0> $\rightarrow$ PCH). The lower example in the figure shows how the PC is loaded during a CALL or gotto instruction (PCLATH<4:3> $\rightarrow$ PCH). FIGURE 2-5: LOADING OF PC IN DIFFERENT SITUATIONS #### 2.3.1 COMPUTED GOTO A computed GOTO is accomplished by adding an offset to the program counter (ADDWF PCL). When doing a table read using a computed GOTO method, care should be exercised if the table location crosses a PCL memory boundary (each 256-byte block). Refer to the application note, AN556, "Implementing a Table Read" (DS00556). #### 2.3.2 STACK The PIC16F87XA family has an 8-level deep x 13-bit wide hardware stack. The stack space is not part of either program or data space and the stack pointer is not readable or writable. The PC is PUSHed onto the stack when a CALL instruction is executed, or an interrupt causes a branch. The stack is POP'ed in the event of a RETURN, RETLW or a RETFIE instruction execution. PCLATH is not affected by a PUSH or POP operation. The stack operates as a circular buffer. This means that after the stack has been PUSHed eight times, the ninth push overwrites the value that was stored from the first push. The tenth push overwrites the second push (and so on). - Note 1: There are no status bits to indicate stack overflow or stack underflow conditions. - There are no instructions/mnemonics called PUSH or POP. These are actions that occur from the execution of the CALL, RETURN, RETLW and RETFIE instructions or the vectoring to an interrupt address. #### 2.4 Program Memory Paging All PIC16F87XA devices are capable of addressing a continuous 8K word block of program memory. The CALL and GOTO instructions provide only 11 bits of address to allow branching within any 2K program memory page. When doing a CALL or GOTO instruction, the upper 2 bits of the address are provided by PCLATH<4:3>. When doing a CALL or GOTO instruction, the user must ensure that the page select bits are programmed so that the desired program memory page is addressed. If a return from a CALL instruction (or interrupt) is executed, the entire 13-bit PC is popped off the stack. Therefore, manipulation of the PCLATH<4:3> bits is not required for the RETURN instructions (which POPs the address from the stack). Note: The contents of the PCLATH register are unchanged after a RETURN or RETFIE instruction is executed. The user must rewrite the contents of the PCLATH register for any subsequent subroutine calls or GOTO instructions. Example 2-1 shows the calling of a subroutine in page 1 of the program memory. This example assumes that PCLATH is saved and restored by the Interrupt Service Routine (if interrupts are used). #### EXAMPLE 2-1: CALL OF A SUBROUTINE IN PAGE 1 FROM PAGE 0 ``` ORG 0x500 BCF PCLATH, 4 BSF PCLATH, 3 ;Select page 1 ; (800h-FFFh) CALL SUB1_P1 ;Call subroutine in ;page 1 (800h-FFFh) ORG 0x900 ;page 1 (800h-FFFh) SUB1 P1 ; called subroutine ;page 1 (800h-FFFh) RETURN return to ;Call subroutine ;in page 0 ; (000h-7FFh) ``` ## 2.5 Indirect Addressing, INDF and FSR Registers The INDF register is not a physical register. Addressing the INDF register will cause indirect addressing. Indirect addressing is possible by using the INDF register. Any instruction using the INDF register actually accesses the register pointed to by the File Select Register, FSR. Reading the INDF register itself, indirectly (FSR = 0) will read 00h. Writing to the INDF register indirectly results in a no operation (although status bits may be affected). An effective 9-bit address is obtained by concatenating the 8-bit FSR register and the IRP bit (Status<7>) as shown in Figure 2-6. A simple program to clear RAM locations 20h-2Fh using indirect addressing is shown in Example 2-2. #### **EXAMPLE 2-2: INDIRECT ADDRESSING** | | MOVLW | 0x20 | ;initialize pointer | |----------|-------|-------|----------------------| | | MOVWF | FSR | ;to RAM | | NEXT | CLRF | INDF | clear INDF register; | | | INCF | FSR,F | ;inc pointer | | | BTFSS | FSR,4 | ;all done? | | | GOTO | NEXT | ;no clear next | | CONTINUE | | | | | | : | | ;yes continue | FIGURE 2-6: DIRECT/INDIRECT ADDRESSING