# Table of Contents

## COMPANY PROFILE

<table>
<thead>
<tr>
<th>SECTION 1. INTRODUCTION</th>
<th>PAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>COMPANY PROFILE</td>
<td>1-1</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>SECTION 1. INTRODUCTION</th>
<th>PAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>Introduction</td>
<td>1-2</td>
</tr>
<tr>
<td>Manual Objective</td>
<td>1-3</td>
</tr>
<tr>
<td>Device Structure</td>
<td>1-4</td>
</tr>
<tr>
<td>Development Support</td>
<td>1-6</td>
</tr>
<tr>
<td>Device Varieties</td>
<td>1-7</td>
</tr>
<tr>
<td>Style and Symbol Conventions</td>
<td>1-12</td>
</tr>
<tr>
<td>Related Documents</td>
<td>1-14</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>1-17</td>
</tr>
<tr>
<td>Revision History</td>
<td>1-18</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>SECTION 2. OSCILLATOR</th>
<th>PAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>Oscillator Configurations</td>
<td>2-4</td>
</tr>
<tr>
<td>Crystal Oscillators/Ceramic Resonators</td>
<td>2-6</td>
</tr>
<tr>
<td>External RC Oscillator</td>
<td>2-15</td>
</tr>
<tr>
<td>HS4</td>
<td>2-18</td>
</tr>
<tr>
<td>Switching to Low Power Clock Source</td>
<td>2-19</td>
</tr>
<tr>
<td>Effects of Sleep Mode on the On-Chip Oscillator</td>
<td>2-23</td>
</tr>
<tr>
<td>Effects of Device Reset on the On-Chip Oscillator</td>
<td>2-23</td>
</tr>
<tr>
<td>Design Tips</td>
<td>2-24</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>2-25</td>
</tr>
<tr>
<td>Revision History</td>
<td>2-26</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>SECTION 3. RESET</th>
<th>PAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>Reset</td>
<td>3-2</td>
</tr>
<tr>
<td>Resets and Delay Timers</td>
<td>3-4</td>
</tr>
<tr>
<td>Registers and Status Bit Values</td>
<td>3-14</td>
</tr>
<tr>
<td>Design Tips</td>
<td>3-20</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>3-21</td>
</tr>
<tr>
<td>Revision History</td>
<td>3-22</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>SECTION 4. ARCHITECTURE</th>
<th>PAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>Clocking Scheme/Instruction Cycle</td>
<td>4-2</td>
</tr>
<tr>
<td>Instruction Flow/Pipelining</td>
<td>4-5</td>
</tr>
<tr>
<td>I/O Descriptions</td>
<td>4-6</td>
</tr>
<tr>
<td>Design Tips</td>
<td>4-7</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>4-14</td>
</tr>
<tr>
<td>Revision History</td>
<td>4-16</td>
</tr>
</tbody>
</table>
Table of Contents

SECTION 5. CPU AND ALU 5-1
- Introduction ................................................. 5-2
- General Instruction Format ............................. 5-5
- Central Processing Unit (CPU) ...................... 5-6
- Instruction Clock ........................................... 5-7
- Arithmetic Logical Unit (ALU) ....................... 5-9
- STATUS Register ......................................... 5-11
- Design Tips .................................................. 5-14
- Related Application Notes ......................... 5-15
- Revision History .......................................... 5-16

SECTION 6. HARDWARE 8X8 MULTIPLIER 6-1
- Introduction .................................................. 6-2
- Operation ...................................................... 6-3
- Design Tips .................................................. 6-6
- Related Application Notes ......................... 6-7
- Revision History .......................................... 6-8

SECTION 7. MEMORY ORGANIZATION 7-1
- Introduction .................................................. 7-2
- Program Memory .......................................... 7-3
- Program Counter (PC) ................................. 7-6
- Lookup Tables ............................................. 7-9
- Stack ............................................................ 7-12
- Data Memory Organization ......................... 7-13
- Return Address Stack .................................. 7-17
- Initialization ............................................... 7-23
- Design Tips .................................................. 7-24
- Related Application Notes ......................... 7-25
- Revision History .......................................... 7-26

SECTION 8. TABLE READ/TABLE WRITE 8-1
- Introduction ................................................. 8-2
- Control Registers ........................................... 8-3
- Program Memory ......................................... 8-6
- Enabling Internal Programming ....................... 8-12
- External Program Memory Operation .............. 8-12
- Initialization .................................................. 8-13
- Design Tips .................................................. 8-14
- Related Application Notes ......................... 8-15
- Revision History .......................................... 8-16
# Table of Contents

## SECTION 9. SYSTEM BUS  
9-1  
Revision History ................................................................. 9-2

## SECTION 10. INTERRUPTS  
10-1  
Introduction ........................................................................................................ 10-2  
Control Registers ........................................................................................................ 10-6  
Interrupt Handling Operation ....................................................................................... 10-19  
Initialization .................................................................................................................. 10-29  
Design Tips ..................................................................................................................... 10-30  
Related Application Notes .......................................................................................... 10-31  
Revision History .......................................................................................................... 10-32

## SECTION 11. I/O PORTS  
11-1  
Introduction ................................................................................................................ 11-2  
PORTA, TRISA, and the LAT Register ........................................................................ 11-6  
PORTB, TRISB, and the LATB Register ....................................................................... 11-12  
PORTC, TRISC, and the LATC Register ....................................................................... 11-16  
PORTD, LATD, and the TRISD Register ..................................................................... 11-19  
PORTE, TRISE, and the LATE Register ........................................................................ 11-21  
PORTF, LATF, and the TRISF Register ....................................................................... 11-23  
PORTG, LATG, and the TRISG Register ....................................................................... 11-25  
PORTH, LATH, and the TRISH Register ..................................................................... 11-27  
PORTJ, LATJ, and the TRISJ Register ........................................................................ 11-29  
PORTK, LATK, and the TRISK Register ..................................................................... 11-31  
PORTL, LATL, and the TRISL Register ....................................................................... 11-33  
Functions Multiplexed on I/O Pins ............................................................................ 11-35  
I/O Programming Considerations ............................................................................... 11-37  
Initialization ................................................................................................................ 11-40  
Design Tips .................................................................................................................. 11-41  
Related Application Notes ......................................................................................... 11-43  
Revision History .......................................................................................................... 11-44

## SECTION 12. PARALLEL SLAVE PORT  
12-1  
Introduction .............................................................................................................. 12-2  
Control Register ........................................................................................................... 12-3  
Operation ....................................................................................................................... 12-5  
Operation in SLEEP Mode ............................................................................................ 12-6  
Effect of a RESET ......................................................................................................... 12-6  
PSP Waveforms ........................................................................................................... 12-6  
Design Tips .................................................................................................................... 12-8  
Related Application Notes .......................................................................................... 12-9  
Revision History .......................................................................................................... 12-10
Table of Contents

SECTION 13. TIMER0 ......................................................... 13-1
  Introduction ................................................................................................................... 13-2
  Control Register ............................................................................................................. 13-3
  Operation ........................................................................................................................ 13-4
  Timer0 Interrupt ............................................................................................................. 13-5
  Using Timer0 with an External Clock .............................................................................. 13-6
  Timer0 Prescaler ............................................................................................................. 13-7
  Initialization ................................................................................................................... 13-9
  Design Tips .................................................................................................................... 13-10
  Related Application Notes ............................................................................................ 13-11
  Revision History ........................................................................................................... 13-12

SECTION 14. TIMER1 ......................................................... 14-1
  Introduction ................................................................................................................... 14-2
  Control Register ............................................................................................................. 14-4
  Timer1 Operation in Timer Mode .................................................................................... 14-5
  Timer1 Operation in Synchronized Counter Mode .......................................................... 14-5
  Timer1 Operation in Asynchronous Counter Mode ......................................................... 14-6
  Reading and Writing of Timer1 ....................................................................................... 14-7
  Timer1 Oscillator ............................................................................................................ 14-10
  Typical Application ....................................................................................................... 14-11
  Sleep Operation ............................................................................................................ 14-12
  Resetting Timer1 Using a CCP Trigger Output ............................................................... 14-12
  Resetting Timer1 Register Pair (TMR1H:TMR1L) .......................................................... 14-13
  Timer1 Prescaler ............................................................................................................ 14-13
  Initialization .................................................................................................................. 14-14
  Design Tips .................................................................................................................... 14-16
  Related Application Notes ............................................................................................ 14-17
  Revision History ........................................................................................................... 14-18

SECTION 15. TIMER2 ......................................................... 15-1
  Introduction ................................................................................................................... 15-2
  Control Register ............................................................................................................. 15-3
  Timer Clock Source ....................................................................................................... 15-4
  Timer (TMR2) and Period (PR2) Registers ...................................................................... 15-4
  TMR2 Match Output ...................................................................................................... 15-4
  Clearing the Timer2 Prescaler and Postscaler ................................................................. 15-4
  Sleep Operation ............................................................................................................ 15-5
  Initialization .................................................................................................................. 15-5
  Design Tips .................................................................................................................... 15-6
  Related Application Notes ............................................................................................ 15-7
  Revision History ........................................................................................................... 15-8
# Table of Contents

## SECTION 16. TIMER3

- Introduction .................................................. 16-2
- Control Registers ............................................. 16-3
- Timer3 Operation in Timer Mode ......................... 16-4
- Timer3 Operation in Synchronized Counter Mode .... 16-5
- Timer3 Operation in Asynchronous Counter Mode ...... 16-6
- Reading and Writing of Timer3 ............................. 16-7
- Timer3 using the Timer1 Oscillator ....................... 16-8
- Timer3 and CCPx Enable .................................... 16-9
- Timer3 Prescaler .............................................. 16-10
- 16-bit Mode Timer Reads/Writes ......................... 16-11
- Typical Application ............................................ 16-12
- Sleep Operation .............................................. 16-13
- Timer3 Prescaler .............................................. 16-14
- Initialization ..................................................... 16-15
- Design Tips ..................................................... 16-16
- Related Application Notes ................................. 16-17
- Revision History .............................................. 16-18

## SECTION 17. COMPARE/CAPTURE/PWM (CCP)

- Introduction .................................................. 17-2
- CCP Control Register ........................................ 17-3
- Capture Mode .................................................. 17-4
- Compare Mode ............................................... 17-5
- PWM Mode ..................................................... 17-6
- Initialization .................................................... 17-7
- Design Tips ..................................................... 17-8
- Related Application Notes ................................. 17-9
- Revision History .............................................. 17-10

## SECTION 18. ECCP

- Introduction .................................................. 18-2
- Control Registers ............................................. 18-3
- SPI Mode ....................................................... 18-4
- Initialization .................................................... 18-5
- Design Tips ..................................................... 18-6
- Related Application Notes ................................. 18-7
- Revision History .............................................. 18-8

## SECTION 19. SYNCHRONOUS SERIAL PORT (SSP)

- Introduction .................................................. 19-2
- Control Registers ............................................. 19-3
- SPI Mode ....................................................... 19-4
- SSP I2C Operation .......................................... 19-5
- Initialization .................................................... 19-6
- Design Tips ..................................................... 19-7
- Related Application Notes ................................. 19-8
- Revision History .............................................. 19-9
# Table of Contents

<table>
<thead>
<tr>
<th>Section</th>
<th>Page</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>SECTION 20. MASTER SSP</strong></td>
<td></td>
</tr>
<tr>
<td>Introduction</td>
<td>20-2</td>
</tr>
<tr>
<td>Control Registers</td>
<td>20-4</td>
</tr>
<tr>
<td>SPI Mode</td>
<td>20-9</td>
</tr>
<tr>
<td>MSSP I²C Operation</td>
<td>20-18</td>
</tr>
<tr>
<td>Design Tips</td>
<td>20-58</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>20-59</td>
</tr>
<tr>
<td>Revision History</td>
<td>20-60</td>
</tr>
<tr>
<td><strong>SECTION 21. ADDRESSABLE USART</strong></td>
<td></td>
</tr>
<tr>
<td>Introduction</td>
<td>21-2</td>
</tr>
<tr>
<td>Control Registers</td>
<td>21-3</td>
</tr>
<tr>
<td>USART Asynchronous Mode</td>
<td>21-9</td>
</tr>
<tr>
<td>USART Synchronous Master Mode</td>
<td>21-18</td>
</tr>
<tr>
<td>USART Synchronous Slave Mode</td>
<td>21-23</td>
</tr>
<tr>
<td>Initialization</td>
<td>21-25</td>
</tr>
<tr>
<td>Design Tips</td>
<td>21-26</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>21-27</td>
</tr>
<tr>
<td>Revision History</td>
<td>21-28</td>
</tr>
<tr>
<td><strong>SECTION 22. CAN</strong></td>
<td></td>
</tr>
<tr>
<td>Introduction</td>
<td>22-2</td>
</tr>
<tr>
<td>Control Registers for the CAN Module</td>
<td>22-3</td>
</tr>
<tr>
<td>CAN Overview</td>
<td>22-28</td>
</tr>
<tr>
<td>CAN Bus Features</td>
<td>22-32</td>
</tr>
<tr>
<td>CAN Module Implementation</td>
<td>22-33</td>
</tr>
<tr>
<td>Frame Types</td>
<td>22-37</td>
</tr>
<tr>
<td>Modes of Operation</td>
<td>22-44</td>
</tr>
<tr>
<td>CAN Bus Initialization</td>
<td>22-48</td>
</tr>
<tr>
<td>Message Reception</td>
<td>22-49</td>
</tr>
<tr>
<td>Transmission</td>
<td>22-60</td>
</tr>
<tr>
<td>Error Detection</td>
<td>22-68</td>
</tr>
<tr>
<td>Baud Rate Setting</td>
<td>22-71</td>
</tr>
<tr>
<td>Interrupts</td>
<td>22-75</td>
</tr>
<tr>
<td>Timestamping</td>
<td>22-77</td>
</tr>
<tr>
<td>CAN Module I/O</td>
<td>22-77</td>
</tr>
<tr>
<td>Design Tips</td>
<td>22-78</td>
</tr>
<tr>
<td>Related Application Notes</td>
<td>22-79</td>
</tr>
<tr>
<td>Revision History</td>
<td>22-80</td>
</tr>
</tbody>
</table>
# Table of Contents

## SECTION 23. COMPATIBLE VOLTAGE REFERENCE

- Introduction ................................................................................................................... 23-2
- Control Register .......................................................................................................... 23-3
- Configuring the Voltage Reference ............................................................................. 23-4
- Voltage Reference Accuracy/Error ........................................................................... 23-5
- Operation During Sleep ............................................................................................... 23-5
- Effects of a Reset ......................................................................................................... 23-5
- Connection Considerations ......................................................................................... 23-6
- Initialization ................................................................................................................ 23-7
- Design Tips .................................................................................................................. 23-8
- Related Application Notes ........................................................................................... 23-9
- Revision History .......................................................................................................... 23-10

## SECTION 24. COMPARATOR

- Introduction .................................................................................................................. 24-2
- Control Register .......................................................................................................... 24-3
- Comparator Configuration ............................................................................................. 24-4
- Comparator Operation ................................................................................................. 24-6
- Comparator Reference ................................................................................................. 24-6
- Comparator Response Time ......................................................................................... 24-8
- Comparator Outputs .................................................................................................... 24-8
- Comparator Interrupts ................................................................................................. 24-9
- Comparator Operation During SLEEP .......................................................................... 24-9
- Effects of a RESET ....................................................................................................... 24-9
- Analog Input Connection Considerations ..................................................................... 24-10
- Initialization ................................................................................................................ 24-11
- Design Tips .................................................................................................................. 24-12
- Related Application Notes ........................................................................................... 24-13
- Revision History .......................................................................................................... 24-14

## SECTION 25. COMPATIBLE 10-BIT A/D CONVERTER

- Introduction .................................................................................................................. 25-2
- Control Register .......................................................................................................... 25-4
- Operation ...................................................................................................................... 25-4
- A/D Acquisition Requirements .................................................................................... 25-8
- Selecting the A/D Conversion Clock ........................................................................... 25-10
- Configuring Analog Port Pins ...................................................................................... 25-11
- A/D Conversions ........................................................................................................... 25-12
- Operation During Sleep ............................................................................................... 25-16
- Effects of a Reset ......................................................................................................... 25-16
- A/D Accuracy/Error ................................................................................................. 25-17
- Connection Considerations ......................................................................................... 25-18
- Transfer Function ....................................................................................................... 25-18
- Initialization ................................................................................................................ 25-19
- Design Tips .................................................................................................................. 25-20
- Related Application Notes ........................................................................................... 25-21
- Revision History .......................................................................................................... 25-22
# Table of Contents

## SECTION 26. 10-BIT A/D CONVERTER

- Introduction ................................................................................................................... 26-2
- Control Register .......................................................................................................... 26-4
- Operation ...................................................................................................................... 26-7
- A/D Acquisition Requirements .................................................................................... 26-8
- Selecting the A/D Conversion Clock ............................................................................. 26-10
- Configuring Analog Port Pins ...................................................................................... 26-11
- A/D Conversions ......................................................................................................... 26-12
- Operation During Sleep ................................................................................................. 26-16
- Effects of a Reset ....................................................................................................... 26-16
- A/D Accuracy/Error ................................................................................................... 26-17
- Connection Considerations ......................................................................................... 26-18
- Transfer Function ....................................................................................................... 26-18
- Initialization ................................................................................................................ 26-19
- Design Tips ............................................................................................................... 26-20
- Related Application Notes ......................................................................................... 26-21
- Revision History ....................................................................................................... 26-22

## SECTION 27. LOW VOLTAGE DETECT

- Introduction ................................................................................................................ 27-2
- Control Register ........................................................................................................ 27-4
- Operation .................................................................................................................... 27-5
- Operation During Sleep ............................................................................................... 27-6
- Effects of a Reset ...................................................................................................... 27-6
- Initialization ............................................................................................................... 27-7
- Design Tips .............................................................................................................. 27-8
- Related Application Notes ....................................................................................... 27-9
- Revision History ..................................................................................................... 27-10

## SECTION 28. WDT AND SLEEP MODE

- Introduction .............................................................................................................. 28-2
- Control Register ...................................................................................................... 28-3
- Watchdog Timer (WDT) Operation ......................................................................... 28-4
- SLEEP (Power-Down) Mode .................................................................................... 28-5
- Initialization ............................................................................................................. 28-11
- Design Tips ........................................................................................................... 28-12
- Related Application Notes .................................................................................... 28-13
- Revision History .................................................................................................. 28-14
Table of Contents

SECTION 29. DEVICE CONFIGURATION BITS ............................................................... 29-1
Introduction .............................................................................................................. 29-2
Configuration Word Bits ......................................................................................... 29-3
Program Verification/Code Protection ................................................................. 29-10
ID Locations ......................................................................................................... 29-11
Device ID .............................................................................................................. 29-11
Design Tips ......................................................................................................... 29-12
Related Application Notes ..................................................................................... 29-13
Revision History ................................................................................................. 29-14

SECTION 30. IN-CIRCUIT SERIAL PROGRAMMING™ (ICSP™) ................................. 30-1
Introduction .............................................................................................................. 30-2
Entering In-Circuit Serial Programming Mode .................................................. 30-3
Application Circuit ................................................................................................. 30-4
Programmer .......................................................................................................... 30-6
Programming Environment .................................................................................. 30-6
Other Benefits .................................................................................................... 30-7
Field Programming of PICmicro OTP MCUs ..................................................... 30-8
Field Programming of FLASH PICmicros ......................................................... 30-10
Design Tips ......................................................................................................... 30-12
Related Application Notes ..................................................................................... 30-13
Revision History ................................................................................................. 30-14

SECTION 31. INSTRUCTION SET ............................................................................. 31-1
Introduction .............................................................................................................. 31-2
Data Memory Map ................................................................................................. 31-3
Instruction Formats ............................................................................................... 31-9
Special Function Registers as Source/Destination ............................................. 31-12
Fast Register Stack ............................................................................................... 31-13
Q Cycle Activity .................................................................................................. 31-13
Instruction Descriptions ...................................................................................... 31-14
Design Tips ......................................................................................................... 31-136
Related Application Notes .................................................................................... 31-137
Revision History ................................................................................................. 31-138
Table of Contents

SECTION 32. ELECTRICAL SPECIFICATIONS 32-1
Introduction ................................................................. 32-2
Absolute Maximums ...................................................... 32-3
Voltage vs Frequency Graph ........................................... 32-4
Device Voltage Specifications ......................................... 32-6
Device Current Specifications ........................................... 32-7
Input Threshold Levels .................................................... 32-10
I/O Current Specifications ............................................... 32-11
Output Drive Levels ....................................................... 32-12
I/O Capacitive Loading .................................................... 32-13
Low Voltage Detect (LVD) ............................................... 32-14
EPROM/FLASH/Data EEPROM ......................................... 32-15
Comparators and Voltage Reference ................................. 32-16
Timing Parameter Symbology ........................................... 32-18
Example External Clock Timing Waveforms and Requirements .................................................... 32-19
Example Phase Lock Loop (PLL) Timing Waveforms and Requirements ............................................. 32-20
Example Power-up and RESET Timing Waveforms and Requirements ................................................. 32-22
Example Timer0 and Timer1 Timing Waveforms and Requirements .................................................... 32-23
Example CCP Timing Waveforms and Requirements ................................................................. 32-24
Example Parallel Slave Port (PSP) Timing Waveforms and Requirements ....................................... 32-25
Example SSP and Master SSP SPI Mode Timing Waveforms and Requirements ................................ 32-26
Example SSP /C Mode Timing Waveforms and Requirements ......................................................... 32-29
Example Master SSP /C Mode Timing Waveforms and Requirements ........................................... 32-30
Example USART/SCI Timing Waveforms and Requirements ..................................................... 32-34
CAN Specifications ......................................................... 32-35
Example 8-bit A/D Timing Waveforms and Requirements ........................................................ 32-36
Example 10-bit A/D Timing Waveforms and Requirements ......................................................... 32-38
Design Tips ................................................................. 32-40
Related Application Notes .................................................. 32-41
Revision History ............................................................. 32-42

SECTION 33. DEVICE CHARACTERISTICS 33-1
Introduction ................................................................. 33-2
Characterization vs. Electrical Specification .............................. 33-2
DC and AC Characteristics Graphs and Tables ......................... 33-2
Revision History ............................................................. 33-26
Table of Contents

SECTION 34. DEVELOPMENT TOOLS 34-1
Introduction ................................................................................................................... 34-2
The Integrated Development Environment (IDE) .......................................................... 34-3
MPLAB® Software Language Support .............................................................................. 34-6
MPLAB-SIM Simulator Software ..................................................................................... 34-8
MPLAB Emulator Hardware Support .............................................................................. 34-9
MPLAB High Performance Universal In-Circuit Emulator with MPLAB IDE ................. 34-9
MPLAB-ICD In-Circuit Debugger .................................................................................... 34-9
MPLAB Programmer Support .......................................................................................... 34-10
Supplemental Tools ......................................................................................................... 34-11
Development Boards ...................................................................................................... 34-12
Development Tools for Other Microchip Products ....................................................... 34-14
Related Application Notes ............................................................................................. 34-15
Revision History ............................................................................................................ 34-16

SECTION 35. CODE DEVELOPMENT 35-1
Overview ....................................................................................................................... 35-2
Good Practice ............................................................................................................... 35-3
Diagnostic Code Techniques .......................................................................................... 35-5
Example Scenario and Implementation ........................................................................... 35-6
Implications of Using a High Level Language (HLL) ................................................... 35-7
Revision History ............................................................................................................ 35-8

SECTION 36. APPENDIX 36-1
Appendix A: I²C Overview ........................................................................................... 36-1
Appendix B: CAN Overview ......................................................................................... 36-12
Appendix C: Module Block Diagrams and Registers ...................................................... 36-13
Appendix D: Register Definitions ............................................................................... 36-14
Appendix E: Migration Tips .......................................................................................... 36-15

SECTION 37. GLOSSARY 37-1
Revision History ............................................................................................................ 37-14

SOURCE CODE

INDEX

© 2000 Microchip Technology Inc.
Since its inception, Microchip Technology has focused its resources on delivering innovative semiconductor products to the global embedded control marketplace. To do this, we have focused our technology, engineering, manufacturing and marketing resources on synergistic product lines: PICmicro® microcontrollers (MCUs), high-endurance Serial EEPROMs, an expanding product portfolio of analog/interface products, RFID tags and KEELOQ® security devices — all aimed at delivering comprehensive, high-value embedded control solutions to a growing base of customers.

Inside Microchip Technology you will find:

- An experienced executive team focused on innovation and committed to listening to our customers
- A focus on providing high-performance, cost-effective embedded control solutions
- Fully integrated manufacturing capabilities
- A global network of manufacturing and customer support facilities
- A unique corporate culture dedicated to continuous improvement
- Distributor network support worldwide including certified distribution FAEs

A Complete Product Solution including:

- RISC OTP, FLASH, EEPROM and ROM MCUs
- A full family of advanced analog MCUs
- KEELOQ security devices featuring patented code hopping technology
- Stand-alone analog and interface products plus microID™ RFID tagging devices
- A complete line of high-endurance Serial EEPROMs
- World-class, easy-to-use development tools
- An Automotive Products Group to engage with key automotive accounts and provide necessary application expertise and customer service

**Business Scope**

Microchip Technology Inc. designs, manufactures, and markets a variety of CMOS semiconductor components to support the market for cost-effective embedded control solutions.

Microchip’s products feature compact size, integrated functionality, ease of development and technical support so essential to timely and cost-effective product development by our customers.
Market Focus

Microchip targets select markets where our advanced designs, progressive process technology and industry-leading product performance enables us to deliver decidedly superior performance. Our Company is positioned to provide a complete product solution for embedded control applications found throughout the consumer, automotive, telecommunication, office automation and industrial control markets. Microchip products are also meeting the unique design requirements of targeted embedded applications including internet, safety and security.

Certified Quality Systems

Microchip’s quality systems have been certified to QS-9000 requirements. Its worldwide headquarters and wafer fabrication facilities in Chandler and Tempe, Arizona, received certification on July 23, 1999. The scope of this certification is the design and manufacture of RISC-based MCUs, related non-volatile memory products and microperipheral devices. The quality systems for Microchip’s product test facility in Bangkok, Thailand, were QS-9000 certified on February 26, 1999. The scope of this certification is the design and testing of integrated circuits. In addition, Microchip’s quality system for the design and manufacture of development systems is ISO 9001 certified.

QS-9000 was developed by Chrysler, Ford and General Motors to establish fundamental quality systems that provide for continuous improvement, emphasizing defect prevention and the reduction of variation and waste in the supply chain. Microchip was audited by QS-9000 registrar Det Norske Veritas Certification Inc. of Houston, the same firm which granted Microchip its ISO 9001 Quality System certification in 1997. QS-9000 certification recognizes Microchip’s quality systems conform to the stringent standards set forth by the automotive industry, benefiting all customers.

Fully Integrated Manufacturing

Microchip delivers fast turnaround and consistent quality through total control over all phases of production. Research and development, design, mask making, wafer fabrication, and the major part of assembly and quality assurance testing are conducted at facilities wholly-owned and operated by Microchip. Our integrated approach to manufacturing along with rigorous use of advanced Statistical Process Control (SPC) and a continuous improvement culture has resulted in high and consistent yields which have positioned Microchip as a quality leader in its global markets. Microchip’s unique approach to SPC provides customers with excellent pricing, quality, reliability and on-time delivery.

A Global Network of Plants and Facilities

Microchip is a global competitor providing local services to the world’s technology centers. The Company’s design and technology advancement facilities, and wafer fabrication sites are located in Chandler and Tempe, Arizona.

The Tempe facility provides an additional 200,000 square feet of manufacturing space that meets the increased production requirements of a growing customer base, and provides production capacity which more than doubles that of Chandler.

Microchip facilities in Bangkok, Thailand, and Shanghai, China, serve as the foundation of Microchip’s extensive assembly and test capability located throughout Asia. The use of multiple fabrication, assembly and test sites, with more than 640,000-square-feet of facilities worldwide, ensures Microchip’s ability to meet the increased production requirements of a fast growing customer base.

Microchip supports its global customer base from direct sales and engineering offices in Asia, North America, Europe and Japan. Offices are staffed to meet the high quality expectations of our customers, and can be accessed for technical and business support. The Company also franchises more than 60 distributors and a network of technical manufacturer’s representatives serving 24 countries worldwide.
Embedded Control Overview

Unlike "processor" applications such as personal computers and workstations, the computing or controlling elements of embedded control applications are embedded inside the application. The consumer is only concerned with the very top-level user interface such as keypads, displays and high-level commands. Very rarely does an end-user know (or care to know) the embedded controller inside (unlike the conscientious PC users, who are intimately familiar not only with the processor type, but also its clock speed, DMA capabilities and so on).

It is, however, most vital for designers of embedded control products to select the most suitable controller and companion devices. Embedded control products are found in all market segments: consumer, commercial, PC peripherals, telecommunications, automotive and industrial. Most embedded control products must meet special requirements: cost effectiveness, low-power, small-footprint and a high level of system integration.

Typically, most embedded control systems are designed around an MCU which integrates on-chip program memory, data memory (RAM) and various peripheral functions, such as timers and serial communication. In addition, these systems usually require complementary Serial EEPROM, analog/interface devices, display drivers, keypads or small displays.

Microchip has established itself as a leading supplier of embedded control solutions. The combination of high-performance PIC12CXXX, PIC16C5X, PIC16CXXX, PIC17CXXX and PIC18CXXX MCU families with Migratable Memory™ technology, along with non-volatile memory products, provide the basis for this leadership. By further expanding our product portfolio to provide precision analog and interface products, Microchip is committed to continuous innovation and improvement in design, manufacturing and technical support to provide the best possible embedded control solutions to you.

PICmicro MCU Overview and Roadmap

Microchip PICmicro MCUs combine high-performance, low-cost, and small package size, offering the best price/performance ratio in the industry. More than one billion of these devices have shipped to customers worldwide since 1990. Microchip offers five families of MCUs to best fit your application needs:

- PIC12CXXX 8-pin 12-bit/14-bit program word
- PIC16C5X 12-bit program word
- PIC16CXXX 14-bit program word
- PIC17CXXX 16-bit program word
- PIC18CXXX enhanced 16-bit program word

All families offer OTP, low-voltage and low-power options, with a variety of package options. Selected members are available in ROM, EEPROM or reprogrammable FLASH versions.

PIC12CXXX: 8-Pin, Family

The PIC12CXXX family packs Microchip's powerful RISC-based PICmicro architecture into 8-pin DIP and SOIC packages. These PIC12CXXX products are available with either a 12-bit or 14-bit wide instruction set, a low operating voltage of 2.5V, small package footprints, interrupt handling, a deeper hardware stack, multiple channels and EEPROM data memory. All of these features provide an intelligence level not previously available in applications because of cost or size considerations.

PIC16C5X: 12-Bit Architecture Family

The PIC16C5X is the well-established base-line family that offers the most cost-effective solution. These PIC16C5X products have a 12-bit wide instruction set and are currently offered in 14-, 18-, 20- and 28-pin packages. In the SOIC and SSOP packaging options, these devices are among the smallest footprint MCUs in the industry. Low-voltage operation, down to 2.0V for OTP MCUs, makes this family ideal for battery operated applications. Additionally, the PIC16HV5XX can operate up to 15 volts for use directly with a battery.

PIC16CXXX: 14-Bit Architecture Family

With the introduction of new PIC16CXXX family members, Microchip now provides the industry's highest performance Analog-to-Digital Converter capability at 12-bits for an MCU. The PIC18CXXX family offers a wide-range of options, from 18- to 68-pin packages as well as low to high levels of peripheral integration. This family has a 14-bit wide instruction set, interrupt handling capability and a deep, 8-level hardware stack. The PIC16CXXX family provides the performance and versatility to meet the more demanding requirements of today's cost-sensitive marketplace for mid-range applications.

PIC17CXXX: 16-Bit Architecture Family

The PIC17CXXX family offers the world's fastest execution performance of any MCU family in the industry. The PIC17CXXX family extends the PICmicro MCU's high-performance RISC architecture with a 16-bit instruction word, enhanced instruction set and powerful vectored interrupt handling capabilities. A powerful array of precise on-chip peripheral features provides the performance for the most demanding applications.

PIC18CXXX: 16-Bit Enhanced Architecture Family

The PIC18CXXX family is a family of high performance, CMOS, fully static, 16-bit MCUs with integrated analog-to-digital (A/D) converter. All PIC18CXXX MCUs incorporate an advanced RISC architecture. The PIC18CXXX has enhanced core features, 32 level-deep stack, and multiple internal and external interrupts sources. The separate instruction and data busses of the Harvard architecture allow a 16-bit wide instruction word with the separate 8-bit wide data. The two-stage instruction pipeline allows all instructions to execute in a single cycle, except for program branches, which require two cycles. A total of 77 instructions (reduced instruction set) are available. Additionally, a large register set gives some of the architectural
innovations used to achieve a very high performance of 10 MIPS for an MCU. The PIC18CXXX family has special features to reduce external components, thus reducing cost, enhancing system reliability and reducing power consumption. These include programmable Low Voltage Detect (LVD) and programmable Brown-Out Detect (BOD).

**The Mechatronics Revolution**

The nature of the revolution is the momentous shift from analog/electro-mechanical timing and control to digital electronics. It is called the Mechatronics Revolution, and it is being staged in companies throughout the world, with design engineers right on the front lines: make it smarter, make it smaller, make it do more, make it cost less to manufacture – and make it snappy.

To meet the needs of this growing customer base, Microchip is rapidly expanding its already broad line of PICmicro MCUs. The PIC12CXXX family’s size opens up new possibilities for product design.

**PICmicro MCU Naming Convention**

The PICmicro architecture offers users a wider range of cost/performance options than any MCU family. In order to identify the families, the following naming conventions have been applied to the PICmicro MCUs:

<table>
<thead>
<tr>
<th>Family</th>
<th>Architectural Features</th>
<th>Name</th>
<th>Technology</th>
</tr>
</thead>
<tbody>
<tr>
<td>PIC18CXXX</td>
<td>8-bit High-Performance MCU Family</td>
<td>PIC18CXX2</td>
<td>OTP program memory with higher resolution analog functions</td>
</tr>
<tr>
<td></td>
<td>• 10 MIPS @ 40 MHz</td>
<td>PIC18FXXX</td>
<td>FLASH program memory</td>
</tr>
<tr>
<td></td>
<td>• 4x PLL clock</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>• 16-bit wide instruction set</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>• C compiler efficient instruction set</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>• Internal/external vectored interrupts</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIC17CXXX</td>
<td>8-bit High-Performance MCU Family</td>
<td>PIC17C4X</td>
<td>OTP program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• 16-bit wide instruction set</td>
<td>PIC17CR4X</td>
<td>ROM program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• Internal/external vectored interrupts</td>
<td>PIC17C7XX</td>
<td>OTP program memory with mixed-signal functions</td>
</tr>
<tr>
<td></td>
<td>• DC - 33 MHz clock speed</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>• 120 ns instruction cycle (@ 33 MHz)</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>• Hardware multiply</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIC16CXXX</td>
<td>8-bit Mid-Range MCU Family</td>
<td>PIC16C4XX</td>
<td>OTP program memory with A/D and D/A functions</td>
</tr>
<tr>
<td></td>
<td>• 14-bit wide instruction set</td>
<td>PIC16C5XX</td>
<td>OTP program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• Internal/external interrupts</td>
<td>PIC16C6XX</td>
<td>OTP program memory with analog functions (i.e. A/D)</td>
</tr>
<tr>
<td></td>
<td>• DC - 20 MHz clock speed (Note 1)</td>
<td>PIC16C8XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td>• 200 ns instruction cycle (@ 20 MHz)</td>
<td>PIC16C9XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C4X2</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C4X4</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C6XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C8XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C9XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C4X2</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C4X4</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C6XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C8XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C9XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C4X2</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C4X4</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C6XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C8XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC16C9XX</td>
<td>OTP program memory with comparators</td>
</tr>
<tr>
<td>PIC16C5X</td>
<td>8-bit Base-Line MCU Family</td>
<td>PIC16C5X</td>
<td>OTP program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• 12-bit wide instruction set</td>
<td>PIC16C5X</td>
<td>ROM program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• DC - 20 MHz clock speed</td>
<td>PIC16C5X5</td>
<td>OTP program memory, digital only, internal 4 MHz oscillator</td>
</tr>
<tr>
<td></td>
<td>• 200 ns instruction cycle (@ 20 MHz)</td>
<td>PIC16C5X5</td>
<td>OTP program memory with high-voltage operation</td>
</tr>
<tr>
<td>PIC12CXXX</td>
<td>8-bit, 8-pin MCU Family</td>
<td>PIC12C5XX</td>
<td>OTP program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• 12- or 14-bit wide instruction set</td>
<td>PIC12C5XX</td>
<td>OTP program memory, digital only, with EEPROM data memory</td>
</tr>
<tr>
<td></td>
<td>• DC - 10 MHz clock speed</td>
<td>PIC12C5XX</td>
<td>ROM program memory, digital only</td>
</tr>
<tr>
<td></td>
<td>• 400 ns instruction cycle (@ 10 MHz)</td>
<td>PIC12C6XX</td>
<td>OTP program memory with analog functions</td>
</tr>
<tr>
<td></td>
<td>• Internal 4 MHz oscillator</td>
<td>PIC12C6XX</td>
<td>OTP program memory with analog functions and EEPROM data memory</td>
</tr>
</tbody>
</table>

*Note 1: The maximum clock speed for some devices is less than 20 MHz.*

*Please check with your local Microchip distributor, sales representative or sales office for the latest product information.*
Development Systems

Microchip is committed to providing useful and innovative solutions to your embedded system designs. Our installed base of application development systems has grown to an impressive 170,000 systems worldwide.

Among support products offered are MPLAB®-ICE 2000 In-Circuit Emulator running under the Windows® environment. This new real-time emulator supports low-voltage emulation, to 2.0 volts, and full-speed emulation. MPLAB, a complete Integrated Development Environment (IDE), is provided with MPLAB-ICE 2000. MPLAB allows the user to edit, compile and emulate from a single user interface, making the developer productive very quickly. MPLAB-ICE 2000 is designed to provide product development engineers with an optimized design tool for developing target applications. This universal in-circuit emulator provides a complete MCU design toolset for PICmicro MCUs in the PIC12CXXX, PIC16C5X, PIC16CXXX, PIC17CXXX and PIC18CXXX families. MPLAB-ICE 2000 is CE compliant.

Microchip’s newest development tool, MPLAB In-Circuit Debugger (ICD) Evaluation Kit, uses the in-circuit debugging capabilities of the PIC16FXXX and PIC18FXXX MCU family and Microchip’s ICSP™ capability to debug source code in the application, debug hardware in real time and program a target PIC16FXXX and PIC18FXXX device.

PRO MATE® II, the full-featured, modular device programmer, enables you to quickly and easily program user software into PICmicro MCUs, HCS products and Serial EEPROMs. PRO MATE II runs under MPLAB IDE and operates as a stand-alone unit or in conjunction with a PC-compatible host system.

The PICSTART® Plus development kit is a low-cost development system for the PIC12CXXX, PIC16C5X, PIC16CXXX and PIC17CXXX MCUs. PICDEM low-cost demonstration boards are simple boards which demonstrate the basic capabilities of the full range of Microchip’s MCUs. Users can program the sample MCUs provided with PICDEM boards, on a PRO MATE II or PICSTART Plus programmer, and easily test firmware. KEELOO Evaluation Tools support Microchip’s HCS Secure Data Products.

The Serial EEPROM Designer’s Kit includes everything necessary to read, write, erase or program special features of any Microchip Serial EEPROMs. The Total Endurance™ Disk is included to aid in trade-off analysis and reliability calculations. The total kit can significantly reduce time-to-market and result in an optimized system.

The FilterLab™ Active Filter Design Tool simplifies active filter design for embedded systems designers. The unique FilterLab software automates the design of the anti-aliasing filter for an analog-to-digital converter-based data acquisition system. FilterLab also provides full schematic diagrams of the filter circuit with component values, a SPICE model, and displays the frequency and phase response.

In addition to the FilterLab Active Filter Design Tool, Microchip offers a second analog development tool, the MXDEV™1 Analog Evaluation System, making it easier for embedded systems designers to evaluate and develop with Microchip’s line of stand-alone analog products. The hardware and software within the MXDEV 1 system is configured device-specific and allows single or continuous conversions off the analog-to-digital converter under evaluation.

The MCP2510 Controller Area Network (CAN) Developer’s Kit makes software developing easy by using a variety of features to manipulate the functionality of the MCP2510. The MCP2510 CAN Developer’s kit provides the ability to read, display and modify all registers of the MCP2510 on a bit-by-bit or a byte-by-byte basis.

The microID™ Developer’s Kit is an easy-to-use tool for design engineers at all skill levels. Available in a variety of configurations, the microID family of RFID tags can be configured to match existing tags and be directly installed - upgrading to contactless programmability at no added cost. This kit includes all the hardware, software, reference designs and samples required to get started in RFID designs.

### TABLE 2: PICmicro SYNERGISTIC DEVELOPMENT TOOLS

<table>
<thead>
<tr>
<th>Development Tool</th>
<th>Name</th>
<th>PIC12CXXX</th>
<th>PIC16C5X</th>
<th>PIC16CXXX</th>
<th>PIC16F87X</th>
<th>PIC17CXXX</th>
<th>PIC18CXXX</th>
</tr>
</thead>
<tbody>
<tr>
<td>Integrated Development Environment (IDE)</td>
<td>MPLAB®</td>
<td>✔</td>
<td>✔</td>
<td>✔</td>
<td>—</td>
<td>✔</td>
<td>✔</td>
</tr>
<tr>
<td>C Compiler</td>
<td>MPLAB-C17</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>✔</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td>MPLAB-C18</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>Full-Featured, Modular In-Circuit Emulator</td>
<td>MPLAB-ICE 2000</td>
<td>✔</td>
<td>✔</td>
<td>✔</td>
<td>—</td>
<td>✔</td>
<td>✔</td>
</tr>
<tr>
<td>In-Circuit Debugger Evaluation Kit</td>
<td>MPLAB-ICD</td>
<td>—</td>
<td>—</td>
<td>✔</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>Full-Featured, Modular Device Programmer</td>
<td>PRO MATE® II</td>
<td>✔</td>
<td>✔</td>
<td>✔</td>
<td>—</td>
<td>✔</td>
<td>✔</td>
</tr>
<tr>
<td>Entry-Level Development Kit with Programmer</td>
<td>PICSTART® Plus</td>
<td>✔</td>
<td>✔</td>
<td>✔</td>
<td>—</td>
<td>✔</td>
<td>✔</td>
</tr>
</tbody>
</table>
Software Support

MPLAB Integrated Development Environment (IDE) is a Windows-based development platform for Microchip’s PICmicro MCUs. MPLAB IDE offers a project manager and program text editor, a user-configurable toolbar containing four pre-defined sets and a status bar which communicates editing and debugging information.

MPLAB-IDE is the common user interface for Microchip development systems tools including MPLAB Editor, MPASM Assembler, MPLAB-SIM Software Simulator, MPLIB, MPLINK, MPLAB-C17 Compiler, MPLAB-C18 Compiler, MPLAB-ICE 2000, PRO MATE II Programmer and PICSTART Plus Development Programmer.

Microchip endeavors at all times to provide the best service and responsiveness possible to its customers. The Microchip Internet site can provide you with the latest technical information, production released software for development tools, application notes and promotional news on Microchip products and technology. The Microchip World Wide Web address is http://www.microchip.com.

Secure Data Products Overview

Microchip’s patented KEELOQ® code hopping technology is the perfect solution for remote keyless entry and logical/physical access control systems. The initial device in the family, the HCS300 encoder, replaces current fixed code encoders in transmitter applications providing a low cost, integrated solution. The KEELOQ family is continuing to expand with the HCS301 (high voltage encoder), HCS200 (low-end, low-cost encoder), and high-end encoders (HCS360 and HCS361) that meet OEM specifications and requirements. The HCS410, a self-powered transponder superset of the HCS360, is the initial device in a new and expanding encoder/transponder family.

Microchip provides flexible decoder solutions by providing optimized routines for Microchip’s PICmicro MCUs. This allows the designer to combine the decoder and system functionality in a MCU. The decoder routines are available under a license agreement. The HCS500, HCS512 and HCS515 are the first decoder devices in the KEELOQ family. These devices are single chip decoder solutions and simplify designs by handling learning and decoding of transmitters.

The KEELOQ product family is expanding to include enhanced encoders and decoders. Typical applications include automotive RKE, alarm and immobilizer systems, garage door openers and home security systems.

<table>
<thead>
<tr>
<th>KEELOQ Encoder Devices</th>
<th>Product</th>
<th>Transmission Code Length Bits</th>
<th>Prog. Encryption-Key Bits</th>
<th>Seed Length</th>
<th>Operating Voltage</th>
</tr>
</thead>
<tbody>
<tr>
<td>HCS101*</td>
<td>66</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>3.0V to 13.0V</td>
</tr>
<tr>
<td>HCS200</td>
<td>66</td>
<td>32</td>
<td>64</td>
<td>32</td>
<td>3.0V to 13.0V</td>
</tr>
<tr>
<td>HCS300*</td>
<td>66</td>
<td>32</td>
<td>64</td>
<td>32</td>
<td>3.5V to 13.0V</td>
</tr>
<tr>
<td>HCS301</td>
<td>66</td>
<td>32</td>
<td>64</td>
<td>32</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS400</td>
<td>66</td>
<td>32</td>
<td>64</td>
<td>32</td>
<td>3.5V to 13.0V</td>
</tr>
<tr>
<td>HCS401</td>
<td>66</td>
<td>32</td>
<td>64</td>
<td>32</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS320</td>
<td>67</td>
<td>32</td>
<td>64</td>
<td>48</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS360</td>
<td>67</td>
<td>32</td>
<td>64</td>
<td>48</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS361</td>
<td>67</td>
<td>32</td>
<td>64</td>
<td>48</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS365*</td>
<td>69</td>
<td>32</td>
<td>2 x 64</td>
<td>60</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS370*</td>
<td>69</td>
<td>32</td>
<td>2 x 64</td>
<td>60</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS410</td>
<td>69</td>
<td>32</td>
<td>64</td>
<td>60</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS412*</td>
<td>69</td>
<td>32</td>
<td>64</td>
<td>60</td>
<td>2.0V to 6.6V</td>
</tr>
<tr>
<td>HCS470*</td>
<td>69</td>
<td>32</td>
<td>2 x 64</td>
<td>60</td>
<td>2.0V to 6.6V</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>KEELOQ Decoder Devices</th>
<th>Product</th>
<th>Reception Code Length Bits</th>
<th>Transmitters Supported</th>
<th>Functions</th>
<th>Operating Voltage</th>
</tr>
</thead>
<tbody>
<tr>
<td>HCS500*</td>
<td>67</td>
<td>1 up to 7</td>
<td>15 serial functions</td>
<td>2.0V to 5.5V</td>
<td></td>
</tr>
<tr>
<td>HCS512</td>
<td>67</td>
<td>Up to 4</td>
<td>15 (ISO, S1, S2, S3); Voice, Serial</td>
<td>3.0V to 6.0V</td>
<td></td>
</tr>
<tr>
<td>HCS515</td>
<td>67</td>
<td>Up to 7</td>
<td>15 serial, 3 parallel</td>
<td>4.0V to 5.5V</td>
<td></td>
</tr>
</tbody>
</table>

*Contact Microchip Technology Inc. for availability.

Analog/Interface Products

Using its technology achievements in developing analog circuitry for its PICmicro MCU family, the Company launched a complementary line of stand-alone analog and interface products. Many of these stand-alone devices support functionality that may not currently available on PICmicro MCUs. Stand-alone analog IC products currently offered include:

- Analog-to-Digital Converters
- Operational Amplifiers
- System Supervisors

Microchip also offers innovative silicon products to support a variety of bus interfaces used to transmit data to and from embedded control systems. The first interface products support Controller Area Network (CAN), a bus protocol highly integrated into a variety of networked applications including automotive.

High-Performance 12-Bit Analog-to-Digital Converters

The MCP320X 12-bit analog-to-digital converter (ADC) family is based on a successive approximation register architecture. The first four members include: MCP3201, MCP3202, MCP3204 and MCP3208. The MCP320X family features 100K samples per second throughput, low power of 400 microamps active and 500 nanamps standby, wide supply voltage of 2.7-5.5 volts, extended industrial temperature range of –40° to 85°, +/- 1 LSB DNL and +/- 1 LSB INL max. at 100 ksp.s., no missing codes, and a serial output with an industry-standard SPI™ bus interface. The MCP320X is available in 1-, 2-, 4-, and 8-input channel versions (the MCP3201, MCP3202, MCP3204 and MCP3208, respectively). The devices
are offered in PDIP, SOIC and TSSOP packages. Applications include data acquisition, instrumentation and measurement, multi-channel data loggers, industrial PCs, motor control, robotics, industrial automation, smart sensors, portable instrumentation, and home medical appliances.

**Operational Amplifiers**

The MCP60X Operational Amplifier family includes four devices: MCP601, MCP602, MCP603 and MCP604. These devices are Microchip's first 2.7 volt single supply operational amplifier products. The MCP606X family offers a gain bandwidth product of 2.8 MHz with low typical operating current of 230 µA. The MCP606X devices use Microchip's advanced CMOS technology which provides low bias current, high speed operation, high open-loop gain and rail-to-rail output swing.

**System Supervisors**

Microchip offers a complete family of system supervisor products. The new devices include the MCP8008/810 and MCP100/101 supervisory circuits with push-pull output and the MCP120/130 supervisory circuits with open drain output. The devices are functionally and pin-out comparable to products from other analog suppliers.

**Controller Area Network (CAN)**

Microchip is enhancing its product portfolio by introducing the CAN Product Family. The MCP2510 is the smallest, easiest-to-use, CAN controller on the market today. Combining the MCP2510 with Microchip's broad range of high-performance PICmicro MCUs enables Microchip to support for virtually all of today's CAN-based applications. Other potential benefits of having a separate CAN controller include the ability for system designers to select from a much wider variety of MCUs for an optimal performance solution.

Additional products planned for Microchip's CAN product portfolio include other CAN peripherals and a family of PICmicro MCUs with integrated CAN support.

**microID™ RFID Tagging Devices**

Only Microchip manufactures world-class components for every application in the radio frequency identification (RFID) system. From the advanced, feature-packed microID family of RFID tags and high-endurance Serial EEPROMs to high performance PICmicro MCUs and KeeLoq code hopping encoders - Microchip's full range of RFID solutions are available for your tag, peripheral and reader application designs.

The microID family can emulate almost any standard on the market today. It provides drop-in compatible solutions to the most commonly used 125 kHz and 13.56 MHz tags and an upgrade migration path for virtually any application with higher performance and new features.

---

**Serial EEPROM Overview**

Microchip’s high-endurance Serial EEPROMs complement the diverse MCU product families. Serial EEPROMs are available in a variety of densities, operating voltages, bus interface protocols, operating temperature ranges and space-saving packages.

**Densities:**

The densities range from 128 bits to 256 Kbits with higher density devices in development.

**Bus Interface Protocols:**

We offer all popular protocols: I^2C™, Microwire® and SPI.

**Operating Voltages:**

In addition to standard 5V devices there are two low voltage families. The “LC” devices operate down to 2.5V, while the breakthrough “AA” family operates, in both read and write mode, down to 1.8V, making these devices highly suitable for alkaline and NiCd battery powered applications.

**Temperature Ranges:**

Like all Microchip devices, many Serial EEPROMs are offered in Commercial (0°C to +70°C), Industrial (-40°C to +85°C) and Extended (-40°C to +125°C) operating temperature ranges.

**Packages:**

Small footprint packages include: industry standard 5-lead SOIC, 8-lead SOIC in JEDEC and EIAJ body widths, and 14-lead SOIC. The SOIC comes in two body widths; 150 mil and 207 mil.

**Technology Leadership:**

Selected Microchip Serial EEPROMs are backed by a 1 million Erase/Write cycle. Microchip's erase/write cycle endurance is among the best in the world, and only Microchip offers such unique and powerful development tools as the Total Endurance disk. This mathematical software model is an innovative tool used by system designers to optimize Serial EEPROM performance and reliability within the application.

Microchip offers Plug-and-Play to the DIMM module market with the 24LC552, a special function single-chip EEPROM that is available in space saving packages. For Plug-and-Play video monitor applications, Microchip offers the 24LC21, a single-chip DDC1™/DDC2™-compatible solution. In addition, Microchip released a high-speed 1 MHz 2-wire Serial EEPROM device ideal for high-performance embedded systems.

Microchip is a high-volume supplier of Serial EEPROMs to all the major markets worldwide. The Company continues to develop new Serial EEPROM solutions for embedded control applications.
OTP EPROM Overview

Microchip’s CMOS EPROM devices are produced in densities from 64K to 512K. Typical applications include computer peripherals, instrumentation, and automotive devices. Microchip’s expertise in surface mount packaging on SOIC and TSOP packages led to the development of the surface mount OTP EPROM market where Microchip is a leading supplier today. Microchip is also a leading supplier of low-voltage EPROMs for battery powered applications.

MIGRATABLE MEMORY™ TECHNOLOGY

Microchip’s innovative Migratable Memory technology (MMT) provides socket and software compatibility among all of its equivalent ROM, OTP and FLASH memory MCUs. MMT allows customers to match the selection of MCU memory technology to the product life cycle of their application, providing an easy migration path to a lower cost solution whenever appropriate.

FLASH memory is an ideal solution for engineers designing products for embedded systems – especially during the development and early stages of the product. In certain products and applications, FLASH memory may be used for the life of the product because of the advantages of field upgradability or where product inventory flexibility is required.

Once the design enters the pre-production stage and continues through introduction and growth stages, OTP program memory provides maximum programming flexibility and minimum inventory scrapage. The OTP device is pin and socket compatible with the FLASH device – providing a lower cost, high-volume flexible solution.

As the design enters a mature stage and program code stabilizes, a lower cost, socket compatible ROM memory device could be used. In some cases, OTP memory may still be used as the most cost-effective memory technology for the product. Compatibility and flexibility are key to the success of the PICmicro MCU product family, and ultimately the success of our customers.

FLEXIBLE PROGRAMMING OPTIONS

To meet the stringent design requirements placed on our customers, the following innovative programming options are offered. These programming options address procurement issues by reducing and limiting work-in-process liability and facilitating finished goods code revisions. Microchip’s worldwide distributors stock reprogrammable and one-time programmable inventory, allowing customers to respond to immediate sales opportunities or accommodate engineering changes off the shelf.

FLASH (electrically reprogrammable)

PICmicro FLASH MCUs allow erase and reprogramming of the MCU program memory. Reprogrammability offers a highly flexible solution to today’s ever-changing market demands – and can substantially reduce time to market. Users can program their systems very late in the manufacturing process or update systems in the field. This allows easy code revisions, system parameterization or customer-specific options with no scrapage. Reprogrammability also reduces the design verification cycle.

One-Time Programmable (OTP)

PICmicro OTP MCUs are manufactured in high volumes without customer specific software and can be shipped immediately for custom programming. This is useful for customers who need rapid time to market and flexibility for frequent software updates.

In-Circuit Serial Programming™ (ICSP™)

Microchip’s PICmicro FLASH and OTP MCUs feature ICSP capability. ICSP allows the MCU to be programmed after being placed in a circuit board, offering tremendous flexibility, reduced development time, increased manufacturing efficiency and improved time to market. This popular technology also enables reduced cost of field upgrades, system calibration during manufacturing, the addition of unique identification codes to the system and system calibration. Requiring only two I/O pins for most devices, Microchip offers the most non-intrusive programming methodology in the industry.

Self Programming

Microchip’s PIC16F87X family features self programming capability. Self programming enables remote upgrades to the FLASH program memory and the end equipment through a variety of medium ranging from Internet and Modem to RF and Infrared. To setup for self programming, the designer programs a simple boot loader algorithm in a code protected area of the FLASH program memory. Through the selected medium, a secure command allows entry into the PIC16F87X MCU through the USART, I²C or SPI serial communication ports. The boot loader is then enabled to reprogram the PIC16F87X FLASH program memory with data received over the desired medium. And, of course, self programming is accomplished without the need for external components and without limitations on the PIC16F87X’s operating speed or voltage.

Quick-Turn Programming (QTP)

Microchip offers a QTP programming service for factory production orders. This service is ideal for customers who choose not to program a medium to high unit volume in their own factories, and whose production code patterns have stabilized.
Serialized Quick-Turn Programming (SQTPSM)

SQTP is a unique, flexible programming option that allows Microchip to program serialized, random or pseudo-random numbers into each device. Serial programming allows each device to have a unique number which can serve as an entry-code, password or ID number.

Masked ROM

Microchip offers Masked ROM versions of many of its most popular PICmicro MCUs, giving customers the lowest cost option for high volume products with stable firmware.

Future Products and Technology

Microchip is constantly developing advanced process technology modules and new products that utilize our advanced manufacturing capabilities. Current production technology utilizes lithography dimensions down to 0.7 micron.

Microchip’s research and development activities include exploring new process technologies and products that have industry leadership potential. Particular emphasis is placed on products that can be put to work in high-performance broad-based markets.

Equipment is continually updated to bring the most sophisticated process, CAD and testing tools online. Cycle times for new technology development are continuously reduced by using in-house mask generation, a high-speed pilot line within the manufacturing facility and continuously improving methodologies.

Objective specifications for new products are developed by listening to our customers and by close co-operation with our many customer-partners worldwide.
NOTES:
HIGHLIGHTS

This section of the manual contains the following major topics:

1.1 Introduction ................................................................. 1-2
1.2 Manual Objective .......................................................... 1-3
1.3 Device Structure .......................................................... 1-4
1.4 Development Support .................................................... 1-6
1.5 Device Varieties .......................................................... 1-7
1.6 Style and Symbol Conventions ....................................... 1-12
1.7 Related Documents ....................................................... 1-14
1.8 Related Application Notes ............................................. 1-17
1.9 Revision History .......................................................... 1-18
1.1 Introduction

Microchip is the Embedded Control Solutions Company. The company’s focus is on products that meet the needs of the embedded control market. We are a leading supplier of:

- 8-bit general purpose microcontrollers (PICmicro® MCUs)
- Speciality and standard non-volatile memory devices
- Security devices (KEELOG®)
- Application specific standard products

Please request a Microchip Product Line Card for a listing of all the interesting products that we have to offer. This literature can be obtained from your local sales office, or downloaded from the Microchip web site (www.microchip.com).

In the past, 8-bit MCU users were fixed on the traditional MCU model for production, a ROM device. Microchip has been the leader in changing this perception by showing that OTP devices can give a better lifetime product cost compared to ROM versions.

Microchip has strength in FLASH and EPROM technology. This makes it the memory technology of choice for the PICmicro MCU’s program memory. Microchip has minimized the cost difference between EPROM and ROM memory technology. Therefore, Microchip can pass these benefits on to our customers. This is not true for other MCU vendors, and is seen in the price difference between their FLASH/EPROM and ROM versions.

The growth of Microchip’s 8-bit MCU market share is a testament to the PICmicro MCU’s ability to meet the needs of many customers. This growth has made the PICmicro architecture one of the top two architectures available in the general market today. This growth was fueled by the Microchip vision of the benefits of a low cost Field Programmable MCU solution. Some of the benefits for the customer include:

- Quick time to market
- Allows code changes to product during production run
- No Non-Recurring Engineering (NRE) charges for Mask Revisions
- Ability to easily serialize the product
- Ability to store calibration data without additional hardware
- Better able to maximize use of PICmicro MCU inventory
- Less risk, since the same device is used for development as well as for production.

Microchip’s PICmicro 8-bit MCUs offer a price/performance ratio that allows them to be considered for any traditional 8-bit MCU application, as well as some traditional 4-bit applications (Base-Line family), low-end 16-bit applications (PIC17CXXX and PIC18CXXX families), dedicated logic replacement and low-end DSP applications (High-End and Enhanced families). These features and price-performance mix make PICmicro MCUs an attractive solution for most applications.
Section 1. Introduction

1.2 Manual Objective

PICmicro devices are grouped by the size of their Instruction Word and their Instruction Set. The four current PICmicro families and their Instruction Word length are:

1. Base-Line: 12-bit Instruction Word length
2. Mid-Range: 14-bit Instruction Word length
3. High-End: 16-bit Instruction Word length
4. Enhanced: 16-bit Instruction Word length

This manual focuses on the Enhanced MCU family of devices, which are also referred to as the PIC18CXXX MCU family.

The operation of the Enhanced MCU family architecture and peripheral modules is explained, but does not cover, the specifics of each device. This manual is not intended to replace the device data sheets, but complement them. In other words, this guide supplies the general details and operation of the PICmicro architecture and peripheral modules, while the data sheet gives the specific details (such as device memory mapping).

Initialization examples are given throughout this manual. These examples sometimes need to be written as device specific as opposed to family generic, though they are valid for most other devices. Some modifications may be required for devices with variations in register file mappings.
1.3 Device Structure

Each part of a device can be placed into one of three groups:

1. Core
2. Peripherals
3. Special Features

1.3.1 Core

The core pertains to the basic features that are required to make the device operate. These include:

1. Oscillator Revision "DS39502A"
2. Reset Revision "DS39503A"
3. Architecture Revision "DS39504A"
4. CPU (Central Processing Unit) and ALU (Arithmetic Logical Unit) Revision "DS39505A"
5. Hardware 8x8 Multiplier Revision "DS31006A"
6. Memory Revision "DS31007A"
7. Table Read / Table Write Revision "DS39508A"
8. System Bus Revision "DS39509A"
9. Interrupts Revision "DS39510A"
10. Instruction Set Revision "DS39522A"

1.3.2 Peripherals

Peripherals are the features that add a differentiation from a microprocessor. These ease in interfacing to the external world (such as general purpose I/O, A/D inputs, and PWM outputs), and internal tasks, such as keeping different time bases (i.e. timers). The peripherals that are discussed are:

1. I/O Revision "DS39511A"
2. Parallel Slave Port (PSP) Revision "DS39512A"
3. Timer0 Revision "DS39513A"
4. Timer1 Revision "DS39514A"
5. Timer2 Revision "DS39515A"
6. Timer3 Revision "DS39516A"
7. Capture/Compare/PWM (CCP) Revision "DS39517A"
8. Serial Slave Port (SSP) Revision "DS39519A"
9. Master Synchronous Serial Port (MSSP) Revision "DS39520A"
10. Addressable USART Revision "DS39521A"
11. CAN Revision "DS39522A"
12. Comparator Voltage Reference Revision "DS31023A"
13. Comparators Revision "DS39525A"
14. Compatible 10-bit A/D Converter Revision "DS31026A"
15. 10-bit A/D Converter Revision "DS31027A"
Section 1. Introduction

1.3.3 Special Features

Special features are the unique features that help to:
- Decrease system cost
- Increase system reliability
- Increase design flexibility

The Enhanced PICmicro MCUs offer several features that help achieve these goals. The special features discussed are:

1. Low Voltage Detect  Revision “DS39528A”
2. WDT and Sleep Operation  Revision “DS31029A”
3. Device Configuration Bits  Revision “DS39530A”
4. In-Circuit Serial Programming™ (ICSP™)  Revision “DS39531A”

1.3.4 Other Sections

This section provides the cross references for the remaining sections of this manual.

1. Introduction  Revision “DS39501A”
2. Electrical Specifications  Revision “DS31033A”
3. Device Characteristics  Revision “DS31034A”
4. Development Tools  Revision “DS31035A”
5. Code Development  Revision “DS31036A”
6. Appendix  Revision “DS39537A”
7. Glossary  Revision “DS39538A”
1.4 Development Support

Microchip offers a wide range of development tools that allow users to efficiently develop and debug application code. Microchip’s development tools can be broken down into four categories:

1. Code generation
2. Hardware/Software debug
3. Device programmer
4. Product evaluation boards

All tools developed by Microchip operate under the MPLAB® Integrated Development Environment (IDE), while some third party tools may not. The code generation tools include:

- MPASM
- MPLAB®-C17 (for PIC17CXXX family only)
- MPLAB®-C18 (for PIC18CXXX family only)

These software development programs include device header files. Each header file defines the register names (as shown in the device data sheet) to the specified address or bit location. Using the header files eases code migration and reduces the tediousness of memorizing a register’s address or a bit’s position in a register.

**Note:** Microchip strongly recommends that the supplied header files be used in the source code of your program. This eases code migration, improves code readability, and increases the quality and depth of the technical support that Microchip can offer.

Tools which ease in debugging software are:

- MPLAB®-ICE In-Circuit Emulator
- PICMASTER® In-Circuit Emulator
- ICEPIC In-Circuit Emulator
- MPLAB®-SIM Software Simulator

After generating and debugging the application software, the device will need to be programmed. Microchip offers two levels of programmers:

1. PICSTART® Plus programmer
2. PRO MATE® II programmer

Demonstration boards allow the developer of software code to evaluate the capability and suitability of the device to the application. The demo boards offered are:

- PICDEM-1
- PICDEM-2 (can be used with PIC18CXX2 devices)
- PICDEM-3
- PICDEM-14A
- PICDEM-17

At the time of publication, only PICDEM-2 could be used with some Enhanced MCU devices.

A full description of each of Microchip’s development tools is discussed in the “Development Tools” section. As new tools are developed, product briefs and user guides may be obtained from the Microchip web site (www.microchip.com) or from your local Microchip Sales Office.

Code development recommendations and techniques are provided in the “Code Development” section.

Microchip offers other reference tools to speed the development cycle. These include:

- Application Notes
- Reference Designs
- Microchip web site
- Local Sales Offices with Field Application Support
- Corporate Support Line

The Microchip web site lists other sites that may be useful references.
Section 1. Introduction

1.5 Device Varieties

Once the functional requirements of the device are specified, other choices need to be made. These include:
- Memory technology
- Operating voltage
- Operating temperature range
- Operating frequency
- Packaging

Microchip has a large number of options and option combinations, one of which should fulfill your requirements.

1.5.1 Memory Varieties

Memory technology has no effect on the logical operation of a device. Due to the different processing steps required, some electrical characteristics may vary between devices with the same feature set/pinout but with different memory technologies. An example is the electrical characteristic \( V_{IL} \) (Input Low Voltage), which may have some difference between a typical EPROM device and a typical ROM device.

Each device has a variety of frequency ranges and packaging options available. Depending on application and production requirements, the proper device options can be identified using the information in the Product Identification System section at the end of each data sheet. When placing orders, please use the “Product Identification System” at the back of the data sheet to specify the correct part number.

When discussing the functionality of the device, the memory technology and the voltage range do not matter. Microchip offers three program memory types. The memory type is designated in the part number by the first letter(s) after the family affiliation designators.

1. \textbf{C}, as in PIC18CXXX. These devices have EPROM type memory.
2. \textbf{CR}, as in PIC18CRXXX. These devices have ROM type memory.
3. \textbf{F}, as in PIC18FXXX. These devices have FLASH type memory.

1.5.1.1 EPROM

Microchip focuses on Erasable Programmable Read Only Memory (EPROM) technology to give customers flexibility throughout their entire design cycle. With this technology, Microchip offers various packaging options as well as services.

1.5.1.2 Read Only Memory (ROM) Devices

Microchip offers a masked Read Only Memory (ROM) version of several of the highest volume parts, thus giving customers a lower cost option for high volume, mature products. ROM devices do not allow serialization information in the program memory space.

For information on submitting ROM code, please contact your local Microchip sales office.

1.5.1.3 FLASH Memory Devices

These devices are electrically erasable, and can therefore be offered in a low cost plastic package. Being electrically erasable, these devices can be erased and reprogrammed without removal from the circuit. A device will have the same specifications whether it is used for prototype development, pilot programs or production.
1.5.2 Operating Voltage Range Options

All Enhanced PICmicro MCUs operate over the standard voltage range. Devices are also offered which operate over an extended voltage range (and reduced frequency range). Table 1-1 shows all possible memory types and voltage range designators for the PIC18CXXX MCU family. The designators are in bold typeface.

Table 1-1: Device Memory Type and Voltage Range Designators

<table>
<thead>
<tr>
<th>Memory Type</th>
<th>Voltage Range</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Standard</td>
</tr>
<tr>
<td>EPROM</td>
<td>PIC18CXXX</td>
</tr>
<tr>
<td>ROM</td>
<td>PIC18CRXXX</td>
</tr>
<tr>
<td>FLASH</td>
<td>PIC18FXXX</td>
</tr>
</tbody>
</table>

*Note:* Not all memory types may be available for a particular device.

As you can see in Table 1-2, Microchip specifies its extended range devices at a more conservative voltage range until device characterization has ensured they will be able to meet the goal of their final design specifications.

Table 1-2: Typical Voltage Ranges for Each Device Type

<table>
<thead>
<tr>
<th>Typical Voltage Range</th>
<th>EPROM</th>
<th>ROM</th>
<th>Flash</th>
</tr>
</thead>
<tbody>
<tr>
<td>Standard</td>
<td>C</td>
<td>CR</td>
<td>F</td>
</tr>
<tr>
<td></td>
<td>4.2 - 5.5V</td>
<td>4.2 - 5.5V</td>
<td>4.2 - 5.5V</td>
</tr>
<tr>
<td>Extended</td>
<td>LC</td>
<td>LCR</td>
<td>LF</td>
</tr>
<tr>
<td>Before device characterization</td>
<td>3.0 - 5.5V</td>
<td>3.0 - 5.5V</td>
<td>3.0 - 5.5V</td>
</tr>
<tr>
<td>Final specification (1)</td>
<td>LC</td>
<td>LCR</td>
<td>LF</td>
</tr>
<tr>
<td></td>
<td>2.5 - 5.5V</td>
<td>2.5 - 5.5V</td>
<td>2.0 - 5.5V</td>
</tr>
</tbody>
</table>

*Note 1:* This voltage range depends on the results of device characterization.

---

*Note:* This voltage range depends on the results of device characterization.
Section 1. Introduction

Figure 1-1 and Figure 1-2 show example Voltage to Frequency Charts for the Enhanced MCU family. The voltages and frequencies of any given device may be different than those shown. These figures are intended to show that with an LC (extended voltage) device, the user can make a trade-off between voltage of operation and the frequency of the device. The device $F_{\text{MAX}}$ is given below the graph. As the voltages and frequencies of the device change, the equation for the device $F_{\text{MAX}}$ will change. Equation 1-1 shows the general equation to determine the device $F_{\text{MAX}}$.

**Equation 1-1: Generic $F_{\text{MAX}}$ Equation**

$$F_{\text{MAX}} = (\text{slope}) \cdot (V_{\text{DDAPPMIN}} - V_{\text{DDMIN}}) + \text{offset}$$

**Figure 1-1:** PIC18CXXX Voltage-Frequency Graph - Example

![Figure 1-1](image1)

**Figure 1-2:** PIC18LCXXX Voltage-Frequency Graph - Example

![Figure 1-2](image2)

Note: $V_{\text{DDAPPMIN}}$ is the minimum voltage of the PICmicro device in the application.
1.5.3 Packaging Varieties

Depending on the development phase of your project, one of three package types would be used.

The first package type is ceramic with an erasure window. The window allows ultraviolet light into the package so that the device inside may be erased. The package is used for the development phase, since the device’s program memory can be erased and reprogrammed many times.

The second package type is a low cost plastic package. This package type is used in production where device cost is to be kept to a minimum.

Lastly, there is the Die option. A Die is an unpackaged device that has been tested. Dies are used in low cost designs and designs where board space is at a minimum. For additional information on die support, please refer to the Die Support Document (DS30258).

Table 1-3 shows a quick summary of the typical use for each package type.

Table 1-3: Typical Package Uses

<table>
<thead>
<tr>
<th>Package Type</th>
<th>Typical Usage</th>
</tr>
</thead>
<tbody>
<tr>
<td>Windowed</td>
<td>Development mode</td>
</tr>
<tr>
<td>Plastic</td>
<td>Production</td>
</tr>
<tr>
<td>Die</td>
<td>Special applications, such as those which require minimum board space</td>
</tr>
</tbody>
</table>

1.5.3.1 UV Erasable Devices

The UV erasable version of EPROM program memory devices is optimal for prototype development and pilot programs.

These devices can be erased and reprogrammed to any of the configuration modes. Third party programmers are available. Refer to Microchip’s Third Party Guide (DS00104) for a list of sources.

The amount of time required to completely erase a UV erasable device depends on the:

- Wavelength of the light
- Intensity of the light
- Distance of the device from the UV source
- Process technology of the device (size of the memory cells).

**Note:** Fluorescent lights and sunlight both emit ultraviolet light at the erasure wavelength. Leaving a UV erasable device’s window uncovered could cause, over time, the device’s memory cells to become erased. The erasure time for a fluorescent light is about three years, while sunlight requires only about one week. To prevent the memory cells from losing data, an opaque label should be placed over the erasure window.

1.5.3.2 One-Time-Programmable (OTP) Devices

The availability of OTP devices is especially useful for customers expecting code changes and updates.

OTP devices in plastic packages permit the user to program them once. Often the system can be designed so that programming may be performed in-circuit (after the device has been mounted on the circuit board).

1.5.3.3 FLASH Devices

These devices are electrically erasable, and can therefore be offered in a low cost plastic package. Being electrically erasable, these devices can be both erased and reprogrammed without removal from the circuit. A device will have the same specifications whether it is used for prototype development, pilot programs, or production.
Section 1. Introduction

1.5.3.4 ROM Devices

ROM devices have their program memory fixed at the time of the silicon manufacture. Since the program memory cannot be changed, the device is usually housed in the low cost plastic package.

1.5.3.5 Die

The Die option allows the board size to become as small as physically possible. The Die Support document (DS30258) explains general information about using and designing with Die. There are also individual specification sheets that detail Die specific information. Manufacturing with Die requires special knowledge and equipment. This means that the number of manufacturing houses that support Die will be limited. If you decide to use the Die option, please research your manufacturing sites to ensure that they will be able to meet the specialized requirements of Die use.

1.5.3.6 Specialized Services

For OTP customers with established code, Microchip offers two specialized services. These two services, Quick Turn Production Programming and Serialized Quick Turn Production Programming, allow customers to shorten their manufacturing cycle time.

1.5.3.6.1 Quick Turn Production (OTP) Programming

Microchip offers this programming service for factory production orders. This service is made available for users who choose not to program a medium to high quantity of units at their factory and whose code patterns have stabilized. The devices are identical to the OTP devices, but with all EPROM locations and configuration options already programmed by Microchip. Certain code and prototype verification procedures apply before production shipments are available. Please contact your local Microchip sales office for more details.

1.5.3.6.2 Serialized Quick Turn Production (SQTP℠) Programming

Microchip offers a unique programming service where a few user-defined locations in each device are programmed with unique numbers. These numbers may be:

- Random numbers
- Pseudo-random numbers
- Sequential numbers

Serial programming allows each device to have a unique number which can serve as an entry-code, password, ID, or serial number.
1.6 Style and Symbol Conventions

Throughout this document, certain style and font format conventions are used. Most format conventions imply a distinction should be made for the emphasized text. The MCU industry has many symbols and non-conventional word definitions/abbreviations. Table 1-4 provides a description for many of the conventions contained in this document. A glossary is provided in the “Glossary” section, which contains more word and abbreviation definitions that are used throughout this manual.

1.6.1 Document Conventions

Table 1-4 defines some of the symbols and terms used throughout this manual.

<table>
<thead>
<tr>
<th>Symbol or Term</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>set</td>
<td>To force a bit/register to a value of logic ‘1’.</td>
</tr>
<tr>
<td>clear</td>
<td>To force a bit/register to a value of logic ‘0’.</td>
</tr>
</tbody>
</table>
| reset          | 1) To force a register/bit to its default state.  
                | 2) A condition in which the device places itself after a device reset occurs. Some bits will be forced to ‘0’ (such as interrupt enable bits), while others will be forced to ‘1’ (such as the I/O data direction bits). |
| 0xnn or nnh     | Designates the number ‘nn’ in the hexadecimal number system. These conventions are used in the code examples. |
| B’bbbbbbbb’     | Designates the number ‘bbbbbbbb’ in the binary number system. This convention is used in the text and in figures and tables. |
| R-M-W           | Read - Modify - Write. This is when a register or port is read, then the value is modified, and that value is then written back to the register or port. This action can occur from a single instruction (such as bit set file, BSF) or a sequence of instructions. |
| : (colon)       | Used to specify a range or the concatenation of registers/bits/pins. One example is TMR1H:TMR1L, which is the concatenation of two 8-bit registers to form a 16-bit timer value, while SSPM3:SSPM0 are 4-bits used to specify the mode of the SSP module. Concatenation order (left-right) usually specifies a positional relationship (MSb to LSb, higher to lower). |
| <>              | Specifies bit(s) locations in a particular register. One example is SSPCON<SSPM3:SSPM0> (or SSPCON<3:0>) which specifies the register and associated bits or bit positions. |
| Courier Font    | Used for code examples, binary numbers and for Instruction Mnemonics in the text. |
| Times Font      | Used for equations and variables. |
| Times, Bold Font, Italic | Used in explanatory text for items called out from a graphic/equa- tion/example. |
| Note            | A Note presents information that we wish to reemphasize, either to help you avoid a common pitfall, or make you aware of operating differences between some device family members. A Note is always in a shaded box (as below), unless used in a table, where it is at the bottom of the table (as in this table). |

Note: This is a Note in a note box.

Caution (1) A caution statement describes a situation that could potentially damage software or equipment.

Warning (1) A warning statement describes a situation that could potentially cause personal harm.

Note 1: The information in a caution or a warning is provided for your protection. Please read each caution and warning carefully.
1.6.2 **Electrical Specifications**

Throughout this manual, there will be references to electrical specification parameter numbers. A parameter number represents a unique set of characteristics and conditions that is consistent between every data sheet, though the actual parameter value may vary from device to device.

The “Electrical Specifications” section shows all the specifications that are documented for all devices. No one device has all these specifications. This section is intended to let you know the types of parameters that Microchip specifies. The value of each specification is device dependent, though we strongly attempt to keep them consistent across all devices.

<table>
<thead>
<tr>
<th>Parameter Number Format</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>DXXX</td>
<td>DC Specification</td>
</tr>
<tr>
<td>AXXX</td>
<td>DC Specification for Analog Peripherals</td>
</tr>
<tr>
<td>XXX</td>
<td>Timing (AC) Specification</td>
</tr>
<tr>
<td>PDXXX</td>
<td>Device Programming DC Specification</td>
</tr>
<tr>
<td>PXXX</td>
<td>Device Programming Timing (AC) Specification</td>
</tr>
</tbody>
</table>

Legend: XXX represents a number.
1.7 Related Documents

Microchip, as well as other sources, offers additional documentation which can aid in your development with PICmicro MCUs. These lists contain the most common documentation, but other documents may also be available. Please check the Microchip website (www.microchip.com) for the latest published technical documentation.

1.7.1 Microchip Documentation

The following documents are available from Microchip. Many of these documents provide application specific information that give actual examples of using, programming and designing with PICmicro MCUs.

1. MPASM and MPLINK (w/ MPLIB) User’s Guide (DS33014)
   This document explains how to use Microchip’s MPASM assembler.

2. MPLAB®-CXX Compiler User’s Guide (DS51217)
   This document explains how to use Microchip’s MPLAB-C17 and MPLAB-C18 compilers.

3. MPLAB® IDE, Simulator, Editor User’s Guide (DS51025)
   This document explains how to use Microchip’s MPLAB Integrated Development Environment.

4. MPLAB®-CXX Reference Guide Libraries and Precompiled Object Files (DS51224)
   This document explains how to use Microchip’s MPLAB Reference Guide Libraries and Precompiled Object Files.

5. PICMASTER® User’s Guide (DS30421)
   This document explains how to use Microchip’s PICMASTER In-Circuit Emulator.

6. PRO MATE® User’s Guide (DS30082)
   This document explains how to use Microchip’s PRO MATE Universal Programmer.

7. PICSTART®-Plus User’s Guide (DS51026)
   This document explains how to use Microchip’s PICSTART-Plus low-cost universal programmer.

8. PICmicro® Mid-Range MCU Family Reference Manual (DS33023)
   This document discusses the operation of PICmicro Mid-Range MCU devices, explaining the detailed operation of the architecture and peripheral modules. It is a compliment to the device data sheets for the Mid-Range family.

   This document contains a plethora of application notes. This is useful for insight on how to use the device (or parts of it), as well as getting started on your particular application due to the availability of extensive code files.

10. Embedded Control Handbook Update 2000 (DS00711)
    This document contains additional application notes.

11. Embedded Control Handbook Volume II Math Library (DS00167)
    This document contains the Math Libraries for PICmicro MCUs.

12. In-Circuit Serial Programming Guide™ (DS30277)
    This document discusses implementing In-Circuit Serial Programming.

13. PICDEM-1 User’s Guide (DS33015)
    This document explains how to use Microchip’s PICDEM-1 demo board.

    This document explains how to use Microchip’s PICDEM-2 demo board.

15. PICDEM-3 User’s Guide (DS51079)
    This document explains how to use Microchip’s PICDEM-3 demo board.

16. PICDEM-14A User’s Guide (DS51097)
    This document explains how to use Microchip’s PICDEM-14A demo board.

17. PICDEM-17 User’s Guide (DS39024)
    This document explains how to use Microchip’s PICDEM-17 demo board.

18. Third Party Guide (DS00104)
    This document lists Microchip’s third parties, as well as various consultants.

19. Die Support (DS30258)
    This document gives information on using Microchip products in Die form.
### Section 1. Introduction

#### 1.7.2 Third Party Documentation

There are several documents available from third party sources around the world. Microchip does not review these documents for technical accuracy. However, they may be a helpful source for understanding the operation of Microchip PICmicro MCU devices. This is not necessarily a complete list, but are the documents that we were aware of at the time of printing. For more information on how to contact some of these sources, as well as any new sources that we become aware of, please visit the Microchip web site (www.microchip.com).

<table>
<thead>
<tr>
<th>DOCUMENT</th>
<th>LANGUAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>The PIC16C5X Microcontroller: A Practical Approach to Embedded Control</td>
<td>English</td>
</tr>
<tr>
<td>Bill Rigby/ Terry Dalby, Tecksystems Inc.</td>
<td>0-9654740-0-3</td>
</tr>
<tr>
<td>Easy PIC’n</td>
<td>English</td>
</tr>
<tr>
<td>David Benson, Square 1 Electronics</td>
<td>0-9654162-0-8</td>
</tr>
<tr>
<td>A Beginner’s Guide to the Microchip PIC®</td>
<td>English</td>
</tr>
<tr>
<td>Nigel Gardner, Bluebird Electronics</td>
<td>1-899013-01-6</td>
</tr>
<tr>
<td>PIC Microcontroller Operation and Applications</td>
<td>English</td>
</tr>
<tr>
<td>DN de Beer, Cape Technikon</td>
<td></td>
</tr>
<tr>
<td>Digital Systems and Programmable Interface Controllers</td>
<td>English</td>
</tr>
<tr>
<td>WP Verburg, Pretoria Technikon</td>
<td></td>
</tr>
<tr>
<td>Mikroprozessor PIC16C5X</td>
<td>German</td>
</tr>
<tr>
<td>Michael Rose, Hüthig</td>
<td>3-7785-2169-1</td>
</tr>
<tr>
<td>Mikroprozessor PIC17C42</td>
<td>German</td>
</tr>
<tr>
<td>Michael Rose, Hüthig</td>
<td>3-7785-2170-5</td>
</tr>
<tr>
<td>Les Microcontrôleurs PIC et mise en oeuvre</td>
<td>French</td>
</tr>
<tr>
<td>Christian Tavernier, Dunod</td>
<td>2-10-0092947-X</td>
</tr>
<tr>
<td>Microcontrôleurs PIC a structure RISC</td>
<td>French</td>
</tr>
<tr>
<td>C.F. Urbain, Publitronic</td>
<td>2-86661-058-X</td>
</tr>
<tr>
<td>New Possibilities with the Microchip PIC</td>
<td>Russian</td>
</tr>
<tr>
<td>RIGA</td>
<td></td>
</tr>
</tbody>
</table>
## PIC18C Reference Manual

<table>
<thead>
<tr>
<th>DOCUMENT</th>
<th>LANGUAGE</th>
</tr>
</thead>
<tbody>
<tr>
<td>PIC16C5X/71/84 Development and Design, Part 1</td>
<td>Chinese</td>
</tr>
<tr>
<td>United Tech Electronic Co. Ltd</td>
<td></td>
</tr>
<tr>
<td>957-21-0907-7</td>
<td></td>
</tr>
<tr>
<td>PIC16C5X/71/84 Development and Design, Part 2</td>
<td>Chinese</td>
</tr>
<tr>
<td>United Tech Electronic Co. Ltd</td>
<td></td>
</tr>
<tr>
<td>957-21-1152-3</td>
<td></td>
</tr>
<tr>
<td>PIC16C5X/71/84 Development and Design, Part 3</td>
<td>Chinese</td>
</tr>
<tr>
<td>United Tech Electronic Co. Ltd</td>
<td></td>
</tr>
<tr>
<td>957-21-1187-6</td>
<td></td>
</tr>
<tr>
<td>PIC16C5X/71/84 Development and Design, Part 4</td>
<td>Chinese</td>
</tr>
<tr>
<td>United Tech Electronic Co. Ltd</td>
<td></td>
</tr>
<tr>
<td>957-21-1251-1</td>
<td></td>
</tr>
<tr>
<td>PIC16C5X/71/84 Development and Design, Part 5</td>
<td>Chinese</td>
</tr>
<tr>
<td>United Tech Electronic Co. Ltd</td>
<td></td>
</tr>
<tr>
<td>957-21-1257-0</td>
<td></td>
</tr>
<tr>
<td>PIC16C84 MCU Architecture and Software Development</td>
<td>Chinese</td>
</tr>
<tr>
<td>ICC Company</td>
<td></td>
</tr>
<tr>
<td>957-8716-79-6</td>
<td></td>
</tr>
</tbody>
</table>
1.8 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (they may be written for the Base-Line, Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to an introduction to Microchip's PICmicro MCUs are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>A Comparison of Low End 8-bit Microcontrollers</td>
<td>AN520</td>
</tr>
<tr>
<td>PIC16C54A EMI Results</td>
<td>AN577</td>
</tr>
<tr>
<td>Continuous Improvement</td>
<td>AN503</td>
</tr>
<tr>
<td>Improving the Susceptibility of an Application to ESD</td>
<td>AN595</td>
</tr>
<tr>
<td>Plastic Packaging and the Effects of Surface Mount Soldering Techniques</td>
<td>AN598</td>
</tr>
<tr>
<td>Migrating Designs from PIC16C74A/74B to PIC18C442</td>
<td>AN716</td>
</tr>
<tr>
<td>PIC17CXXX to PIC18CXXX Migration</td>
<td>AN726</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
1.9 Revision History

Revision A

This is the initial released revision of Enhanced MCU Introduction.
Section 2. Oscillator

HIGHLIGHTS

This section of the manual contains the following major topics:

2.1 Introduction ................................................................. 2-2
2.2 Control Register ............................................................ 2-3
2.3 Oscillator Configurations .............................................. 2-4
2.4 Crystal Oscillators/Ceramic Resonators ....................... 2-6
2.5 External RC Oscillator .................................................. 2-15
2.6 HS4 (HS oscillator with 4xPLL enabled) ....................... 2-18
2.7 Switching to Low Power Clock Source ....................... 2-19
2.8 Effects of Sleep Mode on the On-Chip Oscillator ........... 2-23
2.9 Effects of Device Reset on the On-Chip Oscillator ......... 2-23
2.10 Design Tips ............................................................... 2-24
2.11 Related Application Notes .......................................... 2-25
2.12 Revision History ....................................................... 2-26
2.1 Introduction

The device system clock is required for the device to execute instructions and for the peripherals to function. Four device system clock periods ($T_{SCLK}$) generate one internal instruction clock cycle ($T_CY$).

The device system clock ($T_{SCLK}$) is derived from an external system clock. This external system clock can be generated in one of eight different oscillator modes. The device configuration bits select the oscillator mode. Device configuration bits are nonvolatile memory locations and the operating mode is determined by the value written during device programming. The oscillator modes are:

- **EC**  External Clock
- **ECIO**  External Clock with I/O pin enabled
- **LP**  Low Frequency (Power) Crystal
- **XT**  Crystal/Resonator
- **HS**  High Speed Crystal/Resonator
- **RC**  External Resistor/Capacitor
- **RCIO**  External Resistor/Capacitor with I/O pin enabled
- **HS4**  High Speed Crystal/Resonator with 4x frequency PLL multiplier enabled

Multiple oscillator circuits can be implemented on an Enhanced Architecture device. There is the default oscillator (OSC1), and additional oscillators may be available, such as the Timer1 oscillator. Software may allow these auxiliary oscillators to be switched in as the device oscillator. The Timer1 oscillator is a low frequency (low power) oscillator that is designed to be operated at 32kHz. Figure 2-1 shows a block diagram of the oscillator options.

The output signal of the Timer1 oscillator circuitry is a low frequency (power) clock source ($T_{T1P}$). The source for the device system clock can be switched from the default clock ($T_{SCLK}$) to the 32kHz-clock low power clock source ($T_{T1P}$) under software control. Switching to the 32kHz low frequency (power) clock source from any of the eight default clock sources may allow power saving.

These oscillator options are made available to allow a single device type the flexibility to fit applications with different oscillator requirements. The RC oscillator option saves system cost, while the LP crystal option saves power. The HS4 option allows frequency of incoming crystal oscillator signal to be multiplied by four for higher internal clock frequency. This is useful for customers who are concerned with EMI due to high frequency crystals. The device configuration bits are used to select these various options. For more details on the device configuration bits, see the “Device Configuration Bits” section.

Figure 2-1: Device Clock Sources

![Device Clock Sources Diagram]

---

**PIC18C Reference Manual**

© 2000 Microchip Technology Inc.
## Section 2. Oscillator

### 2.2 Control Register

Register 2-1 shows the OSCCON register which contains the control bit to allow switching of the system clock between the primary oscillator and the Timer1 oscillator.

**Register 2-1: OSCCON Register**

<table>
<thead>
<tr>
<th>bit 7</th>
<th>bit 6</th>
<th>bit 5</th>
<th>bit 4</th>
<th>bit 3</th>
<th>bit 2</th>
<th>bit 1</th>
<th>bit 0</th>
<th>R/W</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>Unimplemented</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>5CS</td>
</tr>
<tr>
<td>bit 7</td>
<td>bit 0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

- **bit 7** Unimplemented: Read as ‘0’
- **bit 0** SCS: System Clock Switch bit
  - when OSCSEN configuration bit = ‘0’ and T1OSCEN bit is set:
    - 1 = Switch to Timer1 Oscillator/Clock pin
    - 0 = Use primary Oscillator/Clock input pin
  - when OSCSEN and T1OSCEN are in other states:
    - bit is forced clear

**Legend**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- n = Value at POR reset
- ‘1’ = Bit is set
- ‘0’ = Bit is cleared
- x = Bit is unknown

**Note:** The Timer1 oscillator must be enabled to switch the system clock source. The Timer1 oscillator is enabled by setting the T1OSCEN bit in the Timer1 control register (T1CON). If the Timer1 oscillator is not enabled, then any write to the SCS bit will be ignored (SCS bit forced cleared) and the main oscillator will continue to be the system clock source.
2.3 Oscillator Configurations

The oscillator selection is configured at time of device programming. The user can program up to three device configuration bits (FOSC2:FOSC0) to select one of eight modes.

2.3.1 Oscillator Types

PIC18CXXX devices can have up to eight different oscillator modes for the default clock source (TSCclk). These eight modes are:

- **EC** External Clock
- **ECIO** External Clock with IO pin enabled
- **LP** Low Frequency (Power) Crystal
- **XT** Crystal/Resonator
- **HS** High Speed Crystal/Resonator
- **RC** External Resistor/Capacitor
- **RCIO** External Resistor/Capacitor with IO pin enabled
- **HS4** High Speed Crystal/Resonator with 4x frequency PLL multiplier enabled

The main difference between the LP, XT and HS modes is the gain of the internal inverter of the oscillator circuit, which allows the different frequency ranges. Table 2-1 gives information to aid in selecting an oscillator mode. In general, use the oscillator option with the lowest possible gain that still meets specifications. This will result in lower dynamic currents (IDD). The frequency range of each oscillator mode is the recommended frequency cutoff, but the selection of a different gain mode is acceptable as long as thorough validation is performed (voltage, temperature, component variations (resistor, capacitor, and internal microcontroller oscillator circuitry).

Switching the system clock source to the alternate clock source is controlled by the application software. The user can switch from any of the eight default clock sources. This is done by setting the SCS (System Clock Switch) bit in the OSCCON register. The requirements for switching to the alternate clock source are:

- Timer1 clock oscillator must be enabled (T1OSCEN is set ‘1’).
- The OSCEN configuration bit must be cleared (‘0’).
## Section 2. Oscillator

### Table 2-1: Selecting the Oscillator Mode for Devices

<table>
<thead>
<tr>
<th>FOSC2:FOSC0 Configuration Bits</th>
<th>OSC Mode</th>
<th>OSC Feedback Inverter Gain</th>
<th>OSC2/CLKO Function</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>1 1 1</td>
<td>RCIO</td>
<td>Zero Gain. Device turned off to save current.</td>
<td>I/O</td>
<td>Least expensive solution for device oscillation (only an external resistor and capacitor is required). Most variation in time-base. Device’s default mode. OSC2/CLKO is configured as general purpose I/O pin. This pin is multiplexed with one of the device’s PORT pins.</td>
</tr>
<tr>
<td>1 1 0</td>
<td>HS4</td>
<td>High Gain — Highest frequency application. This works with the HS oscillator circuit mode and phase lock loop. This mode consumes the most current. The internal phase lock loop circuit multiplies the external oscillator frequency by 4.</td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 0 1</td>
<td>ECIO</td>
<td>Zero Gain. Device turned off to save current.</td>
<td>I/O</td>
<td>External clock mode with OSC2/CLKO configured as general purpose I/O pin. This pin is multiplexed with one of the device’s PORT pins. OSC1/CLKI is hi-impedance and can be driven by CMOS drivers.</td>
</tr>
<tr>
<td>1 0 0</td>
<td>EC</td>
<td>Zero Gain. Device turned off to save current.</td>
<td>Clock out with oscillator frequency divided by 4.</td>
<td>External clock mode with OSC2/CLKO configured with oscillator frequency divided by 4. OSC1/CLKI is hi-impedance and can be driven by CMOS drivers.</td>
</tr>
<tr>
<td>0 1 1</td>
<td>RC</td>
<td>Zero Gain. Device turned off to save current.</td>
<td>Clock out with oscillator frequency divided by 4.</td>
<td>Inexpensive solution for device oscillation. Most variation in time-base. CLKOUT is enabled on OSC2/CLKO with oscillator frequency divided by 4.</td>
</tr>
<tr>
<td>0 1 0</td>
<td>HS</td>
<td>High Gain — High frequency application. Oscillator circuit’s mode consumes the most current of the three crystal modes.</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 0 1</td>
<td>XT</td>
<td>Medium Gain — Standard crystal/resonator frequency.</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 0 0</td>
<td>LP</td>
<td>Low Gain — Low power/frequency applications. Oscillator circuit’s mode consumes the least current of the three crystal modes.</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
2.4 Crystal Oscillators/Ceramic Resonators

In XT, LP, HS and HS4 modes, a crystal or ceramic resonator is connected to the OSC1 and OSC2 pins to establish oscillation (Figure 2-2). The PIC18CXXX oscillator design requires the use of a parallel cut crystal. Using a series cut crystal may give a frequency out of the crystal manufacturer’s specifications. When in EC and ECIO mode, the device can have an external clock source drive the OSC1 pin (Figure 2-3).

See Table 3-1 in the “Reset” section for time-out delays associated with crystal oscillators.

---

**Figure 2-2: Crystal or Ceramic Resonator Operation (HS4, HS, XT or LP Oscillator Mode)**

![Crystal or Ceramic Resonator Operation Diagram]

**Note 1:** A series resistor, $R_s$, may be required for AT strip cut crystals.

**Note 2:** The internal feedback resistor, $R_f$, is typically in the range of 2 to 10 MΩ.

**Note 3:** See Table 2-2 and 2-3 for example values of $C1$ and $C2$.

---

**Figure 2-3: External Clock Input Operation (EC or ECIO Oscillator Modes)**

![External Clock Input Operation Diagram]
Section 2. Oscillator

2.4.1 Oscillator/Resonator Start-up

As the device voltage increases from Vss, the oscillator will start its oscillations. The time required for the oscillator to start oscillating depends on many factors. These include:

- Crystal/resonator frequency
- Capacitor values used (C1 and C2 in Figure2-2)
- Device Vdd rise time
- System temperature
- Series resistor value and type if used (Rs in Figure2-2)
- Oscillator mode selection of device (selects the gain of the internal oscillator inverter)
- Crystal quality
- Oscillator circuit layout
- System noise

Figure2-4 graphs an example oscillator/resonator start-up. The peak-to-peak voltage of the oscillator waveform can be quite low (less than 50% of device Vdd), when the waveform is centered at Vdd/2 (refer to parameters D033 and D043 in the “Electrical Specifications” section).

Figure 2-4: Example Oscillator/Resonator Start-up Characteristics
2.4.2 Component Selection

Figure 2-2 is a diagram of the device’s crystal or ceramic resonator circuitry. The resistance for the feedback resistor, $R_F$, is typically within the 2 to 10 MΩ range. This varies with device voltage, temperature and process variations. A series resistor, $R_s$, may be required if an AT strip cut crystal is used. Be sure to include the device’s operating voltage and the device’s manufacturing process when determining resistor requirements. As you can see in Figure 2-2, the connection to the device’s internal logic is device dependent. See the applicable data sheet for device specifics.

The typical values of capacitors ($C_1$, $C_2$) are given in Table 2-2 and Table 2-3. Each device’s data sheet will give the specific values that we test to at Microchip.

Table 2-2: Example Capacitor Selection for Ceramic Resonators

<table>
<thead>
<tr>
<th>Mode</th>
<th>Frequency</th>
<th>Ranges tested:</th>
<th>C1 (1)</th>
<th>C2 (1)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>XT 455 kHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td></td>
<td>2.0 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td></td>
<td>4.0 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td></td>
<td>HS 8.0 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td></td>
<td>16.0 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
</tbody>
</table>

Resonators used:

<table>
<thead>
<tr>
<th>Frequency</th>
<th>Manufacturer</th>
<th>Tolerance</th>
</tr>
</thead>
<tbody>
<tr>
<td>455 kHz</td>
<td>Panasonic EFO-A45SK04B</td>
<td>±0.3%</td>
</tr>
<tr>
<td>2.0 MHz</td>
<td>Murata Erie CSA2.00MG</td>
<td>±0.5%</td>
</tr>
<tr>
<td>4.0 MHz</td>
<td>Murata Erie CSA4.00MG</td>
<td>±0.5%</td>
</tr>
<tr>
<td>8.0 MHz</td>
<td>Murata Erie CSA8.00MT</td>
<td>±0.5%</td>
</tr>
<tr>
<td>16.0 MHz</td>
<td>Murata Erie CSA16.00MX</td>
<td>±0.5%</td>
</tr>
</tbody>
</table>

Note 1: Recommended values of $C_1$ and $C_2$ are identical to the ranges tested above. Higher capacitance increases the stability of the oscillator but also increases the start-up time. These values are for design guidance only. Since each resonator has its own characteristics, the user should consult the resonator manufacturer for appropriate values of external components or verify oscillator performance.

Note 2: All resonators tested required external capacitors.
## Section 2. Oscillator

### Table 2-3: Example Capacitor Selection for Crystal Oscillator

<table>
<thead>
<tr>
<th>Mode</th>
<th>Frequency</th>
<th>$C_1^{(1)}$</th>
<th>$C_2^{(1)}$</th>
</tr>
</thead>
<tbody>
<tr>
<td>LP</td>
<td>32 kHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td>200 kHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td>XT</td>
<td>200 kHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td>1 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td>4 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td>HS</td>
<td>4.0 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td>8 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td>20 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
<tr>
<td></td>
<td>25 MHz</td>
<td>TBD</td>
<td>TBD</td>
</tr>
</tbody>
</table>

**Crystals used:**

<table>
<thead>
<tr>
<th>Frequency</th>
<th>Manufacturer</th>
<th>Tolerance</th>
</tr>
</thead>
<tbody>
<tr>
<td>32.0 kHz</td>
<td>Epson C-001R32.768K-A</td>
<td>± 20 PPM</td>
</tr>
<tr>
<td>200 kHz</td>
<td>STD XTL 200.000 kHz</td>
<td>± 20 PPM</td>
</tr>
<tr>
<td>1.0 MHz</td>
<td>ECS ECS-10-13-1</td>
<td>± 50 PPM</td>
</tr>
<tr>
<td>4.0 MHz</td>
<td>ECS ECS-40-20-1</td>
<td>± 50 PPM</td>
</tr>
<tr>
<td>8.0 MHz</td>
<td>EPSON CA-301 8.000 M-C</td>
<td>± 30 PPM</td>
</tr>
<tr>
<td>20.0 MHz</td>
<td>EPSON CA-301 20.000 M-C</td>
<td>± 30 PPM</td>
</tr>
</tbody>
</table>

**Note 1:** Higher capacitance increases the stability of the oscillator, but also increases the start-up time. These values are for design guidance only. A series resistor, $R_s$, may be required in HS mode, as well as XT mode, to avoid overdriving crystals with low drive level specification. Since each crystal has its own characteristics, the user should consult the crystal manufacturer for appropriate values of external components or verify oscillator performance.
2.4.3 Tuning the Oscillator Circuit

Since Microchip devices have wide operating ranges (frequency, voltage, and temperature; depending on the part and version ordered) and external components (crystals, capacitors, ...) of varying quality and manufacture, validation of operation needs to be performed to ensure that the component selection will comply with the requirements of the application.

There are many factors that go into the selection and arrangement of these external components. These factors include:

- amplifier gain
- desired frequency
- resonant frequency(s) of the crystal
- temperature of operation
- supply voltage range
- start-up time
- stability
- crystal life
- power consumption
- simplification of the circuit
- use of standard components
- combination which results in fewest components
2.4.3.1 Determining Best Values for Crystals, Clock Mode, C1, C2, and Rs

The best method for selecting components is to apply a little knowledge and a lot of trial, measurement, and testing.

**Crystals** are usually selected by their parallel resonant frequency only, however other parameters may be important to your design, such as temperature or frequency tolerance. Application Note AN588 is an excellent reference if you would like to know more about crystal operation and their ordering information.

The PICmicro’s internal oscillator circuit is a parallel oscillator circuit, which requires that a parallel resonant crystal be selected. The load capacitance is usually specified in the 20 pF to 32 pF range. The crystal will oscillate closest to the desired frequency with capacitance in this range. It may be necessary to sometimes alter these values a bit, as described later, in order to achieve other benefits.

**Clock mode** is primarily chosen by using the Fosc parameter specification (parameter 1A) in the device data sheet, based on frequency. Clock modes (except RC and EC) are simply gain selections; lower gain for lower frequencies, higher gain for higher frequencies. It is possible to select a higher or lower gain, if desired, based on the specific needs of the oscillator circuit.

C1 and C2 should also be initially selected based on the load capacitance as suggested by the crystal manufacturer and the tables supplied in the device data sheet. The values given in the device data sheet can only be used as a starting point, since the crystal manufacturer, supply voltage, and other factors already mentioned may cause your circuit to differ from the one used in the factory characterization process.

Ideally, the capacitance is chosen so that it will oscillate at the highest temperature and lowest VDD that the circuit will be expected to perform under. High temperature and low VDD both have a limiting effect on the loop gain, such that if the circuit functions at these extremes, the designer can be more assured of proper operation at other temperatures and supply voltage combinations. The output sine wave should not be clipped in the highest gain environment (highest VDD and lowest temperature) and the sine output amplitude should be great enough in the lowest gain environment (lowest VDD and highest temperature) to cover the logic input requirements of the clock as listed in the device data sheet.

A method for improving start-up is to use a value of C2 greater than C1. This causes a greater phase shift across the crystal at power-up, which speeds oscillator start-up.

Besides loading the crystal for proper frequency response, these capacitors can have the effect of lowering loop gain if their value is increased. C2 can be selected to affect the overall gain of the circuit. A higher C2 can lower the gain if the crystal is being overdriven (see also discussion on Rs). Capacitance values that are too high can store and dump too much current through the crystal, so C1 and C2 should not become excessively large. Unfortunately, measuring the wattage through a crystal is tricky business, but if you do not stray too far from the suggested values, you should not have to be concerned with this.

A series resistor, Rs, is added to the circuit if, after all other external components are selected to satisfaction, the crystal is still being overdriven. This can be determined by looking at the OSC2 pin, which is the driven pin, with an oscilloscope. Connecting the probe to the OSC1 pin will load the pin too much and negatively affect performance. Remember that a scope probe adds its own capacitance to the circuit, so this may have to be accounted for in your design. (i.e. if the circuit worked best with a C2 of 20 pF and scope probe was 10 pF, a 30 pF capacitor may actually be called for). The output signal should not be clipping or squashed. Overdriving the crystal can also lead to the circuit jumping to a higher harmonic level or even crystal damage.
The OSC2 signal should be a clean sine wave that easily spans the input minimum and maximum of the clock input pin (4V to 5V peak to peak for a 5V VDD is usually good). An easy way to set this is to again test the circuit at the minimum temperature and maximum VDD that the design will be expected to perform in, then look at the output. This should be the maximum amplitude of the clock output. If there is clipping or the sine wave is squashing near VDD and VSS at the top and bottom, increasing load capacitors will risk too much current through the crystal or push the value too far from the manufacturer’s load specification. Add a trimpot between the output pin and C2, and adjust it until the sine wave is clean. Keeping it fairly close to maximum amplitude at the low temperature and high VDD combination will assure this is the maximum amplitude the crystal will see and prevent overdriving. A series resistor, Rs, of the closest standard value can now be inserted in place of the trimpot. If Rs is too high, perhaps more than 20k ohms, the input will be too isolated from the output, making the clock more susceptible to noise. If you find a value this high is needed to prevent overdriving the crystal, try increasing C2 to compensate. Try to get a combination where Rs is around 10k or less and load capacitance is not too far from the 20 pF or 32 pF manufacturer specification.

2.4.3.1 Start-up

The most difficult time for the oscillator to start-up is when waking up from sleep. This is because the load capacitors have both partially charged to some quiescent value, and phase differential at wake-up is minimal. Thus, more time is required to achieve stable oscillation. Remember also that low voltage, high temperatures and the lower frequency clock modes also impose limitations on loop gain, which in turn affects start-up. Each of the following factors makes the start-up time worse:

- a low frequency design (with its low gain clock mode)
- a quiet environment (such as a battery operated device)
- operating in a shielded box (away from the noisy RF area)
- low voltage
- high temperature
- waking up from sleep.

Noise actually helps a design for oscillator start-up, since it helps “kick start” the oscillator.
Section 2. Oscillator

2.4.4 External Clock Input

Two of the oscillator modes use an external clock. These modes are EC and ECIO oscillator modes.

In the EC mode (Figure 2-5), the OSC1 pin can be driven by CMOS drivers. In this mode, the OSC1/CLKI pin is hi-impedance and the OSC2/CLKO pin is the CLKO output (Fosc/4). The output is at a frequency of the selected oscillator divided by 4. This output clock is useful for testing or synchronization purposes. If the power-up timer is disabled, then there is no time-out after a POR, or else there will be a power-up timer. There is always a power-up time after a brown-out reset.

The feedback device between OSC1 and OSC2 is turned off to save current. There is no oscillator start-up time required after wake-up from sleep mode. If the power-up timer is disabled, then there is no time-out after a POR, or else (power-up timer enabled) there will be a power-up timer delay after POR. There is always a power-up timer after a brown-out reset.

Figure 2-5: External Clock Input Operation (EC Oscillator Configuration)

In the ECIO mode (Figure 2-6), the OSC1 pin can be driven by CMOS drivers. In this mode, the OSC1/CLKI pin is hi-impedance and the OSC2/CLKO is now multiplexed with a general purpose I/O pin.

The feedback device between OSC1 and OSC2 is turned off to save current. There is no oscillator start-up time required after wake-up from sleep mode. If the power-up timer is disabled, then there is no time-out after a POR, or else (power-up timer enabled) there will be a power-up timer delay after POR. There is always a power-up timer after a brown-out reset.

Figure 2-6: External Clock Input Operation (ECIO Oscillator Configuration)
2.4.5 External Crystal Oscillator Circuit for Device Clock

Sometimes more than one device needs to be clocked from a single crystal. Since Microchip does not recommend connecting other logic to the PICmicro’s internal oscillator circuit, an external crystal oscillator circuit is recommended. Each device will then have an external clock source, and the number of devices that can be driven will depend on the buffer drive capability. This circuit is also useful when more than one device needs to operate synchronously to each other.

Either a prepackaged oscillator can be used or a simple oscillator circuit with TTL gates can be built. Prepackaged oscillators provide a wide operating range and better stability. A well-designed crystal oscillator will provide good performance with TTL gates. Two types of crystal oscillator circuits can be used; one with series resonance or one with parallel resonance.

Figure 2-7 shows implementation of an external parallel resonant oscillator circuit. The circuit is designed to use the fundamental frequency of the crystal. The 74AS04 inverter performs the 180-degree phase shift that a parallel oscillator requires. The 4.7 kΩ resistor affects the circuit in three ways:

1. Provides negative feedback.
2. Biases the 74AS04 (#1) into the linear region.
3. Bounds the gain of the amplifier.

The 10 kΩ potentiometer is used to prevent overdriving of the crystal. It dissipates the power of the amplifier and allows the requirements of the crystal to be met.

Figure 2-7: External Parallel Resonant Crystal Oscillator Circuit

Figure 2-8 shows an external series resonant oscillator circuit. This circuit is also designed to use the fundamental frequency of the crystal. The inverter performs a 180-degree phase shift in a series resonant oscillator circuit. The 330 kΩ resistors provide the negative feedback to bias the inverters in their linear region.

Figure 2-8: External Series Resonant Crystal Oscillator Circuit

When the device is clocked from an external clock source (as in Figure 2-7 or Figure 2-8) then the microcontroller’s oscillator should be configured for EC or ECIO mode (Figure 2-3).
Section 2. Oscillator

2.5 External RC Oscillator

For timing insensitive applications, the RC and RCIO device options offer additional cost savings. The RC oscillator frequency is a function of the:

- Supply voltage
- External resistor ($R_{EXT}$) values
- External capacitor ($C_{EXT}$) values
- Operating temperature

In addition to this, the oscillator frequency will vary from unit to unit due to normal process parameter variation. Furthermore, the difference in lead frame capacitance between package types will also affect the oscillation frequency, especially for low $C_{EXT}$ values. The user also needs to take into account variation due to tolerance of external $R_{EXT}$ and $C_{EXT}$ components used. Figure 2-9 shows how the RC combination is connected. For $R_{EXT}$ values below 2.2 kΩ, oscillator operation may become unstable, or stop completely. For very high $R_{EXT}$ values (e.g. 1 MΩ), the oscillator becomes sensitive to noise, humidity and leakage. Thus, we recommend keeping $R_{EXT}$ between 3 kΩ and 100 kΩ.

Figure 2-9: RC Oscillator Mode

Although the oscillator will operate with no external capacitor ($C_{EXT} = 0$ pF), we recommend using values above 20 pF for noise and stability reasons. With no or a small external capacitance, the oscillation frequency can vary dramatically due to changes in external capacitances, such as PCB trace capacitance and package lead frame capacitance.

See characterization data for RC frequency variation from part to part due to normal process variation. The variation is larger for larger resistance (since leakage current variation will affect RC frequency more for large R) and for smaller capacitance (since variation of input capacitance will affect RC frequency more).

See characterization data for the variation of oscillator frequency due to $V_{DD}$ for given $R_{EXT}/C_{EXT}$ values, as well as frequency variation due to operating temperature for given $R_{EXT}$, $C_{EXT}$ and $V_{DD}$ values.

The oscillator frequency, divided by 4, is available on the OSC2/CLKO pin, and can be used for test purposes or to synchronize other logic (see Figure 4-3: "Clock/Instruction Cycle" in the "Architecture" section, for waveform).
2.5.1 RC Oscillator with I/O Enabled

The RCIO oscillator mode functions in the exact same manner as the RC oscillator mode. The only difference is that OSC2 pin does not output oscillator frequency divided by 4, but in this mode is configured as an I/O pin.

As in the RC mode, the user needs to take into account any variation of the clock frequency due to tolerance of external REXT and CEXT components used, process variation, voltage, and temperature. Figure 2-10 shows how the RC with the I/O pin combination is connected.

Figure 2-10: RCIO Oscillator Mode

![Diagram of RCIO Oscillator Mode]
2.5.2 RC Start-up

As the device voltage increases, the RC will start its oscillations immediately after the pin voltage levels meet the input threshold specifications (parameters $D032$ and $D042$ in the "Electrical Specifications" section). The time required for the RC to start oscillating depends on many factors. These include:

- Resistor value used
- Capacitor value used
- Device $V_{DD}$ rise time
- System temperature

There is no oscillator start-up time (TOST) regardless of the source of reset or when sleep is terminated. If the power-up timer is disabled, then there is no time-out after a POR, or else (power-up timer enabled) there will be a power-up timer delay after POR. There is always a power-up time after a brown-out reset.
2.6 HS4 (HS oscillator with 4xPLL enabled)

A Phase Locked Loop (PLL) circuit is provided as a programmable option for users that want to multiply the frequency of the incoming crystal oscillator signal by 4. For an input clock frequency of 10 MHz, the internal clock frequency will be multiplied to 40 MHz. This is useful for customers who are concerned with EMI due to high frequency crystals.

The PLL can only be enabled when the oscillator configuration bits are programmed for HS4 mode (FOSC2:FOSC0 = ‘110’). If they are programmed for any other mode, the PLL is not enabled and the system clock will come directly from OSC1. The oscillator mode is specified during device programming.

The PLL is divided into four basic parts (see Figure 2-11):

- Phase comparator
- Loop filter
- VCO (Voltage Controlled Oscillator)
- Feedback divider

When in HS4 mode, the incoming clock is sampled by the phase comparator and is compared to PLL output clock divided by four. If the two are not in phase, the phase comparator drives an input to the loop filter to "pump" the voltage to the VCO, either up or down, depending upon whether the input clock was leading or lagging the output clock. This process continues until the incoming clock on OSC1 and the divide by 4 output clock of the VCO are in phase. The output clock is now "locked" in phase with the incoming clock, and its frequency is four times greater.

A PLL lock timer is used to ensure that the PLL has locked before device execution starts. The PLL lock timer has a time-out that is called TPLL. This delay is shown in Figure 2-14.

Figure 2-11: PLL Block Diagram
Section 2. Oscillator

2.7 Switching to Low Power Clock Source

This feature allows the clock source to switch from the default clock source that is selected by the FOSC2:FOSC0 bits to the Timer1 oscillator clock source. The availability of this feature is device dependent.

2.7.1 Switching Oscillator Mode Option

This feature is enabled by clearing the Oscillator System Clock Switch Enable (OSCSEN) configuration bit. This provides the ability to switch to a low power execution mode if the alternate clock source (such as Timer1) is configured in oscillator mode with a low frequency (32 kHz, for example) crystal.

The enabling of the low power clock source is determined by the state of the SCS control bit in the Oscillator control register (OSCCON). (Register 2-1)

2.7.1.2 System Clock Switch Bit

The system clock switch bit, SCS (OSCCON) controls the switching of the oscillator source. It can be configured for either the Timer1 Oscillator clock source, or the default clock source (selected by the Fosc2:Fosc0 bits). When the SCS bit is set, it enables the Timer1 Oscillator clock source as the system clock. When the SCS bit is cleared, the system clock comes from the clock source specified by the Fosc2:Fosc0 bits. The SCS bit is cleared on all forms of reset.

Note: The Timer1 oscillator must be enabled in order to switch the system clock source. The Timer1 oscillator is enabled by setting the T1OSCEN bit in the Timer1 Control Register (T1CON). If the Timer1 oscillator is not enabled, then any write to the SCS bit will be ignored, and the SCS bit will remain in the default state with the clock source coming from OSC1 or the PLL output.
2.7.2 Oscillator Transitions

Switching from the default clock to the Timer1 Oscillator clock source is controlled as shown in the flow diagram (Figure 2-16). This ensures a clean transition when switching oscillator clocks.

Circuitry is used to prevent "glitches" due to transitions when switching from the default clock source to the low power clock source and vice versa. Essentially, the circuitry waits for eight rising edges of the clock input to which the processor is switching. This ensures that the clock output pulse width will not be less than the shortest pulse width of the two clock sources. No additional delays are required when switching from the default clock source to the low power clock source.

Figure 2-12 through Figure 2-15 show different transition waveforms when switching between the oscillators.

**Figure 2-12: Transition From OSC1 to Timer1 Oscillator Waveform**

- **Note 1:** Delay on internal system clock is eight oscillator cycles for synchronization.
- **Note 2:** The T1OSCEN bit is set.
- **Note 3:** The OSCSEN configuration bit is cleared.

**Figure 2-13: Transition Between Timer1 and OSC1 Waveform (HS, XT, LP)**

- **Note 1:** TOST = 1024TOSC (drawing not to scale).
Section 2. Oscillator

Additional delays may occur before switching from the low power clock source back to the main oscillator. The sequence of events that take place will depend upon the main oscillator setting in the configuration register (the mode of the main oscillator).

If the main oscillator is configured as a RC oscillator (RC, RCIO) or External Clock (EC, ECIO), then there is no oscillator start-up time. The transition from a low power clock to the main oscillator occurs after 8 clock cycles are counted on OSC1.

If the main oscillator is configured as a crystal (HS4, HS, XT or LP), then the transition will take place after an oscillator start-up time (\(T_{OST}\)).

If the main oscillator is configured as a crystal with PLL (HS4) enabled, then the transition will take place after an oscillator start-up time (\(T_{OST}\)) plus an additional PLL time-out, \(T_{PLL}\) (see "Electrical Specifications" section, parameter 32). This is necessary because the crystal oscillator had been powered down until the time of the transition. In order to provide the system with a reliable clock when the change-over has occurred, the clock will not be released to the change-over circuit until the oscillator start-up time has expired. The additional \(T_{PLL}\) time is required after oscillator start-up to allow the phase lock loop ample time to lock to the incoming oscillator frequency from OSC1.
A flow diagram for switching between a low power clock and the default oscillator clock is shown in Figure 2-16.

Figure 2-16: Switching Oscillator Flow Diagram

- **Start**
  - DSCSEN = 0?
    - Yes: No switch to low power clock
    - No: Has SCS changed state?
  - Has SCS changed state?
    - Yes: Begin switch to low power clock
    - No: SCS = 0?
      - Yes: Begin switch to high speed clock
      - No: Transition on Newclk?

- **Transition on Newclk?**
  - Yes: Newclk = T1OSC input
  - No: No switch to low power clock, Set SCS = 0

- **No switch to low power clock, SCS = 0**
  - Transition on Newclk?
    - Yes: Newclk = T1OSC input
    - No: N = N + 1

- **N = 0, hold CPU clock in Q1 state**
  - Transition on Newclk?
    - Yes: Newclk = T1OSC input
    - No: N = N + 1

- **N = 8?**
  - Yes: Sysclk = Newclk, Release Q clocks
  - No: T1OSCEN = 1?
    - Yes: No switch to low power clock, Set SCS = 0
    - No: Fosc2:Fosc0 = XT, LP, or HS?
      - Yes: Fosc2:Fosc0 = XT, HS, LP
      - No: Fosc2:Fosc0 = HS4

- **Fosc2:Fosc0 = XT, HS, LP**
  - Wait T PLL for PLL to lock

- **Fosc2:Fosc0 = HS4**
  - Start OST, wait 1024 oscillations on OSC1
  - Newclk = EC or RC

- **End**
2.8 Effects of Sleep Mode on the On-Chip Oscillator

When the device executes a SLEEP instruction, the on-chip clocks and oscillator are turned off and the device is held at the beginning of an instruction cycle (Q1 state). With the oscillator off, the OSC1 and OSC2 signals will stop oscillating. Since all the transistor switching currents have been removed, SLEEP mode achieves the lowest current consumption of the device (only leakage currents). Enabling any on-chip feature that will operate during SLEEP will increase the current consumed. The user can wake from SLEEP through external reset, Watchdog Timer Reset or through an interrupt.

See Table 3-1 in the “Reset” section for time-outs due to SLEEP and MCLR reset.

Table 2-4: OSC1 and OSC2 Pin States in Sleep Mode

<table>
<thead>
<tr>
<th>OSC Mode</th>
<th>OSC1 Pin State</th>
<th>OSC2 Pin State</th>
</tr>
</thead>
<tbody>
<tr>
<td>RC</td>
<td>Floating, external resistor should pull high</td>
<td>At logic low</td>
</tr>
<tr>
<td>RCIO</td>
<td>Floating, external resistor should pull high</td>
<td>Configured as I/O pin</td>
</tr>
<tr>
<td>ECIO</td>
<td>Floating</td>
<td>Configured as I/O pin</td>
</tr>
<tr>
<td>EC</td>
<td>Floating</td>
<td>At logic low</td>
</tr>
<tr>
<td>LP, XT and HS</td>
<td>Feedback inverter disabled at quiescent voltage level</td>
<td>Feedback inverter disabled at quiescent voltage level</td>
</tr>
<tr>
<td>HS4</td>
<td>Feedback inverter disabled at quiescent voltage level</td>
<td>Feedback inverter disabled at quiescent voltage level</td>
</tr>
</tbody>
</table>

2.9 Effects of Device Reset on the On-Chip Oscillator

Device resets have no effect on the on-chip crystal oscillator circuitry. The oscillator will continue to operate as it does under normal execution. While in RESET, the device logic is held at the Q1 state so that when the device exits RESET, it is at the beginning of an instruction cycle. The OSC2 pin, when used as the external clockout (RC, EC mode), will be held low during RESET, and as soon as the MCLR pin is at V_H (input high voltage), the RC will start to oscillate.

See Table 3-1 in the “Reset” section for time-outs due to SLEEP and MCLR reset.

2.9.1 Power-up Delays

Power-up delays are controlled by two timers, so that no external reset circuitry is required for most applications. The delays ensure that the device is kept in RESET until the device power supply and clock are stable. For additional information on RESET operation, see the “Reset” section.

The Power-up Timer (PWRT) provides a fixed 72 ms delay on power-up due to POR or BOR, and keeps the part in RESET until the device power supply is stable. When a crystal is used (LP, XT, HS), the Oscillator Start-Up Timer (OST) keeps the chip in RESET until the PWRT timer delay has expired, allowing the crystal oscillator to stabilize on power up. The PWRTEN bit must be cleared for this time-out to occur.

When the PLL is enabled (HS4 oscillator mode), the Power-up Timer (PWRT) is used to keep the device in RESET for an extra nominal delay (T_{PLL}) above crystal mode. This delay ensures that the PLL is locked to the crystal frequency.

For additional information on RESET operation, see the “Reset” section.
2.10 Design Tips

Question 1: When looking at the OSC2 pin after power-up with an oscilloscope, there is no clock. What can cause this?

Answer 1:
1. Executing a SLEEP instruction with no source for wake-up (such as, WDT, MCLR, or an Interrupt). Verify that the code does not put the device to SLEEP without providing for wake-up. If it is possible, try waking it up with a low pulse on MCLR. Powering up with MCLR held low will also give the crystal oscillator more time to start-up, but the Program Counter will not advance until the MCLR pin is high.
2. The wrong clock mode is selected for the desired frequency. For a blank device, the default oscillator is RCIO. Most parts come with the clock selected in the default RC mode, which will not start oscillation with a crystal or resonator. Verify that the clock mode has been programmed correctly.
3. The proper power-up sequence has not been followed. If a CMOS part is powered through an I/O pin prior to power-up, bad things can happen (latch up, improper start-up, etc.) It is also possible for brown-out conditions, noisy power lines at start-up, and slow VDD rise times to cause problems. Try powering up the device with nothing connected to the I/O, and power-up with a known, good, fast-rise, power supply. Refer to the power-up information in the device data sheet for considerations on brown-out and power-up sequences.
4. The $C_1$ and $C_2$ capacitors attached to the crystal have not been connected properly or are not the correct values. Make sure all connections are correct. The device data sheet values for these components will usually get the oscillator running; however, they just might not be the optimal values for your design.

Question 2: The PICmicro device starts, but runs at a frequency much higher than the resonant frequency of the crystal.

Answer 2:
The gain is too high for this oscillator circuit. Refer to subsection 2.4 “Crystal Oscillators/Ceramic Resonators” to aid in the selection of $C_2$ (may need to be higher) $R_s$ (may be needed) and clock mode (wrong mode may be selected). This is especially possible for low frequency crystals, like the common 32.768 kHz.

Question 3: The design runs fine, but the frequency is slightly off. What can be done to adjust this?

Answer 3:
Changing the value of $C_1$ has some effect on the oscillator frequency. If a SERIES resonant crystal is used, it will resonate at a different frequency than a PARALLEL resonant crystal of the same frequency call-out. Ensure that you are using a PARALLEL resonant crystal.

Question 4: The board works fine, then suddenly quits or loses time.

Answer 4:
Other than the obvious software checks that should be done to investigate losing time, it is possible that the amplitude of the oscillator output is not high enough to reliably trigger the oscillator input. Look at the $C_1$ and $C_2$ values and ensure that the the device configuration bits are correct for the desired oscillator mode.

Question 5: If I put an oscilloscope probe on an oscillator pin, I don’t see what I expect. Why?

Answer 5:
Remember that an oscilloscope probe has capacitance. Connecting the probe to the oscillator circuitry will modify the oscillator characteristics. Consider using a low capacitance (active) probe.
# Section 2. Oscillator

## 2.11 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is they may be written for the Base-Line, Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the oscillator are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>PICmicro Microcontrollers Oscillator Design Guide</td>
<td>AN588</td>
</tr>
<tr>
<td>Low Power Design using PICmicro Microcontrollers</td>
<td>AN606</td>
</tr>
</tbody>
</table>

**Note:** Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
2.12 Revision History

Revision A

This is the initial released revision of the Enhanced MCU oscillators description.
Section 3. Reset

HIGHLIGHTS

This section of the manual contains the following major topics:

3.1 Introduction ................................................................. 3-2
3.2 Resets and Delay Timers ............................................... 3-4
3.3 Registers and Status Bit Values ................................... 3-14
3.4 Design Tips ................................................................. 3-20
3.5 Related Application Notes .......................................... 3-21
3.6 Revision History .......................................................... 3-22
3.1 Introduction

The reset logic is used to place the device into a known state. The source of the reset can be determined by reading the device status bits. The reset logic is designed with features that reduce system cost and increase system reliability.

Devices differentiate between various kinds of reset:

- **a)** Power-on Reset (POR)
- **b)** MCLR Reset during normal operation
- **c)** MCLR Reset during SLEEP
- **d)** WDT Reset (normal operation)
- **e)** Programmable Brown-out Reset (BOR)
- **f)** **RESET** Instruction
- **g)** Stack Overflow Reset
- **h)** Stack Underflow Reset

Most registers are unaffected by a reset; their status is unknown on POR and unchanged by all other resets. The other registers are forced to a "reset state" on Power-on Reset, MCLR Reset, Brown-out Reset, MCLR Reset during SLEEP and by the **RESET** instruction.

Most registers are not affected by a WDT wake-up, since this is viewed as the resumption of normal operation. Status bits from the RCON register, RI, TO, PD, POR and BOR are set or cleared differently in different reset situations as indicated in Table 3-3. These bits are used in software to determine the nature of the reset. See Table 3-4 for a full description of the reset states of all registers.

A simplified block diagram of the on-chip reset circuit is shown in Figure 3-1. This block diagram is a superset of reset features. To determine the features that are available on a specific device, please refer to the device’s Data Sheet.

**Note:** While the Enhanced MCU is in a reset state, the internal phase clock is held at Q1 (beginning of an instruction cycle).
Section 3. Reset

Figure 3-1: Simplified Block Diagram of On-chip Reset Circuit

Note 1: This is a separate oscillator from the RC oscillator of the CLkin pin.
2: See Table 3-1 for time-out situations.
3.2 Resets and Delay Timers

The device has many sources for a device reset. Depending on the source of the reset, different delays may be initiated. These reset sources and the delays are discussed in the following subsections.

3.2.1 Power-on Reset (POR)

A Power-on Reset pulse is generated on-chip when VDD rise is detected. To take advantage of the POR, just tie the MCLR pin directly (or through a resistor) to VDD as shown in Figure 3-2. This will eliminate external RC components usually needed to create a Power-on Reset delay. A minimum rise time for VDD is required. See parameter D003 and parameter D004 in the “Electrical Specifications” section for details.

Figure 3-2: Using On-Chip POR

![Figure 3-2: Using On-Chip POR](image)

Note 1: The resistor is optional.

When the device exits the reset condition (begins normal operation), the device operating parameters (voltage, frequency, temperature, etc.) must be within their operating ranges, otherwise the device will not function correctly. Ensure the delay is long enough to get all operating parameters within specification.

Figure 3-3 shows a possible POR circuit for a slow power supply ramp up. The external Power-on Reset circuit is only required if the device would exit reset before the device VDD is in the valid operating range. The diode, D, helps discharge the capacitor quickly when VDD powers down.

Figure 3-3: External Power-on Reset Circuit (For Slow VDD Power-up)

![Figure 3-3: External Power-on Reset Circuit (For Slow VDD Power-up)](image)

Note 1: R < 40 kΩ is recommended to ensure that the voltage drop across R does not violate the device's electrical specification.

2: R1 = 100Ω to 1 kΩ will limit any current flowing into MCLR from external capacitor C in the event of MCLR/VPP pin breakdown due to Electrostatic Discharge (ESD) or Electrical Overstress (EOS).
Section 3. Reset

3.2.2 Power-up Timer (PWRT)

The Power-up Timer provides a delay on Power-on Reset (POR) or Brown-out Reset (BOR). See parameter D033 in the “Electrical Specifications” section. The Power-up Timer operates on a dedicated internal RC oscillator. The device is kept in reset as long as the PWRT is active. The PWRT delay allows VDD to rise to an acceptable level. A configuration bit (PWRTEN) is provided to enable/disable the Power-up Timer.

**Note:** Some devices require the Power-up Timer to be enabled when the Brown-out Reset circuitry is enabled. Please refer to the device data sheet for requirements.

The power-up time delay will vary from device to device due to VDD, temperature and process variations. See DC parameters for details.

3.2.3 Oscillator Start-up Timer (OST)

The Oscillator Start-Up Timer (OST) provides a 1024 oscillator cycle delay (from OSC1 input) (parameter 32) after the PWRT delay is over. This ensures that the crystal oscillator or resonator has started and is stable.

The OST time-out is invoked only for XT, LP and HS modes, on Power-on Reset, Brown-out Reset, wake-up from SLEEP, or on a transition from Timer1 input clock as the system clock to the oscillator as the system clock by clearing the SCS bit. The oscillator start-up timer is disabled for all resets and wake-ups in RC and EC modes. (See Table 3-1)

The OST counts the oscillator pulses on the OSC1/CLKIN pin. The counter only starts incrementing after the amplitude of the signal reaches the oscillator input thresholds. This delay allows the crystal oscillator or resonator to stabilize before the device exits the OST delay. The length of the time-out is a function of the crystal/resonator frequency.

Figure 3-4 shows the operation of the OST circuit in conjunction with the power-up timer. For low frequency crystals, this start-up time can become quite long. That is because the time it takes the low frequency oscillator to start oscillating is longer than the power-up timer’s delay. The time from when the power-up timer times out to when the oscillator starts to oscillate is a dead time. There is no minimum or maximum time for this dead time (TDEADTIME), and is dependent on the time for the oscillator circuitry to have “good” oscillations.

**Figure 3-4: Oscillator Start-up Time**

![Diagram of Oscillator Start-up Time](image)

TOSC1 = Time for the crystal oscillator to react to an oscillation level detectable by the Oscillator Start-up Timer (OST).

TOST = 1024Tosc.

3.2.3.1 PLL Lock Time-out

When the PLL is enabled, the time-out sequence following a Power-on Reset is different from other oscillator modes. A portion of the Power-up Timer is used to provide a fixed time-out that is sufficient for the PLL to lock to the main oscillator frequency. This PLL lock time-out TpkL (2ms nominal, Parameter 7 in the “Electrical Specifications” section) follows the Oscillator Start-up Time-out (OST).
3.2.4 Power-up Sequence

On power-up, the time-out sequence is as follows: First the internal POR is detected, then, if enabled, the PWRT time-out is invoked. After the PWRT time-out is over, the OST is activated. The total time-out will vary based on oscillator configuration and PWRTEN bit status. For example, in RC mode with the PWRTEN bit set (PWRT disabled), there will be no time-out at all. Figure 3-5, Figure 3-6 and Figure 3-7 depict time-out sequences.

Since the time-outs occur from the internal POR pulse, if MCLR is kept low long enough, the time-outs will expire. Bringing MCLR high will begin execution immediately (Figure 3-7). This is useful for testing purposes or to synchronize more than one device operating in parallel.

If the device voltage is not within the electrical specifications by the end of a time-out, the MCLR/VPP pin must be held low until the voltage is within the device specification. The use of an external RC delay is sufficient for many of these applications.

On wake-up from sleep, the OST is activated for various oscillator configurations. When the PLL is activated in HS mode, an additional delay called TPLL (2 ms nominal) is added to the OST time-out to allow the necessary lock time for the PLL. See parameter D003 in the “Electrical Specifications" section for details.

Table 3-1 shows the time-outs that occur in various situations, while Figure 3-5 through Figure 3-8 show four different cases that can happen on powering up the device.

### Table 3-1: Time-out in Various Situations

<table>
<thead>
<tr>
<th>Oscillator Configuration</th>
<th>Power-up (2) or Brown-Out (3)</th>
<th>Wake-up from SLEEP or Oscillator Switch</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>PWRTEN = 0</td>
<td>PWRTEN = 1</td>
</tr>
<tr>
<td>HS with PLL enabled (1)</td>
<td>72 ms + 1024Tosc + 2ms</td>
<td>1024Tosc + 2 ms</td>
</tr>
<tr>
<td>HS, XT, LP</td>
<td>72 ms + 1024Tosc</td>
<td>1024Tosc</td>
</tr>
<tr>
<td>EC</td>
<td>72 ms</td>
<td>—</td>
</tr>
<tr>
<td>External RC</td>
<td>72 ms</td>
<td>—</td>
</tr>
</tbody>
</table>

**Note 1:** 2 ms = Nominal time required for the PLL to lock. See the “Electrical Specifications" section.

**Note 2:** 72 ms is the nominal power-up timer delay. See the “Electrical Specifications" section.

**Note 3:** It is recommended that the power-up timer is enabled when using the Brown-out Reset module.

Figure 3-5: Time-out Sequence on Power-up (MCLR Tied to VDD)

---

Note 1: TPWRT only occurs when PWRTEN = '1'.

---

© 2000 Microchip Technology Inc.
Section 3. Reset

Figure 3-6: Time-out Sequence on Power-up (MCLR not Tied to VDD): Case 1

Figure 3-7: Time-out Sequence on Power-up (MCLR not Tied to VDD): Case 2

Figure 3-8: Time-out Sequence on Power-up with Slow Rise Time (MCLR Tied to VDD)

Note 1: T_PWRT only occurs when PWRTEN = '1'.

© 2000 Microchip Technology Inc.
Figure 3-9: Time-out Sequence on POR w/ PLL Enabled (MCLR Tied to VDD)

VDD

MCLR

INTERNAL POR

PWRT TIME-OUT

OST TIME-OUT

PLL TIME-OUT

INTERNAL RESET

\[ T_{PWRT} = \text{PWRT lock time.} \]

\[ T_{PLL} = \text{PLL lock time.} \]

Note 1: \( T_{PWRT} \) only occurs when \( \text{PWRTEN} = '1' \).
Section 3. Reset

3.2.5 Brown-Out Reset (BOR)

On-chip Brown-out Reset circuitry places the device into reset when the device voltage ($V_{DD}$) falls below a trip point ($V_{BOR}$). This ensures that the device does not continue program execution outside the valid voltage operation range of the device. Brown-out resets are typically used in AC line applications (such as appliances) or large battery applications where large loads may be switched in (such as automotive).

Appliances encounter brown-out situations during plug-in and online voltage dip. Automotive electronics encounter brown-out when the ignition key is turned. In these application scenarios, the device voltage temporarily falls below the specified operating minimum.

If the brown-out circuit meets the current consumption requirements of the system, it may also be used as a voltage supervisory function.

**Note:** Before using the on-chip brown-out for a voltage supervisory function (monitor battery decay), please review the electrical specifications to ensure that they meet your requirements.

**Figure 3-10** shows typical brown-out situations. The Brown-out Reset module is enabled by default. To disable the module, the BOREN configuration bit must be cleared at device programming.

**Note 1:** It is recommended that the power-up timer be enabled when using the BOR module. The power-up timer is enabled by programming the PWRTEN configuration bit to '0'.

**Note 2:** Some devices require the Power-up Timer to be enabled when the Brown-out Reset circuitry is enabled. Please refer to the device data sheet for requirements.

**Figure 3-10:** Brown-Out Situations

**Note 1:** The Electrical Specification Parameter (parameter 33) has a typical value of 72 ms.
3.2.5.1 BOR Operation

The BOREN configuration bit can disable (if clear/programmed) or enable (if set) the Brown-out Reset circuitry. If $V_{DD}$ falls below $V_{BOR}$ (parameter D005 in the "Electrical Specifications" section), for greater than the Brown-out Pulse Width Time (T_{BOR}), parameter 35, the brown-out situation will reset the chip. A reset is not guaranteed to occur if $V_{DD}$ falls below $V_{BOR}$ for less than parameter 35.

The chip will remain in Brown-out Reset until $V_{DD}$ rises above $V_{BOR}$. After which, the Power-up Timer is invoked and will keep the chip in reset an additional time delay (parameter 33). If $V_{DD}$ drops below $V_{BOR}$ while the Power-up Timer is running, the chip will go back into Reset and the Power-up Timer will be re-initialized. Once $V_{DD}$ rises above $V_{BOR}$, the Power-up Timer will again start a time delay.

When the BOREN bit is set, all voltages below $V_{BOR}$ will hold the device in the reset state. This includes during the power-up sequence.

The brown-out trip point is user programmable at time of device programming. Figure 3-11 is a block diagram for the BOR circuit.

**Figure 3-11: Block Diagram of BOR Circuit**
Section 3. Reset

The Brown-out Reset circuit has four available reset trip point voltages. The device selected determines which trip points make sense in an application. All devices have the trip points of 4.2V and 4.5V available. PIC18LCXXX devices add two more trip points. The first is 2.7V, while the second is dependent on the minimum operating voltage of that device. This means that the lowest trip point voltage will either be 2.5V or 1.8V. Table 3-2 shows the state of the configuration bits (BORV1:BORV0) and the BOR trip points that they select.

Table 3-2: Example BOR Trip Point Levels

<table>
<thead>
<tr>
<th>BORV1:BORV0 Configuration Bits</th>
<th>Minimum Voltage Trip Point</th>
<th>Maximum Voltage Trip Point</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>1 1</td>
<td>1.8 V</td>
<td>1.86 V</td>
<td>PIC18LCXXX Devices (w/ VDDMIN = 1.8V)</td>
</tr>
<tr>
<td>1 1</td>
<td>2.5 V</td>
<td>2.98 V</td>
<td>PIC18LCXXX Devices (w/ VDDMIN ≥ 2.0V)</td>
</tr>
<tr>
<td>1 0</td>
<td>2.7 V</td>
<td>2.78 V</td>
<td>PIC18LCXXX Devices</td>
</tr>
<tr>
<td>0 0</td>
<td>4.2 V</td>
<td>4.33 V</td>
<td>All Devices</td>
</tr>
<tr>
<td>0 1</td>
<td>4.5 V</td>
<td>4.64 V</td>
<td>All Devices</td>
</tr>
</tbody>
</table>

Note: The minimum voltage at which the Brown-out Reset trip point can occur should be in the valid operating voltage range of the device.

The BOR is programmable to ensure that the BOR can be optimized to the voltage-frequency of the device, since the minimum device VDD value will depend on the frequency of operation. For example, VDD min. at 40 MHz may be 4.2V, whereas at 2 MHz it may be 1.8V.
3.2.5.2 Current Implications for BOR Operation

There are three components to the current consumption of the BOR operation. These are:

1. Current from Internal Reference Voltage
2. Current from BOR comparator
3. Current from resistor ladder

The Internal Reference Voltage is also used by the Low Voltage Detect circuitry and the A/D voltage references. The resistor ladder is also used by the Low Voltage Detect circuitry. If the Low Voltage Detect is enabled, then only the additional current of the comparator is added for enabling the BOR feature.

When the module is enabled, the BOR comparator and voltage divider are enabled and consume static current. The “Electrical Specifications” section parameter 32 gives the current specification.

The Brown-out Comparator circuit consumes current when enabled. To eliminate this current consumption, the Brown-out Reset can be disabled by programming the Brown-out Reset Enable configuration bit (BOREN) to '0'.

3.2.5.3 BOR Initialization

The BOR module must be enabled and programmed through the device configuration bits. These include BOREN, which enables or disables the module, and BORV1:BORV0, which set the BOR voltage.
Section 3. Reset

3.2.5.4 External Brown-Out Reset Circuits

There are some applications where the device's programmable Brown-out Reset trip point levels may still not be at the desired level for the application.

Figure 3-12 shows a circuit for external brown-out protection using the MCP100 device.

Figure 3-13 and Figure 3-14 are two examples of external circuitry that may be implemented. Each option needs to be evaluated to determine if they match the requirements of the application.

Figure 3-12: External Brown-Out Protection Using the MCP100

![Circuit Diagram](image1)

Note 1: Internal Brown-out Reset circuitry should be disabled when using this circuit.
2: Resistors should be adjusted for the characteristics of the transistor.
3: This circuit will activate reset when VDD goes below (Vz + 0.7V) where Vz = Zener voltage.

Figure 3-13: External Brown-Out Protection Circuit 1

![Circuit Diagram](image2)

Note 1: Internal Brown-out Reset circuitry should be disabled when using this circuit.
2: Resistors should be adjusted for the characteristics of the transistor.
3: This circuit will activate reset when VDD goes below (Vz + 0.7V) where Vz = Zener voltage.

Figure 3-14: External Brown-Out Protection Circuit 2

![Circuit Diagram](image3)

Note 1: This circuit is less expensive, but less accurate. Transistor Q1 turns off when VDD is below a certain level such that:
\[ VDD \cdot \frac{R1}{R1 + R2} = 0.7V \]
2: Internal Brown-out Reset circuitry should be disabled when using this circuit.
3: Resistors should be adjusted for the characteristics of the transistor.
3.3 Registers and Status Bit Values

Table 3-3 shows the significance of the device status bits and the initialization conditions for the RCON register. Table 3-4 shows the reset conditions for the Special Function Registers.

Register 3-1 shows the bits of the RCON register and Table 3-3 shows the initialization values.

Register 3-1: RCON Register Bits and Positions

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
</tr>
</thead>
<tbody>
<tr>
<td>IPEN</td>
<td>LWRT</td>
<td>—</td>
<td>RI</td>
<td>TO</td>
<td>PD</td>
<td>POR</td>
<td>BOR</td>
</tr>
</tbody>
</table>

Table 3-3: Status Bits, Their Significance, and the Initialization Condition for RCON Register

<table>
<thead>
<tr>
<th>Condition</th>
<th>Program Counter</th>
<th>RCON Register</th>
<th>RI</th>
<th>TO</th>
<th>PD</th>
<th>POR</th>
<th>BOR</th>
<th>STKFUL</th>
<th>STKUNF</th>
</tr>
</thead>
<tbody>
<tr>
<td>Power-on Reset</td>
<td>0000h</td>
<td>00-1 1100</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>u</td>
<td>u</td>
<td>u</td>
</tr>
<tr>
<td>MCLR Reset during normal operation</td>
<td>0000h</td>
<td>00-u uuuu</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
<tr>
<td>Software Reset during normal operation</td>
<td>0000h</td>
<td>0u-0 uuuu</td>
<td>0</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
<tr>
<td>Stack Overflow Reset during normal operation</td>
<td>0000h</td>
<td>0u-u uu11</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>1</td>
<td></td>
</tr>
<tr>
<td>Stack Underflow Reset during normal operation</td>
<td>0000h</td>
<td>0u-u uu11</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td>1</td>
<td></td>
</tr>
<tr>
<td>MCLR Reset during SLEEP</td>
<td>0000h</td>
<td>00-u 10uu</td>
<td>u</td>
<td>1</td>
<td>0</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
<tr>
<td>WDT Reset</td>
<td>0000h</td>
<td>0u-u 01uu</td>
<td>1</td>
<td>0</td>
<td>1</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
<tr>
<td>WDT Wake-up</td>
<td>PC + 2</td>
<td>uu-u 00uu</td>
<td>u</td>
<td>0</td>
<td>0</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
<tr>
<td>Brown-out Reset</td>
<td>0000h</td>
<td>0u-1 11u0</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
<tr>
<td>Interrupt Wake-up from SLEEP</td>
<td>PC + 2 (f1)</td>
<td>uu-u 00uu</td>
<td>u</td>
<td>1</td>
<td>0</td>
<td>u</td>
<td>u</td>
<td>u</td>
<td></td>
</tr>
</tbody>
</table>

Legend: u = unchanged, x = unknown, - = unimplemented bit read as '0'.

Note 1: When the wake-up is due to an interrupt and the GIEH or GIEL bits are set, the PC is loaded with the interrupt vector (0008h or 0018h).
### Section 3. Reset

#### Table 3-4: Initialization Conditions for SFR Registers

<table>
<thead>
<tr>
<th>Register</th>
<th>Power-on Reset, Brown-out Reset</th>
<th>MCLR Resets</th>
<th>WDT Reset</th>
<th>Wake-up via WDT or Interrupt</th>
</tr>
</thead>
<tbody>
<tr>
<td>TOSU</td>
<td>---0 0000</td>
<td>---0 0000</td>
<td></td>
<td>---0 uuuu (3)</td>
</tr>
<tr>
<td>TOSH</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu (3)</td>
</tr>
<tr>
<td>TOSL</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu (3)</td>
</tr>
<tr>
<td>STKPTR</td>
<td>00-0 0000</td>
<td>00-0 0000</td>
<td></td>
<td>uu uu uu (3)</td>
</tr>
<tr>
<td>PCLATU</td>
<td>---0 0000</td>
<td>---0 0000</td>
<td></td>
<td>uu uu uu</td>
</tr>
<tr>
<td>PCLATH</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PCL</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TBLPTRU</td>
<td>---0 0000</td>
<td>---0 0000</td>
<td></td>
<td>uu uu uu</td>
</tr>
<tr>
<td>TBLPTRH</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TBLPTRL</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TABLAT</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PRODH</td>
<td>xxxxxxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PRODL</td>
<td>xxxxxxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>INTCON</td>
<td>0000 000x</td>
<td>0000 000u</td>
<td></td>
<td>uuuu uuuu (1)</td>
</tr>
<tr>
<td>INTCON2</td>
<td>1111 -1-1</td>
<td>1111 -1-1</td>
<td></td>
<td>uu uu uu (1)</td>
</tr>
<tr>
<td>INTCON3</td>
<td>11-0 0-0-0</td>
<td>11-0 0-0-0</td>
<td></td>
<td>uu uu uu (1)</td>
</tr>
<tr>
<td>INDF0</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>POSTINC0</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>POSTDEC0</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>PREINC0</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>PLUSW0</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>FSR0H</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td></td>
<td>---- uuuu</td>
</tr>
<tr>
<td>FSR0L</td>
<td>xxxxxxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>WREG</td>
<td>xxxxxxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>INDF1</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>POSTINC1</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>POSTDEC1</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
<tr>
<td>PREINC1</td>
<td>N/A</td>
<td>N/A</td>
<td></td>
<td>N/A</td>
</tr>
</tbody>
</table>

Legend: u = unchanged, x = unknown, = unimplemented bit, read as ‘0’, q = value depends on condition.

**Note 1:** One or more bits in the INTCONx or PIRx registers will be affected (to cause wake-up).

1. When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the PC is loaded with the interrupt vector (0008h or 0018h).
2. When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the TOSU, TOSH and TOSL are updated with the current value of the PC. The STKPTR is modified to point to the next location in the hardware stack.
3. The long write enable is only reset on a POR or MCLR reset.
4. The bits in the PIR, PIE, and IPR registers are device dependent. Their function and location may change from device to device.
### Table 3-4: Initialization Conditions for SFR Registers (Continued)

<table>
<thead>
<tr>
<th>Register</th>
<th>Power-on Reset, Brown-out Reset</th>
<th>MCLR Resets WDT Reset Reset Instruction Stack Resets</th>
<th>Wake-up via WDT or Interrupt</th>
</tr>
</thead>
<tbody>
<tr>
<td>PLUSW1</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
</tr>
<tr>
<td>FSR1H</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td>---- uuuu</td>
</tr>
<tr>
<td>FSR1L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>BSR</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td>---- uuuu</td>
</tr>
<tr>
<td>INDF2</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
</tr>
<tr>
<td>POSTINC2</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
</tr>
<tr>
<td>POSTDEC2</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
</tr>
<tr>
<td>PREINC2</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
</tr>
<tr>
<td>PLUSW2</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
</tr>
<tr>
<td>FSR2H</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td>---- uuuu</td>
</tr>
<tr>
<td>FSR2L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>STATUS</td>
<td>---- xxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TMR0H</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TMR0L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TOCON</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>OSCCON</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td>---- uuuu</td>
</tr>
<tr>
<td>LVDCON</td>
<td>---- 0010</td>
<td>---- 0010</td>
<td>---- uuuu</td>
</tr>
<tr>
<td>WDTCON</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td>---- uuuu</td>
</tr>
<tr>
<td>RCON (4)</td>
<td>00-1 11q0</td>
<td>00-1 11q0</td>
<td>uu-u uu-uquu</td>
</tr>
<tr>
<td>TMR1H</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TMR1L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>T1CON</td>
<td>0-00 0000</td>
<td>u-uq uu-u uuu</td>
<td>uu-u uuu</td>
</tr>
<tr>
<td>TMR2</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PR2</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>T2CON</td>
<td>---- 0000</td>
<td>---- 0000</td>
<td>uu-u uu-u</td>
</tr>
<tr>
<td>SSPBUF</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>SSPADD</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>SSPSTAT</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>SSPCON1</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>SSPCON2</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>uuuu uuuu</td>
</tr>
</tbody>
</table>

**Legend:**
- \( u \) = unchanged,
- \( x \) = unknown,
- \( - \) = unimplemented bit, read as '0',
- \( q \) = value depends on condition.

**Note 1:**

One or more bits in the INTCONx or PIRx registers will be affected (to cause wake-up).

2: When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the PC is loaded with the interrupt vector (0008h or 0018h).

3: When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the TOSU, TOSH and TOSL are updated with the current value of the PC. The STKPTR is modified to point to the next location in the hardware stack.

4: The long write enable is only reset on a POR or MCLR reset.

5: The bits in the PIR, PIE, and IPR registers are device dependent. Their function and location may change from device to device.
Section 3. Reset

Table 3-4: Initialization Conditions for SFR Registers (Continued)

<table>
<thead>
<tr>
<th>Register</th>
<th>Power-on Reset, Brown-out Reset</th>
<th>MCLR Resets</th>
<th>WDT Reset</th>
<th>Wake-up via WDT or Interrupt</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>WDT Reset</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Stack Resets</td>
<td></td>
<td></td>
</tr>
<tr>
<td>ADRESH</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>ADRESL</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>ADCON0</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>ADCON1</td>
<td>--0- 0000</td>
<td>--0- 0000</td>
<td>--u- uuuu</td>
<td></td>
</tr>
<tr>
<td>CCP1H</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>CCP1L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>CCP2H</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>CCP2L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>CCP2CON</td>
<td>--0- 0000</td>
<td>--0- 0000</td>
<td>--u- uuuu</td>
<td></td>
</tr>
<tr>
<td>TMR3H</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>TMR3L</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>T3CON</td>
<td>0000 0000</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>SPBRG</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>RCREG</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>TXREG</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>TXSTA</td>
<td>0000 -01x</td>
<td>0000 -01u</td>
<td>uuuu -uuu</td>
<td></td>
</tr>
<tr>
<td>RCSTA</td>
<td>0000 000x</td>
<td>0000 000u</td>
<td>uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>IPR2 (5)</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td></td>
</tr>
<tr>
<td>PIR2 (5)</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>u</td>
</tr>
<tr>
<td>PIE2 (5)</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>u</td>
</tr>
<tr>
<td>IPR1 (5)</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>u</td>
</tr>
<tr>
<td>PIR1 (5)</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>u (1)</td>
</tr>
<tr>
<td>PIE1 (5)</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>u</td>
</tr>
</tbody>
</table>

Legend: u = unchanged, x = unknown, - = unimplemented bit, read as '0', q = value depends on condition.

Note 1: One or more bits in the INTCONx or PIRx registers will be affected (to cause wake-up).

2: When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the PC is loaded with the interrupt vector (0008h or 0018h).

3: When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the TOSU, TOSH and TOSL are updated with the current value of the PC. The STKPTR is modified to point to the next location in the hardware stack.

4: The long write enable is only reset on a POR or MCLR reset.

5: The bits in the PIR, PIE, and IPR registers are device dependent. Their function and location may change from device to device.
Table 3-4: Initialization Conditions for SFR Registers (Continued)

<table>
<thead>
<tr>
<th>Register</th>
<th>Power-on Reset, Brown-out Reset</th>
<th>MCLR Resets</th>
<th>WDT Reset</th>
<th>Wake-up via WDT or Interrupt</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>WDT Reset</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Reset</td>
<td>Instruction</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Stack Resets</td>
<td></td>
<td></td>
</tr>
<tr>
<td>TRISE</td>
<td>0000 -111</td>
<td>0000 -111</td>
<td></td>
<td>uuuu -uuu</td>
</tr>
<tr>
<td>TRISD</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TRISC</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TRISB</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TRIS</td>
<td>-111 1111</td>
<td>-111 1111</td>
<td></td>
<td>-uuu uuuu</td>
</tr>
<tr>
<td>LATE</td>
<td>----- -xxxx</td>
<td>----- -uuu</td>
<td></td>
<td>----- -uuu</td>
</tr>
<tr>
<td>LATD</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>LATC</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>LATB</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>LATA</td>
<td>-xxxx xxxxx</td>
<td>-uuu uuuu</td>
<td></td>
<td>-uuu uuuu</td>
</tr>
<tr>
<td>PORTED</td>
<td>----- -000</td>
<td>----- -000</td>
<td></td>
<td>----- -uuu</td>
</tr>
<tr>
<td>PORTC</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PORTB</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td></td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PORTA</td>
<td>-x0x 0000</td>
<td>-u0u 0000</td>
<td></td>
<td>-uuu uuuu</td>
</tr>
</tbody>
</table>

Legend: \( u \) = unchanged, \( x \) = unknown, \( - \) = unimplemented bit, read as '0', \( q \) = value depends on condition.

Note 1: One or more bits in the INTCONx or PIRx registers will be affected (to cause wake-up).

2: When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the PC is loaded with the interrupt vector (0008h or 0018h).

3: When the wake-up is due to an interrupt and the GIEL or GIEH bit is set, the TOSU, TOSH and TOSL are updated with the current value of the PC. The STKPTR is modified to point to the next location in the hardware stack.

4: The long write enable is only reset on a POR or MCLR reset.

5: The bits in the PIR, PIE, and IPR registers are device dependent. Their function and location may change from device to device.
Section 3. Reset

3.3.1 Reset Control (RCON) Register

The Reset Control (RCON) register contains flag bits to allow differentiation between resets. The Reset Control register has seven bits.

The POR (Power-on Reset) bit is cleared on a Power-on Reset and is unaffected otherwise. The user sets this bit following a Power-on Reset. On subsequent resets, if the POR bit is clear (= '0'), it will indicate that a Power-on Reset must have occurred.

Note: The state of the BOR bit is unknown on Power-on Reset. It must be set by the user and checked on subsequent resets to see if the BOR bit is clear, indicating a brown-out has occurred. The BOR status bit is a "don't care" and is not necessarily predictable if the brown-out circuit is disabled (by clearing the BOREN bit in the Configuration register).

The power-down bit (PD) provides indication if the device was placed into sleep mode. It is set by a power-up, a CLRWDT instruction or by user software. The PD bit is cleared when the SLEEP instruction is executed or by user software.

Register 3-2: RCON Register

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
</tr>
</thead>
<tbody>
<tr>
<td>IPEN</td>
<td>LWRT</td>
<td>—</td>
<td>RF</td>
<td>TO</td>
<td>PB</td>
<td>POR</td>
<td>BOR</td>
</tr>
</tbody>
</table>

bit 7 IPEN: Interrupt Priority Enable bit
1 = Enable priority levels on interrupts
0 = Disable priority levels on interrupts

bit 6 LWRT: Long Write Enable bit
1 = Enable Table Writes to internal program memory
   Once this bit is set, it can only be cleared by a POR or MCLR reset.
0 = Disable Table Writes to internal program memory; Table Writes only to external program memory.

bit 5 Unimplemented: Read as '0'

bit 4 RI: Reset Instruction Flag bit
1 = The RESET instruction was not invoked
0 = The RESET instruction was executed
   (must be set in software after the RESET instruction is executed)

bit 3 TO: Time-out bit
1 = After power-up, CLRWDT instruction or SLEEP instruction
0 = A WDT time-out occurred

bit 2 PD: Power-down bit
1 = After power-up or by the CLRWDT instruction
0 = By execution of the SLEEP instruction

bit 1 POR: Power-on Reset Flag bit
1 = A Power-on Reset has not occurred
0 = A Power-on Reset occurred
   (must be set in software after a Power-on Reset occurs)

bit 0 BOR: Brown-out Reset Flag bit
1 = A Brown-out Reset has not occurred
0 = A Brown-out Reset occurred
   (must be set in software after a Brown-out Reset or Power-on Reset occurs)

Legend
R = Readable bit     W = Writable bit     U = Unimplemented bit, read as '0'
- n = Value at POR reset  '1' = bit is set  '0' = bit is cleared  x = bit is unknown

© 2000 Microchip Technology Inc.
3.4 Design Tips

Question 1:  *With windowed devices, my system resets and operates properly. With an OTP device, my system does not operate properly.*

Answer 1:  
The most common reason for this is that the windowed device has not had its window covered. The background light causes the device to power-up in a different state than would typically be seen in a device where no light is present. In most cases, all the General Purpose RAM and Special Function Registers were not initialized by the application software.
Section 3. Reset

3.5 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent and could be used (with modification and possible limitations). The current application notes related to Resets are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Power-up Trouble Shooting</td>
<td>AN607</td>
</tr>
<tr>
<td>Power-up Considerations</td>
<td>AN522</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
3.6 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Reset description.
Section 4. Architecture

HIGHLIGHTS

This section of the manual contains the following major topics:

4.1 Introduction ............................................................. 4-2
4.2 Clocking Scheme/Instruction Cycle .............................. 4-5
4.3 Instruction Flow/Pipelining ...................................... 4-6
4.4 I/O Descriptions ...................................................... 4-7
4.5 Design Tips .............................................................. 4-14
4.6 Related Application Notes ...................................... 4-15
4.7 Revision History ...................................................... 4-16
4.1 Introduction

The high performance of the PIC18CXXX devices can be attributed to a number of architectural features commonly found in RISC microprocessors. These include:

- Harvard architecture
- Long Word Instructions
- Single Word Instructions
- Single Cycle Instructions
- Instruction Pipelining
- Reduced Instruction Set
- Register File Architecture
- Orthogonal (Symmetric) Instructions

Figure 4-2 shows a general block diagram for PIC18CXXX devices.

Harvard Architecture:

Harvard architecture has the program memory and data memory as separate memories which are accessed from separate buses. This improves bandwidth over traditional von Neumann architecture in which program and data are fetched from the same memory using the same bus. To execute an instruction, a von Neumann machine must make one or more (generally more) accesses across the 8-bit bus to fetch the instruction. Then data may need to be fetched, operated on and possibly written. As can be seen from this description, the bus can become extremely congested. With a Harvard architecture, the instruction is fetched in a single instruction cycle (all 16 bits). While the program memory is being accessed, the data memory is on an independent bus and can be read and written. These separated busses allow one instruction to execute, while the next instruction is fetched. A comparison of Harvard and von Neumann architectures is shown in Figure 4-1.

Figure 4-1: Harvard vs. von Neumann Block Architectures

Long Word Instructions:

Long word instructions have a wider (more bits) instruction bus than the 8-bit data memory bus. This is possible because the two buses are separate. This allows instructions to be sized differently than the 8-bit wide data word and allows a more efficient use of the program memory, since the program memory width is optimized to the architectural requirements.

Single Word Instructions:

Single word instruction opcodes are 16-bits wide making it possible to have all but a few instructions be single word instructions. A 16-bit wide program memory access bus fetches a 16-bit instruction in a single cycle. With single word instructions, the number of words of program memory locations equals the number of instructions for the device. This means that all locations are valid instructions.

Typically in the von Neumann architecture, most instructions are multi-byte. In general, a device with 4 Kbytes of program memory would allow approximately 2K of instructions. This 2:1 ratio is generalized and dependent on the application code. Since each instruction may take multiple bytes, there is no assurance that each location is a valid instruction.
Double Word Instructions:
Some operations require more information than can be stored in the 16 bits of a program memory location. These operations require a double word instruction, and are therefore 32-bits wide. Instructions that require this second instruction word are:

- Memory to memory move instruction (12 bits for each RAM address)
  - MOVFF SourceReg, DestReg
- Literal value to FSR move instruction (12 bits for data and 2 bits for FSR to load)
  - LFSR FSR#, Address
- Call and goto operations (20 bits for address)
  - CALL Address
  - GOTO Address

The first word indicates to the CPU that the next program memory location is the additional information for this instruction and not an instruction. If the CPU tries to execute the second word of an instruction (due to a software modified PC pointing to that location as an instruction), the fetched data is executed as a NOP.

Double word instruction execution is not split between the two T CY cycles by an interrupt request. That is, when an interrupt request occurs during the execution of a double word instruction, the execution of the instruction is completed before the processor vectors to the interrupt address. The interrupt latency is preserved.

Instruction Pipeline:
The instruction pipeline is a two-stage pipeline that overlaps the fetch and execution of instructions. The fetch of the instruction takes one T CY, while the execution takes another T CY. However, due to the overlap of the fetch of current instruction and execution of previous instruction, an instruction is fetched and another instruction is executed every T CY.

Single Cycle Instructions:
With the program memory bus being 16-bits wide, the entire instruction is fetched in a single machine cycle (T CY), except for double word instructions. The instruction contains all the information required and is executed in a single cycle. There may be a one cycle delay in execution if the result of the instruction modified the contents of the program counter. This requires the pipeline to be flushed and a new instruction to be fetched.

Two Cycle Instructions:
Double word instructions require two cycles to execute, since all the required information is in the 32 bits.

Reduced Instruction Set:
When an instruction set is well designed and highly orthogonal (symmetric), fewer instructions are required to perform all needed tasks. With fewer instructions, the whole set can be more rapidly learned.

Register File Architecture:
The register files/data memory can be directly or indirectly addressed. All special function registers, including the program counter, are mapped in the data memory.

Orthogonal (Symmetric) Instructions:
Orthogonal instructions make it possible to carry out any operation on any register using any addressing mode. This symmetrical nature and lack of “special instructions” make programming simple yet efficient. In addition, the learning curve is reduced significantly. The Enhanced MCU instruction set uses only three non-register oriented instructions, which are used for two of the cores features. One is the SLEEP instruction, which places the device into the lowest power use mode. The second is the CLRWDT instruction, which verifies the chip is operating properly by preventing the on-chip Watchdog Timer (WDT) from overflowing and resetting the device. The third is the RESET instruction, which resets the device.
Figure 4-2: General Enhanced MCU Block Diagram

Note 1: Many of the general purpose I/O pins are multiplexed with one or more peripheral module functions. The multiplexing combinations are device dependent.
Section 4 Architecture

4.2 Clocking Scheme/Instruction Cycle

The clock input is internally divided by four to generate four non-overlapping quadrature clocks, namely Q1, Q2, Q3 and Q4. Internally, the program counter is incremented every Q1, and the instruction is fetched from the program memory and latched into the instruction register in Q4. The instruction is decoded and executed during the following Q1 through Q4. The clocks and instruction execution flow are illustrated in Figure 4-3 and Example 4-1.

Figure 4-3: Clock/Instruction Cycle

<table>
<thead>
<tr>
<th>Device Clock</th>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>(OSC1 or T1OSCI)</td>
<td>Q1</td>
<td>Q2</td>
<td>Q3</td>
<td>Q4</td>
</tr>
<tr>
<td>Q1</td>
<td>Q2</td>
<td>Q3</td>
<td>Q4</td>
<td>Q1</td>
</tr>
<tr>
<td>Q2</td>
<td>Q3</td>
<td>Q4</td>
<td>Q1</td>
<td>Q2</td>
</tr>
<tr>
<td>Q3</td>
<td>Q4</td>
<td>Q1</td>
<td>Q2</td>
<td>Q3</td>
</tr>
<tr>
<td>Q4</td>
<td>Q1</td>
<td>Q2</td>
<td>Q3</td>
<td>Q4</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>CLKOUT (RC mode)</th>
<th>PC</th>
<th>PC+2</th>
<th>PC+4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Fetch INST (PC)</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Execute INST (PC)</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Fetch INST (PC+2)</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Execute INST (PC+2)</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

4.2.1 Phase Lock Loop (PLL)

The clock input is multiplied by four by the PLL. Therefore, when it is internally divided by four, it provides an instruction cycle that is the same frequency as the external clock frequency. Four non-overlapping quadrature clocks, namely Q1, Q2, Q3 and Q4 are still generated internally. Internally, the program counter (PC) is incremented every Q1, and the instruction is fetched from the program memory and latched into the instruction register in Q4. The instruction is decoded and executed during the following Q1 through Q4. The clocks and instruction execution flow are illustrated in Figure 4-4 and Example 4-1.

Figure 4-4: Clock/Instruction Cycle with PLL

<table>
<thead>
<tr>
<th>PLL Output</th>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>OSC2/CLKOUT (RC mode)</td>
<td>Q1</td>
<td>Q2</td>
<td>Q3</td>
<td>Q4</td>
</tr>
<tr>
<td>OS1</td>
<td>PC</td>
<td>PC+2</td>
<td>PC+4</td>
<td></td>
</tr>
<tr>
<td>Fetch INST (PC)</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Execute INST (PC)</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Fetch INST (PC+2)</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Execute INST (PC+2)</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Fetch INST (PC+4)</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Execute INST (PC+4)</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

© 2000 Microchip Technology Inc.
4.3 Instruction Flow/Pipelining

An “Instruction Cycle” consists of four Q cycles (Q1, Q2, Q3 and Q4). Fetch takes one instruction cycle, while decode and execute takes another instruction cycle. However, due to pipelining, each instruction effectively executes in one cycle. If an instruction causes the program counter to change (e.g. GOTO instruction), then an extra cycle is required to complete the instruction (See Example 4-1).

The instruction fetch begins with the program counter incrementing in Q1. In the execution cycle, the fetched instruction is latched into the “Instruction Register (IR)” in cycle Q1. This instruction is then decoded and executed during the Q2, Q3 and Q4 cycles. Data memory is read during Q2 (operand read) and written during Q4 (destination write).

Example 4-1 shows the operation of the two stage pipeline for the instruction sequence shown. At time Tcy0, the first instruction is fetched from program memory. During Tcy1, the first instruction executes, while the second instruction is fetched. During Tcy2, the second instruction executes, while the third instruction is fetched. During Tcy3, the fourth instruction is fetched, while the third instruction (CALL SUB_1) is executed. When the third instruction completes execution, the CPU forces the address of instruction four onto the Stack and then changes the Program Counter (PC) to the address of SUB_1. This means that the instruction that was fetched during Tcy3 needs to be “flushed” from the pipeline. During Tcy4, instruction four is flushed (executed as a NOP) and the instruction at address SUB_1 is fetched. Finally during Tcy5, instruction five is executed and the instruction at address SUB_1 + 2 is fetched.

Example 4-1: Instruction Pipeline Flow

<table>
<thead>
<tr>
<th>Tcy0</th>
<th>Tcy1</th>
<th>Tcy2</th>
<th>Tcy3</th>
<th>Tcy4</th>
<th>Tcy5</th>
</tr>
</thead>
<tbody>
<tr>
<td>1. MOVLM 55h</td>
<td>Fetch 1</td>
<td>Execute 1</td>
<td>Fetch 2</td>
<td>Execute 2</td>
<td>Fetch 3</td>
</tr>
</tbody>
</table>

Most instructions are single cycle. Program branches take two cycles, since the fetch instruction is “flushed” from the pipeline while the new instruction is being fetched and then executed.
4.4 I/O Descriptions

Table 4-1 gives a brief description of device pins and the functions that may be multiplexed to a port pin. Multiple functions may exist on one port pin. When multiplexing occurs, the peripheral module's functional requirements may force an override of the data direction (TRIS bit) of the port pin (such as in the A/D and Comparator modules).

Table 4-1: I/O Descriptions

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Pin Type</th>
<th>Buffer Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>A19</td>
<td>O</td>
<td>—</td>
<td>System bus address line 19</td>
</tr>
<tr>
<td>A18</td>
<td>O</td>
<td>—</td>
<td>System bus address line 18</td>
</tr>
<tr>
<td>A17</td>
<td>O</td>
<td>—</td>
<td>System bus address line 17</td>
</tr>
<tr>
<td>A16</td>
<td>O</td>
<td>—</td>
<td>System bus address line 16</td>
</tr>
<tr>
<td>AD15</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 15</td>
</tr>
<tr>
<td>AD14</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 14</td>
</tr>
<tr>
<td>AD13</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 13</td>
</tr>
<tr>
<td>AD12</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 12</td>
</tr>
<tr>
<td>AD11</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 11</td>
</tr>
<tr>
<td>AD10</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 10</td>
</tr>
<tr>
<td>AD9</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 9</td>
</tr>
<tr>
<td>AD8</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 8</td>
</tr>
<tr>
<td>AD7</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 7</td>
</tr>
<tr>
<td>AD6</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 6</td>
</tr>
<tr>
<td>AD5</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 5</td>
</tr>
<tr>
<td>AD4</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 4</td>
</tr>
<tr>
<td>AD3</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 3</td>
</tr>
<tr>
<td>AD2</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 2</td>
</tr>
<tr>
<td>AD1</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 1</td>
</tr>
<tr>
<td>AD0</td>
<td>I/O</td>
<td>TTL</td>
<td>System bus address/data line 0</td>
</tr>
<tr>
<td>ALE</td>
<td>O</td>
<td>—</td>
<td>System bus address latch enable strobe</td>
</tr>
<tr>
<td>AN0</td>
<td>I</td>
<td>Analog</td>
<td>Analog Input Channels</td>
</tr>
<tr>
<td>AN1</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN2</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN3</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN4</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN5</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN6</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN7</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN8</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN9</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN10</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN11</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN12</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN13</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN14</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>AN15</td>
<td>I</td>
<td>Analog</td>
<td></td>
</tr>
<tr>
<td>Avdd</td>
<td>P</td>
<td>P</td>
<td>Analog Power</td>
</tr>
</tbody>
</table>

Legend: TTL = TTL-compatible input  CMOS = CMOS compatible input or output
ST = Schmitt Trigger input with CMOS levels  O = output
PU = Weak internal pull-up  I = input
Analog = Analog input or output  P = Power
Table 4-1: I/O Descriptions (Continued)

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Buffer Type</th>
<th>Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>AVSS</td>
<td>P</td>
<td>P</td>
<td>Analog Ground</td>
</tr>
<tr>
<td>BA0</td>
<td>O</td>
<td></td>
<td>System bus byte address 0</td>
</tr>
<tr>
<td>CANRX</td>
<td>I</td>
<td>ST</td>
<td>CAN bus receive pin</td>
</tr>
<tr>
<td>CANTX0</td>
<td>O</td>
<td></td>
<td>CAN bus transmit</td>
</tr>
<tr>
<td>CANTX1</td>
<td>O</td>
<td></td>
<td>CAN bus complimentary transmit or CAN bus bit time clock</td>
</tr>
<tr>
<td>CCP1</td>
<td>I/O</td>
<td>ST</td>
<td>Capture1 input/Compare1 output/PWM1 output</td>
</tr>
<tr>
<td>CCP2</td>
<td>I/O</td>
<td>ST</td>
<td>Capture2 input/Compare2 output/PWM2 output.</td>
</tr>
<tr>
<td>CK</td>
<td>I/O</td>
<td>ST</td>
<td>USART Synchronous Clock, always associated with TX pin function (See related TX, RX, DT)</td>
</tr>
<tr>
<td>CLKI</td>
<td>I</td>
<td>ST/CMOS</td>
<td>External clock source input. Always associated with pin function OSC1. (See related OSC1/CLKIN, OSC2/CLKOUT pins)</td>
</tr>
<tr>
<td>CLKO</td>
<td>O</td>
<td></td>
<td>Oscillator crystal output. Connects to crystal or resonator in crystal oscillator mode. In RC mode, OSC2 pin outputs CLKOUT which has 1/4 the frequency of OSC1, and denotes the instruction cycle rate. Always associated with OSC2 pin function. (See related OSC2, OSC1)</td>
</tr>
<tr>
<td>CMPA</td>
<td>O</td>
<td></td>
<td>Comparator A output</td>
</tr>
<tr>
<td>CMPB</td>
<td>O</td>
<td></td>
<td>Comparator B output</td>
</tr>
<tr>
<td>CS</td>
<td>I</td>
<td>TTL</td>
<td>Chip select control for parallel slave port (See related RD and WR)</td>
</tr>
<tr>
<td>CVREF</td>
<td>O</td>
<td>Analog</td>
<td>Comparator voltage reference output</td>
</tr>
<tr>
<td>DT</td>
<td>I/O</td>
<td>ST</td>
<td>USART Synchronous Data. Always associated RX pin function. (See related RX, TX, CK)</td>
</tr>
</tbody>
</table>

Legend:
- TTL = TTL-compatible input
- CMOS = CMOS compatible input or output
- ST = Schmitt Trigger input with CMOS levels
- O = output
- PU = Weak internal pull-up
- I = input
- Analog = Analog input or output
- P = Power
### Section 4 Architecture

#### Table 4-1: I/O Descriptions (Continued)

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Pin Type</th>
<th>Buffer Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>INT0</td>
<td>I</td>
<td>ST</td>
<td>External Interrupt0</td>
</tr>
<tr>
<td>INT1</td>
<td>I</td>
<td>ST</td>
<td>External Interrupt1</td>
</tr>
<tr>
<td>INT2</td>
<td>I</td>
<td>ST</td>
<td>External Interrupt2</td>
</tr>
<tr>
<td>LB</td>
<td>O</td>
<td>—</td>
<td>System bus low byte strobe</td>
</tr>
<tr>
<td>LVDIN</td>
<td>I</td>
<td>Analog</td>
<td>Low voltage detect input</td>
</tr>
<tr>
<td>MCLR</td>
<td>I/P</td>
<td>ST</td>
<td>Master clear (reset) input or programming voltage input. This pin is an active low reset to the device.</td>
</tr>
<tr>
<td>NC</td>
<td>—</td>
<td>—</td>
<td>These pins should be left unconnected.</td>
</tr>
<tr>
<td>OE</td>
<td>O</td>
<td>—</td>
<td>System bus output enable strobe</td>
</tr>
<tr>
<td>OSC1</td>
<td>I</td>
<td>ST/CMOS</td>
<td>Oscillator crystal input or external clock source input, ST buffer when configured in RC mode. CMOS otherwise.</td>
</tr>
<tr>
<td>OSC2</td>
<td>O</td>
<td>—</td>
<td>Oscillator crystal output. Connects to crystal or resonator in crystal oscillator mode. In RC mode, OSC2 pin outputs CLKOUT, which has 1/4 the frequency of OSC1, and denotes the instruction cycle rate.</td>
</tr>
<tr>
<td>PSP0</td>
<td>I/O</td>
<td>TTL</td>
<td>Parallel Slave Port for interfacing to a microprocessor port. These pins have TTL input buffers when PSP module is enabled.</td>
</tr>
<tr>
<td>PSP1</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>PSP2</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>PSP3</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>PSP4</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>PSP5</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>PSP6</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>PSP7</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RA0</td>
<td>I/O</td>
<td>TTL</td>
<td>PORTA is a bi-directional I/O port.</td>
</tr>
<tr>
<td>RA1</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RA2</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RA3</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RA4</td>
<td>I/O</td>
<td>ST</td>
<td>RA4 is an open drain when configured as output.</td>
</tr>
<tr>
<td>RA5</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RA6</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RB0</td>
<td>I/O</td>
<td>TTL</td>
<td>PORTB is a bi-directional I/O port. PORTB can be software programmed for internal weak pull-ups on all inputs.</td>
</tr>
<tr>
<td>RB1</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RB2</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RB3</td>
<td>I/O</td>
<td>TTL</td>
<td></td>
</tr>
<tr>
<td>RB4</td>
<td>I/O</td>
<td>TTL</td>
<td>Interrupt on change pin.</td>
</tr>
<tr>
<td>RB5</td>
<td>I/O</td>
<td>TTL</td>
<td>Interrupt on change pin.</td>
</tr>
<tr>
<td>RB6</td>
<td>I/O</td>
<td>TTL/ST</td>
<td>Interrupt on change pin. Serial programming clock. TTL input buffer as general purpose I/O, Schmitt Trigger input buffer when used as the serial programming clock.</td>
</tr>
<tr>
<td>RB7</td>
<td>I/O</td>
<td>TTL/ST</td>
<td>Interrupt on change pin. Serial programming data. TTL input buffer as general purpose I/O, Schmitt Trigger input buffer when used as the serial programming data.</td>
</tr>
</tbody>
</table>

**Legend:**
- TTL = TTL-compatible input
- CMOS = CMOS compatible input or output
- ST = Schmitt Trigger input with CMOS levels
- O = output
- PU = Weak internal pull-up
- I = input
- Analog = Analog input or output
- P = Power
**Table 4-1: I/O Descriptions (Continued)**

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Pin Type</th>
<th>Buffer Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>RC0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTC is a bi-directional I/O port.</td>
</tr>
<tr>
<td>RC1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RC2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RC3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RC4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RC5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RC6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RC7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RD0</td>
<td>I</td>
<td>TTL</td>
<td>Read control for parallel slave port. (See also WR and CS pins.)</td>
</tr>
<tr>
<td>RD1</td>
<td>I/O</td>
<td>ST</td>
<td>PORTD is a bi-directional I/O port.</td>
</tr>
<tr>
<td>RD2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RD3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RD4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RD5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RD6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RD7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTE is a bi-directional I/O port.</td>
</tr>
<tr>
<td>RE1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RE7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTF is a digital input</td>
</tr>
<tr>
<td>RF1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RF7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
</tbody>
</table>

Legend: TTL = TTL-compatible input, CMOS = CMOS compatible input or output, ST = Schmitt Trigger input with CMOS levels, O = output, P = Power.
Section 4 Architecture

Table 4-1: I/O Descriptions (Continued)

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Pin Type</th>
<th>Buffer Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>RG0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTG is a digital input</td>
</tr>
<tr>
<td>RG1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RG2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RG3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RG4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RG5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RG6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RG7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTH is a digital input</td>
</tr>
<tr>
<td>RH1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RH7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTJ is a digital input</td>
</tr>
<tr>
<td>RJ1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RJ7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTK is a digital input</td>
</tr>
<tr>
<td>RK1</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK2</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK3</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK4</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK5</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK6</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
<tr>
<td>RK7</td>
<td>I/O</td>
<td>ST</td>
<td></td>
</tr>
</tbody>
</table>

Legend: TTL = TTL-compatible input
ST = Schmitt Trigger input with CMOS levels
CMOS = CMOS compatible input or output
O = output
I = input
PU = Weak internal pull-up
P = Power
Analog = Analog input or output
### Table 4-1: I/O Descriptions (Continued)

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Pin Type</th>
<th>Buffer Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>RL0</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL1</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL2</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL3</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL4</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL5</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL6</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RL7</td>
<td>I/O</td>
<td>ST</td>
<td>PORTL is a digital input</td>
</tr>
<tr>
<td>RX</td>
<td>I</td>
<td>ST</td>
<td>USART Asynchronous Receive</td>
</tr>
<tr>
<td>SCL</td>
<td>I/O</td>
<td>ST</td>
<td>Synchronous serial clock input/output for I²C mode.</td>
</tr>
<tr>
<td>SCLA</td>
<td>I/O</td>
<td>ST</td>
<td>Synchronous serial clock for I²C interface.</td>
</tr>
<tr>
<td>SCLB</td>
<td>I/O</td>
<td>ST</td>
<td>Synchronous serial clock for I²C interface.</td>
</tr>
<tr>
<td>SDA</td>
<td>I/O</td>
<td>ST</td>
<td>I²C™ Data I/O</td>
</tr>
<tr>
<td>SDAA</td>
<td>I/O</td>
<td>ST</td>
<td>Synchronous serial data I/O for I²C interface</td>
</tr>
<tr>
<td>SDAB</td>
<td>I/O</td>
<td>ST</td>
<td>Synchronous serial data I/O for I²C interface</td>
</tr>
<tr>
<td>SCK</td>
<td>I/O</td>
<td>ST</td>
<td>Synchronous serial clock input/output for SPI mode.</td>
</tr>
<tr>
<td>SDO</td>
<td>O</td>
<td>—</td>
<td>SPI Data Out (SPI mode)</td>
</tr>
<tr>
<td>SS</td>
<td>I</td>
<td>ST</td>
<td>SPI Slave Select input</td>
</tr>
<tr>
<td>T0CKI</td>
<td>I</td>
<td>ST</td>
<td>Timer0 external clock input</td>
</tr>
<tr>
<td>T1CKI</td>
<td>I</td>
<td>ST</td>
<td>Timer1 external clock input</td>
</tr>
<tr>
<td>T1OSO</td>
<td>O</td>
<td>CMOS</td>
<td>Timer1 oscillator output</td>
</tr>
<tr>
<td>T1OSI</td>
<td>I</td>
<td>CMOS</td>
<td>Timer1 oscillator input</td>
</tr>
<tr>
<td>TX</td>
<td>O</td>
<td>—</td>
<td>USART Asynchronous Transmit (See related RX)</td>
</tr>
<tr>
<td>UB</td>
<td>O</td>
<td>—</td>
<td>System bus upper byte strobe</td>
</tr>
</tbody>
</table>

Legend: TTL = TTL-compatible input  
CMOS = CMOS compatible input or output  
ST = Schmitt Trigger input with CMOS levels  
O = output  
P = Power  
PU = Weak internal pull-up  
I = input  
Analog = Analog input or output
### Table 4-1: I/O Descriptions (Continued)

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Pin Type</th>
<th>Buffer Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>VREF</td>
<td>I</td>
<td>Analog</td>
<td>Analog High Voltage Reference input. DR reference voltage output on devices with comparators.</td>
</tr>
<tr>
<td>VREF+</td>
<td>I</td>
<td>Analog</td>
<td>Analog High Voltage Reference input. Usually multiplexed onto an analog pin.</td>
</tr>
<tr>
<td>VREF-</td>
<td>I</td>
<td>Analog</td>
<td>Analog Low Voltage Reference input. Usually multiplexed onto an analog pin.</td>
</tr>
<tr>
<td>VSS</td>
<td>P</td>
<td>—</td>
<td>Ground reference for logic and I/O pins.</td>
</tr>
<tr>
<td>VDD</td>
<td>P</td>
<td>—</td>
<td>Positive supply for logic and I/O pins.</td>
</tr>
<tr>
<td>VPP</td>
<td>P</td>
<td>—</td>
<td>Programming voltage input</td>
</tr>
<tr>
<td>WR</td>
<td>I</td>
<td>TTL</td>
<td>Write control for parallel slave port (See CS and RD pins also).</td>
</tr>
<tr>
<td>WRL</td>
<td>O</td>
<td>—</td>
<td>System bus write low byte strobe</td>
</tr>
<tr>
<td>WRH</td>
<td>O</td>
<td>—</td>
<td>System bus write high byte strobe</td>
</tr>
</tbody>
</table>

Legend: TTL = TTL-compatible input  
CMOS = CMOS compatible input or output  
ST = Schmitt Trigger input with CMOS levels  
O = output  
PU = Weak internal pull-up  
I = input  
Analog = Analog input or output  
P = Power
4.5 Design Tips

No related design tips at this time.
Section 4 Architecture

4.6 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent and could be used (with modification and possible limitations). The current application notes related to Architecture are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related application notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
4.7 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Architecture description.
Section 5. CPU and ALU

HIGHLIGHTS

This section of the manual contains the following major topics:

5.1 Introduction ................................................................................................................ .... 5-2
5.2 General Instruction Format ............................................................................................ 5-6
5.3 Central Processing Unit (CPU) ...................................................................................... 5-7
5.4 Instruction Clock ............................................................................................................ 5-8
5.5 Arithmetic Logical Unit (ALU) ......................................................................................... 5-9
5.6 STATUS Register ......................................................................................................... 5-11
5.7 Design Tips .................................................................................................................. 5-14
5.8 Related Application Notes ............................................................................................ 5-15
5.9 Revision History ........................................................................................................... 5-16
5.1 Introduction

The Central Processing Unit (CPU) is responsible for using the information in the program memory (instructions) to control the operation of the device. Many of these instructions operate on data memory. To operate on data memory, the Arithmetic Logical Unit (ALU) is required. In addition to performing arithmetical and logical operations, the ALU controls the state of the status bits, which are found in the STATUS register. The result of some instructions force status bits to a value depending on the state of the result.

The machine codes that the CPU recognizes are shown in Table 5-1, as well as the instruction mnemonics that the MPASM uses to generate these codes.
## Section 5. CPU and ALU

### Table 5-1: PIC18CXXX Instruction Set

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><strong>BYTE-ORIENTED FILE REGISTER OPERATIONS</strong></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ADDWF</td>
<td>f, d, a</td>
<td>Add WREG and f</td>
<td>1</td>
<td>0010 01da 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>ADDRFC</td>
<td>f, d, a</td>
<td>Add WREG and Carry bit to f</td>
<td>1</td>
<td>0010 00da 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>ADDWF</td>
<td>f, d, a</td>
<td>AND WREG with f</td>
<td>1</td>
<td>0001 101a 0000 0000</td>
<td>Z, N</td>
</tr>
<tr>
<td>CLRF</td>
<td>f, a</td>
<td>Clear f</td>
<td>1</td>
<td>0110 11da 0000 0000</td>
<td>Z, N</td>
</tr>
<tr>
<td>COMF</td>
<td>f, d, a</td>
<td>Complement f</td>
<td>1</td>
<td>0001 11da 0000 0000</td>
<td>Z, N</td>
</tr>
<tr>
<td>CPFSEQ</td>
<td>f, a</td>
<td>Compare f with WREG, skip =</td>
<td>1</td>
<td>0110 011a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>CPFSGT</td>
<td>f, a</td>
<td>Compare f with WREG, skip &gt;</td>
<td>1</td>
<td>0110 010a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>CPFSLT</td>
<td>f, a</td>
<td>Compare f with WREG, skip &lt;</td>
<td>1</td>
<td>0110 000a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>DECF</td>
<td>f, d, a</td>
<td>Decrement f</td>
<td>1</td>
<td>0000 01da 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>DECFSZ</td>
<td>f, d, a</td>
<td>Decrement f, Skip if 0</td>
<td>1</td>
<td>0010 11da 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>DCFSNZ</td>
<td>f, d, a</td>
<td>Decrement f, Skip if Not 0</td>
<td>1</td>
<td>0100 11da 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>INCF</td>
<td>f, d, a</td>
<td>Increment f</td>
<td>1</td>
<td>0010 10da 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>INCFSZ</td>
<td>f, d, a</td>
<td>Increment f, Skip if 0</td>
<td>1</td>
<td>0011 11da 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>INCFSNZ</td>
<td>f, d, a</td>
<td>Increment f, Skip if Not 0</td>
<td>1</td>
<td>0010 10da 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>IRCWF</td>
<td>f, d, a</td>
<td>Inclusive OR WREG with f</td>
<td>1</td>
<td>0001 00da 0000 0000</td>
<td>Z</td>
</tr>
<tr>
<td>MOVE</td>
<td>f, d, a</td>
<td>Move f</td>
<td>1</td>
<td>0101 00da 0000 0000</td>
<td>Z</td>
</tr>
<tr>
<td>MOVFF</td>
<td>f, t, d</td>
<td>Move f, t (source) to</td>
<td>2</td>
<td>1100 000a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>MOVWF</td>
<td>f, a</td>
<td>Move WREG to f</td>
<td>1</td>
<td>0111 000a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>MOVWF</td>
<td>f, a</td>
<td>Move WREG to f</td>
<td>1</td>
<td>0101 000a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>MULWF</td>
<td>f, a</td>
<td>Multiply WREG with f</td>
<td>1</td>
<td>0000 001a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>NEGWF</td>
<td>f, a</td>
<td>Negate f</td>
<td>1</td>
<td>0110 110a 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>RLCF</td>
<td>f, d, a</td>
<td>Rotate Left f through Carry</td>
<td>1</td>
<td>0011 01da 0000 0000</td>
<td>C, DC, Z, N</td>
</tr>
<tr>
<td>RLCNF</td>
<td>f, d, a</td>
<td>Rotate Left f (No Carry)</td>
<td>1</td>
<td>0100 01da 0000 0000</td>
<td>C, DC, Z, N</td>
</tr>
<tr>
<td>RRCF</td>
<td>f, d, a</td>
<td>Rotate Right f through Carry</td>
<td>1</td>
<td>0011 00da 0000 0000</td>
<td>C, DC, Z, N</td>
</tr>
<tr>
<td>RRNCF</td>
<td>f, d, a</td>
<td>Rotate Right f (No Carry)</td>
<td>1</td>
<td>0100 00da 0000 0000</td>
<td>C, DC, Z, N</td>
</tr>
<tr>
<td>SETF</td>
<td>f, a</td>
<td>Set f</td>
<td>1</td>
<td>0110 100a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>SUBWF</td>
<td>f, d, a</td>
<td>Subtract f from WREG with</td>
<td>1</td>
<td>0101 010a 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>SUBWFB</td>
<td>f, d, a</td>
<td>Subtract WREG from f</td>
<td>1</td>
<td>0110 111a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>SUBWFB</td>
<td>f, d, a</td>
<td>Subtract WREG from f</td>
<td>1</td>
<td>0101 011a 0000 0000</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>SWAPF</td>
<td>f, d, a</td>
<td>Swap nibbles in f</td>
<td>1</td>
<td>0011 00da 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>TSTFSZ</td>
<td>f, a</td>
<td>Test f, skip if 0</td>
<td>1</td>
<td>0110 011a 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>XORWF</td>
<td>f, d, a</td>
<td>Exclusive OR WREG with f</td>
<td>1</td>
<td>0001 10da 0000 0000</td>
<td>Z, N</td>
</tr>
</tbody>
</table>

### BIT-ORIENTED FILE REGISTER OPERATIONS

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>BCF</td>
<td>f, b, a</td>
<td>Bit Clear f</td>
<td>1</td>
<td>1001 bbba 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>BSF</td>
<td>f, b, a</td>
<td>Bit Set f</td>
<td>1</td>
<td>1000 bbba 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>BTFSC</td>
<td>f, b, a</td>
<td>Bit Test f, Skip if Clear</td>
<td>1</td>
<td>1011 bbba 0000 0000</td>
<td>None</td>
</tr>
<tr>
<td>BTFSS</td>
<td>f, b, a</td>
<td>Bit Test f, Skip if Set</td>
<td>1</td>
<td>1010 bbba 0000 0000</td>
<td>None</td>
</tr>
</tbody>
</table>

Note 1: When an I/O register is modified as a function of itself (e.g., MOVF PORTB, 1, 0), the value used will be that value present on the pins themselves. For example, if the data latch is '1' for a pin configured as input and is driven low by an external device, the data will be written back with a '0'.

2: If this instruction is executed on the TMR0 register (and, where applicable, d = 1), the prescaler will be cleared if assigned to the Timer0 Module.

3: If Program Counter (PC) is modified or a conditional test is true, the instruction requires two cycles. The second cycle is executed as a NOP.

4: Some instructions are 2 word instructions. The second word of these instructions will be executed as a NOP, unless the first word retrieves the information embedded in these 16 bits. This ensures that all program memory locations have a valid instruction.

5: If the Table Write starts the write cycle to internal memory, the write will continue until terminated.
Table 5-1: PIC18CXXX Instruction Set (Continued)

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles (4)</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td>CONTROL OPERATIONS</td>
<td></td>
<td></td>
<td>MSb</td>
<td>LSb</td>
<td></td>
</tr>
<tr>
<td>BC n, BN n, BNC n, BNN n, BNOV n, BNZ n, BOV n, BRA n, BZ n, CALL n, s</td>
<td>Branch if Carry, Negative, Not Carry, Not Negative, Not Overflow, Not Zero, Overflow, Unconditionally, Zero</td>
<td>1 (2)</td>
<td>1110 0010 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>CLRWDT —</td>
<td>Clear Watchdog Timer</td>
<td>1</td>
<td>0000 0000 0000 0100</td>
<td>TO, PD</td>
<td></td>
</tr>
<tr>
<td>DAW —</td>
<td>Decimal Adjust WREG</td>
<td>1</td>
<td>0000 0000 0000 0111</td>
<td>C</td>
<td></td>
</tr>
<tr>
<td>GOTO n</td>
<td>Go to address</td>
<td>1st word</td>
<td>1110 1110 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>NOP —</td>
<td>No Operation</td>
<td>1st word</td>
<td>1111 kkkk kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>NOP —</td>
<td>No Operation (4)</td>
<td>2nd word</td>
<td>1111 kkkk kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>POP —</td>
<td>Pop top of return stack (TOS)</td>
<td>1</td>
<td>0000 0000 0000 0110</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>PUSH —</td>
<td>Push top of return stack (TOS)</td>
<td>1</td>
<td>0000 0000 0000 0101</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RCALL n</td>
<td>Relative Call</td>
<td>2</td>
<td>1101 1nnn nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RESET —</td>
<td>Software device RESET</td>
<td>1</td>
<td>0000 0000 1111 1111</td>
<td>All</td>
<td></td>
</tr>
<tr>
<td>RETFIE s</td>
<td>Return from interrupt enable</td>
<td>2</td>
<td>0000 0000 0011 000a</td>
<td>GIEH, GIEL</td>
<td></td>
</tr>
<tr>
<td>RETLW k</td>
<td>Return with literal in WREG</td>
<td>2</td>
<td>0000 1100 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RETURN s</td>
<td>Return from Subroutine</td>
<td>2</td>
<td>0000 0000 001i 001s</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>SLEEP —</td>
<td>Go into standby mode</td>
<td>1</td>
<td>0000 0000 0000 0011</td>
<td>TO, PD</td>
<td></td>
</tr>
<tr>
<td>TBLRD m</td>
<td>Table Read * → mm = 00</td>
<td>2</td>
<td>0000 0000 0000 10mm</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TBLWT m</td>
<td>Table Write * → mm = 00</td>
<td>2</td>
<td>0000 0000 0000 11mm</td>
<td>None</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: When an I/O register is modified as a function of itself (e.g., MOVF PORTB, 1, 0), the value used will be that value present on the pins themselves. For example, if the data latch is '1' for a pin configured as input and is driven low by an external device, the data will be written back with a '0'.

2: If this instruction is executed on the TMR0 register (and, where applicable, d = 1), the prescaler will be cleared if assigned to the Timer0 Module.

3: If Program Counter (PC) is modified or a conditional test is true, the instruction requires two cycles. The second cycle is executed as a NOP.

4: Some instructions are 2 word instructions. The second word of these instructions will be executed as a NOP, unless the first word retrieves the information embedded in these 16 bits. This ensures that all program memory locations have a valid instruction.

5: If the Table Write starts the write cycle to internal memory, the write will continue until terminated.
Section 5. CPU and ALU

Table 5-1: PIC18CXXX Instruction Set (Continued)

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>MSb</td>
<td>LSb</td>
<td></td>
</tr>
<tr>
<td>LITERAL OPERATIONS</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ADDLW k</td>
<td>Add literal and WREG</td>
<td>1</td>
<td>0000</td>
<td>1111  kkkk kkkk</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>ANDLW k</td>
<td>AND literal with WREG</td>
<td>1</td>
<td>0000</td>
<td>1011  kkkk kkkk</td>
<td>Z, N</td>
</tr>
<tr>
<td>IORLW k</td>
<td>Inclusive OR literal with WREG</td>
<td>1</td>
<td>0000</td>
<td>1001  kkkk kkkk</td>
<td>Z, N</td>
</tr>
<tr>
<td>MOVLB k</td>
<td>Move literal to BSR&lt;3:0&gt;</td>
<td>1</td>
<td>0000</td>
<td>0001  0000  kkkk</td>
<td>None</td>
</tr>
<tr>
<td>LFSR f, k</td>
<td>Move literal (12-bit) to FSRx 2nd word</td>
<td>2</td>
<td>1110</td>
<td>1110  00ff  kkkk</td>
<td>None</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1111</td>
<td>0000  kkkk kkkk</td>
<td></td>
</tr>
<tr>
<td>MOVW k</td>
<td>Move literal to WREG</td>
<td>1</td>
<td>0000</td>
<td>1110  kkkk kkkk</td>
<td>None</td>
</tr>
<tr>
<td>MULLW k</td>
<td>Multiply literal with WREG</td>
<td>1</td>
<td>0000</td>
<td>1101  kkkk kkkk</td>
<td>None</td>
</tr>
<tr>
<td>RETLW k</td>
<td>Return with literal in WREG</td>
<td>2</td>
<td>0000</td>
<td>1100  kkkk kkkk</td>
<td>None</td>
</tr>
<tr>
<td>SUBLW k</td>
<td>Subtract WREG from literal</td>
<td>1</td>
<td>0000</td>
<td>1000  kkkk kkkk</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>XORLW k</td>
<td>Exclusive OR literal with WREG</td>
<td>1</td>
<td>0000</td>
<td>1010  kkkk kkkk</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: When an I/O register is modified as a function of itself (e.g., MOVF PORTB, 1, 0), the value used will be that value present on the pins themselves. For example, if the data latch is '1' for a pin configured as input and is driven low by an external device, the data will be written back with a '0'.

2: If this instruction is executed on the TMR0 register (and, where applicable, d = 1), the prescaler will be cleared if assigned to the Timer0 Module.

3: If Program Counter (PC) is modified or a conditional test is true, the instruction requires two cycles. The second cycle is executed as a NOP.

4: Some instructions are 2 word instructions. The second word of these instructions will be executed as a NOP unless the first word retrieves the information embedded in these 16 bits. This ensures that all program memory locations have a valid instruction.

5: If the Table Write starts the write cycle to internal memory, the write will continue until terminated.
5.2 General Instruction Format

The Enhanced family instructions can be broken down into five general formats as shown in Figure 5-1. As can be seen, the opcode for the instruction varies from 4 bits to 8 bits. This variable opcode size is what allows 77 instructions to be implemented.

Figure 5-1: General Format for Instructions

### Byte-oriented file register operations

15 10 9 8 7 0

<table>
<thead>
<tr>
<th>Opcode</th>
<th>d</th>
<th>a</th>
<th>f (FILE #)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>0</td>
<td>WREG</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>file register (f)</td>
</tr>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>Access Bank</td>
</tr>
</tbody>
</table>

- **Example Instruction**: ADDWF MYREG, W, a

### Byte to Byte move operations (2-word)

15 12 11 0

<table>
<thead>
<tr>
<th>Opcode</th>
<th>f (Source FILE #)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>MOVFF MYREG1, MYREG2</td>
</tr>
</tbody>
</table>

### Bit-oriented file register operations

15 12 11 9 8 7 0

<table>
<thead>
<tr>
<th>Opcode</th>
<th>b (BIT #)</th>
<th>a</th>
<th>f (FILE #)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>0</td>
<td>Access Bank</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>BSR to select bank</td>
</tr>
</tbody>
</table>

- **Example Instruction**: BSF MYREG, bit, a

### Literal operations

15 8 7 0

<table>
<thead>
<tr>
<th>Opcode</th>
<th>k (literal)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>MOVLW 0x7F</td>
</tr>
</tbody>
</table>

- **Example**: MOVLW 0x7F

### Control operations

CALL, GOTO and Branch operations

15 8 7 0

<table>
<thead>
<tr>
<th>Opcode</th>
<th>n&lt;7:0&gt; (literal)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>GOTO Label</td>
</tr>
<tr>
<td></td>
<td>CALL MYFUNC</td>
</tr>
<tr>
<td></td>
<td>BRA MYFUNC</td>
</tr>
<tr>
<td></td>
<td>BC MYFUNC</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Opcode</th>
<th>n&lt;19:8&gt; (literal)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>CALL MYFUNC</td>
</tr>
<tr>
<td></td>
<td>BRA MYFUNC</td>
</tr>
<tr>
<td></td>
<td>BC MYFUNC</td>
</tr>
</tbody>
</table>

- **Example**: GOTO Label
- **Example**: CALL MYFUNC
- **Example**: BRA MYFUNC
- **Example**: BC MYFUNC
Section 5. CPU and ALU

5.3 Central Processing Unit (CPU)

The CPU can be thought of as the “brains” of the device. It is responsible for fetching the correct instruction for execution, decoding that instruction and then executing that instruction.

The CPU sometimes works in conjunction with the ALU to complete the execution of the instruction (in arithmetic and logical operations).

The CPU controls the program memory address bus, the data memory address bus and accesses to the stack.
5.4 Instruction Clock

There are three oscillator clock sources from which the device can operate. These are:

1. External System Clock (TOSC)
2. Phase Lock Loop (PLL)
3. Timer1 Oscillator (TT1P)

Figure 5-2 shows these clock inputs and the device clock output (T SCLK). T SCLK is the system clock. Four T SCLK cycles are an instruction cycle (T CY).

The external system clock (TOSC) goes into the device and is input into a multiplexer and a 4 x Phase Lock Loop (PLL). The output of the PLL also enters the multiplexer and has a name TOSC/4. Some devices may also have an alternate oscillator called Timer1 oscillator (see “Timer1” section), which can provide another system clock. Timer1 has a cycle time called TT1P. This clock source also enters into the multiplexer.

Each instruction cycle (T CY) is comprised of four Q cycles (Q1-Q4). The Q cycle time is the same as the system clock cycle time (T SCLK). The Q cycles provide the timing/designation for the Decode, Read, Process Data, Write, etc., of each instruction cycle.

The four Q cycles that make up an instruction cycle (T CY) are shown in Figure 5-3. The relationship of the Q cycles to the instruction cycle can be generalized as:

- Q1: Instruction Decode Cycle or forced No Operation (NOP)
- Q2: Instruction Read Data Cycle or No Operation (NOP)
- Q3: Process the Data
- Q4: Instruction Write Data Cycle or No Operation (NOP)

Each instruction description will show a detailed Q cycle operation for the instruction.
5.5 Arithmetic Logical Unit (ALU)

PICmicro devices contain an 8-bit ALU and an 8-bit working register (WREG). The ALU is a general purpose arithmetic and logical unit. It performs arithmetic and Boolean functions between the data in the working register and any register file. The WREG register is directly addressable and in the SFR memory map.

Figure 5-4: Operation of the ALU and WREG Register

The ALU is 8-bits wide and is capable of addition, subtraction, multiplication, shift and logical operations. Unless otherwise mentioned, arithmetic operations are two's complement in nature. In two-operand instructions, typically one operand is the working register (WREG register). The other operand is a file register or an immediate constant. In single operand instructions, the operand is either the WREG register or a file register.

The 8x8 multiplier operates in a single cycle, placing the 16-bit result in the PRODH:PRODL register pair.

Depending on the instruction executed, the ALU may affect the values of the Carry (C), Digit Carry (DC), Zero (Z), Overflow (OV), and Negative (N) bits in the STATUS register. The C and DC bits operate as a borrow bit and a digit borrow out bit, respectively, in subtraction. See the SUBLW and SUBWF instructions in the “Instruction Set” section for examples.
Signed arithmetic is comprised of a magnitude and a sign bit. The overflow bit indicates if the magnitude overflows and causes the sign bit to change state when the result of an 8-bit signed operation is greater than 127 (7FH) or less than -128 (80h).

Signed math can have greater than 7-bit values (magnitude), if more than one byte is used. The overflow bit only operates on bit6 (MSb of magnitude) and bit7 (sign bit) of each byte value in the ALU. That is, the overflow bit is not useful if trying to implement signed math where the magnitude, for example, is 11 bits.

If the signed math values are greater than 7 bits (such as 15, 24 or 31 bits), the algorithm must ensure that the low order bytes of the signed value ignore the overflow status bit.

Example 5-1 shows two cases of doing signed arithmetic. The Carry (C) bit and the Overflow (OV) bit are the most important status bits for signed math operations.

Example 5-1: 8-bit Math Addition

**Case 1:**

<table>
<thead>
<tr>
<th>Hex Value</th>
<th>Signed Values</th>
<th>Unsigned Values</th>
</tr>
</thead>
<tbody>
<tr>
<td>FFh</td>
<td>0x00</td>
<td>255</td>
</tr>
<tr>
<td>+ 01h</td>
<td>0x01</td>
<td>0x01</td>
</tr>
<tr>
<td>= 00h</td>
<td>0x00 (FEx)</td>
<td>0x00</td>
</tr>
<tr>
<td>C bit = 1</td>
<td>C bit = 1</td>
<td>C bit = 1</td>
</tr>
<tr>
<td>OV bit = 0</td>
<td>OV bit = 0</td>
<td>OV bit = 0</td>
</tr>
<tr>
<td>DC bit = 1</td>
<td>DC bit = 1</td>
<td>DC bit = 1</td>
</tr>
<tr>
<td>Z bit = 1</td>
<td>Z bit = 1</td>
<td>Z bit = 1</td>
</tr>
<tr>
<td>N bit = 0</td>
<td>N bit = 0</td>
<td>N bit = 0</td>
</tr>
</tbody>
</table>

**Case 2:**

<table>
<thead>
<tr>
<th>Hex Value</th>
<th>Signed Values</th>
<th>Unsigned Values</th>
</tr>
</thead>
<tbody>
<tr>
<td>7Fh</td>
<td>127</td>
<td>127</td>
</tr>
<tr>
<td>+ 01h</td>
<td>128</td>
<td>0x00</td>
</tr>
<tr>
<td>= 80h</td>
<td>128</td>
<td>0x80</td>
</tr>
<tr>
<td>C bit = 0</td>
<td>C bit = 0</td>
<td>C bit = 0</td>
</tr>
<tr>
<td>OV bit = 1</td>
<td>OV bit = 1</td>
<td>OV bit = 1</td>
</tr>
<tr>
<td>DC bit = 1</td>
<td>DC bit = 1</td>
<td>DC bit = 1</td>
</tr>
<tr>
<td>Z bit = 0</td>
<td>Z bit = 0</td>
<td>Z bit = 0</td>
</tr>
<tr>
<td>N bit = 1</td>
<td>N bit = 1</td>
<td>N bit = 1</td>
</tr>
</tbody>
</table>

The Negative bit is used to indicate if the MSb of the result is set or cleared.
Section 5. CPU and ALU

5.6 STATUS Register

The STATUS register, shown in Register 5-1, contains the arithmetic status of the ALU. 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, C, OV or N bits, then the write to these five bits is disabled. These bits are set or cleared according to the device logic. 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 0001uu (where u = unchanged).

It is recommended, therefore, that only BCF, BSF, SWAPF, MOVFF, and MOVWF instructions are used to alter the STATUS register, because these instructions do not affect the Z, C, DC, OV or N bits of the STATUS register. For other instructions, not affecting any status bits, see Table 5-1.

Note 1: The C and DC bits operate as a borrow and digit borrow bit, respectively, in subtraction.
## Register 5-1: STATUS Register

| bit 7-5 | Unimplemented: Read as '0' |
| bit 4  | N: Negative bit |
|        | This bit is used for signed arithmetic (2's complement). It indicates whether the result was negative. (ALU MSb = 1). |
|        | 1 = Result was negative |
|        | 0 = Result was positive |
| bit 3  | OV: Overflow bit |
|        | This bit is used for signed arithmetic (2's complement). It indicates an overflow of the 7-bit magnitude, which causes the sign bit (bit7) to change state. |
|        | 1 = Overflow occurred for signed arithmetic (in this arithmetic operation) |
|        | 0 = No overflow occurred |
| 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 |
|        | For ADDWF, ADDLW, SUBLW, and SUBWF instructions |
|        | 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 |
|        | **Note:** For borrow, the polarity is reversed. A subtraction is executed by adding the 2's complement of the second operand. For rotate (RRF, RLF) instructions, this bit is loaded with either the bit4 or bit3 of the source register. |
| bit 0  | C: Carry/Borrow bit |
|        | For ADDWF, ADDLW, SUBLW, and SUBWF instructions |
|        | 1 = A carry-out from the most significant bit of the result occurred |
|        | 0 = 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 2'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 reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown

| bit 7-5 | Unimplemented: Read as '0' |
| bit 4  | N: Negative bit |
|        | This bit is used for signed arithmetic (2's complement). It indicates whether the result was negative. (ALU MSb = 1). |
|        | 1 = Result was negative |
|        | 0 = Result was positive |
| bit 3  | OV: Overflow bit |
|        | This bit is used for signed arithmetic (2's complement). It indicates an overflow of the 7-bit magnitude, which causes the sign bit (bit7) to change state. |
|        | 1 = Overflow occurred for signed arithmetic (in this arithmetic operation) |
|        | 0 = No overflow occurred |
| 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 |
|        | For ADDWF, ADDLW, SUBLW, and SUBWF instructions |
|        | 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 |
|        | **Note:** For borrow, the polarity is reversed. A subtraction is executed by adding the 2's complement of the second operand. For rotate (RRF, RLF) instructions, this bit is loaded with either the bit4 or bit3 of the source register. |
| bit 0  | C: Carry/Borrow bit |
|        | For ADDWF, ADDLW, SUBLW, and SUBWF instructions |
|        | 1 = A carry-out from the most significant bit of the result occurred |
|        | 0 = 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 2'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. |
## Section 5. CPU and ALU

### 5.6.1 RCON Register

The Reset Control (RCON) register contains flag bit(s) that allow the user to differentiate between the device resets.

**Note 1:** If the BOREN configuration bit is set, BOR is '0' on Power-on Reset. If the BOREN configuration bit is clear, BOR is unknown on Power-on Reset.

The BOR status bit is a "don’t care" and is not necessarily predictable if the brown-out circuit is disabled (the BOREN configuration bit is clear).

**Note 2:** It is recommended that the POR bit be set after a Power-on Reset has been detected, so that subsequent Power-on Resets may be detected.

#### Register 5-2: RCON Register

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>IPEN</td>
<td>LWRT</td>
</tr>
</tbody>
</table>

- **IPEN:** Interrupt Priority Enable bit
  - 0: Disable priority levels on interrupts (PIC16CXXX compatibility mode)
  - 1: Enable priority levels on interrupts

- **LWRT:** Long Write Enable
  - 0: Disable Table Writes to internal program memory; Table Writes only to external program memory
  - 1: Enable Table Writes to internal program memory

- **RI:** Reset Instruction Flag bit
  - 0: The Reset instruction was executed (must be set in software after a Brown-out Reset occurs)
  - 1: The Reset instruction was not executed to cause the device reset

- **TO:** Watchdog Time-out Flag bit
  - 0: A WDT time-out occurred
  - 1: After Power-up, CLRWDT instruction, or SLEEP instruction

- **PD:** Power-down Detection Flag bit
  - 0: By execution of the SLEEP instruction
  - 1: After Power-up or by the CLRWDT instruction

- **POR:** Power-on Reset Status bit
  - 0: A Power-on Reset occurred
  - 1: A Power-on Reset has not occurred

- **BOR:** Brown-out Reset Status bit
  - 0: A Brown-out Reset occurred
  - 1: A Brown-out Reset has not occurred

**Legend**

- **R:** Readable bit
- **W:** Writable bit
- **U:** Unimplemented bit, read as '0'
- **n:** Value at POR reset
- **'1':** bit is set
- **'0':** bit is cleared
- **x:** bit is unknown
5.7 Design Tips

Question 1: My program algorithm does not seem to function correctly.

Answer 1:

There are many possible reasons for this. A couple of possibilities are:

1. The destination of the instruction may be specifying the WREG register (d = 0) instead of the file register (d = 1).
2. The access bit may be specifying the Virtual RAM Bank instead of the desired bank of RAM.

When possible, the use of an In-Circuit Emulator (such as MPLAB-ICE) or a simulator (such as MPLAB-SIM) can assist in locating the reason for the unexpected execution flow.

Question 2: I cannot seem to modify the STATUS register flags.

Answer 2:

If the STATUS register is the destination for an instruction that affects the Z, DC, C, OV or N bits, then the write to those bits is disabled. These bits are set or cleared based on device logic. Therefore, to modify bits in the STATUS register, it is recommended to use the BCF and BSF instructions.
5.8 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent and could be used (with modification and possible limitations). The current application notes related to the CPU or the ALU are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>IEEE 754 Compliant Floating Point Routines</td>
<td>AN575</td>
</tr>
<tr>
<td>Fixed Point Routines</td>
<td>AN617</td>
</tr>
<tr>
<td>Floating Point Math Functions</td>
<td>AN660</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
5.9 Revision History

Revision A
This is the initial released revision of the Enhanced MCU CPU and ALU description.
Section 6. Hardware 8x8 Multiplier

HIGHLIGHTS

This section of the manual contains the following major topics:

6.1 Introduction ................................................................. 6-2
6.2 Operation ................................................................. 6-3
6.3 Design Tips ............................................................... 6-6
6.4 Related Application Notes .............................................. 6-7
6.5 Revision History ......................................................... 6-8
6.1 Introduction

An 8 x 8 hardware multiplier is included in the ALU of the devices. By making the multiplication a hardware operation, it completes in a single instruction cycle. This is an unsigned multiplication that gives a 16-bit result. The result is stored into the 16-bit Product register (PRODH:PRODL). The multiplier does not affect any flags in the ALUST A register.

Making the 8 x 8 multiplier execute in a single cycle gives the following advantages:

- Higher computational throughput
- Reduces code size requirements for multiplication algorithms

The performance increase allows the device to be used in applications previously reserved for Digital Signal Processors.

Table 6-1 shows a performance comparison between devices using the single cycle hardware multiplier and performing the same function without the hardware multiplier.

<table>
<thead>
<tr>
<th>Routine</th>
<th>Multiply Method</th>
<th>Program Memory (Words)</th>
<th>Cycles (Max)</th>
<th>Time @ 40 MHz</th>
<th>Time @ 10 MHz</th>
<th>Time @ 4 MHz</th>
</tr>
</thead>
<tbody>
<tr>
<td>8 x 8 unsigned</td>
<td>Without hardware multiplier</td>
<td>13</td>
<td>69</td>
<td>6.9 µs</td>
<td>27.6 µs</td>
<td>69 µs</td>
</tr>
<tr>
<td></td>
<td>Hardware multiply</td>
<td>1</td>
<td>1</td>
<td>100 ns</td>
<td>400 ns</td>
<td>1 µs</td>
</tr>
<tr>
<td>8 x 8 signed</td>
<td>Without hardware multiplier</td>
<td>33</td>
<td>91</td>
<td>9.1 µs</td>
<td>36.4 µs</td>
<td>91 µs</td>
</tr>
<tr>
<td></td>
<td>Hardware multiply</td>
<td>6</td>
<td>6</td>
<td>600 ns</td>
<td>2.4 µs</td>
<td>6 µs</td>
</tr>
<tr>
<td>16 x 16 unsigned</td>
<td>Without hardware multiplier</td>
<td>21</td>
<td>242</td>
<td>24.2 µs</td>
<td>96.8 µs</td>
<td>242 µs</td>
</tr>
<tr>
<td></td>
<td>Hardware multiply</td>
<td>24</td>
<td>24</td>
<td>2.4 µs</td>
<td>9.6 µs</td>
<td>24 µs</td>
</tr>
<tr>
<td>16 x 16 signed</td>
<td>Without hardware multiplier</td>
<td>52</td>
<td>254</td>
<td>25.4 µs</td>
<td>102.6 µs</td>
<td>254 µs</td>
</tr>
<tr>
<td></td>
<td>Hardware multiply</td>
<td>36</td>
<td>36</td>
<td>3.6 µs</td>
<td>14.4 µs</td>
<td>36 µs</td>
</tr>
</tbody>
</table>
6.2 Operation

Example 6-1 shows the sequence to do an 8 x 8 unsigned multiply. Only one instruction is required when one argument of the multiply is already loaded in the WREG register.

Example 6-2 shows the sequence to do an 8 x 8 signed multiply. To account for the sign bits of the arguments, each argument’s most significant bit (MSb) is tested and the appropriate subtractions are done.

Example 6-1: 8 x 8 Unsigned Multiply Routine

```
MOVFF ARG1, WREG ;
MULWF ARG2     ; ARG1 * ARG2 ->
                 ; PRODH:PRODL
```

Example 6-2: 8 x 8 Signed Multiply Routine

```
MOVFF ARG1, WREG
MULWF ARG2     ; ARG1 * ARG2 ->
                 ; PRODH:PRODL
BTFSC ARG2, SB ; Test Sign Bit
SUBWF PRODH, F ; PRODH = PRODH
                 ; - ARG1
MOVFF ARG2, WREG
BTFSC ARG1, SB ; Test Sign Bit
SUBWF PRODH, F ; PRODH = PRODH
                 ; - ARG2
```

Example 6-3 shows the sequence to do a 16 x 16 unsigned multiply. Equation 6-1 shows the algorithm that is used. The 32-bit result is stored in four registers, RES3, RES2, RES1 and RES0.

Equation 6-1: 16 x 16 Unsigned Multiplication Algorithm

```
RES3:RES2:RES1:RES0 = ARG1H:ARG1L • ARG2H:ARG2L
                   = (ARG1H • ARG2H • 2^16) +
                     (ARG1H • ARG2L • 2^8) +
                     (ARG1L • ARG2H • 2^8) +
                     (ARG1L • ARG2L)
```
Example 6-3: 16 x 16 Unsigned Multiply Routine

Example 6-4 shows the sequence to do a 16 x 16 signed multiply. Equation 6-2 shows the algorithm used. The 32-bit result is stored in four registers, RES3, RES2, RES1 and RES0. To account for the sign bits of the arguments, each argument pairs’ most significant bit (MSb) is tested and the appropriate subtractions are done.

Equation 6-2: 16 x 16 Signed Multiplication Algorithm

\[
\begin{align*}
\text{RES3:RES2:RES1:RES0} &= \text{ARG1H:ARG1L} \cdot \text{ARG2H:ARG2L} \\
&= (\text{ARG1H} \cdot \text{ARG2H} \cdot 2^{16}) + \\
&\quad (\text{ARG1H} \cdot \text{ARG2L} \cdot 2^8) + \\
&\quad (\text{ARG1L} \cdot \text{ARG2H} \cdot 2^8) + \\
&\quad (-1 \cdot \text{ARG2H<7>} \cdot \text{ARG1H:ARG1L} \cdot 2^{16}) + \\
&\quad (-1 \cdot \text{ARG1H<7>} \cdot \text{ARG2H:ARG2L} \cdot 2^{16})
\end{align*}
\]
Section 6. Hardware 8x8 Multiplier

Example 6-4: 16 x 16 Signed Multiply Routine

```
; MOVFF ARG1L, WREG
MULWF ARG2L ; ARG1L * ARG2L -> ; PRODH:PRODL
; MOVFF PRODH, RES1 ;
MOVFF PRODL, RES0 ;
; MOVFF ARG1H, WREG
MULWF ARG2H ; ARG1H * ARG2H -> ; PRODH:PRODL
; MOVFF PRODH, RES3 ;
MOVFF PRODL, RES2 ;
; MOVFF ARG1L, WREG
MULWF ARG2H ; ARG1L * ARG2H -> ; PRODH:PRODL
; MOVFF PRODL, WREG ;
ADDWF RES1, F ; Add cross
MOVFF PRODH, WREG ; products
ADDWF RES2, F ;
CLRWF WREG, F ;
ADDWF RES3, F ;
;
MOVFF ARG1H, WREG ;
MULWF ARG2L ; ARG1H * ARG2L -> ; PRODH:PRODL
MOVEF PRODH, RES1 ;
ADDWF RES1, F ; Add cross
MOVEF PRODL, RES0 ;
ADDWF RES2, F ;
CLRWF WREG, F ;
ADDWF RES3, F ;
;
BTFSS ARG2H, 7 ; ARG2H:ARG2L neg?
GOTO SIGN_ARG1 ; no, check ARG1
MOVEF ARG1L, WREG ;
SUBWF RES2 ;
MOVEF ARG1H, WREG ;
SUBWFB RES3 ;
;
SIGN_ARG1
BTFSS ARG1H, 7 ; ARG1H:ARG1L neg?
GOTO CONT_CODE ; no, done
MOVEF ARG2L, WREG ;
SUBWF RES2 ;
MOVEF ARG2H, WREG ;
SUBWFB RES3 ;
;
CONT_CODE
;```

6.3 Design Tips

None.
Section 6. Hardware 8x8 Multiplier

6.4 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the H/W Multiplier modules are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>IEEE 754 Compliant Floating Point Routines</td>
<td>AN575</td>
</tr>
<tr>
<td>Fixed Point Routines</td>
<td>AN617</td>
</tr>
<tr>
<td>Floating Point Math Functions</td>
<td>AN660</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
6.5 Revision History

Revision A

This is the initial released revision of the Enhanced MCE Hardware Multiplier module description.
Section 7. Memory Organization

HIGHLIGHTS

This section of the manual contains the following major topics:

7.1 Introduction ................................................................. 7-2
7.2 Program Memory .......................................................... 7-3
7.3 Program Counter (PC) .................................................... 7-6
7.4 Lookup Tables .............................................................. 7-9
7.5 Stack ............................................................................. 7-12
7.6 Data Memory Organization ........................................... 7-13
7.7 Return Address Stack ..................................................... 7-17
7.8 Initialization ................................................................. 7-23
7.9 Design Tips ................................................................. 7-24
7.10 Related Application Notes ............................................. 7-25
7.11 Revision History ......................................................... 7-26
7.1 Introduction

There are two memory blocks in the memory map: program memory and data memory. Each block has its own bus, so that access to each block can occur during the same instruction cycle. The data memory can further be broken down into General Purpose RAM and the Special Function Registers (SFRs). The operation of the SFRs that control the “core” are described here. The SFRs used to control the peripheral modules are described in the section discussing each individual peripheral module.

In addition, there are other registers used that are neither part of the program nor data memory spaces. These registers are not directly addressable and include:

- return address stack
- fast return stack

Table 7-1 shows the program memory space used depending on the memory allocated, and Table 7-2 shows the data memory space used.

<table>
<thead>
<tr>
<th>Program Memory</th>
<th>Program Memory Address Range</th>
<th>Data Memory</th>
<th>Banks</th>
</tr>
</thead>
<tbody>
<tr>
<td>1K x 8</td>
<td>0000h – 3FFh</td>
<td>64</td>
<td>0, 15</td>
</tr>
<tr>
<td>2K x 8</td>
<td>0000h – 7FFh</td>
<td>128</td>
<td>0, 15</td>
</tr>
<tr>
<td>4K x 8</td>
<td>0000h – FFFh</td>
<td>256</td>
<td>0, 15</td>
</tr>
<tr>
<td>8K x 8</td>
<td>0000h – 1FFFh</td>
<td>512</td>
<td>0-1, 15</td>
</tr>
<tr>
<td>12K x 8</td>
<td>0000h – 2FFFFh</td>
<td>640</td>
<td>0-2, 15</td>
</tr>
<tr>
<td>16K x 8</td>
<td>0000h – 3FFFFh</td>
<td>768</td>
<td>0-2, 15</td>
</tr>
<tr>
<td>24K x 8</td>
<td>0000h – 5FFFFh</td>
<td>1024</td>
<td>0-3, 15</td>
</tr>
<tr>
<td>32K x 8</td>
<td>0000h – 7FFFFh</td>
<td>1280</td>
<td>0-4, 15</td>
</tr>
<tr>
<td>48K x 8</td>
<td>0000h – BFFFFh</td>
<td>1536</td>
<td>0-5, 15</td>
</tr>
<tr>
<td>64K x 8</td>
<td>0000h – FFFFFh</td>
<td>1792</td>
<td>0-6, 15</td>
</tr>
<tr>
<td>96K x 8</td>
<td>0000h – 17FFFh</td>
<td>2048</td>
<td>0-7, 15</td>
</tr>
<tr>
<td>128K x 8</td>
<td>0000h – 1FFFFFh</td>
<td>2304</td>
<td>0-8, 15</td>
</tr>
<tr>
<td>160K x 8</td>
<td>0000h – 27FFFh</td>
<td>2560</td>
<td>0-9, 15</td>
</tr>
<tr>
<td>192K x 8</td>
<td>0000h – 2FFFFFh</td>
<td>2816</td>
<td>0-10, 15</td>
</tr>
<tr>
<td>256K x 8</td>
<td>0000h – 3FFFFFh</td>
<td>3072</td>
<td>0-11, 15</td>
</tr>
<tr>
<td>384K x 8</td>
<td>0000h – 5FFFFFh</td>
<td>3328</td>
<td>0-12, 15</td>
</tr>
<tr>
<td>512K x 8</td>
<td>0000h – 7FFFFFh</td>
<td>3584</td>
<td>0-13, 15</td>
</tr>
<tr>
<td>768K x 8</td>
<td>0000h – BFFFFFh</td>
<td>3840</td>
<td>0-14, 15</td>
</tr>
<tr>
<td>1024K x 8</td>
<td>0000h – FFFFFFh</td>
<td>3968</td>
<td>0-15</td>
</tr>
<tr>
<td>1536K x 8</td>
<td>0000h – 17FFFFFh</td>
<td></td>
<td></td>
</tr>
<tr>
<td>2048K x 8</td>
<td>0000h – 1FFFFFh</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
7.2 Program Memory

Enhanced MCU devices have a 21-bit program counter capable of addressing 2 Mbytes (1Mwords) of program memory space. The program memory contains instructions for execution and data tables for storing fixed data. Data tables may be written once using table write instructions and read as required, using the table read instructions.

The program space is implemented as a single contiguous block. The reset vector is at address 000000h, the high priority interrupt vector is at address 000008h, and the low priority interrupt vector is at address 000018h (Figure 7-1).

CALL and GOTO instructions can address any location in the memory map, while the BRA and RCALL instructions have a limited program memory reach (+1024, -1023 program memory word locations). To allow the CALL and GOTO instructions to contain the entire address, it requires that these instructions use 2 program memory words (2 word instruction).

Instructions are also available to move information between the data memory and the program memory areas. These are called table operations. Table operations work with byte entities. This is discussed in detail in the “Table Read/Table Write” section.

Figure 7-1: Program Memory Map and Stack for PIC18CXXX

- CALL, BSUB, RETURN, RETFIE, RETLW
- Stack Level 1
- Stack Level 31
- Reset Vector 0000h
- High Priority Interrupt Vector 0008h
- Low Priority Interrupt Vector 0018h
- On-chip Program Memory
- External/Unimplemented Program Memory (Read as ‘0’ in microcontroller mode)
- User Memory Space
- PC<20>
7.2.1 Reset Vector

On any Enhanced MCU device, a reset forces the Program Counter (PC) to address 0h. This is known as the "Reset Vector Address", since this is the address that program execution will branch to when a device reset occurs.

Any reset will also clear the contents of the PCLATU and PCLATH registers.

7.2.2 Interrupt Vectors

Two interrupt vectors are implemented; one for interrupts programmed as high priority and the other for the interrupts programmed as low priority. The vector addresses are 08h for high priority interrupts and 18h for low priority interrupts. If the interrupt priority is not used, all interrupts are treated as high priority.

When an interrupt is acknowledged, the PC is forced to address 0008h or 0018h. This is known as the “Interrupt Vector Address”. When the PC is forced to the interrupt vector, the PCLATU and PCLATH registers are not modified. Once in the service interrupt routine (ISR), before any write to the PC, the PCLATH register should be written with the value that will specify the desired location in program memory. Before the PCLATH register is modified by the Interrupt Service Routine (ISR), the contents of the PCLATH may need to be saved so it can be restored before returning from the ISR.

7.2.3 Calibration Information

Some devices have calibration information stored in their program memory. This information is programmed by Microchip when the device is under final test. The use of these values allows the application to achieve better results. The calibration information is typically at the end of program memory. These bytes can be accessed with the table read instructions.

Note: For windowed devices, write down all calibration values BEFORE erasing. This allows the device’s calibration values to be restored when the device is re-programmed. When possible, writing the values on the package is recommended.
Section 7. Memory

7.2.4 Instructions in Program Memory

The program memory is addressed in bytes. Instructions are stored as two bytes or four bytes in program memory. The least significant byte of an instruction word is always stored in a program memory location with an even address (LSb = '0'). Figure 7-2 shows an example of how instruction words are stored in the program memory. To maintain alignment with instruction boundaries, the PC increments in steps of 2 and the LSb will always read '0'.

The CALL and GOTO instructions have an absolute program memory address embedded into the instruction. Since instructions are always stored on word boundaries, the data contained in the instruction is a word address. The word address is written to PC<20:1>, which accesses the desired byte address in program memory. Instruction #2 in Figure 7-2 shows how the instruction "GOTO 000006h" is encoded in the program memory. Program branch instructions which encode a relative address offset operate in the same manner. The offset value stored in a branch instruction represents the number of single word instructions that the PC will be offset by. The "Instruction Set" section provides further details of the instruction set.

Figure 7-2: Instructions in Program Memory

<table>
<thead>
<tr>
<th>Program Memory Byte Locations</th>
<th>Word Address</th>
</tr>
</thead>
<tbody>
<tr>
<td>High Byte</td>
<td>Low Byte</td>
</tr>
<tr>
<td>000000h</td>
<td></td>
</tr>
<tr>
<td>000002h</td>
<td></td>
</tr>
<tr>
<td>000004h</td>
<td></td>
</tr>
<tr>
<td>000006h</td>
<td></td>
</tr>
</tbody>
</table>

Instruction 1: MOVlw 055h

<table>
<thead>
<tr>
<th>0Fh</th>
<th>55h</th>
<th>000008h</th>
</tr>
</thead>
</table>

Instruction 2: GOTO 000006h

<table>
<thead>
<tr>
<th>EFh</th>
<th>03h</th>
<th>00000Ah</th>
</tr>
</thead>
<tbody>
<tr>
<td>F0h</td>
<td>00h</td>
<td>00000Ch</td>
</tr>
</tbody>
</table>

Instruction 3: MOVFF 123h, 456h

<table>
<thead>
<tr>
<th>C1h</th>
<th>23h</th>
<th>00000Eh</th>
</tr>
</thead>
<tbody>
<tr>
<td>F4h</td>
<td>56h</td>
<td>000010h</td>
</tr>
<tr>
<td></td>
<td></td>
<td>000012h</td>
</tr>
<tr>
<td></td>
<td></td>
<td>000014h</td>
</tr>
</tbody>
</table>
7.3 Program Counter (PC)

The Program Counter (PC) specifies the address of the instruction to fetch for execution. The PC is 21-bits wide and addresses each byte (rather than words) in the program memory. The low byte is called the PCL register (PC<7:0>). This register is readable and writable. The high byte is called the PCH register (PC<15:8>). This register is not directly readable or writable. Updates to the PCH register may be performed through the PCLATH register.

The upper byte is called the PCU register (PC<20:16>). The PCU register is not directly readable or writable. Updates to the PCU register may be performed through the PCLATU register. The PC structure is PCU<4:0>:PCH<7:0>:PCL<7:0> and is equivalent to PC<20:0>.

Figure 7-3 shows the interaction of the PCU, PCH, and PCL registers with the PCLATU and PCLATH registers.

Figure 7-3: Program Counter Structure

![Program Counter Structure Diagram]

The low byte of the PC (PCL<7:0>) is mapped in the data memory. PCL is readable and writable just as is any other register. PCU and PCH are the upper and high bytes of the PC respectively, and are not directly addressable. Registers PCLATU<4:0> (PC upper latch) and PCLATH<7:0> (PC high latch) are used as holding latches for the high bytes of the PCU and PCH, and are mapped into data memory. The user can read and write PCH through PCLATH and PCU through PCLATU. Any time PCL is read, the current contents of PCH and PCU are transferred to PCLATH and PCLATU, respectively. Any time PCL is written to, the contents of PCLATH and PCLATU are transferred to PCH and PCU, respectively.

Note: The values in PCLATU and PCLATH do not always reflect the current value in PCU and PCH. When needing to modify the current Program Counter (PC) value, first read the PCL register to update the values in the PCLATU and PCLATH registers.

The PC addresses bytes rather than words in the program memory. Because the PC must access the instructions in program memory on an even byte boundary, the LSB of the PC is a forced '0' and the PC increments by two for each instruction. The LSB bit of the PCL is readable but not writable. Any write to the LSB is ignored.

Figure 7-4 shows the four situations for the loading of the PC. Situation 1 shows how the PC is loaded on a write to PCL (PCLATH<4:0> → PCH). Situation 2 shows how the PC is loaded during a GOTO instruction (PCLATH<4:3> → PCH). Situation 4 shows how the PC is loaded during a CALL instruction (PCLATH<4:3> → PCH), with the PC loaded (PUSHed) onto the Top of Stack. Situation 6 shows how the PC is loaded during one of the return instructions where the PC is loaded (POPed) from the Top of Stack.
Section 7. Memory

Figure 7-4: Loading of PC In Different Situations

Situation 1 - Instruction with PCL as destination

STACK (21-bits x 31)

Top of STACK

Situation 2 - GOTO Instruction

STACK (21-bits x 31)

Top of STACK

Situation 3 - BRA Instruction in Conditional Branch Instruction

STACK (21-bits x 31)

Top of STACK

Situation 4 - CALL Instruction

STACK (21-bits x 31)

Top of STACK

Note: PCLATU and PCLATH are not updated with the contents of PCH.
Figure 7-4: Loading of PC in Different Situations (Continued)

**Situation 5 - RCALL Instruction**

Offset from Instruction

**Situation 6 - RETURN, RETFIE, or RETLW Instruction**

Note: PCLATH and PCLATU are not updated with the contents of PCH.
Section 7. Memory

7.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) and the PCH memory boundary (each 64Kbyte block).

A lookup table can be formed with an ADDWF PCL instruction and a group of RETLW 0xnn instructions. WREG is loaded with an offset into the table before executing a call to that table. The first instruction of the called routine is the ADDWF PCL instruction. The next instruction executed will be one of the RETLW 0xnn instructions that returns the value 0xnn to the calling function.

Since the Program Counter is a byte counter (instead of a word counter), adds to the PCL allow a table size of 128 entries before the PCLA TH needs to be modified.

Note: Since the Program Counter is 21-bits, the uppercase PCLATU register may also need to be modified when doing computed gotos.

In this method of storing tables in PIC18CXXX devices, only one data value may be stored in each instruction location, and room on the return address stack is required. A better method of storing data in program memory is through the use of table reads and writes. Two bytes of data can now be stored in each instruction location.

Note: Any write to the Program Counter (PCL) will cause the contents of PCLATU and PCLATH to be loaded into PCU and PCH, respectively.

7.4 Lookup Tables

Look-up tables instructions are implemented two ways in the PIC18CXXX devices. The computed goto is compatible with the PIC16CXXX and PIC17CXXX parts. Code written for those devices will run on the PIC18CXXX devices with minor modifications.

Table read instructions are implemented on the PIC17CXXX and PIC18CXXX devices. However, table operations on the PIC18CXXX work differently than on the PIC17CXXX.

7.4.1 Table Reads/Table Writes

Lookup table data may be stored 2 bytes per program word. By using TBLPTR and TABLAT, data may be retrieved from program memory one byte at a time as required.

Table writes to program memory can be executed as many times as desired. Remember that the technology of the program memory determines the outcome of the table write. Table writes to EPROM memory allow the program memory cell to go from a '1' state to a '0' state, but not the other direction. FLASH memory allows the cell to go from a '1' to a '0' and a '0' to a '1' (though typically a program memory word or block location is always written).
### Example 7-1: PIC18CXXX Table Lookup

```assembly
MOVLW BYTE_COUNT    ; Load the Byte Count value
MOVWF CNTR          ; into CNTR

;; MOVLW UPPER(TBL_ADDR) ; Load the Table Address
;; MOVWF TBLPTRH       ; (on POR TBLPTRU = 0, so
;; ; ; ; loading TBLPTRU is not
;; ;; ; ; required for conversions)

MOVLW HIGH(TBL_ADDR) ; Load the Table Address
MOVWF TBLPTRH       
MOVWF LOW(TBL_ADDR) 
MOVWF TBLPTRL       
LOOP1 TBLRD*+       ; Read value into TABLAT,
                     ; Increment TBLPTR
MOVFF TABLAT, POSTINC0 ; Copy byte to RAM @ FSR0
                     ; Increment FSR0
DECFSZ CNTR          ; Read Byte Count locations
GOTO LOOP1           ; Read next Byte
```

### Example 7-2: PIC17CXXX Table Lookup

```assembly
MOVLW WORD_COUNT     ; Load the Word Count value
MOVWF CNTR           ; into CNTR

MOVLW HIGH(TBL_ADDR) ; Load the Table Address
MOVWF TBLPTRH       
MOVWF LOW(TBL_ADDR) 
MOVWF TBLPTRL       
TABLRD 0, 1, DUMMY   ; Dummy read,
                     ; Update TABLATH
                     ; Increments TBLPTR
LOOP1 TLRD 1, INDF0  ; Read HI byte in TABLATH
TABLRD 0, 1, INDF0   ; Read LO byte in TABLATL,
                     ; Update TABLATH:TABLATL,
                     ; and increment TBLPTR
DECFSZ CNTR          ; Read Word Count locations
GOTO LOOP1           ; Read next word
```
Example 7-3: PIC16CXXX Table Lookup

```
CLRF CNTR ;
TABLELP

MOVF CNTR, W ; Place value in
             ; WREG register
CALL TABLE1
MOVF INDF
INCF FSR
INCF CNTR
BTFSS CNTR, 3 ; CNTR = 00001000b?
GOTO TABLE_LP

TABLE1

ADDWF PCL ; Ensure that table does
           ; not cross 256 byte
           ; page boundary.

RETLW 'G'
RETLW 'O'
RETLW ' ' 
RETLW 'M'
RETLW 'H'
RETLW '!' 
RETLW 'P'
RETLW '!' 
```
7.5 Stack

The stack allows a combination of up to 31 program calls and interrupts to occur. The stack contains the return address from this branch in program execution.

Enhanced MCU devices have an 31-level deep x 21-bit wide hardware stack. The stack space is not part of either program or data space and the stack pointer is not readable nor writable. The PC is PUSHed onto the stack when a CALL instruction is executed or an interrupt causes a branch. The stack is POPed in the event of a RETURN, RETLW or a RETFIE instruction execution. PCLATH is not modified when the stack is PUSHed or POPed.

After the PC is PUSHed onto the stack 31 times (without POPing any values off the stack), the 32nd PUSH over-writes the value from the 31st PUSH and sets the STKFUL bit while the STKPTR remains at 11111b. The 33rd PUSH overwrites the 32nd PUSH (and so on) while STKPTR remains 11111b. When the stack overflow enable bit is enabled a device reset will occur.

Figure 7-5: Stack Modification

Whenever the program branches, the return address is saved to the stack. Such branches include CALL, RCALL, or an interrupt. The stack pointer is incremented and PC<20:1> is PUSHed onto the return stack. PC<0> is always assumed to be 0. When a branch return is executed, the top of the stack is POPed to the Program Counter and the stack pointer is decremented. PCLATU and PCLATH are not affected during these branches.

The PC is word incremented by 2 after each instruction fetch during Q1 unless:

- Modified by a GOTO, CALL, RCALL, RETURN, RETLW, RETFIE, or branch instruction
- Modified by an interrupt response
- Due to a write to PCL by an instruction

Skips are equivalent to a forced NOP cycle at the skipped address.

Note1: The stack pointer value does not increment past 11111b.
Section 7. Memory

7.6 Data Memory Organization

Data memory is made up of the Special Function Registers (SFR) area and the General Purpose Registers (GPR) area. The SFRs are used for control and status of the microcontroller and peripheral functions, while GPRs are the general area for user data storage and scratch pad operations.

Each register has a 12-bit address. This allows up to 4096 bytes of data memory. This memory is partitioned into 16 banks of 256 bytes that contain the General Purpose Registers (GPRs) and Special Function Registers (SFRs).

The data memory can be banked for both the GPR and SFR areas. Banking requires the use of BSR Register. Figure 7-6 shows the data memory map organizations, while Table 7-2 shows which banks will be used depending on the memory size of the devices.

SFRs start at the last location of Bank 15 (0xFF) and work up. Once the SFR space ends, any lower locations in that bank may be implemented as GPRs. GPRs start at the first location of Bank 0 (0h) and work down. Any read of an unimplemented location will read as '0's.

The Instruction set and architecture allows operations across all banks. To move values from one register to another register, the MOVFF instruction can be used. This is a two word / two cycle instruction.

The entire data memory can be accessed either directly or indirectly. Direct addressing may require the use of the BSR register. Indirect addressing requires the use of the File Select Registers (FSRs). Each FSR holds a 12-bit value that can access any location in the Data Memory map.

To ensure that commonly used registers (SFRs and select GPRs) can be accessed in a single cycle, regardless of the current BSR values, an Access Bank is implemented. This is explained in Section 7.6.1.

The GPR area is banked to allow greater than 256 bytes of general purpose RAM to be addressed. SFRs are for the registers that control the peripheral and core functions.

Figure 7-6: The Data Memory Map and the Access Bank
7.6.1 Access Bank

The Access Bank is an architectural enhancement that is very useful for C compiler code optimization. The techniques used by the C compiler may also be useful for programs written in assembly.

This data memory region can be used for:

- Intermediate computational values
- Local variables of subroutines
- Faster context saving/switching of variables
- Common variables
- Faster evaluation/control of SFRs (no banking)

The Access Bank is comprised of the upper portion of Bank 15 (SFRs) and the lower portion of Bank 0 (GPR). These two sections will be referred to as Access RAM High and Access RAM Low, respectively. Figure 7-6 indicates the Access RAM areas. The actual size of memory used from Bank 0 and Bank 15 depends on the specific device. When appropriate, devices will use 128 bytes from Bank 0 (GPR) and 128 bytes from Bank 15 (SFR). In larger devices with more SFRs, the GPR Access bank size may be reduced to allocate that space to SFRs Access space.

A bit in the instruction word specifies if the operation is to occur in the bank specified by the BSR register or in the Access Bank. This bit is denoted by the 'a' bit (for access bit).

When forced in the Access Bank (a = '0'), the last address in Access RAM Low is followed by the first address in Access RAM High. Access RAM High maps the Special Function Registers so that these registers can be accessed without any software overhead. This is useful for testing status flags, modifying control bits, software stacks, and context saving of registers.

7.6.2 General Purpose Registers (GPR)

Enhanced MCU devices may have banked memory in the GPR area. GPRs are not initialized by a Power-on Reset and are unchanged on all other resets.

The register file can be accessed either directly, or indirectly, using the File Select Register (FSR). Some devices have areas that are shared across the data memory banks, so a read/write to that area will appear as the same location (value), regardless of the current bank. We refer to this area as the Common RAM.

Data RAM is available for use as GPR registers by all instructions. Most banks of data memory contain only GPR registers starting with bank 0. The top half of bank 15 (0x80 to 0xFFF) contains SFRs.

Each data memory bank has 256 locations and can be addressed using an 8-bit address.
7.6.3 Special Function Registers

The SFRs are used by the CPU and peripheral modules for controlling the desired operation of the device. These registers are implemented as static RAM.

The SFRs can be classified into two sets; those associated with the "core" function and those related to the peripheral functions. Those registers related to the "core" are described in this section, while those related to the operation of the peripheral features are described in the section of that peripheral feature.

The SFRs are typically distributed among the peripherals whose functions they control.

If the SFRs do not use all the available locations on a particular device, the unused locations will be unimplemented and read as '0's. In devices that have a high integration of features, some of the SFRs may be in banks other than bank 15. See Figure 7-7 for addresses for the SFRs. As new devices are introduced with new SFRs, this register map will be updated. Please refer to the device data sheet for each device's register map.

Figure 7-7: Special Function Register Map

<table>
<thead>
<tr>
<th>Address</th>
<th>Register</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FFFh</td>
<td>TOSU</td>
<td></td>
</tr>
<tr>
<td>FFEh</td>
<td>TOSH</td>
<td></td>
</tr>
<tr>
<td>FFDh</td>
<td>TOSL</td>
<td></td>
</tr>
<tr>
<td>FFCh</td>
<td>STKPTR</td>
<td></td>
</tr>
<tr>
<td>FFdh</td>
<td>PCLATU</td>
<td></td>
</tr>
<tr>
<td>FFeh</td>
<td>PCLATH</td>
<td></td>
</tr>
<tr>
<td>FFFh</td>
<td>PCL</td>
<td></td>
</tr>
<tr>
<td>FF6h</td>
<td>TBLPTRU</td>
<td></td>
</tr>
<tr>
<td>FF7h</td>
<td>TBLPTRH</td>
<td></td>
</tr>
<tr>
<td>FF6h</td>
<td>TBLPTRL</td>
<td></td>
</tr>
<tr>
<td>FF5h</td>
<td>TABLAT</td>
<td></td>
</tr>
<tr>
<td>FF4h</td>
<td>PRODH</td>
<td></td>
</tr>
<tr>
<td>FF3h</td>
<td>PRODL</td>
<td></td>
</tr>
<tr>
<td>FF2h</td>
<td>INTCON1</td>
<td></td>
</tr>
<tr>
<td>FF1h</td>
<td>INTCON2</td>
<td></td>
</tr>
<tr>
<td>FF0h</td>
<td>INTCON3</td>
<td></td>
</tr>
<tr>
<td>FEFh</td>
<td>PROD</td>
<td></td>
</tr>
<tr>
<td>FE6h</td>
<td>POSTINC0</td>
<td></td>
</tr>
<tr>
<td>FE5h</td>
<td>POSTDEC0</td>
<td></td>
</tr>
<tr>
<td>FE4h</td>
<td>PREINC0</td>
<td></td>
</tr>
<tr>
<td>FE3h</td>
<td>PLUSW0</td>
<td></td>
</tr>
<tr>
<td>FE2h</td>
<td>FSPIH</td>
<td></td>
</tr>
<tr>
<td>FE1h</td>
<td>FSRIL</td>
<td></td>
</tr>
<tr>
<td>FE0h</td>
<td>WREG</td>
<td></td>
</tr>
<tr>
<td>FE9h</td>
<td>IND0(2)</td>
<td></td>
</tr>
<tr>
<td>FE8h</td>
<td>POSTINC2</td>
<td></td>
</tr>
<tr>
<td>FE7h</td>
<td>POSTDEC2</td>
<td></td>
</tr>
<tr>
<td>FE6h</td>
<td>PREINC2</td>
<td></td>
</tr>
<tr>
<td>FE5h</td>
<td>PLUSW1</td>
<td></td>
</tr>
<tr>
<td>FE4h</td>
<td>FSPIH</td>
<td></td>
</tr>
<tr>
<td>FE3h</td>
<td>FSRIL</td>
<td></td>
</tr>
<tr>
<td>FE2h</td>
<td>BSR</td>
<td></td>
</tr>
<tr>
<td>FE1h</td>
<td>FSR1H</td>
<td></td>
</tr>
<tr>
<td>FE0h</td>
<td>BSR</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: Unimplemented registers are read as '0'.
Note 2: This is not a physical register.
7.6.4 Bank Select Register (BSR)

The need for a large general purpose memory space dictated a general purpose RAM banking scheme. A Special Function Register (named BSR) selects the currently active general purpose RAM bank. Only the lower middle of the BSR register (BSR<3:0>) is used. This allows access to potentially 16 banks. Direct long addressing mode is limited to 12-bit addresses. This also allows accesses to any of the 16 banks. BSR<7:4> will always read 0's, and writes have no effect. All data memory is implemented as static RAM.

A MOVLB bank instruction has been provided in the instruction set to assist in selecting banks. If the currently selected bank is not implemented, any read will return all '0's, and all writes are ignored and the STATUS register bits will be set/cleared as appropriate for the instruction performed.
### 7. Memory

#### 7.7 Return Address Stack

The Return Address Stack allows any combination of up to 31 program calls and interrupts to occur. The PC (Program Counter) is PUSHed onto the stack when a CALL or RCALL instruction is executed or an interrupt is acknowledged. The PC value is pulled off the stack on a RETURN, RETLW, or a RETFIE instruction. PCLATU and PCLATH are not affected by any of the return instructions.

The stack operates as a 31 word by 21-bit RAM and a 5-bit stack pointer (STKPTR), with the stack pointer initialized to 00000b after all resets. There is no RAM associated with stack pointer location 00000b. This is only a reset value. During a CALL type instruction causing a PUSH onto the stack, the stack pointer is first incremented and the RAM location pointed to by the stack pointer is written with the contents of the PC. During a RETURN type instruction causing a POP from the stack, the contents of the RAM location pointed to by the STKPTR register are transferred to the PC and then the stack pointer is decremented.

The stack space is not part of either program or data space and the stack pointer is neither readable nor writable. The address on the top of the stack is readable and writable through SFR registers. Data can also be PUSHed to or POPed from the stack using the top-of-stack SFRs. Status bits indicate if the stack pointer attempts to exceed the 31 levels provided. The stack does not wrap when the stack is PUSHed greater than 31 times.

#### 7.7.1 Top-Of-Stack Access

The Top-of-Stack (TOS) is readable and writable. Three register locations, TOSU, TOSH and TOSL, hold the contents of the stack location pointed to by the STKPTR register. This allows users to implement a software stack if necessary. After a CALL, RCALL or interrupt, the software can read the PUSHed value by reading the TOSU, TOSH and TOSL registers. These values can be placed on a user defined software stack. At return time, the software can replace the TOSU, TOSH and TOSL and do a return.

The user must disable the global interrupt enable bits during this time to prevent inadvertent stack operations.

**Note:** The user must disable interrupts when manipulating the stack.
7.7.2 PUSH and POP Instructions

Since the Top-of-Stack (TOS) is readable and writable, the ability to PUSH values onto the stack and pull values off the stack without disturbing normal program execution is a desirable option. To PUSH the current PC value onto the stack, a PUSH instruction can be executed. This will increment the stack pointer and load the current PC value onto the stack. TOSU, TOSH, and TOSL can then be modified to place data on the stack instead of a return address. This data may be a new return address. This may be done in the operation of a Real Time Operating System (RTOS).

The ability to pull the TOS value off of the stack and replace it with the value that was previously PUSHed onto the stack, without disturbing normal execution, is achieved by using the POP instruction. The POP instruction discards the current TOS by decrementing the stack pointer. The previous value PUSHed onto the stack then becomes the TOS value.

Example 7-4: Using the PUSH Instruction

```
MOVLW Dummy_TOSU ;
MOVWF TOSU ;
MOVLW Dummy_TOSH ;
MOVWF TOSH ;
MOVLW Dummy_TOSL ;
MOVWF TOSL ;
PUSH ;
```

Example 7-5: Using the POP Instruction

```
MOVF TOSU, PREINC1 ;
MOVF TOSH, PREINC1 ;
MOVF TOSL, PREINC1 ;
POP ;
```
Section 7. Memory

7.7.3 Return Stack Pointer (STKPTR)

The STKPTR register contains the return stack pointer value and the overflow and underflow bits. The stack overflow bit (STKFUL) and underflow bit (STKUNF) allow software verification of a stack condition. The STKFUL and STKUNF bits are cleared only after a POR reset.

After the PC is PUSHed onto the stack 31 times (without POPing any values off the stack), the 32nd PUSH over-writes the value from the 31st PUSH and sets the STKFUL bit, while the STKPTR remains at 11111b. The 33rd PUSH overwrites the 32nd PUSH (and so on), while STKPTR remains 11111b.

After the stack is POPed enough times to unload the stack, the next POP will return a value of zero to the PC and set the STKUNF bit while the STKPTR remains at 00000b. The next POP returns zero again (and so on), while STKPTR remains 00000b.

The stack pointer can be accessed through the STKPTR register. The user may read and write the stack pointer values. This can be used by a Real Time Operating System (RTOS) for return stack maintenance. Figure 7-8 shows the STKPTR register. The value of the stack pointer will be 0 through 31. At reset, the stack pointer value will be 0. The stack pointer will increment when PUSHing and will decrement when POPing.

7.7.4 Stack Full/Underflow Resets

At the user’s option, the overflow and underflow can cause a device reset to interrupt the program code. The reset is enabled with a configuration bit, STVREN. When the STVREN bit is disabled, a full or underflow condition will set the appropriate STKFUL or STKUNF bit but not cause a reset. When the STVREN bit is enabled, a overflow or underflow will set the appropriate STKFUL or STKUNF bit and then cause a device reset very similar in nature to the WDT reset. In either case, the STKFUL or STKUNF bits are only cleared by user software or a POR reset.

7.7.5 Fast Register Stack

A “fast interrupt return” option is available for interrupts. A fast register stack is provided for the STATUS, WREG, and BSR registers and are only one in depth. The stack is neither readable nor writable and is loaded with the current value of the corresponding register when the processor vectors for an interrupt. The values in the registers are then loaded back into the working registers if the fast return instruction is used to return from the interrupt.

Low or high priority interrupt PUSHes values into the stack registers. If both low and high priority interrupts are enabled, the stack registers cannot be used reliably for low priority interrupts. A high priority interrupt, if one occurs while servicing a low priority interrupt, will overwrite the stack registers stored by the low priority interrupt.

If high priority interrupts are not disabled during low priority interrupts, users must save the key registers in software during a low priority interrupt.

If no interrupts are used, the fast register stack can be used to restore the STATUS, WREG and BSR registers at the end of a subroutine call. To use the fast register stack for a subroutine call, a fast call instruction must be executed.

Note: Returning a zero to the PC on an underflow has the effect of vectoring the program to the reset vector, where the stack conditions can be verified and appropriate actions can be taken.
Indirect addressing is a mode of addressing data memory where the data memory address in the instruction is not fixed. An SFR register is used as a pointer to the data memory location that is to be read or written. Since this pointer is in RAM, the contents can be modified by the program. This can be useful for data tables in the data memory and for software stacks. Figure 7-9 shows the operation of indirect addressing. This shows the moving of the value to the data memory address specified by the value of the FSR register.

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). The FSR register contains a 12-bit address, which is shown in Figure 7-9.

There are three indirect addressing registers. To address the entire Data Memory space (4096 bytes), these registers are 12 bits wide. To store the 12 bits of addressing information, two 8-bit registers are required. These indirect addressing registers are:

1. FSR0: composed of FSR0H:FSR0L
2. FSR1: composed of FSR1H:FSR1L
3. FSR2: composed of FSR2H:FSR2L

In addition, there are registers INDF0, INDF1 and INDF2, which are not physically implemented. Reading or writing to these registers activates indirect addressing, with the value in the corresponding FSR register being the address of the data.

If an instruction writes a value to INDF0, the value will be written to the address pointed to by FSR0H:FSR0L. A read from INDF1 actually reads the data from the address pointed to by FSR1H:FSR1L. INDFn can be used in code anywhere an operand can be used.

If the INDF0, INDF1 or INDF2 register is read indirectly via an FSR, all '0's are read (Zero bit is set). Similarly, if INDF0, INDF1 or INDF2 is written to indirectly, the operation will be equivalent to a NOP, and the STATUS bits are not affected.
Section 7. Memory

7.7.6.1 Indirect Addressing Operation

Each FSR register has an INDF register plus four addresses associated with it. The same INDFn, and FSRnH:FSRnL registers are used, but depending on the INDFn address selected, the FSRnH:FSRnL registers may be modified.

When a data access is done to one of the five INDFn locations, the address selected will configure the FSRn register to:

- Do nothing to FSRn after an indirect access (no change) - INDFn
- Auto-decrement FSRn after an indirect access (post-decrement) - POSTDECn
- Auto-increment FSRn after an indirect access (post-increment) - POSTINCn
- Use the value in the WREG register as an offset to FSRn. Do not modify the value of the WREG or the FSRn register after an indirect access (no change) - PLUSWn

When using the auto-increment or auto-decrement features, the effect on the FSR is not reflected in the STATUS register. For example, if the indirect address causes the FSR to equal '0', the Z bit will not be set.

Incrementing or decrementing an FSR affects all 12 bits. That is, when FSRnL overflows from an increment, FSRnH will be incremented automatically.

Adding these features allows the FSRn to be used as a stack pointer in addition to its uses for table operations in data memory.

Each FSR has an address associated with it that performs an indexed indirect access. When a data access to this INDFn location (PLUSWn) occurs, the FSRn is configured to add the signed value in the WREG register and the value in FSR to form the address before an indirect access. The FSR value is not changed.

**Note:** Accessing the PLUSWn address causes indexed indirect access. The addressed register is the addition of the value in the FSRn register and the SIGNED value in the WREG register.

If an FSR register contains a value that points to one of the INDFn, an indirect read will read 00h (Zero bit is set), while an indirect write will be equivalent to a NOP (STATUS bits are not affected).

If an indirect addressing operation is done where the target address is an FSRnH or FSRnL register, the write operation will dominate over the pre- or post-increment/decrement functions.

Figure 7-10 shows the interaction of the FSR register and the data memory.
Example 7-6 shows a simple use of indirect addressing to clear RAM (locations 20h-2Fh) in a minimum number of instructions. A similar concept could be used to move a defined number of bytes (block) of data to the USART transmit register (TXREG). The starting address of the block of data to be transmitted could easily be modified by the program.

Example 7-6: Indirect Addressing

```
; Clear High byte of FSR
CLRF FSR1H
; Load Low byte of 20h
MOVLW 0x20
; Clear register and the
MOVWF FSR1L
; increment
NEXT
; Clear register FSR1
CLRF POSTINC1
; Load FSR1, 4
BTFSS FSR1L, 4
; GOTO NEXT
GOTO NEXT

; CONTINUE
```
7.8 Initialization

Example 7-7 shows how the bank switching occurs for direct addressing, while Example 7-8 shows some code to do initialization (clearing) of General Purpose RAM.

Example 7-7: Bank Switching

```
CLRF BSR ; Clear BSR register (Bank0)
:
BSF BSR, 0 ; Bank1
:
BCF BSR, 0 ; Bank0
:
MOVLR 0x06 ;
MOVWF BSR ; Bank6
:
BCF BSR, 2 ; Bank2
:
BCF BSR, 1 ; Bank0
```

Example 7-8: RAM Initialization

```
CLRF FSR1H
CLRF FSR1L
CLR_LP CLRF POSTINC1 ; Clear location, increment
MOVLR 0x0F ; Bank is FSR1H:FSR1L
SUBWF FSR1H,W ; Are we now in Bank 15?
BNZ CLR_LP ; NO, continue to clear GPRs
CONTINUE ;
```
7.9 Design Tips

Question 1: I need to initialize RAM to '0's. What is an easy way to do that?

Answer 1:
Example 7-8 shows this. If the device you are using does not use all 15 data memory banks, the value to compare FSR1H against will need to be modified.

Question 2: I want to convert from a PIC16C77 which has 368 bytes of RAM (across 4 banks). What is the best way to remap this memory?

Answer 2:
In devices where the Access GPR region is greater or equal to 128 bytes and Bank 1 contains 256 bytes of GPR, the RAM should be partitioned with the RAM in Bank0 at locations 0x00 to 0x7F and all of Bank1. This allows a total of 384 bytes, which is larger than the 368 bytes in the PIC16C77. Now the BSR can be loaded with 0x01 (pointing to Bank1). All memory access are now either in Bank1 or the Access RAM, so no bank switching is required.
Section 7. Memory

7.10 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the Memory Organization are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Implementing a Table Read</td>
<td>AN556</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
7.11 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Memory Organization description.
Section 8. Table Read/Table Write

HIGHLIGHTS

This section of the manual contains the following major topics:

8.1 Introduction ................................................................. 8-2
8.2 Control Registers ......................................................... 8-3
8.3 Program Memory ......................................................... 8-6
8.4 Enabling Internal Programming ................................. 8-12
8.5 External Program Memory Operation ......................... 8-12
8.6 Initialization ............................................................. 8-13
8.7 Design Tips .............................................................. 8-14
8.8 Related Application Notes ........................................... 8-15
8.9 Revision History ......................................................... 8-16
8.1 Introduction

Enhanced devices have two memory spaces. The program memory space and the data memory space. The program memory space is 16 bits wide, while the data memory space is 8 bits wide. Table Reads and Table Writes have been provided to move data between these two memory spaces through an 8-bit register (TABLAT).

The operations that allow the processor to move data between the data and program memory spaces are:
- Table Read (TBLRD)
- Table Write (TBLWT)

Table Read operations retrieve data from program memory and place it into the data memory space. Figure 8-1 shows the operation of a Table Read with program and data memory.

Table Write operations store data from the data memory space into program memory. Figure 8-2 shows the operation of a Table Write with program and data memory.

Table operations work with byte entities. A table block containing data is not required to be word aligned, so a table block can start and end at any byte address. If Enhanced MCU instructions are being written to program memory, these instructions must be word aligned.

Figure 8-1: Table Read Operation

Figure 8-2: Table Write Operation
Section 8. Table Read/Table Write

8.2 Control Registers

Several control registers are used in conjunction with the TBLRD and TBLWT instructions. These include the:
- RCON register
- MEMCON register
- TBLPTR registers
- TABLAT register

8.2.1 RCON Register

The LWRT bit specifies the operation of Table Writes to internal memory when the VPP voltage is applied to the MCLR pin. When the LWRT bit is set, the controller continues to execute user code, but long Table Writes are allowed (for programming internal program memory) from user mode. The LWRT bit can be cleared only by performing either a Power-On Reset (POR) or MCLR reset. The other bits of the RCON register do not relate to Table Read nor Table Write operation.

Register 8-1: RCON Register

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>IPEN</td>
<td>LWRT</td>
<td>—</td>
<td>RI</td>
<td>TO</td>
<td>PD</td>
<td>POR</td>
<td>BOR</td>
</tr>
</tbody>
</table>

bit 7  IPEN: Interrupt Priority Enable
0 = Disable priority levels on interrupts (PIC16CXXX compatibility mode)

bit 6  LWRT: Long Write Enable
0 = Disable TABLE WRITE to internal program memory

Note 1: Only cleared on a POR or MCLR reset.

Note 2: This bit has no effect on TBLWT instructions to external program memory.

bit 5  Unimplemented: Read as ‘0’

bit 4  RI: Reset Instruction Flag bit
0 = A Reset instruction occurred

bit 3  TO: Time-out bit
0 = A WDT time-out occurred

bit 2  PD: Power-down bit
0 = By execution of the SLEEP instruction

bit 1  POR: Power-On Reset Status bit
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
0 = A Brown-out Reset occurred

Legend
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown
8.2.2 MEMCON Register

This register is only available on devices with a system bus to interface to external program memory. The MEMCON register is used to specify the operation of the 16-bit external system bus for Table Write operations. For additional information see the "System Bus" section. This register is not implemented in devices without a System Bus.

Register 8-2: MEMCON Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Name</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>EBDIS</td>
<td>External bus disable bit</td>
</tr>
<tr>
<td>6</td>
<td>PGRM</td>
<td>Program RAM enable bit</td>
</tr>
<tr>
<td>5-4</td>
<td>WAIT1</td>
<td>Wait Cycle count bits</td>
</tr>
<tr>
<td>3-2</td>
<td>WAIT0</td>
<td>Table reads and writes bus cycle wait count</td>
</tr>
<tr>
<td>1-0</td>
<td>WM1:WM0</td>
<td>Write Mode bit</td>
</tr>
</tbody>
</table>

Legend

R = Readable bit   W = Writable bit   U = Unimplemented bit, read as '0'
- n = Value at POR reset '1' = bit is set   '0' = bit is cleared  x = bit is unknown

Note: Any register that has a protected bit requires that the entire register is protected. To write to a protected register requires the proper write sequence on the CMLK1:CMLK0 bits before this register can be updated.
Section 8. Table Read/Table Write

8.2.3 TABLAT - Table Latch Register

The Table Latch (TABLAT) is an 8-bit register mapped into the SFR space. The Table Latch is used to hold 8-bit data during data transfers between program memory and data memory.

8.2.4 TBLPTR - Table Pointer Register

The Table Pointer (TBLPTR) addresses a byte within the program memory. The TBLPTR is comprised of three SFR registers (Table Pointer Upper byte, High byte, and Low byte). These three registers (TBLPTRU:TBLPTRH:TBLPTRL) join to form a 22-bit wide pointer. The low order 21-bits allows the device to address up to 2M bytes (or 1M words) of program memory space. The 22nd bit allows access to the Device ID, the User ID, and the Configuration bits.

The Table Pointer, TBLPTR, is used by the TBLRD and TBLWT instructions. These instructions can update the TBLPTR in one of four ways, based on the table operation. These operations are shown in Table 8-1. These operations on the TBLPTR only affect the low order 21-bits.

Table 8-1: Table Pointer Operations with TBLRD and TBLWT Instructions

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operation on Table Pointer</th>
</tr>
</thead>
<tbody>
<tr>
<td>TBLRD*</td>
<td>TBLPTR is not modified</td>
</tr>
<tr>
<td>TBLRD*+</td>
<td>TBLPTR is incremented after the read/write</td>
</tr>
<tr>
<td>TBLRD*--</td>
<td>TBLPTR is decremented after the read/write</td>
</tr>
<tr>
<td>TBLRD*+*</td>
<td>TBLPTR is incremented before the read/write</td>
</tr>
</tbody>
</table>
Section 8. Table Read/Table Write

8.3 Program Memory

The program memory can be either internal or external. External program memory requires that the device has the system bus interface. The operation of Table Reads and Table Writes is different depending on the location of the program memory (internal or external).

For the device to access external memory, the device needs to be operating in either extended microcontroller mode (some program memory is also internal), or microprocessor mode (no program memory is internal).

In this section, the discussion of Table Read and Table Write operation will be limited to the operation with internal program memory. For operation with external program memory, please refer to the “System Bus” section.

8.3.1 Internal Program Memory

The device selected will determine the program memory technology used. The Internal Program Memory can currently be one of three different memory technologies:

1. EPROM
2. FLASH
3. ROM

Depending on the memory technology the following statements can be made.

For EPROM devices:
- All unprogrammed memory locations will read back 0xFF (all bits set)
- Any bit that is set can be programmed clear
- Locations with data can be reprogrammed only if 1’s are changed to 0’s
- No cleared bit can be set unless the entire device is erased, which is only possible with windowed parts.

For FLASH devices:
- Any bit can be modified on a byte/block basis (individual bits cannot be modified)
- All writes occur with the write of an entire write block
- The size of the write block is device dependent

For ROM devices:
- All unprogrammed memory locations will read back 0xFF (all bits set)
- No program memory location can be modified
8.3.2 Internal Program Memory Operation

Table Read operation is independent of internal Program Memory Technology used. Table Write operation will be dependent on the memory technology of the internal Program Memory.

The notation "TBLRD instruction" means any of the four Table Read forms (TBLRD*, TBLRD**, TBLRD*, TBLRD**+) and the notation "TBLWT instruction" means any of the four Table Write forms (TBLWT*, TBLWT**, TBLWT*, TBLWT**+). For additional details on the operation of these instructions refer to the "Instruction Set" section.

8.3.2.1 Table Read Overview (TBLRD)

The TBLRD instructions are used to read data from program memory to data memory.

The Table Reads from program memory are performed one byte at a time. The TBLPTR points to a byte address in program space. Executing a TBLRD instruction moves the contents of the addressed byte into the TABLAT register. In addition, the TBLPTR can be modified automatically for the next Table Read operation.

The TBLPTR can be automatically modified, depending on the form of the TBLRD instruction (see Table 8-1). All of the TBLRD instructions require two instruction cycles (Tcy) to execute.

8.3.2.1.1 Effects of a Reset

The TABLAT register will retain the value read from program memory (if the instruction completed). The Table Pointer registers will not change, and the RCON register will be forced to the appropriate reset state.

8.3.2.2 Table Write Overview (TBLWT)

The TBLWT instructions are used to write data from data memory to program memory.

For devices with EPROM Program Memory, Table Writes are performed in pairs, one byte at a time. Table Writes to an even program memory address (TBLPTR<0> is clear) will load an internal memory latch from TABLAT, and is known as a short write. Table Writes to an odd program memory address (TBLPTR<0> is set) will start long writes. (TBLAT is programmed to the program word high byte, and the internal memory latch is programmed to the same word low byte).

For devices using another program memory technology, the operation may be different. Please refer to the device data sheet.
8.3.2.2.1 Memory Write Block Size

Depending on the device used, the write block size will be different. The larger the block size, the
tfaster the entire internal program memory space can be programmed. This is because the
EPROM/FLASH write time is much longer than the time to load the holding registers. For internal
program memory, the write block size can vary from 2 bytes to many bytes (in FLASH devices).
A write to the Most Significant byte (MSB) of the block causes the entire block to be written to
program memory. Writes to all other locations in the write block only modify the contents of the
specified holding register.

After a write cycle has completed, the contents of the Write Block Holding Registers are forced
set (‘1’s). This eases the writing of only a single byte within a program memory block. Figure 8-3
shows the write block for EPROM program memory. The write block size is 2 bytes.

If a single byte is to be programmed, the low (even) byte of the destination program word should
be read using TBLRD*, then modified if required, and written back to the same address using
TBLWT*. Then the high (odd) byte should be read using TBLRD*, modified if required, and writ-
ten back to the same address using a TBLMT instruction. The write to an odd address will cause
a long write to begin (in EPROM program memory devices). This process ensures that existing
data in either byte will not be changed unless desired.

Figure 8-3: Holding Registers and the Write Block (EPROM Program Memory)

![Diagram of Holding Registers and the Write Block](image-url)
Section 8. Table Read/Table Write

8.3.2.2 EPROM Program Memory Operation

The long write is what actually programs the data into the internal memory. When a tblwt instruction to the MSB of the write block occurs, instruction execution is halted. During this time, programming voltage and the data stored in internal latches is applied to program memory.

The sequence of steps to program the internal program memory is:

1. MCLR/VPP pin must be at the programming voltage
2. LWRT bit must be set
3. TBLWT to the address of the MSB of the write block

If the LWRT bit is clear, a short write will occur and program memory will not be changed. If the TBLWT is not to the MSB of the write block, then the programming phase is not initiated.

Setting the LWRT bit enables long writes when the MCLR pin is taken to VPP voltage. Once the LWRT bit is set, it can be cleared only by performing a Power-On Reset (POR) or MCLR reset.

To ensure that the memory location has been well programmed, a minimum programming time is required. The long write can be terminated after the programming time has expired by any event that can wake the controller from SLEEP. This may be a reset or an interrupt that operates during sleep. Having only one interrupt source enabled to terminate the long write ensures that no unintended interrupts will prematurely terminate the long write. Usable interrupt sources include:

- WDT
- A/D
- External Interrupts (INT0, INT1, or INT2)
- PORTB interrupt on change
- USART on address detect
- Timer1 in async counter mode or async external clock mode.
- Timer3 in async counter mode or async external clock mode.
- Capture
- SPI

8.3.2.3 Sequence of events

The sequence of events for programming an internal program memory location should be:

1. Enable the interrupt that terminates the long write. Disable all other interrupts.
2. Clear the source interrupt flag.
3. If Interrupt Service Routine (ISR) execution is desired when the device wakes, enable global interrupts (GIE, GIEH, or GIEL).
4. Set LWRT bit in RCON register.
5. Raise MCLR/VPP pin to the programming voltage, VPP.
6. Clear the WDT (if enabled).
7. Set the interrupt source to interrupt at the required time.
8. Load the desired Table Pointer Address.
9. Execute the Table Write for the lower (even) byte. This will be a short write.
10. Execute the Table Write for the upper (odd) byte. This will be a long write. The controller will halt instruction execution while programming. The interrupt wakes the controller.
11. If GIE was set, service the interrupt request.
12. If more locations to program, go to step 2.
13. Lower MCLR/VPP pin to VDD.
14. Verify the memory location (Table Read).
8.3.2.4 Interrupts

The long write must be terminated by a RESET or any interrupt that can wake the controller from SLEEP mode. Usable interrupt sources include:

- WDT
- A/D
- INT0
- INT1
- INT2
- RB<7:4> interrupt on change
- USART on address detect
- Timer1 in asynchronous counter mode or asynchronous external clock mode
- Timer3 in asynchronous counter mode or asynchronous external clock mode

The interrupt source must have its interrupt enable bit set. When the source sets its interrupt flag, programming will terminate. This will occur regardless of the settings of interrupt priority bits, the GIE/GIEH bit, or the PIE/GIEL bit.

Depending on the states of interrupt priority bits, the GIE/GIEH bit, or the PIE/GIEL bit, program execution can either be vectored to the high or low priority Interrupt Service Routine (ISR), or resume program execution.

In either case, the interrupt flag will not be automatically cleared when programming is terminated, and will need to be cleared by the software.

<table>
<thead>
<tr>
<th>Interrupt Source</th>
<th>GIE/GIEH</th>
<th>PIE/GIEL</th>
<th>Priority</th>
<th>Interrupt Enable</th>
<th>Interrupt Flag</th>
<th>Action</th>
</tr>
</thead>
<tbody>
<tr>
<td>Any interrupt source that operates during SLEEP</td>
<td>X</td>
<td>X</td>
<td>X</td>
<td>0 (default)</td>
<td>X</td>
<td>SLEEP mode continues even if interrupt flag becomes set during SLEEP.</td>
</tr>
<tr>
<td>0 (default)</td>
<td>0 (default)</td>
<td>X</td>
<td>1</td>
<td>0</td>
<td>X</td>
<td>SLEEP mode continues, will wake when interrupt flag is set.</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>Wakes controller, terminates long write, executes next instruction. Interrupt flag not cleared.</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>0 (default)</td>
<td>0</td>
<td>low</td>
<td>1</td>
<td>1</td>
<td>Wakes controller, terminates long write, executes next instruction. Interrupt flag not cleared.</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>0</td>
<td>low</td>
<td>1</td>
<td>1</td>
<td>Wakes controller, terminates long write, branches to low priority interrupt vector. Interrupt flag can be cleared by ISR.</td>
</tr>
<tr>
<td>1</td>
<td>0 (default)</td>
<td>1</td>
<td>high priority (default)</td>
<td>1</td>
<td>1</td>
<td>Wakes controller, terminates long write, branches to high priority interrupt vector. Interrupt flag can be cleared by ISR.</td>
</tr>
</tbody>
</table>
Section 8. Table Read/Table Write

8.3.2.5 Unexpected Termination of Write Operations

If a table write operation is terminated by an unplanned event, such as loss of power, an unexpected reset, or an interrupt that was not disabled, the memory location just programmed should be verified and reprogrammed if needed.

For applications where a loss of power could occur, a Brown-out reset circuit is recommended to ensure that the write operation is terminated cleanly. This reduces the possibility that a programmed location could not be reprogrammed to the desired value.

8.3.2.6 Effects of a RESET

A device reset during a long write may cause the location being programmed to be incompletely programmed. The location should be verified and reprogrammed if needed. A device reset during a write (short) will not effect the value in the TABLAT register (if the write cycle completed). The RCON register will be forced into the appropriate reset state.
8.4 Enabling Internal Programming

There is a combination of actions that need to occur to enable programming of the internal program memory. These modes are entered by applying the $V_{IH}$ voltage on the MCLR/VPP pin and either setting the LWRT bit (self-programming) or having RB6 and RB7 at a low level when $V_{IH}$ was detected (ICSP mode).

8.4.1 Programming Modes

Table 8-3 shows the device operating modes depending on the state of the MCLR pin, the LWRT bit, and the RB7:RB6.

<table>
<thead>
<tr>
<th>MCLR/VPP Voltage</th>
<th>LWRT</th>
<th>RB6</th>
<th>RB7</th>
<th>OPERATING MODE</th>
</tr>
</thead>
<tbody>
<tr>
<td>VPP</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>ICSP</td>
</tr>
<tr>
<td>VPP</td>
<td>0</td>
<td>1</td>
<td>X</td>
<td>Reserved</td>
</tr>
<tr>
<td>VPP</td>
<td>0</td>
<td>X</td>
<td>1</td>
<td>Reserved</td>
</tr>
<tr>
<td>VPP</td>
<td>1</td>
<td>X</td>
<td>X</td>
<td>Normal execution, long Table Writes enabled</td>
</tr>
<tr>
<td>VDD</td>
<td>1</td>
<td>X</td>
<td>X</td>
<td>Normal execution, short Table Writes only</td>
</tr>
<tr>
<td>VSS</td>
<td>0 (1)</td>
<td>X</td>
<td>X</td>
<td>In Device Reset</td>
</tr>
</tbody>
</table>

Legend: X = Don’t care.

Note 1: The LWRT bit is cleared by any device reset.

8.5 External Program Memory Operation

Regardless of the system bus mode selected, Table Reads and Table Writes to external memory execute in 2 Tcy. For information regarding the modes and waveforms of the System Bus, see the “System Bus” section. All further details of external program memory and table operations will be described in that section.
Section 8. Table Read/Table Write

8.6 Initialization

Example 8-1 shows the initialization of the Table pointer and the FSR0 register to read the value from Program memory to a RAM buffer (starting at address RAMBUFADDR).

Example 8-2 shows the sequence to initialize these same registers and then write a word to the internal EPROM Program memory.

Example 8-1: Initialization to do a Table Read

```assembly
LFSR  FSR0, RAMBUFADDR ;
MOVLW  UPPER (Read Table) ;
MOVWF  TBLPTRU ;
MOVLW  HIGH (Read Table) ;
MOVWF  TBLPTRH ;
MOVLW  LOW (Read Table) ;
MOVWF  TBLPTRL ;
TBLRD++ ; Read location and then increment
          ; the table pointer
MOVFF  TABLAT, POSTINC0 ; Copy contents of table latch to the
                      ; indirect address and then
                      ; increment the indirect address
                      ; pointer.
```

Example 8-2: Initialization to do a Table Write (to internal EPROM memory)

```assembly
; Set up interrupts to terminate
; long write
LFSR  FSR0, RAMBUFADDR ;
MOVLW  UPPER (Read Table) ;
MOVWF  TBLPTRU ;
MOVLW  HIGH (Read Table) ;
MOVWF  TBLPTRH ;
MOVLW  LOW (Read Table) ;
MOVWF  TBLPTRL ;
MOVFF  POSTINC0, TABLAT ; Load table latch with value
                      ; to write
TBLWT++ ; Write to holding register
MOVFF  POSTINC0, TABLAT ; Load second byte to table latch
TBLWT++ ; Write to MSB, (odd address)
          ; start long write
```
8.7 Design Tips

Question 1: The location I programmed does not contain the value I wrote. What is wrong?

Answer 1:
There are several possibilities. These include, but are not limited to:

• The maximum time requirement of the long write time was not met (for internal program memory).
• The address of the location to program was changed during the write cycle.
• A device reset occurred during the write cycle.
• If the program memory is EPROM or EOTP, then required overprogramming was not done.
• An Interrupt flag may have been set, so the long write cycle was immediately completed (violated long write time specification).

Question 2: Occasionally the device hangs. What is this?

Answer 2:
Your program may have executed a long write, and the device may not have the interrupts/modules set up to terminate this long write.

Question 3: When programming the program memory are there any required algorithm algorithms to follow?

Answer 3:
The programming algorithm will be dependent on the memory technology of the device. For complete information on device programming please refer to the device’s programming specifications.
Section 8. Table Read/Table Write

8.8 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the oscillator are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related application notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
8.9 Revision History

This is the initial released revision of the Enhanced MCU Table Read/Table Write description.
Section 9. System Bus

Please check the Microchip web site for Revision B of the System Bus Section.
9.1 Revision History

Revision A

This is the initial released revision of the Enhanced MCU System Bus module description.
Section 10. Interrupts

HIGHLIGHTS

This section of the manual contains the following major topics:

10.1 Introduction ............................................................................................................... 10-2
10.2 Control Registers ......................................................................................................... 10-6
10.3 Interrupt Handling Operation ...................................................................................... 10-19
10.4 Initialization ............................................................................................................. 10-29
10.5 Design Tips ................................................................................................................ 10-30
10.6 Related Application Notes .......................................................................................... 10-31
10.7 Revision History ......................................................................................................... 10-32
10.1 Introduction

Interrupts can come from many sources. These sources currently include:

- External interrupt from the INT, INT1, and INT2 pins
- Change on RB7:RB4 pins
- TMR0 Overflow
- TMR1 Overflow
- TMR2 Overflow
- TMR3 Overflow
- USART Interrupts
  - Receive buffer full
  - Transmit buffer empty
- SSP Interrupt
- SSP PC bus collision interrupt
- A/D conversion complete
- CCP interrupt
- LVD Interrupt
- Parallel Slave Port
- CAN interrupts
  - Receive buffer 1 full
  - Receive buffer 2 full
  - Receive invalid
  - Transmit buffer 0 empty
  - Transmit buffer 1 empty
  - Transmit buffer 2 empty
  - Bus wakeup
  - Bus invalid error

As other peripheral modules are developed, they will have interrupt sources. These sources will map into the 10 registers used in the control and status of interrupts. These registers are:

- INTCON
- INTCON1
- INTCON2
- INTCON3
- PIR1
- PIR2
- PIE1
- PIE2
- IPR1
- IPR2

The INTCON register contains the GIE/GIEH bit. This is the Global Interrupt Enable bit. When this bit is set, all interrupts are enabled. If needed for any single device, additional INTCON, PIR, PIE, and IPR registers will be defined.
Section 10. Interrupts

10.1.1 Interrupt Priority

There are two interrupt vectors. One interrupt vector is for high priority interrupts and is located at address 000008h. The other interrupt vector is for low priority interrupts and is located at address 000018h.

When a valid interrupt occurs, program execution vectors to one of these interrupt vector addresses and the corresponding Global Interrupt Enable bit (GIEH, GIEH, or GIEL) is automatically cleared. In the interrupt service routine, the source(s) of the interrupt can be determined by testing the interrupt flag bits. The interrupt flag bit(s) must be cleared before re-enabling interrupts to avoid infinite interrupt requests. Most flag bits are required to be cleared by the application software. There are some flag bits that are automatically cleared by the hardware.

When an interrupt condition is met, that individual interrupt flag bit will be set regardless of the status of its corresponding mask bit.

For external interrupt events, such as the RB0/INT0 pin or PORTB change interrupt, the interrupt latency will be three or four instruction cycles. The exact latency depends when the interrupt event occurs. The interrupt latency is the same for one or two cycle instructions.

The "return from interrupt" instruction, RETFIE, can be used to mark the end of the interrupt service routine. When this instruction is executed, the stack is “POPed” and the GIE bit is set (to re-enable interrupts).

Figure 10-1: Interrupt Logic High Level Block Diagram
Figure 10-2: High Priority Interrupt Logic Block Diagram

PSPIF  PSPIE  PSPIP
ADIF   ADIE   ADIP
RCIF   RCIE   RCIP
TXIF   TXIE   TXIP
SSIF   SSIE   SSIP
CCPIF  CCPIE  CCPIP
CCPIE  CCPIP
TMR1IF  TMR1IE  TMR1IP
TMR2IF  TMR2IE  TMR2IP
TMR3IF  TMR3IE  TMR3IP
BCLIF  BCLIE  BCLIP
LVdif  LVdie  LVdip
INT0IF  INT0IE  INT0IP
RBIF   RBIE   RBIP
INT2IF  INT2IE  INT2IP
IPEN   PEIE/GIE
PEIE/GIEH

Wake-up (If in SLEEP mode)
Interrupt to CPU Vector to Location 000008h
To low priority interrupt logic
Section 10. Interrupts

Figure 10-3: Low Priority Interrupt Logic Block Diagram

- PSPIE
- PSPIP
- ADIF
- ADIF
- RCCF
- RCCF
- TXIF
- TXIF
- SSPIF
- SSPIF
- CCP1IF
- CCP1IF
- CCP2IF
- CCP2IF
- TMR1IF
- TMR1IF
- TMR2IF
- TMR2IF
- TMR3IF
- TMR3IF
- BCLIF
- BCLIF
- LV DIF
- LV DIF
- TMR0IF
- TMR0IF
- INT0IF
- INT0IF
- INT1IF
- INT1IF
- INT2IF
- INT2IF
- RBIF
- RBIF

High priority interrupt initiated signal
(Disable Low Priority Interrupts)

Wake-up (If in SLEEP Mode)

Interrupt to CPU
Vector to Location 00018h
(Low Priority Interrupt Vector Address)

GIEL

IPEN
10.2 Control Registers

Generally devices have a minimum of four registers associated with interrupts. The INTCON register contains the Global Interrupt Enable bit, GIE, as well as the Peripheral Interrupt Enable bit, PEIE, the PIE / PIR register pair that enables the peripheral interrupts and displays the interrupt flag status, and the Interrupt Priority Register (IPR) that controls whether the interrupt source is a high priority or low priority interrupt.
Section 10. Interrupts

10.2.1 INTCON Register

The INTCON Registers are readable and writable registers that contain various enable, priority, and flag bits.

Register 10-1: INTCON Register

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>GIE/GIEH</td>
<td>RBIF</td>
</tr>
<tr>
<td>PEIE/GEIL</td>
<td>INT0IF</td>
</tr>
<tr>
<td>TMR0IE</td>
<td>RBIE</td>
</tr>
<tr>
<td>INT0IE</td>
<td>TMR0IF</td>
</tr>
<tr>
<td>bit 7</td>
<td>bit 0</td>
</tr>
</tbody>
</table>

- **bit 7 GIE/GIEH**: Global Interrupt Enable bit  
  - When IPEN = 0:
    - 1 = Enables all un-masked interrupts
    - 0 = Disables all interrupts
  - When IPEN = 1:
    - 1 = Enables all interrupts
    - 0 = Disables all interrupts

- **bit 6 PEIE/GEIL**: Peripheral Interrupt Enable bit  
  - When IPEN = 0:
    - 1 = Enables all un-masked peripheral interrupts
    - 0 = Disables all peripheral interrupts
  - When IPEN = 1:
    - 1 = Enables all low peripheral interrupts
    - 0 = Disables all priority peripheral interrupts

- **bit 5 TMR0IE**: TMR0 Overflow Interrupt Enable bit  
  - 1 = Enables the TMR0 overflow interrupt
  - 0 = Disables the TMR0 overflow interrupt

- **bit 4 INT0IE**: INT0 External Interrupt Enable bit  
  - 1 = Enables the INT0 external interrupt
  - 0 = Disables the INT0 external interrupt

- **bit 3 RBIE**: RB Port Change Interrupt Enable bit  
  - 1 = Enables the RB port change interrupt
  - 0 = Disables the RB port change interrupt

- **bit 2 TMR0IF**: TMR0 Overflow Interrupt Flag bit  
  - 1 = TMR0 register has overflowed (must be cleared in software)
  - 0 = TMR0 register did not overflow

- **bit 1 INT0IF**: INT0 External Interrupt Flag bit  
  - 1 = The INT0 external interrupt occurred (must be cleared in software)
  - 0 = The INT0 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 (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 reset  
- ‘1’ = bit is set  
- ‘0’ = bit is cleared  
- x = bit is unknown

**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. User software should ensure the appropriate interrupt flag bits are clear prior to enabling an interrupt. This feature allows for software polling.
## Register 10-2: INTCON2 Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Name</th>
<th>Description</th>
<th>Value</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>RBPU (R/W-1)</td>
<td>PORTB Pull-up Enable bit</td>
<td>1</td>
<td>All PORTB pull-ups are disabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>PORTB pull-ups are enabled by individual port latch values</td>
</tr>
<tr>
<td>6</td>
<td>INTEDG0 (R/W-1)</td>
<td>External Interrupt0 Edge Select bit</td>
<td>1</td>
<td>Interrupt on rising edge</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Interrupt on falling edge</td>
</tr>
<tr>
<td>5</td>
<td>INTEDG1 (R/W-1)</td>
<td>External Interrupt1 Edge Select bit</td>
<td>1</td>
<td>Interrupt on rising edge</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Interrupt on falling edge</td>
</tr>
<tr>
<td>4</td>
<td>INTEDG2 (R/W-1)</td>
<td>External Interrupt2 Edge Select bit</td>
<td>1</td>
<td>Interrupt on rising edge</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Interrupt on falling edge</td>
</tr>
<tr>
<td>3</td>
<td></td>
<td>Unimplemented: Read as '1'</td>
<td></td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>TMR0IP (R/W-1)</td>
<td>TMR0 Overflow Interrupt Priority bit</td>
<td>1</td>
<td>TMR0 Overflow Interrupt is a high priority event</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>TMR0 Overflow Interrupt is a low priority event</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td>Unimplemented: Read as '1'</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>RBIP (R/W-1)</td>
<td>RB Port Change Interrupt Priority bit</td>
<td>1</td>
<td>RB Port Change Interrupt is a high priority event</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>RB Port Change Interrupt is a low priority event</td>
</tr>
</tbody>
</table>

**Legend:**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **n** = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown

**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. User software should ensure the appropriate interrupt flag bits are clear prior to enabling an interrupt. This feature allows for software polling.
## Section 10. Interrupts

### Register 10-3: INTCON3 Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Description</th>
<th>Value at POR reset</th>
<th>Default Value</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>INT2IP: INT2 External Interrupt Priority bit</td>
<td>1</td>
<td>0</td>
<td>1 = INT2 External Interrupt is a high priority event</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 = INT2 External Interrupt is a low priority event</td>
</tr>
<tr>
<td>6</td>
<td>INT1IP: INT1 External Interrupt Priority bit</td>
<td>1</td>
<td>0</td>
<td>1 = INT1 External Interrupt is a high priority event</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 = INT1 External Interrupt is a low priority event</td>
</tr>
<tr>
<td>5</td>
<td>Unimplemented: Read as '0'</td>
<td></td>
<td>0</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td>INT2IE: INT2 External Interrupt Enable bit</td>
<td>1</td>
<td>0</td>
<td>1 = Enables the INT2 external interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 = Disables the INT2 external interrupt</td>
</tr>
<tr>
<td>3</td>
<td>INT1IE: INT1 External Interrupt Enable bit</td>
<td>1</td>
<td>0</td>
<td>1 = Enables the INT1 external interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 = Disables the INT1 external interrupt</td>
</tr>
<tr>
<td>2</td>
<td>Unimplemented: Read as '0'</td>
<td></td>
<td>0</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>INT2IF: INT2 External Interrupt Flag bit</td>
<td>1</td>
<td>0</td>
<td>1 = The INT2 external interrupt occurred (must be cleared in software)</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 = The INT2 external interrupt did not occur</td>
</tr>
<tr>
<td>0</td>
<td>INT1IF: INT1 External Interrupt Flag bit</td>
<td>1</td>
<td>0</td>
<td>1 = The INT1 external interrupt occurred (must be cleared in software)</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 = The INT1 external interrupt did not occur</td>
</tr>
</tbody>
</table>

### Legend

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **- n** = Value at POR reset
- **'1'** = bit is set
- **'0'** = bit is cleared
- **x** = bit is unknown

### 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. User software should ensure the appropriate interrupt flag bits are clear prior to enabling an interrupt. This feature allows for software polling.
10.2.2 PIE Register(s)

Depending on the number of peripheral interrupt sources, there may be multiple Peripheral Interrupt Enable registers (such as PIE1 and PIE2). These registers contain the individual enable bits for the peripheral interrupts. These registers will be generically referred to as PIE.

Note: If the device has a PIE register and IPEN = 0, the PEIE bit must be set to enable any of the peripheral interrupts.

Although the PIE register bits have a general bit location with each register, future devices may not have consistent placement. Bit location inconsistencies will not be a problem if you use the supplied Microchip Include files for the symbolic use of these bits. This will allow the Assembler/Compiler to automatically take care of the placement of these bits by specifying the correct Register number and bit name.

Register 10-4: PIE Peripheral Interrupt Enable Registers

<table>
<thead>
<tr>
<th>R/W</th>
<th>0</th>
</tr>
</thead>
<tbody>
<tr>
<td>bit 7</td>
<td>bit 0</td>
</tr>
</tbody>
</table>

- **TMR1IE**: TMR1 Overflow Interrupt Enable bit
  - 1 = Enables the TMR1 overflow interrupt
  - 0 = Disables the TMR1 overflow interrupt

- **TMR2IE**: TMR2 to PR2 Match Interrupt Enable bit
  - 1 = Enables the TMR2 to PR2 match interrupt
  - 0 = Disables the TMR2 to PR2 match interrupt

- **TMR3IE**: TMR3 Overflow Interrupt Enable bit
  - 1 = Enables the TMR3 overflow interrupt
  - 0 = Disables the TMR3 overflow interrupt

- **CCPxIE**: CCPx Interrupt Enable bit
  - 1 = Enables the CCPx interrupt
  - 0 = Disables the CCPx interrupt

- **ECCPxIE**: Enhanced CCPx Interrupt Enable bit
  - 1 = Enables the CCPx interrupt
  - 0 = Disables the CCPx interrupt

- **SSPIE**: Synchronous Serial Port Interrupt Enable bit
  - 1 = Enables the SSP interrupt
  - 0 = Disables the SSP interrupt

- **MSSPIE**: Master Synchronous Serial Port Interrupt Enable bit
  - 1 = Enables the MSSP interrupt
  - 0 = Disables the MSSP interrupt

- **RCIE**: USART Receive Interrupt Enable bit
  - 1 = Enables the USART receive interrupt
  - 0 = Disables the USART receive interrupt

- **TXIE**: USART Transmit Interrupt Enable bit
  - 1 = Enables the USART transmit interrupt
  - 0 = Disables the USART transmit interrupt

- **IRXIE**: CAN Invalid Received message Interrupt Enable bit
  - 1 = Enable invalid message received interrupt
  - 0 = Disable invalid message received interrupt

- **WAKIE**: CAN Bus Activity Wake-up Interrupt Enable bit
  - 1 = Enable CAN bus Activity Wake-up Interrupt
  - 0 = Disable CAN bus Activity Wake-up Interrupt

- **ERRIE**: CAN bus Error Interrupt Enable bit
  - 1 = Enable CAN bus Error Interrupt
  - 0 = Disable CAN bus Error Interrupt
Section 10. Interrupts

- **TXB2IE**: CAN Transmit Buffer 2 Interrupt Enable bit
  - 1 = Enable Transmit Buffer 2 Interrupt
  - 0 = Disable Transmit Buffer 2 Interrupt

- **TXB1IE**: CAN Transmit Buffer 1 Interrupt Enable bit
  - 1 = Enable Transmit Buffer 1 Interrupt
  - 0 = Disable Transmit Buffer 1 Interrupt

- **TXB0IE**: CAN Transmit Buffer 0 Interrupt Enable bit
  - 1 = Enable Transmit Buffer 0 Interrupt
  - 0 = Disable Transmit Buffer 0 Interrupt

- **RXB1IE**: CAN Receive Buffer 1 Interrupt Enable bit
  - 1 = Enable Receive Buffer 1 Interrupt
  - 0 = Disable Receive Buffer 1 Interrupt

- **RXB0IE**: CAN Receive Buffer 0 Interrupt Enable bit
  - 1 = Enable Receive Buffer 0 Interrupt
  - 0 = Disable Receive Buffer 0 Interrupt

- **ADIE**: A/D Converter Interrupt Enable bit
  - 1 = Enables the A/D interrupt
  - 0 = Disables the A/D interrupt

- **PSPIE**: Parallel Slave Port Read/Write Interrupt Enable bit
  - 1 = Enables the PSP read/write interrupt
  - 0 = Disables the PSP read/write interrupt

- **EEIE**: EE Write Complete Interrupt Enable bit
  - 1 = Enables the EE write complete interrupt
  - 0 = Disables the EE write complete interrupt

- **CMIE**: Comparator Interrupt Enable bit
  - 1 = Enables the Comparator interrupt
  - 0 = Disables the Comparator interrupt

- **BCLIE**: Bus Collision Interrupt Enable bit
  - 1 = Enabled
  - 0 = Disabled

- **LVDIE**: Low-voltage Detect Interrupt Enable bit
  - 1 = Enabled
  - 0 = Disabled

**Legend**
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown

**Note 1:** The bit position of the enable bits is device dependent. Please refer to the device data sheet for bit placement.
10.2.3 PIR Register(s)

Depending on the number of peripheral interrupt sources, there may be multiple Peripheral Inter-
rupt Flag registers (PIR1, PIR2). These registers contain the individual flag bits for the peripheral
interrupts. These registers will be generically referred to as PIR.

| Note 1: | 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>). |
| Note 2: | User software should ensure the appropriate interrupt flag bits are cleared prior to
|         | enabling an interrupt and after servicing that interrupt. |

Although the PIR bits have a general bit location within each register, future devices may not
have consistent placement. It is recommended that you use the supplied Microchip Include files
for the symbolic use of these bits. This will allow the Assembler/Compiler to automatically take
care of the placement of these bits within the specified register.
Section 10. Interrupts

Register 10-5: PIR Register

<table>
<thead>
<tr>
<th></th>
<th>bit 7</th>
<th>bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td><strong>TMR1IF</strong>: TMR1 Overflow Interrupt Flag bit</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = TMR1 register overflowed (must be cleared in software)</td>
<td>0 = TMR1 register did not overflow</td>
</tr>
<tr>
<td><strong>TMR2IF</strong>: TMR2 to PR2 Match Interrupt Flag bit</td>
<td>1 = TMR2 to PR2 match occurred (must be cleared in software)</td>
<td>0 = No TMR2 to PR2 match occurred</td>
</tr>
<tr>
<td><strong>TMR3IF</strong>: TMR3 Overflow Interrupt Flag bit</td>
<td>1 = TMR3 register overflowed (must be cleared in software)</td>
<td>0 = TMR3 register did not overflow</td>
</tr>
<tr>
<td><strong>CCPxIF</strong>: CCPx Interrupt Flag bit</td>
<td></td>
<td></td>
</tr>
<tr>
<td>Capture Mode</td>
<td>1 = A TMR1 register capture occurred</td>
<td>0 = No TMR1 register capture occurred</td>
</tr>
<tr>
<td>Compare Mode</td>
<td>1 = A TMR1 register compare match occurred (must be cleared in software)</td>
<td>0 = No TMR1 register compare match occurred</td>
</tr>
<tr>
<td>PWM Mode</td>
<td>Unused in this mode</td>
<td></td>
</tr>
<tr>
<td><strong>ECCPxIF</strong>: Enhanced CCPx Interrupt Flag bit</td>
<td></td>
<td></td>
</tr>
<tr>
<td>Capture Mode</td>
<td>1 = A TMR1 register capture occurred (must be cleared in software)</td>
<td>0 = No TMR1 register capture occurred</td>
</tr>
<tr>
<td>Compare Mode</td>
<td>1 = A TMR1 register compare match occurred (must be cleared in software)</td>
<td>0 = No TMR1 register compare match occurred</td>
</tr>
<tr>
<td>PWM Mode</td>
<td>Unused in this mode</td>
<td></td>
</tr>
</tbody>
</table>
bit IRXIF: CAN Invalid Received message Interrupt Flag bit
1 = An invalid message has occurred on the CAN bus
0 = No invalid message on CAN bus

bit WAKIF: CAN Bus Activity Wake-up Interrupt Flag bit
1 = Activity on CAN bus has occurred
0 = No activity on CAN bus

bit ERRIF: CAN bus Error Interrupt Flag bit
1 = An error has occurred in the CAN module (multiple sources)
0 = No CAN module errors

bit TXB2IF: CAN Transmit Buffer 2 Interrupt Flag bit
1 = Transmit Buffer 2 has completed transmission of a message, and may be re-loaded
0 = Transmit Buffer 2 has not completed transmission of a message

bit TXB1IF: CAN Transmit Buffer 1 Interrupt Flag bit
1 = Transmit Buffer 1 has completed transmission of a message, and may be re-loaded
0 = Transmit Buffer 1 has not completed transmission of a message

bit TXB0IF: CAN Transmit Buffer 0 Interrupt Flag bit
1 = Transmit Buffer 0 has completed transmission of a message, and may be re-loaded
0 = Transmit Buffer 0 has not completed transmission of a message

bit RXB1IF: CAN Receive Buffer 1 Interrupt Flag bit
1 = Receive Buffer 1 has received a new message
0 = Receive Buffer 1 has not received a new message

bit RXB0IF: CAN Receive Buffer 0 Interrupt Flag bit
1 = Receive Buffer 0 has received a new message
0 = Receive Buffer 0 has not received a new message

bit SSPIF: Synchronous Serial Port Interrupt Flag bit
1 = The transmission/reception is complete
   (must be cleared in software)
0 = Waiting to transmit/receive

bit MSSPIF: Master Synchronous Serial Port Interrupt Flag bit
1 = The transmission/reception is complete
   (must be cleared in software)
0 = Waiting to transmit/receive

bit RCIF: USART Receive Interrupt Flag bit
1 = The USART receive buffer, RCREG, is full
   (cleared when RCREG is read)
0 = The USART receive buffer is empty

bit TXIF: USART Transmit Interrupt Flag bit
1 = The USART transmit buffer, TXREG, is empty
   (cleared when TXREG is written)
0 = The USART transmit buffer is full
Section 10. Interrupts

bit **ADIF**: A/D Converter Interrupt Flag bit
1 = An A/D conversion completed
   (must be cleared in software)
0 = The A/D conversion is not complete

bit **PSPIF**: Parallel Slave Port Read/Write Interrupt Flag bit
1 = A read or a write operation has taken place
   (must be cleared in software)
0 = No read or write has occurred

bit **EEIF**: EE Write Complete Interrupt Flag bit
1 = The data EEPROM write operation is complete
   (must be cleared in software)
0 = The data EEPROM write operation is not complete

bit **CMIF**: Comparator Interrupt Flag bit
1 = Comparator input has changed
   (must be cleared in software)
0 = Comparator input has not changed

bit **BCLIF**: Bus Collision Interrupt Flag bit
1 = A Bus Collision occurred
   (must be cleared in software)
0 = No Bus Collision occurred

bit **LVDIF**: Low-voltage Detect Interrupt Flag bit
1 = A Low Voltage condition occurred
   (must be cleared in software)
0 = The device voltage is above the Low Voltage Detect trip point

Legend
R = Readable bit  W = Writable bit  U = Unimplemented bit, read as '0'
- n = Value at POR reset  '1' = bit is set  '0' = bit is cleared  x = bit is unknown

**Note 1:** The bit position of the enable bits is device dependent. Please refer to the device data sheet for bit placement.
10.2.4 IPR Register

Depending on the number of peripheral interrupt sources, there may be multiple Peripheral Interrupt Priority registers (such as IPR1 and IPR2). These registers contain the individual priority bits for the peripheral interrupts. These registers will be generically referred to as IPR. If the device has an IPR register and IPEN = 0, the PEIE bit must be set to enable any of these peripheral interrupts.

<table>
<thead>
<tr>
<th>Bit</th>
<th>Description</th>
<th>Value 1</th>
<th>Value 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>TMR1IP</td>
<td>TMR1 Overflow Interrupt Priority bit</td>
<td>TMR1 Overflow Interrupt is a high priority event</td>
<td>TMR1 Overflow Interrupt is a low priority event</td>
</tr>
<tr>
<td>TMR2IP</td>
<td>TMR2 to PR2 Match Interrupt Priority bit</td>
<td>TMR2 to PR2 Match Interrupt is a high priority event</td>
<td>TMR2 to PR2 Match Interrupt is a low priority event</td>
</tr>
<tr>
<td>TMR3IP</td>
<td>TMR3 Overflow Interrupt Priority bit</td>
<td>TMR3 Overflow Interrupt is a high priority event</td>
<td>TMR3 Overflow Interrupt is a low priority event</td>
</tr>
<tr>
<td>CCPxIP</td>
<td>CCPx Interrupt Priority bit</td>
<td>CCPx Interrupt is a high priority event</td>
<td>CCPx Interrupt is a low priority event</td>
</tr>
<tr>
<td>ECCPxIP</td>
<td>Enhanced CCPx Interrupt Priority bit</td>
<td>Enhanced CCPx Interrupt is a high priority event</td>
<td>Enhanced CCPx Interrupt is a low priority event</td>
</tr>
<tr>
<td>MSSPIP</td>
<td>Master Synchronous Serial Port Interrupt Priority bit</td>
<td>Master Synchronous Serial Port Interrupt is a high priority event</td>
<td>Master Synchronous Serial Port Interrupt is a low priority event</td>
</tr>
<tr>
<td>SSPIP</td>
<td>Synchronous Serial Port Interrupt Priority bit</td>
<td>Synchronous Serial Port Interrupt is a high priority event</td>
<td>Synchronous Serial Port Interrupt is a low priority event</td>
</tr>
<tr>
<td>RCIP</td>
<td>USART Receive Interrupt Priority bit</td>
<td>USART Receive Interrupt is a high priority event</td>
<td>USART Receive Interrupt is a low priority event</td>
</tr>
<tr>
<td>TXIP</td>
<td>USART Transmit Interrupt Priority bit</td>
<td>USART Transmit Interrupt is a high priority event</td>
<td>USART Transmit Interrupt is a low priority event</td>
</tr>
<tr>
<td>ADIP</td>
<td>A/D Converter Interrupt Priority bit</td>
<td>A/D Converter Interrupt is a high priority event</td>
<td>A/D Converter Interrupt is a low priority event</td>
</tr>
<tr>
<td>PSPIP</td>
<td>Parallel Slave Port Read/Write Interrupt Priority bit</td>
<td>Parallel Slave Port Read/Write Interrupt is a high priority event</td>
<td>Parallel Slave Port Read/Write Interrupt is a low priority event</td>
</tr>
</tbody>
</table>

Note: The IP bit specifies the priority of the peripheral interrupt.

Although the IPR register bits have a general bit location with each register, future devices may not have consistent placement. Bit location inconsistencies will not be a problem if you use the supplied Microchip Include files for the symbolic use of these bits. This will allow the Assembler/Compiler to automatically take care of the placement of these bits by specifying the correct register and bit name.
Section 10. Interrupts

bit  IRXIP: CAN Invalid Received message Interrupt Priority bit
     1 = CAN Invalid Received message Interrupt is a high priority event
     0 = CAN Invalid Received message Interrupt is a low priority event

bit  WAKIP: CAN Bus Activity Wake-up Interrupt Priority bit
     1 = CAN Bus Activity Wake-up Interrupt is a high priority event
     0 = CAN Bus Activity Wake-up Interrupt is a low priority event

bit  ERRIP: CAN bus Error Interrupt Priority bit
     1 = CAN bus Error Interrupt is a high priority event
     0 = CAN bus Error Interrupt is a low priority event

bit  TXB2IP: CAN Transmit Buffer 2 Interrupt Priority bit
     1 = CAN Transmit Buffer 2 Interrupt is a high priority event
     0 = CAN Transmit Buffer 2 Interrupt is a low priority event

bit  TXB1IP: CAN Transmit Buffer 1 Interrupt Priority bit
     1 = CAN Transmit Buffer 1 Interrupt is a high priority event
     0 = CAN Transmit Buffer 1 Interrupt is a low priority event

bit  TXB0IP: CAN Transmit Buffer 0 Interrupt Priority bit
     1 = CAN Transmit Buffer 0 Interrupt is a high priority event
     0 = CAN Transmit Buffer 0 Interrupt is a low priority event

bit  RXB1IP: CAN Receive Buffer 1 Interrupt Priority bit
     1 = CAN Receive Buffer 1 Interrupt is a high priority event
     0 = CAN Receive Buffer 1 Interrupt is a low priority event

bit  RXB0IP: CAN Receive Buffer 0 Interrupt Priority bit
     1 = CAN Receive Buffer 0 Interrupt is a high priority event
     0 = CAN Receive Buffer 0 Interrupt is a low priority event

bit  EEIP: EE Write Complete Interrupt Priority bit
     1 = EE Write Complete Interrupt is a high priority event
     0 = EE Write Complete Interrupt is a low priority event

bit  CMIP: Comparator Interrupt Priority bit
     1 = Comparator Interrupt is a high priority event
     0 = Comparator Interrupt is a low priority event

bit  BCLIP: Bus Collision Interrupt Priority bit
     1 = Bus Collision Interrupt is a high priority event
     0 = Bus Collision Interrupt is a low priority event

bit  LVDIP: Low-voltage Detect Interrupt Priority bit
     1 = Low-voltage Detect Interrupt is a high priority event
     0 = Low-voltage Detect Interrupt is a low priority event

---

Legend
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- n = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown

Note 1: The bit position of the priority bits is device dependent. Please refer to the device data sheet for bit placement.
10.2.5 RCON Register

The RCON register contains the bit that is used to enable prioritized interrupts (IPEN) as well as status bits to indicate the cause of a device reset, if the device was in sleep mode and if long writes to internal memory are enabled.

Register 10-7: RCON Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Description</th>
<th>Value at POR reset</th>
<th>Bit Operation</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>IPEN: Interrupt Priority Enable bit</td>
<td>0</td>
<td>R/W-0</td>
<td>1 = Enable priority levels (high and low) on interrupts</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>R/W-1</td>
<td>0 = Disable priority levels (all peripherals are high) on interrupts</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>U-0</td>
<td>(PIC16CXXX compatibility)</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>R/W-1</td>
<td>(This causes the Interrupt Priority (IP) bits to be ignored)</td>
</tr>
<tr>
<td>6</td>
<td>LWRT: Long Write Enable</td>
<td>0</td>
<td>R/W-0</td>
<td>For details of bit operation see description of RCON register bit in Register 3-2</td>
</tr>
<tr>
<td>5</td>
<td>Unimplemented: Read as '0'</td>
<td></td>
<td>R/W-1</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td>RI: Reset Instruction Flag</td>
<td>0</td>
<td>R/W-0</td>
<td>For details of bit operation see description of RCON register bit in Register 3-2</td>
</tr>
<tr>
<td>3</td>
<td>TO: Watchdog Time-out Flag</td>
<td>0</td>
<td>R/W-0</td>
<td>For details of bit operation see description of RCON register bit in Register 3-2</td>
</tr>
<tr>
<td>2</td>
<td>PD: Power-down Detection Flag</td>
<td>0</td>
<td>R/W-0</td>
<td>For details of bit operation see description of RCON register bit in Register 3-2</td>
</tr>
<tr>
<td>1</td>
<td>POR: Power-on Reset Status</td>
<td>0</td>
<td>R/W-0</td>
<td>For details of bit operation see description of RCON register bit in Register 3-2</td>
</tr>
<tr>
<td>0</td>
<td>BOR: Brown-out Reset Status</td>
<td>0</td>
<td>R/W-0</td>
<td>For details of bit operation see description of RCON register bit in Register 3-2</td>
</tr>
</tbody>
</table>

Legend

R = Readable bit     W = Writable bit     U = Unimplemented bit, read as '0'
- n = Value at POR reset    '1' = bit is set    '0' = bit is cleared    x = bit is unknown
Section 10. Interrupts

10.3 Interrupt Handling Operation

The interrupts are controlled and monitored using several Special Function Registers. These may include the following register types:

- INTCON registers
- PIR registers
- PIE registers
- IPR registers

The PIR registers contain the interrupt flag bits, the PIE registers contain the enable bits and the IPR registers contain the priority bits. The number of PIR, PIE, and IPR registers depends on the number of interrupt sources on the device.

10.3.1 Interrupt Priority

Each interrupt can be assigned a priority level by clearing or setting the corresponding interrupt priority bit. The priority bits are located in the interrupt priority registers (IPR1, IPR2, IPR3, INTCON2 and INTCON3). A ‘1’ in the priority register assigns high priority to the corresponding interrupt. A ‘0’ in the register assigns low priority to the interrupt. All interrupt priority bits are reset to ‘1’, meaning that all interrupts are assigned high priority at reset. The IPEN bit in the RCON register enables priority levels for interrupts. If clear, all priorities are set to high.

10.3.1.1 High Priority Interrupts

A global interrupt enable bit, GIE/GIEH (INTCON<7>) enables (if set) all un-masked interrupts or disables (if cleared) all interrupts. When bit GIE/GIEH is enabled and an interrupt’s flag bit and enable bit are set while the priority is high, the interrupt will vector immediately. Individual interrupts can be disabled through their corresponding enable bits in various registers. Individual interrupt flag bits are set, regardless of the status of the GIE/GIEH bit. The GIE/GIEH bit is cleared on reset.

When a high priority interrupt is responded to, the GIE/GIEH bit is automatically cleared to disable any further interrupts, the return address is pushed onto the stack, and the PC is loaded with 000008h. Once in the interrupt service routine, the source of the interrupt can be determined by polling the interrupt flag bits. The interrupt flag bit(s) must be cleared before re-enabling interrupts to avoid recursive interrupts. Most flag bits are required to be cleared by the application software. There are some flag bits that are automatically cleared by the hardware.

The "return from interrupt" instruction, RETFIE, exits the interrupt routine and sets the GIE/GIEH bit, which re-enables high priority interrupts.

10.3.1.2 Low Priority Interrupts

Low priority interrupts are defined by having a “0” in an interrupt priority register IPRx. To enable low priority interrupts, the IPEN bit must be set.

When the IPEN is set, the PEIE/GIEL bit (INTCON<6>) is no longer used to enable peripheral interrupts. Its new function is to globally enable and disable low priority interrupts only. When the service routine for a low priority interrupt is vectored to, the PEIE/GIEL bit is automatically cleared in hardware to disable any further low priority interrupts.

The return address is pushed onto the stack and the PC is loaded with 000018h instead of 000008h (all low priority interrupts will vector to 000018h). Once in the interrupt service routine, the source(s) of the low priority interrupt can be determined by polling the low priority interrupt flag bits. The interrupt flag bit(s) must be cleared before re-enabling interrupts to avoid recursive interrupts. Most flag bits are required to be cleared by the application software. There are some flag bits that are automatically cleared by the hardware. The RETFIE instruction will reset the PEIE/GIEL bit on return from low priority interrupts.

The GIE/GIEH bit’s function has not changed in that it still enables/disables all interrupts, however, it is only cleared by hardware when servicing a high priority interrupt.
10.3.1.3 High Priority Interrupts Interrupting a Low Priority ISR

If a high priority interrupt flag and enable bits are set while servicing a low priority interrupt, the high priority interrupt will cause the low priority ISR to be interrupted (regardless of the state of the PEIE/GIEL bit), because it is used to disable/enable low priority interrupts only. The GIE/GIEH bit is cleared by hardware to disable any further high and low priority interrupts, the return address is pushed onto the stack, and the PC is loaded with 000008h (the high priority interrupt vector). Once in the interrupt service routine, the source of the high priority interrupt can be determined by polling the interrupt flag bits. The interrupt flag bit(s) must be cleared in software before re-enabling interrupts to avoid recursive interrupts.

Figure 10-4 shows a high priority interrupt interrupting a low priority ISR. Figure 10-5 shows a high priority FSR with a low priority interrupt pending.

Note: The GIEH bit, when cleared, will disable all interrupts regardless of priority.
Figure 10-6 and Figure 10-7 show the two cases where a low priority interrupt has occurred and then a high priority interrupt occurs before the low priority ISR can begin execution. Figure 10-8 shows the first instruction of the low priority interrupt (at address 18h) beginning execution, when the high priority interrupt causes the program counter to be forced to the high priority interrupt vector address (08h).

Figure 10-6: Low Interrupt With High Interrupt Within 1 Cycle

Figure 10-7: Low Interrupt With High Interrupt Within 2 Cycles
Figure 10-8: Low Interrupt With High Interrupt Within 3 Cycles

The diagram illustrates the timing and sequence of events for a low interrupt followed by a high priority interrupt within three cycles. The key elements include:

- **OSC1**: Oscillator Input
- **CLKOUT**: Clock Output
- **INT0 pin**: Input Pin for Interrupt 0
- **INT0IF**: Interrupt 0 Flag
- **GIE/GIEH**: Global Interrupt Enable
- **INT2 pin**: Input Pin for Interrupt 2
- **INT2IF**: Interrupt 2 Flag
- **PEIE/GIEL**: Peripheral Interrupt Enable

The diagram shows the execution flow, starting with an instruction fetched and executed, followed by a vector to a low priority interrupt, then a vector to a high priority interrupt, and finally a return from interrupt to the low priority interrupt service routine (ISR).

The timing is critical, with specific values for cycles and addresses highlighted to demonstrate the sequence of operations and the priority handling between interrupts.
Section 10. Interrupts

10.3.1.4 Low Priority Interrupts Interrupting a High Priority ISR

A low priority interrupt cannot interrupt a high priority ISR. The low priority interrupt will be served after all high priority interrupts have been served.

10.3.1.5 Simultaneous High and Low Priority Interrupts

If a high priority interrupt and a low priority interrupt are sampled at the same time, the high priority interrupt service routine is always serviced first. The GIE/GIEH bit is cleared by the hardware and the device vectors to location 000008h to the high priority ISR. After the interrupt is serviced, the corresponding interrupt flag should be cleared to avoid a recursive interrupt. The RETFIE instruction resets the GIE/GIEH bit, and if no other high priority interrupts are pending, the low priority interrupt is serviced.

10.3.1.6 Fast Context Saving During High Priority Interrupts

A “fast interrupt service” option is available for high priority interrupts. This is done by creating shadow registers for a few key registers (WREG, BSR, and STATUS). Shadow registers are provided for the STATUS, WREG, and BSR registers and are only 1 deep. The shadow registers are not readable and are loaded with the current value of their corresponding register when the processor vectors for a high priority interrupt. The values in the shadow registers are then loaded back into the actual register if the fast return instruction (RETFIE 0x01) is used to return from the interrupt. An example for fast context saving is shown in Example 10-1.

Example 10-1: Fast Context Saving

```
ORG 0x08
; Interrupt Service Routine (ISR) code. WREG, BSR and STATUS need
; to be saved upon entering the high priority interrupt service routine
; RETFIE 0x01 ; WREG, BSR and STATUS will be restored
```

Note: Fast interrupt saving cannot be used reliably if high and low priority interrupts are enabled. See Section 10.3.1.7.
10.3.1.7 Context Saving During Low Priority Interrupts

Low priority interrupts may use the shadow registers. Any interrupt pushes values into the shadow registers. If both low and high priority interrupts are enabled, the shadow registers cannot be used reliably for low priority interrupts, as a high priority interrupt event will overwrite the shadow registers.

Users must save the key registers in software during a low priority interrupt. For example:

a) Store the STATUS, WREG and BSR registers on a software stack.
b) Execute the ISR code.
c) Restore the STATUS, WREG and BSR registers from the software stack.
Section 10. Interrupts

Example 10-2 shows example service routine code for when high and low priority interrupts are enabled.

Example 10-2: Interrupt Service Routine Template

```assembly
ORG 0x08 ; high priority ISR
PUSH_REG_H MOVWF WREG_TEMP_HIGH
MOVFF BSR, BSR_TEMP_HIGH
MOVFF STATUS, STATUS_TEMP_HIGH

; High Priority Interrupt Service Routine (ISR) Code goes here

POP_REG_H MOVFF BSR_TEMP_HIGH, BSR
MOVF WREG_TEMP_HIGH, W
MOVFF STATUS_TEMP_HIGH, STATUS
RETFIE 0x00

; Low Priority Interrupt Service Routine (ISR) Code goes here

ORG 0x18 ; low priority ISR
PUSH_REG_L MOVWF WREG_TEMP_LOW
MOVFF BSR, BSR_TEMP_LOW
MOVFF STATUS, STATUS_TEMP_LOW

; Low Priority Interrupt Service Routine (ISR) Code goes here

POP_REG_L MOVFF BSR_TEMP_LOW, BSR
MOVF WREG_TEMP_LOW
MOVFF STATUS_TEMP_LOW, STATUS
RETFIE 0x00
```
10.3.1.8 Interrupt Latency

For external interrupt events, such as the RB0/INT0 pin or PORTB change interrupt, the interrupt latency will be three or four instruction cycles. The exact latency depends when the interrupt event occurs. The interrupt latency is the same for one or two cycle instructions.

10.3.1.8.1 Interrupt Latency For One Cycle Instructions

Figure 10-9 shows the timing when an external interrupt is asserted during a one cycle instruction. The interrupt is sampled on Q4. The interrupt is then acknowledged on the Q2 cycle of the following instruction cycle when instruction PC is executed. This is followed by a forced NOP (dummy cycle) and the contents of the PC are stored on the stack during the Q3 cycle of this machine cycle. By the Q3/Q4 boundary of instruction cycle two, the interrupt vector is placed into the PC, and is presented on the program memory bus on the following cycle. This cycle is also a dummy cycle executing a forced NOP (\texttt{FNOP}) so that the CPU can fetch the first instruction from the interrupt service routine.

**Figure 10-9: Interrupt Flow on a 1 Cycle Instruction**

<table>
<thead>
<tr>
<th>PC</th>
<th>PC+2</th>
<th>PC+2</th>
<th>0008h</th>
<th>000Ah</th>
<th>000Ch</th>
</tr>
</thead>
<tbody>
<tr>
<td>INST(PC)</td>
<td>INST(PC+2)</td>
<td>INST(PC+2)</td>
<td>INST(0008h)</td>
<td>INST(000Ah)</td>
<td></td>
</tr>
<tr>
<td>INST(PC-1)</td>
<td>INST(0008h)</td>
<td>INST(000Ah)</td>
<td>INST(000Ah)</td>
<td></td>
<td></td>
</tr>
<tr>
<td>INST Execute</td>
<td>INST Execute</td>
<td>INST Execute</td>
<td>INST Execute</td>
<td></td>
<td></td>
</tr>
<tr>
<td>INST Execute</td>
<td>INST Execute</td>
<td>INST Execute</td>
<td>INST Execute</td>
<td></td>
<td></td>
</tr>
<tr>
<td>INTxIF flag</td>
<td>FNOP</td>
<td>FNOP</td>
<td>FNOP</td>
<td>FNOP</td>
<td></td>
</tr>
<tr>
<td>GIE/GIEH bit</td>
<td>Executed here</td>
<td>Executed here</td>
<td>Executed here</td>
<td>Executed here</td>
<td>Executed here</td>
</tr>
<tr>
<td>STACK</td>
<td>RAM</td>
<td>register</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
Section 10. Interrupts

10.3.1.8.2 Interrupt Latency For Two Cycle Instructions

Figure 10-10 shows the timing when an external interrupt is asserted during a two cycle instruction. The interrupt is sampled on Q4. The interrupt is then acknowledged on the Q1 of the following instruction cycle when instruction PC is executed. This is followed by the second cycle of the instruction and the contents of the PC are stored on the stack during Q3 of this machine cycle. For all two cycle instructions, the PC may be updated with a new PC value due to execution control instructions like GOTO and CALL. The reason for the forced NOP (dummy cycle) is to maintain consistent interrupt latency between one and two cycle instructions. Two cycle instructions require this cycle for the update of the PC to a new PC value, because all two cycle instructions with the exception of MOVFF and MOVLF are execution control type instructions that update the PC with a new value (i.e., GOTO and CALL). The MOVFF and MOVLF instructions will increment the PC by 2 in this cycle because an operand fetch takes place in the second cycle. By Q3/Q4 the interrupt vector 000008h is placed into the PC and is presented on the program memory bus on the following cycle. This cycle is a dummy cycle executing a forced NOP (FNOP) so that the CPU can fetch the first instruction from the interrupt service routine.

Note: When using the MOVFF instruction with any one of the PCL, TOSU, TOSH, and TOSL registers as destination, all interrupts have to be disabled.

Figure 10-10: Interrupt Flow on a 2 Cycle or 2 Word Instruction
10.3.1.9 InTerrupts During Table Write Operations (Long Writes)

The long write is necessary for programming the internal EPROM. Instruction execution is halted while in a long write cycle. The long write will be terminated by any enabled interrupt. To ensure that the EPROM location has been well programmed, a minimum programming time is required. Typically, a Timer interrupt is used to time and terminate the long write. Having only one interrupt enabled to terminate the long write ensures that no unintended interrupts will prematurely terminate the long write.

Figure 10-11: INT0, INT1, and INT2 Pin Interrupt Timing (High Priority Shown)

Note 1: INTxIF flag is sampled here (every Q1).
Note 2: Interrupt latency = 3-4T CY where T CY = instruction cycle time. Latency is the same whether Inst (PC) is a single cycle or a 2-cycle instruction.
Note 3: CLKOUT is available only in RC oscillator mode.
Note 4: For minimum width of INT pulse, refer to AC specs.
Note 5: INTxIF is enabled to be set anytime during the Q1 cycle.
Section 10. Interrupts

10.4 Initialization

Example 10-3 enables high and low priority interrupts. The priority level for the peripherals is loaded into the IRP1 register (IRP1_VALUE) and the peripherals that are enabled depend on the value of PIE1_VALUE, which is loaded into the PIE1 register.

Example 10-3: Generic Initialization Example

```
MOVLW RCON_VALUE ; RCON_VALUE = 1??????b
MOVF RCON ;
MOVLW IPR1_VALUE ; Peripherals with high priority
; have a ‘1’ in their bit
; position.
; Those with a low priority have
; a ‘0’ in their bit position.
MOVF IRF1 ;
CLRF PIR1 ; Clear all flag bits
MOVLW PIE1_VALUE ; Enable desired peripheral
; interrupts by setting their
; bit position.
; Disable others by clearing their
; bit position.
MOVF PIE1 ;
CLRF INTCN3 ;
CLRF INTCN3 ;
MOVLW 0xC0 ; Enable high and low global
; interrupts.
MOVWF INTCN ;
```
10.5 Design Tips

Question 1: My code does not seem to execute properly.

Answer 1:

There are many possible reasons. A couple of possibilities related to interrupts are:

• Interrupts are not enabled, so the code cannot execute your expected ISR.
• The Interrupt may not be set to the priority level where your ISR code is located.
Section 10. Interrupts

10.6 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the interrupts are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related application notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
10.7 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Interrupt description.
Section 11. I/O Ports

HIGHLIGHTS

This section of the manual contains the following major topics:

11.1 Introduction ............................................................................................................... 11-2
11.2 PORTA, TRISA, and the LAT A Register ................................................................... 11-8
11.3 PORTB, TRISB, and the LAT B Register ................................................................. 11-12
11.4 PORTC, TRISC, and the LAT C Register ............................................................... 11-16
11.5 PORTD, LATD, and the TRISD Register ............................................................... 11-19
11.6 PORTE, TRISE, and the LATE Register ................................................................ 11-21
11.7 PORTF, LATF, and the TRISF Register .................................................................. 11-23
11.8 PORTG, LATG, and the TRISG Register .................................................................. 11-25
11.9 PORTH, LATH, and the TRISH Register .................................................................. 11-27
11.10 PORTJ, LATJ, and the TRISJ Register .................................................................. 11-29
11.11 PORTK, LATK, and the TRISK Register ............................................................... 11-31
11.12 PORTL, LATL, and the TRISL Register ................................................................ 11-33
11.14 I/O Programming Considerations ......................................................................... 11-37
11.15 Initialization ............................................................................................................ 11-40
11.16 Design Tips ............................................................................................................ 11-41
11.17 Related Application Notes .................................................................................... 11-43
11.18 Revision History .................................................................................................... 11-44
11.1 Introduction

General purpose I/O pins can be considered the simplest of peripherals. They allow the PICmicro to monitor and control other devices. To add flexibility and functionality to a device, some pins are multiplexed with an alternate function(s). These functions depend on which peripheral features are on the device. In general, when a peripheral is functioning, that pin may not be used as a general purpose I/O pin.

For most ports, the I/O pin’s direction (input or output) is controlled by the data direction register, called the TRIS register. TRIS<<X>> controls the direction of PORT<<X>>. A ‘1’ in the TRIS bit corresponds to that pin being an input, while a ‘0’ corresponds to that pin being an output. An easy way to remember is that a ‘1’ looks like an I (input) and a ‘0’ looks like an O (output).

The PORT register is the latch for the data to be output. When the PORT is read, the device reads the levels present on the I/O pins (not the latch). This means that care should be taken with read-modify-write commands on the ports and changing the direction of a pin from an input to an output.

Figure 11-1 shows a typical I/O port. This does not take into account peripheral functions that may be multiplexed onto the I/O pin. Reading the PORT register reads the status of the pins whereas writing to it will write to the port latch. All write operations (such as BSF and BCF instructions) are read-modify-write operations. Therefore, a write to a port implies that the port pins are read, this value is modified, and then written to the port data latch.

**Figure 11-1: Typical I/O Port**

![Typical I/O Port Diagram]

Note: I/O pins have protection diodes to VDD and VSS.
When peripheral functions are multiplexed onto general I/O pins, the functionality of the I/O pins may change to accommodate the requirements of the peripheral module. An example of this is the Analog to Digital converter module which forces the I/O pin to the peripheral function when the device is reset. This prevents the device from consuming excess current if any analog levels were on the A/D pins after a reset occurred.

With some peripherals, the TRIS bit is overridden while the peripheral is enabled. Therefore, read-modify-write instructions (BSF, BCF, XORWF) with TRIS as destination should be avoided. The user should refer to the corresponding peripheral section for the correct TRIS bit settings.

PORT pins may be multiplexed with analog inputs and analog V_REF inputs. The operation of each of these pins is selected, to be an analog input or digital I/O, by clearing/setting the control bits in other Special Function registers (SFRs). An example of this is the ADCON1 register for the 10-bit A/D module. Currently, when devices have pins selected as an analog input, these pins will read as '0's.

The TRIS registers control the direction of the port pins, even when they are being used as analog inputs. The user must ensure the TRIS bits are maintained set when using the pins as analog inputs.

Note 1: If pins are multiplexed with analog inputs, then on a Power-on Reset these pins are configured as analog inputs, as controlled by the ADCON1 register. Reading port pins configured as analog inputs read a '0'.

Note 2: If pins are multiplexed with comparator inputs, then on a Power-on Reset these pins are configured as analog inputs, as controlled by the CMCON register. Reading port pins configured as analog inputs read a '0'.

Note 3: Pins may be multiplexed with the Parallel Slave Port (PSP). For the PSP to function, the I/O pins must be configured as digital inputs and the PSPMODE bit must be set.

Note 4: At present, the Parallel Slave Port (PSP) is only multiplexed onto PORTD and PORTE. The PSP port becomes enabled when the PSPMODE bit is set. In this mode, the user must make sure that the TRISE bits are set (pins are configured as digital inputs) and that PORTE is configured for digital I/O. PORTD will override the values in the TRISD register. In this mode, the PORTD and PORTE input buffers are TTL. The control bits for the PSP operation are located in TRISE.
11.1.1 Multiplexed Peripherals

Pins may be configured as either digital inputs or digital outputs. Digital inputs are either TTL buffers or Schmitt Triggers. Outputs are CMOS drivers except for pin RA4, which is an open-drain output.

All pins also support one or more peripheral modules. When configured to operate with a peripheral, a pin may not be used for general input or output. In many cases, a pin must still be configured for input or output, although some peripherals override the TRIS configuration.

Peripherals supported include:
- Analog to Digital Converter Modules (A/D)
- Timer Modules
  - Timer0
  - Timer1
  - Timer2
  - Timer3
- Capture/Compare/Pulse Width Modulation (CCP) modules
- External Interrupts
- Interrupt On Change pins
- Parallel Slave Port (PSP) module
- In Circuit Serial Programming
- System Oscillator
- Weak Pull-Up sources
- Synchronous Serial Port (SSP) module
  - Serial Peripheral Interface (SPI)
  - \(^2\text{C}\)
- Master Synchronous Serial Port (MSSP) module
  - Serial Peripheral Interface (SPI)
  - \(^2\text{C}\) with full hardware Master mode support
- Addressable USART module
- Controller Area Network (CAN) module
- Comparator modules
- Voltage Reference modules
- Low Voltage Detect (LVD) module
Section 11. I/O Ports

11.1.2 Output Data Latches (LATx) and Data Direction Register (TRISx)

All port pins have an output data latch. Writing to a port writes to that latch (LATx). The data latch may also be read from and written to directly. If the pin is not being used by a peripheral, and is configured as an output by its TRIS bit, data in the latch will be output to the pin.

All port pins have corresponding Data Direction Register bits (TRISx register) which configure each pin as an input or output. Clearing a bit in a TRIS register (bit=0) configures the corresponding pin for output, and drives the contents of the output data latch (LATx) to the selected pin. Setting a TRIS register bit (bit=1) configures the corresponding pin as an input, and puts the corresponding output driver in a high impedance state. After a reset, all pins are configured as inputs.

Example 11-1 shows that writing the value in the WREG register to PORTB actually writes the value to the LATB register.

Example 11-1: Writing to PORTB actually writes to LATB

Example 11-2 demonstrates the difference between reading a PORT and reading the output latch of the PORT.

Example 11-2: Reading PORTB compared to reading LATB

Reading the PORTx register reads the status of the pins whereas writing to it will write to the port data latch (LATx). A write to LATx can also be performed.

Example 11-3 shows the result of simply reading the PORT register. In this example, RB0 is being overdrive low and RB1 is being overdriven high. This is NOT recommended, and may actually violate device specifications, but is shown to give insight to the operation of an instruction which reads the I/O port with respect to the I/O ports data latch.

Example 11-3: Reading PORTB reads the state of RB7:RB0
Example 11-4 shows what effects can occur when writing to the PORT registers.

**Example 11-4: Writing to PORTB**

| ; TRISB = 1111 0000  l=input 0=output |
| ; W_REG = 1011 0110 |
| ; LATB = 1100 0011 |
| ; RB<7:0> = 1001 0011 |
| movwf PORTB ; writes W_REG to LATB |
| ; LATB = 1011 0110 |
| ; RB<7:0> = 1001 0110 low nibble only |
| ; is output |

Example 11-5 shows what effects can occur when writing to the LAT registers.

**Example 11-5: Writing to LATB**

| ; TRISB = 1111 0000  l=input 0=output |
| ; W_REG = 1011 0110 |
| ; LATB = 1100 0011 |
| ; RB<7:0> = 1001 0011 |
| movwf LATB ; writes W_REG to LATB |
| ; LATB = 1011 0110 |
| ; RB<7:0> = 1001 0110 same result as |
| ; 'movwf PORTB' |

Any instruction that performs a write operates internally as a read-modify-write operation. Caution must be used when these instructions are applied to a port where pins are switching between input and output states.

For example, a BSF PORTB, 5 instruction will cause all eight bits of PORTB to be read into the CPU. Then the instruction sets bit 5 and the resulting data is written to LATB.

If the RB7 pin is used for bi-directional I/O and is defined as an input when BSF PORTB, 5 executes, the input signal present on the pin itself would be read into the CPU and be written to LATB<7>, overwriting the previous contents. As long as the RB7 pin stays in the input mode, no problem occurs. However, if the RB7 pin is switched to an output, the contents of the data latch may be in an unintended state, causing the RB7 pin to be in the wrong state.

Example 11-6 shows how read-modify-write operations can affect the PORT register or the TRIS register.

**Example 11-6: Read-modify-write of PORTB, and TRISB change toggles RB7**

| ; RB<7:0> = 0001 0110 |
| ; LATB = 1001 0110 |
| ; TRISB = 1100 0000 |
| bsf PORTB,5 ; read-modify-write operation. |
| ; LATB = 0011 0110 bit 7 cleared |
| ; RB<7:0> = 1011 0110 RB7 changes to high speed |
| bcf TRISB,7 ; changes RB7 from input to output |
| ; TRISB = 0100 0000 |
| ; RB<7:0> = 0011 0110 RB7 in now driven low |

A better solution would be to use the data latch instead. A BSF LATB, 5 instruction will read the bits in the output latch, set bit 5, and write the results back to the output latch. LATB<7> will never be at risk of being changed.
Section 11. I/O Ports

Example 11-7 shows that doing read-modify-writes on the LATx register and TRISx register may not cause the voltage level on the pin to change.

Example 11-7: Read-modify-write of LATB, and TRISB change has no effect on RB7

```assembly
bsf LATB,5 ; read-modify-write operation
LATB = 1011 0110 bit 7 has not changed
bcf TRISB,7 ; changes RB7 from input to output
TRISB = 0100 0000
RB<7:0> = 1011 0110 RB7 remains high
```
11.2 PORTA, TRISA, and the LATA Register

PORTA is a 6-bit, or 7-bit latch depending upon the oscillator configuration selected by the Fosc configuration bits. The corresponding data direction register is TRISA, the data output latch is LATA, and the pins are PORTA. Except for RA4, all PORTA pins have TTL input buffers and full CMOS output drivers. All pins are configured as inputs on a reset.

The RA4 pin is a Schmitt Trigger input and an open drain output. All other RA port pins have TTL input levels and full CMOS output drivers. All pins have data direction bits (TRIS registers) which can configure these pins as output or input.

Setting a TRISA register bit puts the corresponding output driver in a hi-impedance mode. Clearing a bit in the TRISA register puts the contents of the output latch on the selected pin(s).

Example 11-8: Initializing PORTA

```
CLRF PORTA          ; Initialize PORTA by clearing output
                   ; data latches
; CLRF LATA          ; Alternate method to initialize PORTA
MOVWF TRISA        ; PORTA<3:0> = inputs PORTA<5:4> = outputs
                   ; TRISA<7:6> always read as '0'
```

11.2.1 PORTA multi-plexed with Analog inputs

PORTA may be multiplexed with the AD module. When used as analog inputs, the TRISA must configure the corresponding pins as digital inputs (‘1’ on TRIS bit). On all resets, the PORTA pins are configured as analog inputs and a read of the digital inputs will result in read values of ‘0’.
Section 11. I/O Ports

Figure 11-2: Block Diagram of RA3:RA0 and RA5 Pins

Note: I/O pins have protection diodes to VDD and VSS.
11.2.2 RA4 / Timer0 Clock Input

The RA4/T0CKI pin is a Schmitt Trigger input and an open drain output. Pin RA4 may be multiplexed with the peripheral module. All other PORTA pins have TTL input levels and CMOS output drivers.

Figure 11-3: Block Diagram of RA4 Pin

Note: I/O pins have protection diodes to Vss only.
Table 11-1: PORTA Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit #</th>
<th>Buffer</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RA0</td>
<td>bit0</td>
<td>TTL</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RA1</td>
<td>bit1</td>
<td>TTL</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RA2</td>
<td>bit2</td>
<td>TTL</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RA3</td>
<td>bit3</td>
<td>TTL</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RA4/T0CKI</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin or external clock input for Timer0</td>
</tr>
<tr>
<td>RA5</td>
<td>bit5</td>
<td>TTL</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RA6</td>
<td>bit6</td>
<td>TTL</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: TTL = TTL input, ST = Schmitt Trigger input.

Table 11-2: Summary of Registers Associated with PORTA

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISA</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>111 1111</td>
<td>111 1111</td>
</tr>
<tr>
<td>PORTA</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>00x 0000</td>
<td>00u 0000</td>
</tr>
<tr>
<td>LAT0</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>xxx xxxx</td>
<td>uuu uuuu</td>
</tr>
<tr>
<td>ADCON0</td>
<td>ADFM</td>
<td>ADCS2</td>
<td>—</td>
<td>—</td>
<td>PCFG3</td>
<td>PCFG2</td>
<td>PCFG1</td>
<td>PCFG0</td>
<td>00-- 0000</td>
<td>00-- 0000</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged, - = unimplemented locations read as '0'.

Shaded cells are not used by PORTA. ST = Schmitt Trigger input, TTL = TTL input.
11.3 PORTB, TRISB, and the LATB Register

PORTB is an 8-bit wide bidirectional port. The corresponding data direction register is TRISB, the data output latch is LATB, and the pins are PORTB. All pins have TTL inputs.

Setting a bit in the TRISB register puts the corresponding output driver in a hi-impedance input mode. Clearing a bit in the TRISB register puts the contents of the output latch on the selected pin. All pins are configured as inputs on a reset.

Four of the PORTB pins have a weak internal pull-up. Clearing the RBPU bit (INTCON2<7>) turns on pull-ups on all pins. The weak pull-up is automatically turned off when the port pin is configured as an output. The pull-ups are disabled on a reset.

Example 11-9: Initializing PORTB

```
CLRF PORTS ; Initialize PORTS by clearing output
; data latches
CLRF LATB ; Alternate method to initialize data latches
MOVLW 0xCF ; Value used to initialize data direction
MOVWF TRISB ; PORTB<3:0> = inputs, PORTB<5:4> = outputs
; PORTB<7:6> = inputs
```

11.3.1 RB2:RB0 / External Interrupts INT2:INT0

RB2:RB0 pins can also function as external interrupt sources INT2:INT0 while working as digital inputs. These interrupts are edge triggered on the edges selected by the bits. If enabled prior to entering sleep mode, these interrupts can wake the controller.

INT2:INT0 inputs have Schmitt trigger inputs, while the RB2:RB0 inputs have TTL buffer inputs.

Figure 11-4: Block Diagram of RB3:RB0 Pins

Note 1: I/O pins have diode protection to VDD and VSS.
Note: To enable weak pull-ups, set the appropriate TRIS bit(s) and clear the RBPU bit.
Section 11. I/O Ports

Four of PORTB's pins, RB7:RB4, have an interrupt on change feature. Only pins configured as inputs can cause this interrupt to occur (i.e., any RB7:RB4 pin configured as an output is excluded from the interrupt on change comparison). The input pins (of RB7:RB4) are compared with the old value latched on the last read of PORTB. The present inputs of RB7:RB4 and their previous values are XOR'ed together to detect a "mismatch" condition and set the RB Port change interrupt flag bit RBIF. When enabled, this flag will generate an interrupt that can wake the device from SLEEP.

This interrupt can wake the device from SLEEP. The user, in the interrupt service routine, can clear the interrupt in the following manner:

a) Any read or write of PORTB will end the mismatch condition, except a write using the MOVFF instruction.
b) Clear flag bit RBIF.

The MOVFF instruction will not end the mismatch condition if PORTB is used only as the destination register. The contents of the destination register are not automatically read by this instruction in the second cycle. All other reads, writes, and bit operations will read the port during execution.

A mismatch condition will continue to set flag bit RBIF. Reading PORTB will end the mismatch condition, and allow flag bit RBIF to be cleared.

This interrupt on change (i.e., mismatch) feature, together with software configurable pull-ups on these four pins allow easy interface to a keypad and make it possible for wake-up on key-depression.

The interrupt on change feature is recommended for wake-up on key depression and operations where PORTB is only used for the interrupt on change feature. Polling of PORTB is not recommended while using the interrupt on change feature.
11.3.2 RB7:RB6 - In Circuit Serial Programming

If ICSP is implemented in the target application, some means of isolating RB7:RB6 from the rest of the circuit should be provided. The ISCP inputs have Schmitt Triggers while the RB7:RB6 inputs have TTL inputs.

Note 1: I/O pins have diode protection to VDD and VSS.
Note 2: To enable weak pull-ups, set the appropriate TRIS bit(s) and clear the RBPU bit.
Section 11. I/O Ports

Table 11-3: PORTB Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit</th>
<th>Buffer</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RB0/INT0</td>
<td>bit0</td>
<td>TTL/ST (1)</td>
<td>Input/output port pin or external interrupt0 input. Internal software programmable weak pull-up.</td>
</tr>
<tr>
<td>RB1/INT1</td>
<td>bit1</td>
<td>TTL/ST (1)</td>
<td>Input/output port pin or external interrupt1 input. Internal software programmable weak pull-up.</td>
</tr>
<tr>
<td>RB2/INT2</td>
<td>bit2</td>
<td>TTL/ST (1)</td>
<td>Input/output port pin or external interrupt2 input. Internal software programmable weak pull-up.</td>
</tr>
<tr>
<td>RB3/CCP2</td>
<td>bit3</td>
<td>TTL/ST (4)</td>
<td>Input/output port pin or Capture2 input/Compare2 output/PWM2 output if CCP2MX is enabled in the configuration register. Internal software programmable weak pull-up.</td>
</tr>
<tr>
<td>RB4</td>
<td>bit4</td>
<td>TTL</td>
<td>Input/output port pin (with interrupt on change). Internal software programmable weak pull-up.</td>
</tr>
<tr>
<td>RB5</td>
<td>bit5</td>
<td>TTL</td>
<td>Input/output port pin (with interrupt on change). Internal software programmable weak pull-up.</td>
</tr>
<tr>
<td>RB6</td>
<td>bit6</td>
<td>TTL/ST (2)</td>
<td>Input/output port pin (with interrupt on change). Internal software programmable weak pull-up. Serial programming (CLOCK).</td>
</tr>
<tr>
<td>RB7</td>
<td>bit7</td>
<td>TTL/ST (2)</td>
<td>Input/output port pin (with interrupt on change). Internal software programmable weak pull-up. Serial programming (DATA).</td>
</tr>
</tbody>
</table>

Legend: TTL = TTL input, ST = Schmitt Trigger input.

Note 1: This buffer is a Schmitt Trigger input when configured as the external interrupt.
Note 2: This buffer is a Schmitt Trigger input when used in serial programming mode.
Note 3: The CCP2 input is only multiplexed on the RB3 pin if the CCP2MX configuration bit is '0'.
Note 4: The CCP2 input is a Schmitt Trigger if the CCP2MX configuration bit is '0'.

Table 11-4: Summary of Registers Associated with PORTB

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISB</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111 1111 1111</td>
<td></td>
</tr>
<tr>
<td>PORTB</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxxxxx xxxxxxxx xxxxxxxx xxxxxxxx</td>
<td></td>
</tr>
<tr>
<td>LATB</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxxxxx xxxxxxxx xxxxxxxx xxxxxxxx</td>
<td></td>
</tr>
<tr>
<td>INTCON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
<tr>
<td>INTCON2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 -1-1 1111 -1-1</td>
<td></td>
</tr>
<tr>
<td>INTCON3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>11-0 0-00 11-0 0-00</td>
<td></td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged, - = unimplemented locations read as '0'.
Shaded cells are not used by PORTB.
11.4 PORTC, TRISC, and the LATC Register

PORTC is an 8-bit bi-directional port. Each pin is individually configurable as an input or output through the TRISC register. The data output latch is LATC. PORTC pins have Schmitt Trigger input buffers.

When enabling peripheral functions, care should be taken in defining TRIS bits for each PORTC pin. Some peripherals override the TRIS bit to make a pin an output, while other peripherals override the TRIS bit to make a pin an input, and other peripherals may not override the TRIS bits (requires that TRIS bits are configured for proper peripheral operation). The user should refer to the corresponding peripheral section for the correct TRIS bit settings.

Example 11-10: Initializing PORTC

```
CLRF PORTC ; Initialize PORTC
; by clearing output data latches
; CLRF LATC ; Alternate method
; to clear output latch
MOVlw 0xCF ; Value used to initialize data direction
MOVWF TRISC ; PORTC<3:0> = inputs,
; PORTC<5:4> = outputs,
; PORTC<7:6> = inputs
```

Figure 11-6: PORTC Block Diagram (Peripheral Output Override)

Note 1: Port/Peripheral select signal selects between port data and peripheral output. Peripheral OE (output enable) is only activated if peripheral select is active. I/O pins have diode protection to VDD and VSS.
Section 11. I/O Ports

All PORTC pins have Schmitt Trigger input buffers. When a peripheral uses a pin for output, the peripheral will override the TRIS and force the pin to be an output. Conversely, when a peripheral uses a pin for input, the peripheral will override the TRIS and force the pin to be an input. The TRIS is ignored while the peripheral controls the pin.

A read from the TRISC register bits will always yield the value contained in the TRISC latch whether or not a peripheral TRIS override is being asserted. This will allow a user to read the status of the TRISC bits at all times.

Since the TRIS bit override is in effect when the peripheral is enabled, read-modify-write instructions (BSF, BCF and others) with TRIS as destination should be used with care.

These instructions will have no effect on the current state of the pin. However, prior to disabling the peripheral and returning the pin to general use, the user should ensure that the TRIS bit is correctly set for that pin.

When a peripheral uses a pin for output, the peripheral will override the TRIS and force the pin to be an output. Conversely, when a peripheral uses a pin for input, the peripheral will override the TRIS and force the pin to be an input. The TRIS is ignored while the peripheral controls the pin.

A read from the TRISC register bits will always yield the value contained in the TRISC latch whether or not a peripheral TRIS override is being asserted. This will allow a user to read the status of the TRISC bits at all times.

Since the TRIS bit override is in effect when the peripheral is enabled, read-modify-write instructions (BSF, BCF, and others) with TRIS as destination should be used with care.

These instructions will have no effect on the current state of the pin. However, prior to disabling the peripheral and returning the pin to general use, the user should ensure that the TRIS bit is correctly set for that pin.

Figure 11-7: PORTC Block Diagram (Peripheral Output Override)
11.4.1 RC1 / CCP2 Input / Output

The RC1 pin can be multiplexed with the CCP2 module input/output. To achieve this, the CCP2MX Configuration bit must be programmed to a ‘1’.

Table 11-5: PORTC Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RC0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin or Timer1 oscillator output or Timer1/Timer3 clock input</td>
</tr>
<tr>
<td>RC1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin or Timer1 oscillator input</td>
</tr>
<tr>
<td>RC2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin or Capture1 input/Compare1 output/PWM1 output</td>
</tr>
<tr>
<td>RC3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RC4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RC5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RC6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RC7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input.

Table 11-6: Summary of Registers Associated with PORTC

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISC</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>PORTC</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxxxxxx</td>
<td>xxxxxxxx</td>
</tr>
<tr>
<td>LATC</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxxxxxx</td>
<td>xxxxxxxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.
Section 11. I/O Ports

11.5 PORTD, LATD, and the TRISD Register

PORTD is an 8-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output.

All PORTD pins have latch bits (LATD register). The LATD register, when read, will yield the contents of the PORTD latch, and when written, will modify the contents of the PORTD latch. This modifies the value driven out on a pin if the corresponding TRISD bit is configured for output. This can be used in read-modify-write instructions that allow the user to modify the contents of the latch register regardless of the status of the corresponding pins.

Example 11-11: Initializing PORTD

```assembly
CLRF PORTD ; Initialize PORTD
; by clearing output data latches
; CLRF LATD ; Alternate method to initialize
; data output latch
MOVWF TRISD ; PORTD<3:0> = inputs,
; PORTD<5:4> = outputs,
; PORTD<7:6> = inputs
```

Figure 11-8: Typical PORTD Block Diagram (in I/O Port Mode)

Note 1: I/O pins have protection diodes to VDD and VSS.
### Table 11-7: PORTD Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RD0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RD7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input, TTL = TTL input.

### Table 11-8: Summary of Registers Associated with PORTD

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>PORTD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxxxx</td>
<td>xxxxxxx xxxxx</td>
</tr>
<tr>
<td>LATD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxxxx</td>
<td>xxxxxxx xxxxx</td>
</tr>
<tr>
<td>PSPCON (1)</td>
<td>IBF</td>
<td>OBF</td>
<td>IBOV</td>
<td>PSPMODE</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>0000 xxxx</td>
<td>0000 xxxxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.

**Note 1:** In some devices, the four bits in the PSPCON register may be located in the upper four bits of the TRISE register.
Section 11. I/O Ports

11.6 PORTE, TRISE, and the LATE Register

PORTE can be up to an 8-bit port with Schmitt Trigger input buffers. Each pin is individually configurable as an input or output.

Example 11-12: Initializing PORTE

```
CLRF PORTE   ; Initialize PORTE by clearing output
; CLRF LATE   ; Alternate method to initialize
MOVlw 0x03    ; data output latch
MOVwf TRISE   ; PORTE<1:0> = inputs,
; PORTE<7:2> = outputs

CLRF PORTE ; Initialize PORTE by clearing output
```

Figure 11-9: Typical PORTE Block Diagram (in I/O Port Mode)

Note 1: I/O pins have protection diodes to VDD and VSS.

Note: On some devices with PORTE, the upper bits of the TRISE register are used for the Parallel Slave Port control and status bits.
### Table 11-9: PORTE Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RE0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RE1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RE2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input, TTL = TTL input.

### Table 11-10: Summary of Registers Associated with PORTE

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISE</td>
<td>IBF</td>
<td>OBF</td>
<td>IBOV</td>
<td>PSP-MODE</td>
<td>—</td>
<td>PORTE Data Direction Bits</td>
<td>0000 111</td>
<td>0000 111</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PORTE</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>RE2</td>
<td>RE1</td>
<td>RE0</td>
<td>—000 —000</td>
<td>— — — —</td>
</tr>
<tr>
<td>LATE</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—xxx —xxx</td>
<td>— — — —</td>
</tr>
<tr>
<td>ADCON1</td>
<td>ADFM</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>PCFG2</td>
<td>PCFG1</td>
<td>PCFG0</td>
<td>—0 0000 000</td>
<td>—0 0000</td>
</tr>
<tr>
<td>PSPCON (1)</td>
<td>IBF</td>
<td>OBF</td>
<td>IBOV</td>
<td>PSP-MODE</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>0000 xxxx 0000</td>
<td>xxxx xxxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.

**Note 1:** In some devices, the four bits in the PSPCON register may be located in the upper four bits of the TRISE register.
Section 11. I/O Ports

11.7 PORTF, LATF, and the TRISF Register

PORTF is an 8-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output.

All PORTF pins have latch bits (LATF register). The LATF register, when read, will yield the contents of the PORTF latch, and when written, will modify the contents of the PORTF latch. This modifies the value driven out on a pin if the corresponding TRISF bit is configured for output. This can be used in read-modify-write instructions that allow the user to modify the contents of the latch register regardless of the status of the corresponding pins.

PORTF pins are multiplexed with analog inputs, system bus address bits, chip enables, and the UB and LB external bus control signals. The operation of each analog pin is selected by clearing/setting the control bits in the ADCON0 and ADCON1 register.

The TRISF register controls the direction of the RF pins, even when they are being used as analog inputs. The user must ensure the bits in the TRISF register are maintained set when using them as analog inputs.

Note: On all forms of Reset, the RF2:RF0 are configured as analog inputs and read as ‘0’.

Figure 11-10: RF1:RF0 Block Diagram

Note 1: I/O pins have protection diodes to VDD and VSS.
### Table 11-11: PORTF Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RF0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RF7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input.

### Table 11-12: Summary of Registers Associated with PORTF

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>PORTF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxx xx00</td>
<td>uuuu u000</td>
</tr>
<tr>
<td>LATF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>uuuu u000</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged. Shaded cells are not used by Port F.
PORTG, LATG, and the TRISG Register

PORTG is a 5-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output.

All PORTG pins have latch bits (LATG register). The LATG register, when read, will yield the contents of the PORTG latch, and when written, will modify the contents of the PORTG latch. This modifies the value driven out on a pin if the corresponding TRISG bit is configured for output. This can be used in read-modify-write instructions that allow the user to modify the contents of the latch register regardless of the status of the corresponding pins.

Figure 11-11: PORTG Block Diagram

Note 1: I/O pins have protection diodes to VDD and VSS.
### Table 11-13: PORTG Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RG0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RG7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input.

### Table 11-14: Summary of Registers Associated with PORTG

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>---1 1111</td>
<td>---1 1111</td>
</tr>
<tr>
<td>PORTG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>---x xxxx</td>
<td>---u uuuu</td>
</tr>
<tr>
<td>LATG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>---x xxxx</td>
<td>---u uuuu</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.
11.9 PORTH, LATH, and the TRISH Register

PORTH is an 8-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output. All PORTH pins have latch bits (LATH register). The LATH register, when read, will yield the contents of the PORTH latch, and when written, will modify the contents of the PORTH latch. This modifies the value driven out on a pin if the corresponding TRISH bit is configured for output. This can be used in read-modify-write instructions that allow the user to modify the contents of the latch register regardless of the status of the corresponding pins.

Figure 11-12: PORTH Block Diagram

Note 1: I/O pins have protection diodes to VDD and VSS.
## Table 11-15: PORTH Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RH0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RH7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: TTL = TTL input.

## Table 11-16: Summary of Registers Associated with PORTH

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>PORTH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 xxxx</td>
<td>0000 uuuu</td>
</tr>
<tr>
<td>LATH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxx x xxx</td>
<td>xxx xxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged, - = unimplemented. Shaded cells are not used by Port H.
11.10 PORTJ, LATJ, and the TRISJ Register

PORTJ is an 8-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output.

All PORTJ pins have latch bits (LATJ register). The LATJ register, when read, will yield the contents of the PORTJ latch, and when written, will modify the contents of the PORTJ latch. This modifies the value driven out on a pin if the corresponding TRISJ bit is configured for output. This can be used in read-modify-write instructions that allow the user to modify the contents of the latch register regardless of the status of the corresponding pins.

Figure 11-13: PORTJ Block Diagram

Note 1: I/O pins have protection diodes to VDD and VSS.
### Table 11-17: PORTJ Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RJ0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RJ7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: **ST** = Schmitt Trigger input.

### Table 11-18: Summary of Registers Associated with PORTJ

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISJ</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>PORTJ</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx xxxxx</td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>LATJ</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx</td>
<td>xxxxx xxxxx</td>
<td>xxxxx xxxxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.
11.11 PORTK, LATK, and the TRISK Register

PORTK is an 8-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output.

Figure 11-14: PORTK Block Diagram

Note 1: I/O pins have protection diodes to VDD and VSS.
# PIC18C Reference Manual

## Table 11-19: PORTK Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RK0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RK7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input.

## Table 11-20: Summary of Registers Associated with PORTK

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>PORTK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxx xxxx</td>
<td>xxxx xxxx</td>
</tr>
<tr>
<td>LATK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxx xxxx</td>
<td>xxxx xxxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.
11.12 PORTL, LATL, and the TRISL Register

PORTL is a 8-bit port with Schmitt Trigger input buffers. Each pin is individually configured as an input or output.

Figure 11-15: Block Diagram of PORTL Pins

Note 1: I/O pins have protection diodes to VDD and VSS.
Table 11-21: PORTL Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit#</th>
<th>Buffer Type</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RL0</td>
<td>bit0</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL1</td>
<td>bit1</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL2</td>
<td>bit2</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL3</td>
<td>bit3</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL4</td>
<td>bit4</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL5</td>
<td>bit5</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL6</td>
<td>bit6</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
<tr>
<td>RL7</td>
<td>bit7</td>
<td>ST</td>
<td>Input/output port pin</td>
</tr>
</tbody>
</table>

Legend: ST = Schmitt Trigger input.

Table 11-22: Summary of Registers Associated with PORTL

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TRISL</td>
<td>1111</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111</td>
<td>1111  1111</td>
<td>1111  1111</td>
</tr>
<tr>
<td>PORTL</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxxx xxxxx uuuu uuuu</td>
<td>xxxxx xxxxx uuuu uuuu</td>
</tr>
<tr>
<td>LATL</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxxx xxxxx uuuu uuuu</td>
<td>xxxxx xxxxx uuuu uuuu</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged.
Section 11. I/O Ports

11.13 Functions Multiplexed on I/O Pins

This section discusses a couple of functions that are multiplexed on I/O pins that are new concepts when compared to the Mid-Range family.

11.13.1 Oscillator Configuration

If the system oscillator uses RCIO or ECIO mode, then the OSC2 pin may be used as a general purpose I/O pin. If any other oscillator mode is used, the I/O pin multiplexed with OSC2 is disabled and will read '0', as will the TRIS bit and LAT bit associated with the I/O pin. Writes to I/O pin will have no effect. See Table 11-23.

If the system oscillator uses RC or EC mode, then the I/O pin is configured as OSC2 and outputs Fosc/4.

Table 11-23: RA6 Configuration for Oscillator Configuration

<table>
<thead>
<tr>
<th>Oscillator Configuration</th>
<th>TRIS</th>
<th>PORT</th>
<th>LAT</th>
<th>OSC2 / I/O Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>RCIO / ERIO</td>
<td>Read / Write</td>
<td>Read / Write</td>
<td>Read / Write</td>
<td>General I/O</td>
</tr>
<tr>
<td>RC / EC</td>
<td>Disabled (reads 0)</td>
<td>Disabled (reads 0)</td>
<td>Disabled (reads 0)</td>
<td>FOSC/4</td>
</tr>
<tr>
<td>Other system oscillator modes</td>
<td>Disabled (reads 0)</td>
<td>Disabled (reads 0)</td>
<td>Disabled (reads 0)</td>
<td>OSC2</td>
</tr>
</tbody>
</table>

Figure 11-16: Block Diagram of I/O

Note 1: I/O pins have protection diodes to Vcc and Vss.
11.13.2 CCP2 Pin Multiplexing

In the PIC18CXX2 devices, the RB3 pin can be multiplexed with the CCP2 module input/output. To achieve this, the CCP2MX configuration bit must be programmed to a ‘0’.

Figure 11-17: Block Diagram of RB3

Note: I/O pin has diode protection to VDD and VSS.
Note: To enable weak pull-ups, set the appropriate TRIS bit(s) and clear the RBPU bit (INTCON2).
Note: The CCP2 input/output is multiplexed with RB3 if the CCP2MX bit is enabled (= ‘0’) in the configuration register.
Section 11. I/O Ports

11.14 I/O Programming Considerations

When using the ports as I/O, design considerations need to be taken into account to ensure that the operation is as intended.

11.14.1 Bi-directional I/O Ports

Any instruction that performs a write operation, actually does a read followed by a write operation. The BCF and BSF instructions, for example, read the register into the CPU, execute the bit operation and write the result back to the register. Caution must be used when these instructions are applied to a port with both inputs and outputs defined. For example, a BSF operation on bit5 of PORTB will cause all eight bits of PORTB to be read into the CPU. Then the BSF operation takes place on bit5 and PORTB is written to the output latches. If another bit of PORTB is used as a bi-directional I/O pin (e.g., bit0) and it is defined as an input at this time, the input signal present on the pin itself would be read into the CPU and rewritten to the data latch of this particular pin, overwriting the previous content. As long as the pin stays in the input mode, no problem occurs. However, if bit0 is switched to an output, the content of the data latch may now be unknown.

Reading the port register, reads the values of the port pins. Writing to the port register writes the value to the port latch. When using read-modify-write instructions (e.g., BCF, BSF, etc.) on a port, the value of the port pins is read, the desired operation is performed on this value, and the value is then written to the port latch.

Example 11-13 shows the effect of two sequential read-modify-write instructions on an I/O port.

Example 11-13: Read-Modify-Write Instructions on an I/O Port

```
; Initial PORT settings: PORTB<7:4> Inputs
; PORTB<3:0> Outputs
; PORTB<7:6> have external pull-ups and are not connected to other circuitry
;; PORT latch PORT pins
;; --------------------------
BCF PORTB, 7 ; 01pp pppp 11pp pppp
BCF PORTB, 6 ; 10pp pppp 11pp pppp
BCF TRISB, 7 ; 10pp pppp 10pp pppp
BCF TRISB, 6 ; 10pp pppp 10pp pppp
;; Note that the user may have expected the pin values to be 00pp ppp.
;; The 2nd BCF caused RB7 to be latched as the pin value (high).
```

A pin configured as an output, actively driving a Low or High, should not be driven from external devices at the same time in order to change the level on this pin ("wired-or", "wired-and"). The resulting high output currents may damage the chip.
11.14.2 Successive Operations on an I/O Port

The actual write to an I/O port happens at the end of an instruction cycle, whereas for reading, the data must be valid at the beginning of the instruction cycle (Figure 11-18). Therefore, care must be exercised if a write followed by a read operation is carried out on the same I/O port. The sequence of instructions should be such to allow the pin voltage to stabilize (load dependent) before the next instruction that causes that file to be read into the CPU is executed. Otherwise, the previous state of that pin may be read into the CPU rather than the new state. When in doubt, it is better to separate these instructions with a NOP or another instruction not accessing this I/O port.

This example shows a write to PORTB followed by a read from PORTB.

Note: Data setup time = (0.25TCY - TPD), where TCY = instruction cycle, TPD = propagation delay.

Therefore, at higher clock frequencies, a write followed by a read may be problematic due to external capacitance.

Figure 11-18: Successive I/O Operation
Section 11. I/O Ports

Figure 11-19 shows the I/O model that causes this situation. As the effective capacitance (C) becomes larger, the rise/fall time of the I/O pin increases. As the device frequency increases or the effective capacitance increases, the possibility of this subsequent PORTx read-modify-write instruction issue increases. This effective capacitance includes the effects of the board traces.

A way to address this is to add a series resistor at the I/O pin. This resistor allows the I/O pin to get to the desired level before the next instruction.

The use of NOP instructions between the subsequent PORTx read-modify-write instructions, is a lower cost solution, but has the issue that the number of NOP instructions is dependent on the effective capacitance C and the frequency of the device.

**Figure 11-19: I/O Connection Issues**

Note 1: This is not a capacitor to ground, but the effective capacitive loading on the trace.
11.15 Initialization

See the section describing each port for examples of initialization of the ports.

**Note:** It is recommended that when initializing the port, the PORT data latch (LAT or PORT register) should be initialized first, and then the data direction (TRIS register). This will eliminate a possible pin glitch, since the LAT register (PORT data latch values) power up in a random state.
Section 11. I/O Ports

11.16 Design Tips

Question 1: **Code will not toggle any I/O ports, but the oscillator is running. What can I be doing wrong?**

Answer 1:
1. Have the TRIS registers been initialized properly? These registers can be written to directly in the access bank (Bank15).
2. Is there a peripheral multiplexed onto those pins that are enabled?
3. Is the Watchdog Timer enabled (done at programming)? If it is enabled, is it being cleared properly with a **CLRWDT** instruction at least every 9 ms (or more if prescaled)?
4. Are you using the correct instructions to write to the port? More than one person has used the **MOVF** command when they should have used **MOVWF**.
5. For parts with interrupts, are the interrupts disabled? If not, try disabling them to verify they are not interfering.

Question 2: **When my program reads a port, I get a different value than what I put in the port register. What can cause this?**

Answer 2:
1. When a port is read, it is always the pin that is read, regardless of its being set to input or output. So if a pin is set to an input, you will read the value on the pin regardless of the register value.
2. If a pin is set to output, for instance, it has a one in the data latch; if it is shorted to ground, you will still read a zero on the pin. This is very useful for building fault tolerant systems, or handling I^2C bus conflicts. (The I^2C bus is only driven low, and the pin is high impedance for a one. If the pin is low and you are not driving it, some other device is trying to take the bus).
3. Enhanced devices all have at least one open drain (or open collector) pin. These pins can only drive a zero or high impedance. For most Enhanced devices, this is pin RA4. Open drain pins must have a pull-up resistor to have a high state. This pin is useful for driving odd voltage loads. The pull-up can be connected to a voltage (typically less than VDD) which becomes the high state.
4. Some analog modules, when enabled, will force a read value of '0' from the pin, regardless of the voltage level on the pin.

Question 3: **I have a PIC18CXX2 with pin RB0 configured as an interrupt input, but am not getting interrupted. When I change my routine to poll the pin, it reads the high input and operates fine. What is the problem?**

Answer 3:
PORTB accepts TTL input levels (on most parts), so when you have an input of say 3V (with VDD = 5V), you will read a '1'. However, the buffer to the interrupt structure from pin RB0 is a Schmitt Trigger, which requires a higher voltage (than TTL input) before the high input is registered. So it is possible to read a '1', but not get the interrupt. The interrupt was given a Schmitt Trigger input with hysteresis to minimize noise problems. It is one thing to have short noise spikes on a pin that is a data input that can potentially cause bad data, but quite another to permit noise to cause an interrupt, hence the difference.
**Question 4:** When I perform a *BCF* instruction, other pins get cleared in the port. Why?

**Answer 4:**

1. Another case where a read-modify-write instruction may seem to change other pin values unexpectedly can be illustrated as follows: Suppose you make PORTC all outputs, and drive the pins low. On each of the port pins is an LED connected to ground, such that a high output lights it. Across each LED is a 100 µF capacitor. Let's also suppose that the processor is running very fast, say 20 MHz. Now if you go down the port, setting each pin in order; \texttt{BSF PORTC,0} then \texttt{BSF PORTC,1} then \texttt{BSF PORTC,2} and so on, you may see that only the last pin was set, and only the last LED actually turns on. This is because the capacitors take a while to charge. As each pin was set, the pin before it was not charged yet, and so was read as a zero. This zero is written back out to the port latch (r-m-w, remember), which clears the bit you just tried to set in the previous instruction. This is usually only a concern at high speeds and for successive port operations, but it can happen, so take it into consideration.

2. If this is on a PIC18CXXX device with A/D, you have not configured the I/O pins properly in the ADCON1 register. If a pin is configured for analog input, any read of that pin will read a zero, regardless of the voltage on the pin. This is an exception to the normal rule that the pin state is always read. You can still configure an analog pin as an output in the TRIS register, and drive the pin high or low by writing to it, but you will always read a zero. Therefore, if you execute a Read-Modify-Write instruction (see previous question), all analog pins are read as zero; those not directly modified by the instruction will be written back to the port latch as zero. A pin configured as analog is expected to have values that may be neither high nor low to a digital pin, or floating. Floating inputs on digital pins are a no-no, and can lead to high current draw in the input buffer, so the input buffer is disabled.
Section 11. I/O Ports

11.17 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Baseline, the Midrange, or High-end families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to I/O ports are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Improving the Susceptibility of an Application to ESD</td>
<td>AN595</td>
</tr>
<tr>
<td>Clock Design using Low Power/Cost Techniques</td>
<td>AN615</td>
</tr>
<tr>
<td>Implementing Wake-up on Keystroke</td>
<td>AN528</td>
</tr>
<tr>
<td>Interfacing to AC Power Lines</td>
<td>AN521</td>
</tr>
<tr>
<td>Multiplexing LED Drive and a 4 x 4 Keypad Sampling</td>
<td>AN529</td>
</tr>
<tr>
<td>Using PIC16C5X as an LCD Drivers</td>
<td>AN563</td>
</tr>
<tr>
<td>Serial Port Routines Without Using TMR0</td>
<td>AN593</td>
</tr>
<tr>
<td>Implementation of an Asynchronous Serial I/O</td>
<td>AN510</td>
</tr>
<tr>
<td>Using the PORTB Interrupt on Change Feature as an External Interrupt</td>
<td>AN566</td>
</tr>
<tr>
<td>Implementing Wake-up on Keystroke</td>
<td>AN522</td>
</tr>
<tr>
<td>Apple Desktop Bus</td>
<td>AN591</td>
</tr>
<tr>
<td>Software Implementation of Asynchronous Serial I/O</td>
<td>AN555</td>
</tr>
<tr>
<td>Communicating with the I²C Bus using the PIC16C5X</td>
<td>AN515</td>
</tr>
<tr>
<td>Interfacing 93CX6 Serial EEPROMs to the PIC16C5X Microcontrollers</td>
<td>AN530</td>
</tr>
<tr>
<td>Logic Powered Serial EEPROMs</td>
<td>AN535</td>
</tr>
<tr>
<td>Interfacing 24LCXXB Serial EEPROMs to the PIC16C54</td>
<td>AN567</td>
</tr>
<tr>
<td>Using the 24XX65 and 24XX32 with Stand-alone PIC16C54 Code</td>
<td>AN558</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
PIC18C Reference Manual

11.18 Revision History

Revision A

This is the initial released revision of the Enhanced MCU I/O Ports description.
Section 12. Parallel Slave Port

HIGHLIGHTS

This section of the manual contains the following major topics:

12.1 Introduction ............................................................................................................... 12-2
12.2 Control Register ........................................................................................................... 12-3
12.3 Operation .................................................................................................................. 12-5
12.4 Operation in SLEEP Mode ........................................................................................... 12-6
12.5 Effect of a RESET ........................................................................................................ 12-6
12.6 PSP Waveforms ......................................................................................................... 12-6
12.7 Design Tips ............................................................................................................... 12-8
12.8 Related Application Notes ............................................................................................ 12-9
12.9 Revision History ......................................................................................................... 12-10
12.1 Introduction

Some devices have an 8-bit wide Parallel Slave Port (PSP). This port is multiplexed onto one of the device’s I/O ports. The port operates as an 8-bit wide Parallel Slave Port, or microprocessor port, when the PSPMODE control bit is set. In this mode, the input buffers are TTL.

In slave mode, the module is asynchronously readable and writable by the external world through the RD control input pin and the WR control input pin.

It can directly interface to an 8-bit microprocessor data bus. The external microprocessor can read or write the PORT latch as an 8-bit latch. Setting the PSPMODE bit enables port pins to be the RD input, the WR input, and the CS (chip select) input.

**Note 1:** At present the Parallel Slave Port (PSP) is only multiplexed onto PORTD and PORTE. The microprocessor port becomes enabled when the PSPMODE bit is set. In this mode, the user must make sure that PORTD and PORTE are configured as digital I/O. That is, peripheral modules multiplexed onto the PSP functions are disabled (such as the A/D). When PORTE is configured for digital I/O, PORTD will override the values in the TRISD register.

2: In this mode the PORTD and PORTE input buffers are TTL. The control bits for the PSP operation are located in TRISE.

There are actually two 8-bit latches, one for data-out (from the PICmicro) and one for data input. The user writes 8-bit data to the PORT data latch and reads data from the port pin latch (note that they have the same address). In this mode, the TRIS register is ignored, since the microprocessor is controlling the direction of data flow.

Register 12-1 shows the block diagram for the PSP module.

**Figure 12-1: PORTD and PORTE Block Diagram (Parallel Slave Port)**

Note: I/O pins have protection diodes to Vdd and Vss.
Section 12. Parallel Slave Port

12.2 Control Register

Register 12-1 is the PSP control register (PSPCON). The TRISE register (Register 12-2) contains the 4 bits for the PSP module found in some devices (such as PIC18C4X2) for compatibility with 40-pin midrange devices.

Register 12-1: PSPCON Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Description</th>
<th>0</th>
<th>1</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>IBF: Input Buffer Full Status bit</td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td>OBF: Output Buffer Full Status bit</td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td>IBOV: Input Buffer Overflow Detect bit (in microprocessor mode)</td>
<td></td>
<td></td>
</tr>
<tr>
<td>4</td>
<td>PSPMODE: Parallel Slave Port Mode Select bit</td>
<td></td>
<td></td>
</tr>
<tr>
<td>3:0</td>
<td>Unimplemented: Read as '0'</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Legend

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as '0'
- n = Value at POR reset  '1' = bit is set  '0' = bit is cleared  x = bit is unknown
## Register 12-2: TRISE Register

<table>
<thead>
<tr>
<th>R-0</th>
<th>R-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>R/W-1</th>
<th>R/W-1</th>
<th>R/W-1</th>
</tr>
</thead>
<tbody>
<tr>
<td>IBF</td>
<td>OBF</td>
<td>IBOV</td>
<td>PSPMODE</td>
<td>—</td>
<td>TRISE2</td>
<td>TRISE1</td>
<td>TRISE0</td>
</tr>
</tbody>
</table>

**bit 7**
- **IBF**: Input Buffer Full Status bit
  - 1 = A word has been received and waiting to be read by the CPU
  - 0 = No word has been received

**bit 6**
- **OBF**: Output Buffer Full Status bit
  - 1 = The output buffer still holds a previously written word
  - 0 = The output buffer has been read

**bit 5**
- **IBOV**: Input Buffer Overflow Detect bit (in microprocessor mode)
  - 1 = A write occurred when a previously input word has not been read (must be cleared in software)
  - 0 = No overflow occurred

**bit 4**
- **PSPMODE**: Parallel Slave Port Mode Select bit
  - 1 = Parallel slave port mode
  - 0 = General purpose I/O mode

**bit 3**
- **Unimplemented**: Read as '0'

**bit 2**
- **TRISE2**: RE2 Direction Control bit
  - 1 = Input
  - 0 = Output

**bit 1**
- **TRISE1**: RE1 Direction Control bit
  - 1 = Input
  - 0 = Output

**bit 0**
- **TRISE0**: RE0 Direction Control bit
  - 1 = Input
  - 0 = Output

Legend:
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **n** = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown
12.3 Operation

A write to the PSP from the external system occurs when both the \( CS \) and \( WR \) lines are first detected low. When either the \( CS \) or \( WR \) lines become high (edge triggered), the Input Buffer Full status flag bit IBF is set on the Q4 clock cycle following the next Q2 cycle. This signals that the write is complete. The interrupt flag bit, PSPIF, is also set on the same Q4 clock cycle. The IBF flag bit is inhibited from being cleared for additional \( T_{CY} \) cycles (see parameter 66 in the "Electrical Specifications" section). If the IBF flag bit is cleared by reading the PORTD input latch, then this has to be a read-only instruction (i.e., \texttt{MOVF}) and not a read-modify-write instruction. The Input Buffer Overflow status flag bit IBOV is set if a second write to the Parallel Slave Port is attempted when the previous byte has not been read out of the buffer.

A read of the PSP from the external system occurs when both the \( CS \) and \( RD \) lines are first detected low. The Output Buffer Full status flag bit OBF is cleared immediately indicating that the PORTD latch was read by the external bus. When either the \( CS \) or \( RD \) pin becomes high (edge triggered), the interrupt flag bit, PSPIF, is set on the Q4 clock cycle following the next Q2 cycle, indicating that the read is complete. OBF remains low until data is written to PORTD by the user firmware.

Input Buffer Full Status Flag bit, IBF, is set if a received word is waiting to be read by the CPU. Once the PORT input latch is read, the IBF bit is cleared. The IBF bit is a read only status bit.

Output Buffer Full Status Flag bit, OBF, is set if a word written to the PORT latch is waiting to be read by the external bus. Once the PORT output latch is read by the microprocessor, OBF is cleared. Input Buffer Overflow Status Flag bit, IBOV, is set if a second write to the microprocessor port is attempted when the previous word has not been read by the CPU (the first word is retained in the buffer).

When not in Parallel Slave Port mode, the IBF and OBF bits are held clear. However, if the IBOV bit was previously set, it must be cleared in the software.

An interrupt is generated and latched into flag bit PSPIF when a read or a write operation is completed. Interrupt flag bit PSPIF must be cleared by user software and the interrupt can be disabled by clearing interrupt enable bit PSPIE.

Table 12-1: PORTE Functions

<table>
<thead>
<tr>
<th>Name</th>
<th>Function</th>
</tr>
</thead>
</table>
| RD   | Read Control Input in parallel slave port mode:  \( RD \)  
|      | \( 1 \) = Not a read operation  
|      | \( 0 \) = Read operation. Reads PORTD register (if chip selected)  
| WR   | Write Control Input in parallel slave port mode:  \( WR \)  
|      | \( 1 \) = Not a write operation  
|      | \( 0 \) = Write operation. Writes PORTD register (if chip selected)  
| CS   | Chip Select Control Input in parallel slave port mode:  \( CS \)  
|      | \( 1 \) = Device is not selected  
|      | \( 0 \) = Device is selected  

Note: The PSP may have other functions multiplexed onto the same pins. For the PSP to operate, the pins must be configured as digital I/O.
12.4 Operation in SLEEP Mode

When in SLEEP mode, the microprocessor may still read and write the Parallel Slave Port. These actions will set the PSPIF bit. If the PSP interrupts are enabled, this will wake the processor from SLEEP mode so that the PSP data latch may be either read, or written with the next value for the microprocessor.

12.5 Effect of a RESET

After any RESET, the PSP is disabled and PORTD and PORTE are forced to their default mode.

12.6 PSP Waveforms

Register 12-2 shows the waveform for a write from the microprocessor to the PSP, while
Register 12-3 shows the waveform for a read of the PSP by the microprocessor.

Figure 12-2: Parallel Slave Port Write Waveforms

Figure 12-3: Parallel Slave Port Read Waveforms
# Section 12. Parallel Slave Port

## Table 12-2: Registers Associated with Parallel Slave Port

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>PORTD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>XXXX XXXX</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>LATD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>XXXX XXXX</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TRISD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>PORTE</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>----- -000</td>
<td>----- -000</td>
</tr>
<tr>
<td>LATE</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>----- -xxx</td>
<td>----- -xxx</td>
</tr>
<tr>
<td>TRISE</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0000 -111</td>
<td>0000 -111</td>
</tr>
<tr>
<td>PSPCON</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0000 -----</td>
<td>0000 -----</td>
</tr>
<tr>
<td>INTCON</td>
<td>GIE/</td>
<td>PEIE/</td>
<td>TMROIF</td>
<td>INT0IE</td>
<td>RBIE</td>
<td>TMR0IF</td>
<td>INT0IF</td>
<td>RBIF</td>
<td>0000 000x</td>
<td>0000 000u</td>
</tr>
<tr>
<td></td>
<td>GIEH</td>
<td>GIEL</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIR1</td>
<td>PSPIF</td>
<td>ADIF</td>
<td>RCPF</td>
<td>TXIF</td>
<td>SSPIF</td>
<td>CCP1IF</td>
<td>TMR2IF</td>
<td>TMR1IF</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PIE1</td>
<td>PSPIE</td>
<td>ADIE</td>
<td>RCIE</td>
<td>TXIE</td>
<td>SSPIE</td>
<td>CCP1E</td>
<td>TMR2IE</td>
<td>TMR1IE</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>IPR1</td>
<td>PSPIP</td>
<td>ADIP</td>
<td>RCIP</td>
<td>TXIP</td>
<td>SSPIP</td>
<td>CCP1P</td>
<td>TMR2IP</td>
<td>TMR1IP</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>ADCON1</td>
<td>ADFM</td>
<td>ADCS2</td>
<td></td>
<td></td>
<td>PCFG3</td>
<td>PCFG2</td>
<td>PCFG1</td>
<td>PCFG0</td>
<td>--0-- -000</td>
<td>--0-- -000</td>
</tr>
</tbody>
</table>

Legend:  
- \_ = unknown, \_ = unchanged, - = unimplemented read as '0'.  

Shaded cells are not used by the Parallel Slave Port.  

**Note 1:** On some devices the entire PORTE will be implemented with I/O functions. In these devices, the TRIS register will contain the eight data direction bits and the PSP bits will be located in the PSPCON register.
12.7 Design Tips

Question 1: Migrating from the PIC16C74 to the PIC18CXX2, the operation of the PSP seems to have changed.

Answer 1:
Yes, a design change was made so the PIC18CXX2 is edge sensitive (while the PIC16C74 was level sensitive).
## Section 12. Parallel Slave Port

### 12.8 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the Parallel Slave Port are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using the 8-bit Parallel Slave Port</td>
<td>AN579</td>
</tr>
</tbody>
</table>

*Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:*

![http://www.microchip.com/10/faqs/codeex/](http://www.microchip.com/10/faqs/codeex/)
12.9 Revision History

Revision A

This is the initial released revision of the Parallel Slave Port description.
# Section 13. Timer0

## HIGHLIGHTS

This section of the manual contains the following major topics:

<table>
<thead>
<tr>
<th>Topic</th>
<th>Page</th>
</tr>
</thead>
<tbody>
<tr>
<td>13.1 Introduction</td>
<td>13-2</td>
</tr>
<tr>
<td>13.2 Control Register</td>
<td>13-3</td>
</tr>
<tr>
<td>13.3 Operation</td>
<td>13-4</td>
</tr>
<tr>
<td>13.4 Timer0 Interrupt</td>
<td>13-5</td>
</tr>
<tr>
<td>13.5 Using Timer0 with an External Clock</td>
<td>13-6</td>
</tr>
<tr>
<td>13.6 Timer0 Prescaler</td>
<td>13-7</td>
</tr>
<tr>
<td>13.7 Initialization</td>
<td>13-9</td>
</tr>
<tr>
<td>13.8 Design Tips</td>
<td>13-10</td>
</tr>
<tr>
<td>13.9 Related Application Notes</td>
<td>13-11</td>
</tr>
<tr>
<td>13.10 Revision History</td>
<td>13-12</td>
</tr>
</tbody>
</table>
13.1 Introduction

The Timer0 module has the following features:
- Software selectable as an 8-bit or 16-bit timer/counter
- Readable and writable
- Dedicated 8-bit software programmable prescaler
- Clock source selectable to be external or internal
- Interrupt on overflow from FFh to 00h (FFFFh to 0000h in 16-bit mode)
- Edge select for external clock

Figure 13-1 shows a simplified block diagram of the Timer0 module in 8-bit mode and Figure 13-2 shows a simplified block diagram of the Timer0 module in 16-bit mode.

Note 1: TOCS, T0SE, PSA, T0PS2:T0PS0 (T0CON<5:0>),
2: Upon reset, Timer0 is enabled in 8-bit mode, with clock input from T0CKI, max. prescale.

Note 1: TOCS, T0SE, PSA, T0PS2:T0PS0 (T0CON<5:0>),
2: Upon reset, Timer0 is enabled in 8-bit mode, with clock input from T0CKI, max. prescale.
Section 13. Timer0

13.2 Control Register

The T0CON register is a readable and writable register that controls all the aspects of Timer0, including the prescale selection.

Register 13-1: T0CON: Timer0 Control Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Name</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>TMR0ON</td>
<td>Timer0 On/Off Control bit</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1 = Enables Timer0</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0 = Stops Timer0</td>
</tr>
<tr>
<td>6</td>
<td>T08BIT</td>
<td>Timer0 8-bit/16-bit Control bit</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1 = Timer0 is configured as an 8-bit timer/counter</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0 = Timer0 is configured as a 16-bit timer/counter</td>
</tr>
<tr>
<td>5</td>
<td>T0CS</td>
<td>Timer0 Clock Source Select bit</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1 = Transition on T0CKI pin is clock (counter mode)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0 = Internal instruction cycle is clock (timer mode)</td>
</tr>
<tr>
<td>4</td>
<td>T0SE</td>
<td>Timer0 Source Edge Select bit</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1 = Increment on high-to-low transition on T0CKI pin</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0 = Increment on low-to-high transition on T0CKI pin</td>
</tr>
<tr>
<td>3</td>
<td>PSA</td>
<td>Timer0 Prescaler Assignment bit</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1 = Timer0 prescaler is NOT assigned. Timer0 clock input bypasses prescaler.</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0 = Timer0 prescaler is assigned. Timer0 clock input comes from prescaler.</td>
</tr>
<tr>
<td>2-0</td>
<td>T0PS2:T0PS0</td>
<td>Timer0 Prescaler Select bits</td>
</tr>
<tr>
<td></td>
<td></td>
<td>These bits are ignored if PSA = 1</td>
</tr>
<tr>
<td></td>
<td></td>
<td>111 = 1:256 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>110 = 1:128 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>101 = 1:64 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>100 = 1:32 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>011 = 1:16 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>010 = 1:8 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>001 = 1:4 prescale value</td>
</tr>
<tr>
<td></td>
<td></td>
<td>000 = 1:2 prescale value</td>
</tr>
</tbody>
</table>

Legend

- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- n = Value at POR reset
- ’1’ = bit is set
- ’0’ = bit is cleared
- x = bit is unknown
13.3 Operation

When initializing Timer0, several options need to be specified. This is done by programming the appropriate bits in the T0CON register.

13.3.1 8-Bit/16-Bit Modes

Timer0 can be configured as an 8-bit or a 16-bit counter. The default state for Timer0 is an 8-bit counter. To configure the timer as a 16-bit counter, the T08BIT bit (T0CON register) must be cleared.

If the timer is configured as an 8-bit timer, the MSB of TMR0 (TMR0H) is held clear and will read 00h.

Normally once the mode of the timer is selected, it is not changed. Some applications may require the ability to switch back and forth between 8-bit and 16-bit modes. The two cases are:

1. Changing from 8-bit to 16-bit mode
2. Changing from 16-bit to 8-bit mode

The condition when bit 7 of the Timer0 rolls over must be addressed.

If Timer0 is configured as an 8-bit timer and is changed to a 16-bit timer on the same cycle as a rollover occurs, no interrupt is generated.

If Timer0 is configured as a 16-bit timer and is changed to an 8-bit timer on the same cycle as a rollover occurs, the TMR0IF bit will be set.

13.3.1.1 16-Bit Mode Timer Reads

TMR0H is not the high byte of the timer/counter, but actually a buffered version of the high byte of Timer0. The high byte of the Timer0 counter/timer is not directly readable or writable. TMR0H is updated with the contents of the high byte of Timer0 during a read of TMR0L. This provides a user with the ability to read all 16 bits of Timer0 without having to verify that the read of the high and low byte were valid due to a rollover between successive reads of the high and low byte. The user simply reads the low byte of Timer0, followed by a read of TMR0H, which contains the value in the high byte of Timer0 at the time that the low byte was read.

13.3.1.2 16-Bit Mode Timer Write

A write to the high byte of Timer0 must also take place through the TMR0H buffer register. Timer0 high byte is updated with the contents of TMR0H when a write occurs to TMR0L. This allows a user to update all 16 bits to both the high and low bytes of Timer0 at once (see Figure 13-2).

When performing a write of TMR0, the carry is held off during the write of the TMR0L register. Writes to the TMR0H register only modify the holding latch, not the timer (TMR0<15:8>).

Steps to write to the TMR0:

1. Load the TMR0H register.
2. Write to the TMR0L register.

13.3.1.3 16-Bit Read/Modify Write

Read-modify-write instructions like BSF or BCF, read the contents of a register, make the appropriate changes, and then the result back into the register. The read cycle of a read-modify-write instruction of TMR0L will not update the contents of the TMR0H buffer. The TMR0H buffer will remain unchanged. When the write cycle (to TMR0L) of the instruction takes place, the contents of TMR0H are placed into the high byte of Timer0.
13.3.2 Timer/Counter Modes

Timer mode is selected by clearing the T0CS bit (T0CON register). In timer mode, the Timer0 module will increment every instruction cycle (without prescaler). If the TMR0 register is written, the increment is inhibited for the following two instruction cycles. The user can work around this by writing an adjusted value to the TMR0 register.

Counter mode is selected by setting the T0CS bit (T0CON register). In counter mode, Timer0 will increment either on every rising or falling edge of the T0CKI pin. The incrementing edge is determined by the Timer0 Source Edge Select bit T0SE (T0CON register). Clearing the T0SE bit selects the rising edge. Restrictions on the external clock input are discussed in detail in Section 13.5.1.

13.4 Timer0 Interrupt

The TMR0 interrupt flag bit is set when the TMR0 register overflows. When TMR0 is in 8-bit mode, this means the overflow from FFh to 00h. When TMR0 is in 16-bit mode, this means the overflow from FFFFh to 0000h.

This overflow sets the TMR0IF bit (INTCON register). The interrupt can be disabled by clearing the TMR0IE bit (INTCON register). The TMR0IF bit must be cleared in software by the interrupt service routine. The TMR0 interrupt cannot awaken the processor from SLEEP, since the timer is shut off during SLEEP. See Figure 13-3 for Timer0 interrupt timing.

Figure 13-3: TMR0 Interrupt Timing

Note 1: Interrupt flag bit TMR0IF is sampled here (every Q1).
2: Interrupt latency = 4TCy where TCy = instruction cycle time.
3: CLKO is available only in RC oscillator mode.
13.5 Using Timer0 with an External Clock

When an external clock input is used for Timer0, it must meet certain requirements as detailed in 13.5.1 “External Clock Synchronization”. The requirements ensure the external clock can be synchronized with the internal phase clock (T_SCLK). Also, there is a delay in the actual incrementing of Timer0 after synchronization.

13.5.1 External Clock Synchronization

When no prescaler is used, the external clock input is used instead of the prescaler output. The synchronization of T0CKI with the internal phase clocks is accomplished by sampling the prescaler output on the Q2 and Q4 cycles of the internal phase clocks (Figure 13-4). Therefore, it is necessary for T0CKI to be high for at least 2T_SCLK (and a small RC delay) and low for at least 2T_SCLK (and a small RC delay). Refer to parameters 40, 41 and 42 in the electrical specification of the desired device.

When a prescaler is used, the external clock input is divided by the prescaler so that the prescaler output is symmetrical. For the external clock to meet the sampling requirement, the ripple-counter must be taken into account. Therefore, it is necessary for T0CKI to have a period of at least 4T_SCLK (and a small RC delay) divided by the prescaler value. The only requirement on T0CKI high and low time is that they do not violate the minimum pulse width requirement. Refer to parameters 40, 41 and 42 in the electrical specification of the desired device.

13.5.2 TMR0 Increment Delay

Since the prescaler output is synchronized with the internal clocks, there is a small delay from the time the external clock edge occurs to the time the Timer0 module is actually incremented. Figure 13-4 shows the delay from the external clock edge to the timer incrementing.

**Figure 13-4: Timer0 Timing with External Clock**

Note 1: Delay from clock input change to Timer0 increment is 3Tosc to 7Tosc. (Duration of Q = Tosc). Therefore, the error in measuring the interval between two edges on Timer0 input = ±4Tosc max.

2: External clock if no prescaler selected, prescaler output otherwise.

3: The arrows indicate the points in time where sampling occurs.
Section 13. Timer0

13.6 Timer0 Prescaler

An 8-bit counter is available as a prescaler for the Timer0 module (Figure 13-5).

The PSA and T0PS2:T0PS0 bits (T0CON register) are the prescaler enable and prescale select bits.

All instructions that write to the Timer0 (TMR0) register (such as: CLRF TMR0; BSF TMR0, x; MOVWF TMR0; .... etc.) will clear the prescaler if enabled. The prescaler is not readable or writable.

 Writes to TMR0H do not clear the Timer0 prescaler in 16-bit mode, because a write to TMR0H only modifies the Timer0 latch and does not change the contents of Timer0. The prescaler is only cleared on writes to TMR0L.

Figure 13-5: Block Diagram of the Timer0 Prescaler

The prescaler for Timer0 is enabled or disabled in software by the PSA bit (T0CON register). Setting the PSA bit will enable the prescaler. The prescaler can be modified under software control through the T0PS2:T0PS0 bits. This allows the prescaler reload value to be readable and writable. The prescaler count value (the contents of the prescaler) cannot be read or written. When the prescaler is enabled, prescale values of 1:2, 1:4, ..., 1:256 are selectable.

Note: T0CS, T0SE, PSA, T0PS2:T0PS0 are located in the T0CON register.

The prescaler for Timer0 is enabled or disabled in software by the PSA bit (T0CON register). Setting the PSA bit will enable the prescaler. The prescaler can be modified under software control through the T0PS2:T0PS0 bits. This allows the prescaler reload value to be readable and writable. The prescaler count value (the contents of the prescaler) cannot be read or written. When the prescaler is enabled, prescale values of 1:2, 1:4, ..., 1:256 are selectable.
Any write to the TMR0 register will cause a 2 instruction cycle (2TCY) inhibit. That is, after the TMR0 register has been written with the new value, TMR0 will not be incremented until the third instruction cycle later (Figure 13-6). When the prescaler is assigned to the Timer0 module, any write to the TMR0 register will immediately update the TMR0 register and clear the prescaler. The incrementing of Timer0 (TMR0 and Prescaler) will also be inhibited 2 instruction cycles (TCY). So if the prescaler is configured as 2, then after a write to the TMR0 register, TMR0 will not increment for 4 Timer0 clocks (Figure 13-7). After that, TMR0 will increment every prescaler number of clocks later.

Figure 13-6: Timer0 Timing: Internal Clock/No Prescale

Figure 13-7: Timer0 Timing: Internal Clock/Prescale 1:2

Table 13-1: Registers Associated with Timer0

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TMR0L</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>XXXX XAAA</td>
<td>UUUU UUUU</td>
</tr>
<tr>
<td>TMR0H</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>INTCN</td>
<td>GIE/GIEH</td>
<td>PEIE/GIEL</td>
<td>TMR0IE</td>
<td>INT0IE</td>
<td>RBIE</td>
<td>TMR0IF</td>
<td>INT0IF</td>
<td>RBIF</td>
<td>0000 0000x</td>
<td>0000 0000x</td>
</tr>
<tr>
<td>T0CON</td>
<td>TMR0ON</td>
<td>T08BIT</td>
<td>T0CS</td>
<td>T0SE</td>
<td>PSA</td>
<td>T0PS2</td>
<td>T0PS1</td>
<td>T0PS0</td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>TRISA</td>
<td></td>
<td></td>
<td>PORTA Data Direction Register</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>--11 1111</td>
<td>--11 1111</td>
</tr>
</tbody>
</table>

Legend:  
- = unknown, u = unchanged, - = unimplemented locations read as '0'.  
Shaded cells are not used by Timer0.
13.7 Initialization

Since Timer0 has a software programmable clock source, there are two examples to show the initialization of Timer0 with each source. Example 13-1 shows the initialization for the internal clock source (timer mode), while Example 13-2 shows the initialization for the external clock source (counter mode).

Example 13-1: Timer0 Initialization (Internal Clock Source)

```
CLRF TMR0 ; Clear Timer0 register
CLRF INTCON ; Disable interrupts and clear T0IF
BCF INTCON2, RBPU ;
MOVLW 0x80 ; PortB pull-ups are disabled,
MOVWF T0CON ; Interrupt on rising edge of RB0,
; TMR0 = 16-Bit Time
; Timer0 increment from internal clock
; with a prescaler of 1:2.
;** BSF INTCON, T0IE ; Enable TMR0 interrupt
;** BSF INTCON, GIE ; Enable all interrupts
;
; The TMR0 interrupt is enabled, do polling on the overflow bit
;
T0_OVFL_WAIT
BTFSS INTCON, T0IF
GOTO T0_OVFL_WAIT
; Timer has overflowed
```

Example 13-2: Timer0 Initialization (External Clock Source)

```
CLRF TMR0 ; Clear Timer0 register
CLRF INTCON ; Disable interrupts and clear T0IF
BCF INTCON2, RBPU ;
MOVLW 0xBF ; PortB pull-ups are enabled,
MOVWF T0CON ; Interrupt on falling edge of RB0
; Timer0 increment from external clock
; on the high-to-low transition
; of T0CKI
; with a prescaler of 1:256.
;** BSF INTCON, T0IE ; Enable TMR0 interrupt
;** BSF INTCON, GIE ; Enable all interrupts
;
; The TMR0 interrupt is enabled, do polling on the overflow bit
;
T0_OVFL_WAIT
BTFSS INTCON, T0IF
GOTO T0_OVFL_WAIT
; Timer has overflowed
```
13.8 Design Tips

Question 1: I am implementing a counter/clock, but the clock loses time or is inaccurate.

Answer 1:
If you are polling TMR0 to see if it has rolled over to zero, you could do this by executing:

```assembly
wait   MOVF    TMR0,W ; read the timer into W
       BTFSS   STATUS,Z ; see if it was zero, if so,
                      ; break from loop
       GOTO    wait ; if not zero yet, keep waiting
```

Two possible scenarios to lose clock cycles are:

1. If you are incrementing TMR0 from the internal instruction clock (or an external source that is about as fast), the overflow could occur during the two cycle \texttt{GOTO}, so you could miss it. In this case, the TMR0 source should be prescaled.

2. When writing to TMR0, two instruction clock cycles are lost. Often you have a specific time period you want to count, say 100 decimal. In that case, you might put 156 into TMR0 (256 - 100 = 156). However, since two instruction cycles are lost when you write to TMR0 (for internal logic synchronization), you should actually write 158 to the timer.
Section 13. Timer0

13.9 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to Timer0 are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Frequency Counter Using PIC16C5X</td>
<td>AN592</td>
</tr>
<tr>
<td>A Clock Design using the PIC16C54 for LED Display and Switch Inputs</td>
<td>AN590</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand-alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faq/codeex/
13.10 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Timer0 Module description.
Section 14. Timer1

HIGHLIGHTS

This section of the manual contains the following major topics:

14.1 Introduction ............................................................................................................... 14-2
14.2 Control Register ....................................................................................................... 14-4
14.3 Timer1 Operation in Timer Mode .......................................................................... 14-5
14.4 Timer1 Operation in Synchronized Counter Mode ................................................. 14-5
14.5 Timer1 Operation in Asynchronous Counter Mode ............................................... 14-6
14.6 Reading and Writing of Timer1 .............................................................................. 14-7
14.7 Timer1 Oscillator ................................................................................................... 14-10
14.8 Typical Application .................................................................................................. 14-11
14.9 Sleep Operation ...................................................................................................... 14-12
14.10 Resetting Timer1 Using a CCP Trigger Output ..................................................... 14-12
14.11 Resetting Timer1 Register Pair (TMR1H:TMR1L) ................................................. 14-13
14.12 Timer1 Prescaler .................................................................................................. 14-13
14.13 Initialization ......................................................................................................... 14-14
14.14 Design Tips .......................................................................................................... 14-16
14.15 Related Application Notes .................................................................................... 14-17
14.16 Revision History .................................................................................................. 14-18
14.1 Introduction

The Timer1 module is a 16-bit timer/counter consisting of two 8-bit registers (TMR1H and TMR1L) that are readable and writable. The TMR1 register pair (TMR1H:TMR1L) increments from 0000h to FFFFh and rolls over to 0000h. If enabled, the Timer1 Interrupt is generated on overflow that is latched in the TMR1IF interrupt flag bit. This interrupt can be enabled/disabled by setting/clearing the TMR1IE interrupt enable bit.

Timer1 can operate in one of three modes:
- As a synchronous timer
- As a synchronous counter
- As an asynchronous counter

The operating mode is determined by clock select bit, TMR1CS (T1CON register), and the synchronization bit, T1SYNC (Figure 14-1).

In timer mode, Timer1 increments every instruction cycle. In counter mode, it increments on every rising edge of the external clock input pin T1OSI.

Timer1 can be turned on and off using the TMR1ON control bit (T1CON register).

Timer1 also has an internal "reset input", which can be generated by a CCP module.

Timer1 has the capability to operate off an external crystal. When the Timer1 oscillator is enabled (T1OSCEN is set), the T1OSI and T1OSO pins become inputs, so their corresponding TRIS values are ignored.

Figure 14-1: Timer1 Block Diagram

Note 1: When enable bit T1OSCEN is cleared, the inverter and feedback resistor are turned off. This eliminates power drain.
Section 14. Timer1

Figure 14-2: Timer1 Block Diagram 16-Bit Read/Write Mode

Note 1: When enable bit T1OSCEN is cleared, the inverter and feedback resistor are turned off. This eliminates power drain.
14.2 Control Register

Register 14-1 shows the Timer1 Control register. This register controls the operating mode of the Timer1 module and contains the Timer1 oscillator enable bit (T1OSCEN).

Register 14-1: T1CON: Timer1 Control Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>RD16</td>
<td>16-bit Read/Write Mode Enable bit</td>
</tr>
<tr>
<td>T1CKPS1:T1CKPS0</td>
<td>Timer1 Input Clock Prescale Select bits</td>
</tr>
<tr>
<td>T1OSCEN</td>
<td>Timer1 Oscillator Enable bit</td>
</tr>
<tr>
<td>T1SYNC</td>
<td>Timer1 External Clock Input Synchronization Select bit</td>
</tr>
<tr>
<td>TMR1CS</td>
<td>Timer1 Clock Source Select bit</td>
</tr>
<tr>
<td>TMR1ON</td>
<td>Timer1 On bit</td>
</tr>
</tbody>
</table>

Legend

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as ‘0’
- n = Value at POR reset  ‘1’ = bit is set  ‘0’ = bit is cleared  x = bit is unknown
Section 14. Timer1

14.3 Timer1 Operation in Timer Mode

Timer mode is selected by clearing the TMR1CS (T1CON register) bit. In this mode, the input clock to the timer is Fosc/4. The synchronize control bit, T1SYNC (T1CON register), has no effect since the internal clock is always synchronized.

14.4 Timer1 Operation in Synchronized Counter Mode

Counter mode is selected by setting the TMR1CS bit. In this mode, the timer increments on every rising edge of clock input on the T1OSI pin when the Timer1 oscillator enable bit (T1OSCEN) is set, or the T1OSO/T13CKI pin when the T1OSCEN bit is cleared.

If the T1SYNC bit is cleared, then the external clock input is synchronized with internal phase clocks. The synchronization is done after the prescaler stage. The prescaler operates asynchronously.

The timer increments at the Q4:Q1 edge.

In this configuration, during SLEEP mode, Timer1 will not increment even if the external clock is present, since the synchronization circuit is shut off. The prescaler however will continue to increment.

14.4.1 External Clock Input Timing for Synchronized Counter Mode

When an external clock input is used for Timer1 in synchronized counter mode, it must meet certain requirements. The external clock requirement is due to internal phase clock (Tosc) synchronization. Also, there is a delay in the actual incrementing of TMR1 after synchronization.

When the prescaler is 1:1, the external clock input is the same as the prescaler output. The synchronization of T1CKI with the internal phase clocks is accomplished by sampling the prescaler output on alternating TscLK clocks of the internal phase clocks. Therefore, it is necessary for the T1CKI pin to be high for at least 2TscLK (and a small RC delay) and low for at least 2TscLK (and a small RC delay). Refer to parameters 45, 46, and 47 in the “Electrical Specifications” section.

When a prescaler other than 1:1 is used, the external clock input is divided by the asynchronous prescaler so that the prescaler output is symmetrical. In order for the external clock to meet the sampling requirement, the prescaler counter must be taken into account. Therefore, it is necessary for the T1CKI pin to have a period of at least 4TscLK (and a small RC delay) divided by the prescaler value. Another requirement on the T1CKI pin high and low time is that they do not violate the minimum pulse width requirements). Refer to parameters 40, 42, 45, 46, and 47 in the “Electrical Specifications” section.
14.5 Timer1 Operation in Asynchronous Counter Mode

If T1SYNC (T1CON register) is set, the external clock input is not synchronized. The timer continues to increment asynchronously to the internal phase clocks. The timer will continue to run during SLEEP and can generate an interrupt on overflow that will wake-up the processor. However, special precautions in software are needed to read/write the timer (Subsection 14.6.4 “Reading and Writing Timer1 in Asynchronous Counter Mode with RD16 = 0”). Since the counter can operate in sleep, Timer1 can be used to implement a true real-time clock.

The timer increments at the Q4:Q1 and Q2:Q3 edges.

In asynchronous counter mode, Timer1 cannot be used as a time-base for capture or compare operations.

14.5.1 External Clock Input Timing with Unsynchronized Clock

If the T1SYNC control bit is set, the timer will increment completely asynchronously. The input clock must meet certain minimum high time and low time requirements. Refer to the Device Data Sheet “Electrical Specifications” section, timing parameters 45, 46, and 47.
14.6 Reading and Writing of Timer1

Timer1 has modes that allow the 16-bit timer register to be read/written as two 8-bit registers or one 16-bit register. The mode depends on the state of the RD16 bit. The following subsections discuss this operation.

14.6.1 Timer1 and 16-bit Read/Write Modes

Timer1 can be configured for 16-bit reads and writes. When the RD16 control bit (T1CON register) is set, the address for TMR1H is mapped to a buffer register for the high byte of Timer1. A read from TMR1L will load the contents of the high byte of Timer1 into the Timer1 high byte buffer. This provides the user with the ability to accurately read all 16 bits of Timer1 without having to determine whether a read of the high byte followed by a read of the low byte is valid due to a rollover between reads.

14.6.2 16-bit Mode Timer Write

A write to the high byte of Timer1 must also take place through the TMR1H buffer register. Timer1 high byte is updated with the contents of TMR1H when a write occurs to TMR1L. This allows a user to write all 16 bits to both the high and low bytes of Timer1 at once (See Figure 14-3).

The high byte of Timer1 is not directly readable or writable in this mode. All reads and writes must take place through the Timer1 high byte buffer register.

Writes to TMR1H do not clear the Timer1 prescaler. The prescaler is only cleared on writes to TMR1L.

14.6.3 16-bit Read-Modify-Write

Read-modify-write instructions like BSF or BCF will read the contents of a register, make the appropriate changes, and place the result back into the register. In the case of Timer1 when configured in 16-bit mode, the read portion of a read-modify-write instruction of TMR1L will not update the contents of the TMR1H buffer. The TMR1H buffer will remain unchanged. When the write of TMR1L portion of the instruction takes place, the contents of TMR1H will be placed into the high byte of Timer1.

Figure 14-3: Timer1 Block Diagram When Configured in 16-bit Read/Write Mode

Note 1: When enable bit T1OSCEN is cleared, the inverter and feedback resistor are turned off. This eliminates power drain.
14.6.4 Reading and Writing Timer1 in Asynchronous Counter Mode with RD16 = 0

Reading TMR1H or TMR1L while the timer is running from an external asynchronous clock will ensure a valid read (taken care of in hardware). However, the user should keep in mind that reading the 16-bit timer in two 8-bit values itself poses certain problems, since the timer may overflow between the reads.

For writes, it is recommended that the user simply stop the timer and write the desired values. A write contention may occur by writing to the timer registers while the register is incrementing. This may produce an unpredictable value in the timer register.

Reading the 16-bit value requires some care, since two separate reads are required to read the entire 16-bits. Example 14-1 shows why this may not be a straight forward read of the 16-bit register.

Example 14-1: Reading 16-bit Register Issues

<table>
<thead>
<tr>
<th>TMR1</th>
<th>Sequence 1</th>
<th>Sequence 2</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Action</td>
<td>TMPH:TMPL</td>
</tr>
<tr>
<td>04FFh</td>
<td>READ TMR1L</td>
<td>xxxxh</td>
</tr>
<tr>
<td>0500h</td>
<td>Store in TMPL</td>
<td>xxFFh</td>
</tr>
<tr>
<td>0501h</td>
<td>READ TMR1H</td>
<td>xxFFh</td>
</tr>
<tr>
<td>0502h</td>
<td>Store in TMPH</td>
<td>05FFh</td>
</tr>
</tbody>
</table>
Section 14. Timer1

Example 14-2 shows a routine to read the 16-bit timer value without experiencing the issues shown in Example 14-1. This is useful if the timer cannot be stopped.

Example 14-2: Reading a 16-bit Free-Running Timer

```assembly
; All interrupts are disabled
MOVF TMR1H, W ; Read high byte
MOVWF TMPH ;
MOVF TMR1L, W ; Read low byte
MOVWF TMPL ;
MOVF TMR1H, W ; Read high byte
SUBWF TMPH, W ; Sub 1st read with 2nd read
BTFSC STATUS,Z ; Is result = 0
GOTO CONTINUE ; Good 16-bit read
;
; TMR1L may have rolled over between the read of the high and low bytes.
; Reading the high and low bytes now will read a good value.
;
MOVF TMR1H, W ; Read high byte
MOVWF TMPH ;
MOVF TMR1L, W ; Read low byte
MOVWF TMPL ;
; Re-enable the Interrupt (if required)
CONTINUE ; Continue with your code
```

Writing a 16-bit value to the 16-bit TMR1 register is straightforward. First the TMR1L register is cleared to ensure that there are many Timer1 clock/oscillator cycles before there is a rollover into the TMR1H register. The TMR1H register is then loaded, and finally the TMR1L register is loaded. Example 14-3 shows a routine that does a 16-bit write to a Free Running Timer.

Example 14-3: Writing a 16-bit Free Running Timer

```assembly
; All interrupts are disabled
CLRF TMR1L ; Clear Low byte, Ensures no rollover into TMR1H
MOVLW HI_BYTE ; Value to load into TMR1H
MOVWF TMR1H, F ; Write High byte
MOVLW LO_BYTE ; Value to load into TMR1L
MOVWF TMR1H, F ; Write Low byte
; Re-enable the Interrupt (if required)
CONTINUE ; Continue with your code
```
14.7 Timer1 Oscillator

An alternate crystal oscillator circuit is built into the device. The output of this oscillator can be selected as the input into Timer1. The Timer1 oscillator is primarily intended to operate as a time-base for the timer modules; therefore, the oscillator is primarily intended for a 32 kHz crystal, which is an ideal frequency for real-time keeping. In real-time applications, the timer needs to increment during SLEEP, so SLEEP does not disable the Timer1 oscillator. For many applications, power consumption is also an issue, so the oscillator is designed to minimize power consumption.

The Timer1 oscillator is enabled by setting the T1OSCEN control bit (T1CON register). After the Timer1 oscillator is enabled, the user must provide a software time delay to ensure proper oscillator start-up.

Table 14-1 shows the capacitor selection for the Timer1 oscillator.

<table>
<thead>
<tr>
<th>Osc Type</th>
<th>Freq</th>
<th>C1</th>
<th>C2</th>
</tr>
</thead>
<tbody>
<tr>
<td>LP</td>
<td>32 kHz</td>
<td>33 pF</td>
<td>33 pF</td>
</tr>
<tr>
<td></td>
<td>100 kHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
<tr>
<td></td>
<td>200 kHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
</tbody>
</table>

Crystals Tested:

- 32.768 kHz Epson C-001R32.768K-A ± 20 PPM
- 100 kHz Epson C-2 100.00 KC-P ± 20 PPM
- 200 kHz STD XTL 200.000 kHz ± 20 PPM

**Note:** The Timer1 oscillator allows the counter to operate (increment) when the device is in sleep mode. This allows Timer1 to be used as a real-time clock.

1: Higher capacitance increases the stability of oscillator but also increases the start-up time.
2: Since each resonator/crystal has its own characteristics, the user should consult the resonator/crystal manufacturer for appropriate values of external components.
14.8 Typical Application

In Figure 14-4 an example application is given, where Timer1 is driven from an external 32 kHz oscillator. The external 32 kHz oscillator is typically used in applications where real-time needs to be kept, but it is also desirable to have the lowest possible power consumption. The Timer1 oscillator allows the device to be placed in sleep while the timer continues to increment. When Timer1 overflows, the interrupt wakes up the device so that the appropriate registers can be updated.

Figure 14-4: Timer1 Application

In this example, a 32 kHz crystal is used as the time base for the Real Time Clock. If the clock needs to be updated at 1 second intervals, then the Timer1 must be loaded with a value to allow the Timer1 to overflow at the desired rate. In the case of a 1 second Timer1 overflow, the TMR1H register should be loaded with a value of 80k after each overflow.

Note: The TMR1L register should never be modified, since an external clock is asynchronous to the system clock. Writes to the TRM1L register may corrupt the real time counter value causing inaccuracies.
14.9 Sleep Operation

When Timer1 is configured for asynchronous operation, the TMR1 registers will continue to increment for each timer clock (or prescale multiple of clocks). When the TMR1 register overflows, the TMR1IF bit will get set. If enabled, this will generate an interrupt that will wake the processor from sleep mode.

The Timer1 oscillator will add a delta current, due to the operation of this circuitry. That is, the power-down current will no longer only be the leakage current of the device, but also the active current of the Timer1 oscillator and other circuitry.

14.10 Resetting Timer1 Using a CCP Trigger Output

If a CCP module is configured in compare mode to generate a “Special Event Trigger” (CCP1M3:CCP1M0 = 1011), this signal resets Timer1.

**Note:** The special event trigger from the CCP module does not set interrupt flag bit TMR1IF.

Timer1 must be configured for either timer or synchronized counter mode to take advantage of the special event trigger feature. If Timer1 is running in asynchronous counter mode, this reset operation may not work and should not be used.

In the event that a write to Timer1 coincides with a special event trigger from the CCP module, the write will take precedence.

In this mode of operation, the CCPRxH:CCPRxL register pair effectively becomes the period register for Timer1.

14.10.1 CCP Trigger and A/D Module

Some devices that have the CCP Trigger capability also have an A/D module. These devices may be able to be configured, so the “Special Event Trigger” not only resets the Timer1 registers, but will start an A/D conversion. This allows a constant sampling rate for the A/D, as specified by the value of the compare registers.
Section 14. Timer1

14.11 Resetting Timer1 Register Pair (TMR1H:TMR1L)

TMR1H and TMR1L registers are not cleared on any reset, only by the CCP special event triggers.

T1CON register is reset to 00h on a Power-on Reset or a Brown-out Reset. In any other reset, the register is unaffected.

Timer1 is the default time base for the CCP1 and CCP2 modules. The timer can be disabled as the time base for either CCP1, CCP2, or both, and Timer3 can be substituted. This is achieved by setting control bits in the Timer3 control register.

This is explained in Section 16.8 - Timer3 and CCPx Enable.

When Timer1 is disabled as a time base for a CCP, the reset on Compare will have no effect on Timer1.

14.12 Timer1 Prescaler

The prescaler counter is cleared on writes to the TMR1H or TMR1L registers.

14.12.1 Timer1 Prescaler 16-bit Read/WriteMode

Writes to TMR1H do not clear the Timer1 prescaler in 16-bit read/write mode, because a write to TMR1H only modifies the Timer1 latch and does not change the contents of Timer1. The prescaler is only cleared on writes to TMR1L.

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE</td>
<td>PEIE</td>
<td>T0IE</td>
<td>INTE</td>
<td>RBIE</td>
<td>T0IF</td>
<td>INTF</td>
<td>RBIF</td>
<td>0000 000x</td>
<td>0000 000u</td>
</tr>
<tr>
<td>PIR</td>
<td>TMR1IE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>00</td>
<td>00</td>
</tr>
<tr>
<td>PIE</td>
<td></td>
<td>TMR1IE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>00</td>
<td>00</td>
</tr>
<tr>
<td>PR</td>
<td></td>
<td></td>
<td>TMR1IP</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>00</td>
<td>00</td>
</tr>
<tr>
<td>TMR1L</td>
<td></td>
<td>Holding register for the Least Significant Byte of the 16-bit TMR1 register</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxx xxxx</td>
<td>xxxx xxxxxx</td>
</tr>
<tr>
<td>TMR1H</td>
<td></td>
<td>Holding register for the Most Significant Byte of the 16-bit TMR1 register</td>
<td></td>
<td>xxxx xxxx</td>
<td></td>
<td></td>
<td>xxxx xxxx</td>
<td></td>
<td></td>
<td>xxxx xxxx xxxx xxxx xxxxx</td>
</tr>
<tr>
<td>T1CON</td>
<td>RD16</td>
<td></td>
<td>T1CKPS1</td>
<td>T1CKPS0</td>
<td>T1OSCEN</td>
<td>T1SYNC</td>
<td>TMR1CS</td>
<td>TMR1ON</td>
<td>--00 0000 0000</td>
<td>--00 0000 0000 0000</td>
</tr>
</tbody>
</table>

Legend:  
- x = unknown, u = unchanged, - = unimplemented read as '0'.  
Shaded cells are not used by the Timer1 module.

Note 1: The placement of this bit is device dependent.
14.13 Initialization

Since Timer1 has a software programmable clock source, there are three examples to show the initialization of each mode. Example 14-4 shows the initialization for the internal clock source, Example 14-5 shows the initialization for the external clock source, and Example 14-6 shows the initialization of the external oscillator mode.

Example 14-4: Timer1 Initialization (Internal Clock Source)

```
; Stop Timer1, Internal Clock Source,
; T1 oscillator disabled, 
; prescaler = 1:1
CLRF T1CON

; Clear Timer1 High byte register
CLRF TMR1H

; Clear Timer1 Low byte register
CLRF TMR1L

; Disable interrupts
CLRF INTCON

; Disable peripheral interrupts
CLRF PIE1

; Clear peripheral interrupts Flags
CLRF PIR1

; Internal Clock source
MOVLW 0x30

; with 1:8 prescaler
MOVWF T1CON

; Timer1 starts to increment
BSF T1CON, TMR1ON

; The Timer1 interrupt is disabled, do polling on the overflow bit

; T1_OVFL_WAIT
BTFSS PIR1, TMR1IF
GOTO T1_OVFL_WAIT

; Timer has overflowed

BCF PIR1, TMR1IF
```
Section 14. Timer1

Example 14-6: Timer1 Initialization (External Oscillator Clock Source)

```
CLRF T1CON ; Stop Timer1, Internal Clock Source,  
; T1 oscillator disabled,  
; prescaler = 1:1
CLRF TMR1H ; Clear Timer1 High byte register
CLRF TMR1L ; Clear Timer1 Low byte register
CLRF INTCON ; Disable interrupts
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupt Flags
MOVLW 0x3E ; External Clock source  
; with oscillator
MOVWF T1CON ; circuitry, 1:8 prescaler,  
; Clock source is  
; asynchronous to device  
; Timer1 is stopped
BSF T1CON, TMR1ON ; Timer1 starts to increment
; The Timer1 interrupt is disabled, do polling on the overflow bit
; T1_OVFL_WAIT
BTFSS PIR1, TMR1IF
GOTO T1_OVFL_WAIT
; Timer has overflowed
; BCF PIR1, TMR1IF
```
14.14 Design Tips

Question 1: **Timer1 does not seem to be keeping accurate time.**

Answer 1:

There are a few reasons that this could occur:

1. You should never write to Timer1 where that could cause the loss of time. In most cases, that means you should not write to the TMR1L register, but if the conditions are OK, you may write to the TMR1H register. Normally, you write to the TMR1H register if you want the Timer1 overflow interrupt to be sooner than the full 16-bit time-out.

2. You should ensure that your layout uses good PCB layout techniques so noise does not couple onto the Timer1/TMR3 oscillator lines.
Section 14. Timer1

14.15 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Baseline, the Midrange, or High-end families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to Timer1 are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using Timer1 in Asynchronous Clock Mode</td>
<td>AN580</td>
</tr>
<tr>
<td>Low Power Real Time Clock</td>
<td>AN582</td>
</tr>
<tr>
<td>Yet another Clock using the PIC16C92X</td>
<td>AN649</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
14.16 Revision History

Revision A

This is the initial released revision of the Timer1 module description.
Section 15. Timer2

HIGHLIGHTS

This section of the manual contains the following major topics:

15.1 Introduction ............................................................................................................... 15-2
15.2 Control Register ....................................................................................................... 15-3
15.3 Timer Clock Source .................................................................................................. 15-4
15.4 Timer (TMR2) and Period (PR2) Registers ............................................................ 15-4
15.5 TMR2 Match Output ................................................................................................. 15-4
15.6 Clearing the Timer2 Prescaler and Postscaler ....................................................... 15-4
15.7 Sleep Operation ....................................................................................................... 15-4
15.8 Initialization ............................................................................................................. 15-5
15.9 Design Tips ............................................................................................................. 15-6
15.10 Related Application Notes .................................................................................... 15-7
15.11 Revision History ................................................................................................... 15-8
15.1 Introduction

Timer2 is an 8-bit timer with a prescaler, a postscaler and a period register. Using the prescaler and postscaler at their maximum settings, the overflow time is the same as a 16-bit timer. Timer2 is the PWM time-base when the CCP module(s) is used in the PWM mode.

Figure 15-1 shows a block diagram of Timer2. The postscaler counts the number of times that the TMR2 register matched the PR2 register. This can be useful in reducing the overhead of the interrupt service routine on the CPU performance.

Figure 15-1: Timer2 Block Diagram

Note 1: TMR2 register output can be software selected by the SSP Module as a baud clock.
Section 15. Timer2

15.2 Control Register

Register 15-1 shows the Timer2 control register. The prescaler and postscaler selection of Timer2 are controlled by this register.

Register 15-1: T2CON: Timer2 Control Register

<table>
<thead>
<tr>
<th>U-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>—</td>
<td>TOUTPS3</td>
<td>TOUTPS2</td>
<td>TOUTPS1</td>
<td>TOUTPS0</td>
<td>TMR2ON</td>
<td>T2CKPS1</td>
<td>T2CKPS0</td>
</tr>
</tbody>
</table>

- **bit 7**: Unimplemented: Read as ‘0’
- **bit 6-3**: TOUTPS3:TOUTPS0: Timer2 Output Postscale Select bits
  - 0000 = 1:1 Postscale
  - 0001 = 1:2 Postscale
  - ...
  - 1111 = 1:16 Postscale

- **bit 2**: TMR2ON: Timer2 On bit
  - 1 = Timer2 is on
  - 0 = Timer2 is off

- **bit 1-0**: T2CKPS1:T2CKPS0: Timer2 Clock Prescale Select bits
  - 00 = Prescaler is 1
  - 01 = Prescaler is 4
  - 1x = Prescaler is 16

Legend

- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- -n = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown
15.3 Timer Clock Source

The Timer2 module has one source of input clock, the device clock (FOSC/4). A prescale option of 1:1, 1:4 or 1:16 is software selected by control bits T2CKPS1:T2CKPS0 (T2CON register).

15.4 Timer (TMR2) and Period (PR2) Registers

The TMR2 register is readable and writable, and is cleared on all device resets. Timer2 increments from 00h until it matches PR2 and then resets to 00h on the next increment cycle. PR2 is a readable and writable register.

The TMR2 register is cleared and the PR2 register is set when a WDT, POR, MCLR or a BOR reset occurs.

Timer2 can be shut off (disabled from incrementing) by clearing the TMR2ON control bit (T2CON register). This minimizes the power consumption of the module.

Note: If the PR2 register = 00h, the TMR2 register will not increment (Timer2 cleared).

15.5 TMR2 Match Output

The match output of TMR2 goes to two sources:
1. Timer2 Postscaler
2. SSP Clock Input

There are 4-bits which select the postscaler. This allows the postscaler a 1:1 to 1:16 scaling (inclusive). After the postscaler overflows, the TMR2 interrupt flag bit (TMR2IF) is set to indicate the Timer2 overflow. This is useful in reducing the software overhead of the Timer2 interrupt service routine, since it will only execute once every postscaler # of matches.

The match output of TMR2 is also routed to the Synchronous Serial Port module, which may select this via software, as the clock source for the shift clock.

15.6 Clearing the Timer2 Prescaler and Postscaler

The prescaler and postscaler counters are cleared when any of the following occurs:
- a write to the TMR2 register
- a write to the T2CON register

Note: When T2CON is written, TMR2 does not clear.
- any device reset (Power-on Reset, MCLR reset, Watchdog Timer Reset, Brown-out Reset)

15.7 Sleep Operation

During sleep, TMR2 will not increment. The prescaler will retain the last prescale count, ready for operation to resume after the device wakes from sleep.

Table 15-1: Registers Associated with Timer2

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE</td>
<td>PEIE</td>
<td>TMR0IE</td>
<td>INTE</td>
<td>RBIE</td>
<td>TMR0IF</td>
<td>INTF</td>
<td>RBIF</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>IPR</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TMR2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>T2CON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
</tbody>
</table>

Legend: x = unknown; u = unchanged; - = unimplemented read as ‘0’.

Shaded cells are not used by the Timer2 module.

Note 1: The position of this bit is device dependent.
15.8 Initialization

Example 15-1 shows how to initialize the Timer2 module, including specifying the Timer2 prescaler and postscaler.

Example 15-1: Timer2 Initialization

```
CLRF T2CON ; Stop Timer2, Prescaler = 1:1, 
             ; Postscaler = 1:1
CLRF TMR2 ; Clear Timer2 register
CLRF INTCON ; Disable interrupts
LRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts flags
MOVLW 0x72 ; Postscaler = 1:15, Prescaler = 1:16
MOVWF T2CON ; Timer2 is off
MOVLF PR2VALUE ; This is the value
MOVWF PR2 ; to load into the PR2 register.
BSF T2CON, TMR2ON ; Timer2 starts to increment

; The Timer2 interrupt is disabled, do polling on the overflow bit
; T2_OVFL_WAIT
BTFSS PIR1, TMR2IF ; Has Timer2 interrupt occurred?
GOTO T2_OVFL_WAIT ; NO, continue loop

; Timer has overflowed

; BCF PIR1, TMR2IF ; YES, clear flag and continue.
```
15.9 Design Tips

Question 1: Timer2 never seems to increment?

Answer 1:

Ensure that the Timer2 Period register (PR2) is not 0h. This is because when a period match occurs, the TMR2 register is cleared on the next cycle so Timer2 will never increment.
Section 15. Timer 2

15.10 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the Timer2 Module are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using the CCP Module</td>
<td>AN594</td>
</tr>
<tr>
<td>Air Flow Control using Fuzzy Logic</td>
<td>AN600</td>
</tr>
<tr>
<td>Adaptive Differential Pulse Code Modulation</td>
<td>AN643</td>
</tr>
<tr>
<td>using the PIC16/17</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
15.11 Revision History

Revision A

This is the initial released revision of the Timer2 module description.
Section 16. Timer3

HIGHLIGHTS

This section of the manual contains the following major topics:

16.1 Introduction ............................................................................................................... 16-2
16.2 Control Registers ......................................................................................................... 16-3
16.3 Timer3 Operation in Timer Mode .............................................................................. 16-4
16.4 Timer3 Operation in Synchronized Counter Mode .................................................... 16-4
16.5 Timer3 Operation in Asynchronous Counter Mode ................................................... 16-5
16.6 Reading and Writing of Timer3 ................................................................................ 16-6
16.7 Timer3 using the Timer1 Oscillator .......................................................................... 16-9
16.8 Timer3 and CCPx Enable ......................................................................................... 16-10
16.9 Timer3 Prescaler ...................................................................................................... 16-10
16.10 16-bit Mode Timer Reads/Writes .......................................................................... 16-11
16.11 Typical Application ............................................................................................... 16-12
16.12 Sleep Operation ...................................................................................................... 16-13
16.13 Timer3 Prescaler .................................................................................................... 16-13
16.14 Initialization ............................................................................................................ 16-14
16.15 Design Tips ............................................................................................................ 16-16
16.16 Related Application Notes .................................................................................... 16-17
16.17 Revision History .................................................................................................... 16-18
16.1 Introduction

The Timer3 module is a 16-bit timer/counter consisting of two 8-bit registers (TMR3H and TMR3L) that are readable and writable. The TMR3 register pair (TMR3H:TMR3L) increments from 0000h to FFFFh and rolls over to 0000h. The Timer3 Interrupt, if enabled, is generated on overflow, which is latched in the TMR3IF interrupt flag bit. This interrupt can be enabled/disabled by setting/clearing the TMR3IE interrupt enable bit.

Timer3 can operate in one of three modes:
- As a synchronous timer
- As a synchronous counter
- As an asynchronous counter

Some features of Timer3 include:
- TMR3 also has an internal “reset input”, which can be generated by a CCP module.
- TMR3 has the capability to operate off an external crystal/clock.
- TMR3 is the alternate time base for capture/compare

In timer mode, Timer3 increments every instruction cycle. In counter mode, it increments on every rising edge of the external clock input.

The Timer3 increment can be enabled/disabled by setting/clearing control bit TMR3ON (T3CON register).

Timer3 also has an internal “reset input”. This reset can be generated by a CCP special event trigger (Capture/Compare/PWM) module. See the CCP (Capture/Compare/PWM) section for details.

When the Timer1 oscillator is enabled (T1OSCEN, in T1CON, is set), the T1OSCI1 and T1OSO2 pins are configured as oscillator input and output, so the corresponding values in the TRIS register are ignored.

The Timer3 module also has a software programmable prescaler.

The operating mode is determined by clock select bit, TMR3CS (T3CON register), and the synchronization bit, T3SYNC (Figure 16-1).

**Figure 16-1: Timer3 Block Diagram**

![Timer3 Block Diagram](image-url)
### Section 16. Timer3

#### 16.2 Control Registers

Register 16-1 shows the Timer3 control register. This register controls the operating mode of the Timer3 module and contains the function of the CCP Special Event Trigger.

**Register 16-1: T3CON: Timer3 Control Register**

<table>
<thead>
<tr>
<th></th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>RD16</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T3CCP2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T3CKPS1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T3CKPS0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T3CCP1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T3SYNC</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR3CS</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR3ON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

**Legend**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **- n** = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- *x* = bit is unknown

**bit 7** RD16: 16-bit Read/Write Mode Enable

- 1 = Enables register Read/Write of Timer3 in one 16-bit operation
- 0 = Enables register Read/Write of Timer3 in two 8-bit operations

**bit 6,3** T3CCP2:T3CCP1: Timer3 and Timer1 to CCPx Enable bits

- **1x** = Timer3 is the clock source for compare/capture of the CCP modules
- **01** = Timer3 is the clock source for compare/capture of CCP2, Timer1 is the clock source for compare/capture of CCP1
- **00** = Timer1 is the clock source for compare/capture of the CCP modules

**bit 5:4** T3CKPS1:T3CKPS0: Timer3 Input Clock Prescale Select bits

- **11** = 1:8 Prescale value
- **10** = 1:4 Prescale value
- **01** = 1:2 Prescale value
- **00** = 1:1 Prescale value

**bit 2** T3SYNC: Timer3 External Clock Input Synchronization Control bit

- (Not usable if the system clock comes from Timer1/Timer3)

**When TMR3CS = 1:**

- 1 = Do not synchronize external clock input
- 0 = Synchronize external clock input

**When TMR3CS = 0:**

- This bit is ignored. Timer3 uses the internal clock when TMR3CS = 0.

**bit 1** TMR3CS: Timer3 Clock Source Select bit

- **1** = External clock input from T1OSI or T1CKI (on the rising edge after the first falling edge)
- **0** = Internal clock (Fosc/4)

**bit 0** TMR3ON: Timer3 On bit

- **1** = Enables Timer3
- **0** = Stops Timer3
16.3 Timer3 Operation in Timer Mode

Timer mode is selected by clearing the TMR3CS (T3CON register) bit. In this mode, the input clock to the timer is \( \frac{F_{OSC}}{4} \). The synchronize control bit, T3SYNC (T3CON register), has no effect since the internal clock is always synchronized.

16.4 Timer3 Operation in Synchronized Counter Mode

Counter mode is selected by setting bit TMR3CS. In this mode, the timer increments on every rising edge of input on the T1OSI pin (when enable bit T1OSCEN is set) or the T13CKI pin (when bit T1OSCEN is cleared).

If the T3SYNC bit is cleared, then the external clock input is synchronized with internal phase clocks. The synchronization is done after the prescaler stage. The prescaler operates asynchronously.

The timer increments at the Q4-Q1 edge.

In this configuration, during SLEEP mode, Timer3 will not increment even if an external clock is present, since the synchronization circuit is shut off. The prescaler, however, will continue to increment.

16.4.1 External Clock Input Timing for Synchronized Counter Mode

When an external clock input is used for Timer3 in synchronized counter mode, it must meet certain requirements. The external clock requirement is due to internal phase clock (TSClk) synchronization. Also, there is a delay in the actual incrementing of TMR3 after synchronization.

When the prescaler is 1:1, the external clock input is the same as the prescaler output. There is synchronization of T1OSI/T13CKI with the internal phase clocks. Therefore, it is necessary for T1OSI/T13CKI to be high for at least 2TSClk (and a small RC delay) and low for at least 2TSClk (and a small RC delay). Refer to parameters 45, 46, and 47 in the “Electrical Specifications” section.

When a prescaler other than 1:1 is used, the external clock input is divided by the asynchronous prescaler, so that the prescaler output is symmetrical. In order for the external clock to meet the sampling requirement, the prescaler counter must be taken into account. Therefore, it is necessary for T1OSI/T1CKI to have a period of at least 4TSClk (and a small RC delay) divided by the prescaler value. The only requirement on T1OSI/T1CKI high and low time is that they do not violate the minimum pulse width requirements. Refer to parameters 40, 42, 45, 46, and 47 in the “Electrical Specifications” section.
Section 16. Timer3

16.5 Timer3 Operation in Asynchronous Counter Mode

If T3SYNC (T3CON register) is set, the external clock input is not synchronized. The timer continues to increment asynchronously to the internal phase clocks. The timer will continue to run during SLEEP and can generate an interrupt on overflow that will wake-up the processor. However, special precautions in software are needed to read/write the timer (Subsection 16.6.4 “Reading and Writing Timer3 in Asynchronous Counter Mode with RD16 = 0” ). Since the counter can operate in sleep, Timer1 can be used to implement a true real-time clock.

The timer increments at the Q4:Q1 and Q2:Q3 edges.

In asynchronous counter mode, Timer3 cannot be used as a time-base for capture or compare operations.

| Note: | The control bit T3SYNC is not usable when the system clock source comes from the same source as the Timer1/Timer3 clock input, because the T1CKI input will be sampled at one quarter the frequency of the incoming clock. |

16.5.1 External Clock Input Timing with Unsynchronized Clock

If the T3SYNC control bit is set, the timer will increment completely asynchronously. The input clock must meet certain minimum high time and low time requirements. Refer to the Device Data Sheet “Electrical Specifications” section, timing parameters 45, 46, and 47.
16.6 Reading and Writing of Timer3

Timer3 has modes that allow the 16-bit timer register to be read/written as two 8-bit registers or one 16-bit register. The mode depends on the state of the RD16 bit. The following subsections discuss this operation.

16.6.1 Timer3 and 16-bit Read/Write Modes

Timer3 can be configured for 16-bit reads. When the RD16 control bit (T3CON register) is set, the address for TMR3H is mapped to a buffer register for the high byte of Timer3. A read from TMR3L will load the contents of the high byte of Timer3 into the Timer3 high byte buffer. This provides the user with the ability to accurately read all 16-bits of Timer3 without having to determine whether a read of the high byte followed by a read of the low byte is valid due to a rollover between reads.

16.6.2 16-bit Mode Timer Write

A write to the high byte of Timer3 must also take place through the TMR3H buffer register. Timer3 high byte is updated with the contents of TMR3H when a write occurs to TMR3L. This allows a user to write all 16 bits to both the high and low bytes of Timer3 at once (See Figure 16-2).

The high byte of Timer3 is not directly readable or writable in this mode. All reads and writes must take place through the Timer3 high byte buffer register.

Writes to TMR3H do not clear the Timer3 prescaler. The prescaler is only cleared on writes to TMR3L.

16.6.3 16-bit Read-Modify-Write

Read-modify-write instructions like BSF or BCF will read the contents of a register, make the appropriate changes, and place the result back into the register. In the case of Timer3 when configured in 16-bit mode, the read portion of a read-modify-write instruction of TMR3L will not update the contents of the TMR3H buffer. The TMR3H buffer will remain unchanged. When the write of TMR3L portion of the instruction takes place, the contents of TMR3H will be placed into the high byte of Timer3.

Figure 16-2: Timer3 Block Diagram When Configured in 16-bit Read/Write Mode
Section 16. Timer3

16.6.4 Reading and Writing Timer3 in Asynchronous Counter Mode with RD16 = 0

Reading TMR3H or TMR3L while the timer is running from an external asynchronous clock will ensure a valid read (taken care of in hardware). However, the user should keep in mind that reading the 16-bit timer in two 8-bit values poses certain problems, since the timer may overflow between the reads.

For writes, it is recommended that the user simply stop the timer and write the desired values. A write contention may occur by writing to the timer registers, while the register is incrementing. This may produce an unpredictable value in the timer register.

Reading the 16-bit value requires some care, since two separate reads are required to read the entire 16-bits. Example 16-1 shows why this may not be a straightforward read of the 16-bit register.

Example 16-1: Reading 16-bit Register Issues

<table>
<thead>
<tr>
<th>TMR3</th>
<th>Sequence 1</th>
<th>Sequence 2</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Action</td>
<td>TMPH:TMPL</td>
</tr>
<tr>
<td>04FFh</td>
<td>READ TMR3L</td>
<td>xxxxh</td>
</tr>
<tr>
<td>0500h</td>
<td>Store in TMPL</td>
<td>xxFFh</td>
</tr>
<tr>
<td>0501h</td>
<td>READ TMR3H</td>
<td>xxFFh</td>
</tr>
<tr>
<td>0502h</td>
<td>Store in TMPH</td>
<td>05FFh</td>
</tr>
</tbody>
</table>
Example 16-2 shows a routine to read the 16-bit timer value without experiencing the issues shown in Example 16-1. This is useful if the timer cannot be stopped.

**Example 16-2: Reading a 16-bit Free-Running Timer**

```assembly
; All interrupts are disabled
MOVF TMR3H, W ; Read high byte
MOVWF TMPH ;
MOVF TMR3L, W ; Read low byte
MOVWF TMPL ;
MOVF TMR3H, W ; Read high byte
SUBWF TMPH, W ; Sub 1st read with 2nd read
BTFSC STATUS,Z ; Is result = 0
GOTO CONTINUE ; Good 16-bit read
;
; TMR3L may have rolled over between the read of the high and low bytes.
; Reading the high and low bytes now will read a good value.
;
MOVF TMR3H, W ; Read high byte
MOVWF TMPH ;
MOVF TMR3L, W ; Read low byte
MOVWF TMPL ;
; Re-enable the Interrupt (if required)
CONTINUE ; Continue with your code
```

Writing a 16-bit value to the 16-bit TMR3 register is straightforward. First the TMR3L register is cleared to ensure that there are many Timer3 clock/oscillator cycles before there is a rollover into the TMR3H register. The TMR3H register is then loaded, and finally, the TMR3L register is loaded. Example 16-3 shows a routine that accomplishes this:

**Example 16-3: Writing a 16-bit Free-Running Timer**

```assembly
; All interrupts are disabled
CLRF TMR3L ; Clear Low byte, Ensures no rollover into TMR3H
MOVLW HI_BYTE ; Value to load into TMR3H
MOVWF TMR3H, F ; Write High byte
MOVLW LO_BYTE ; Value to load into TMR3L
MOVWF TMR3L, F ; Write Low byte
; Re-enable the Interrupt (if required)
CONTINUE ; Continue with your code
```
Section 16. Timer3

16.7 Timer3 using the Timer1 Oscillator

An alternate crystal oscillator circuit is built into the device. The output of this oscillator can be selected as the input into Timer3. The Timer1 oscillator is primarily intended to operate as a time-base for the timer modules; therefore, the oscillator is primarily intended for a 32 kHz crystal, which is an ideal frequency for real-time keeping. In real-time applications, the timer needs to increment during SLEEP, so SLEEP does not disable the Timer1 oscillator. For many applications, power consumption is also an issue, so the oscillator is designed to minimize consumption.

The Timer1 oscillator is enabled by setting the T1OSCEN control bit (T1CON register). After the Timer1 oscillator is enabled, the user must provide a software time delay to ensure proper oscillator start-up.

Table 16-1 shows the capacitor selection for the Timer1 oscillator.

Note: The Timer1 oscillator allows the counter to operate (increment) when the device is in sleep mode. This allows Timer1 to be used as a real-time clock.

<table>
<thead>
<tr>
<th>Osc Type</th>
<th>Freq</th>
<th>C1</th>
<th>C2</th>
</tr>
</thead>
<tbody>
<tr>
<td>LP</td>
<td>32 kHz</td>
<td>33 pF</td>
<td>33 pF</td>
</tr>
<tr>
<td></td>
<td>100 kHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
<tr>
<td></td>
<td>200 kHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
</tbody>
</table>

Crystals Tested:
- 32.768 kHz: Epson C-001R32.768K-A ± 20 PPM
- 100 kHz: Epson C-2 100.00 KC-P ± 20 PPM
- 200 kHz: STD XTL 200.000 kHz ± 20 PPM

Note 1: Higher capacitance increases the stability of oscillator, but also increases the start-up time.

2: Since each resonator/crystal has its own characteristics, the user should consult the resonator/crystal manufacturer for appropriate values of external components.
16.8 Timer3 and CCPx Enable

Timer3 can be configured as the time base for capture and compare for either CCP2 or both CCP1 and CCP2. Timer3 cannot be used as the time base for CCP1 only. Timer1 can be used as the time base for CCP1 or both CCP1 and CCP2, but not CCP2 only. Control for the assignment of each of the time bases is given by configuring the corresponding T3CCP2 and T3CCP1 bits in the Timer3 control register, and is described in Table 16-2.

<table>
<thead>
<tr>
<th>T3CCP2:T3CCP1</th>
<th>Time base for CCP1</th>
<th>Time base for CCP2</th>
</tr>
</thead>
<tbody>
<tr>
<td>00</td>
<td>TMR1</td>
<td>TMR1</td>
</tr>
<tr>
<td>01</td>
<td>TMR1</td>
<td>TMR3</td>
</tr>
<tr>
<td>10</td>
<td>TMR3</td>
<td>TMR3</td>
</tr>
<tr>
<td>11</td>
<td>TMR3</td>
<td>TMR3</td>
</tr>
</tbody>
</table>

After reset, Timer1 defaults as the time base for compare and capture for both CCP’s.

16.8.1 Resetting Timer3 Using a CCP Trigger Output

If the T3CCP2 or T3CCP1 bit is set and CCP1 or CCP2 is configured in Compare mode to generate a “Special Event Trigger” (CCPxM3:CCPxM0 = 1 0 1 1), this signal will reset Timer3. Timer3 must be configured for either timer or synchronized counter mode to take advantage of this feature. If Timer3 is running in asynchronous counter mode, this reset operation may not work.

In the event that a write to Timer3 coincides with a special event trigger from a CCP module, the write will take precedence.

In this mode of operation, the CCPRxH:CCPRxL register pair effectively becomes the period register for the Timer3 module.

16.8.2 Resetting of TMR3 Register Pair (TMR3H:TMR3L)

TMR3H and TMR3L registers are not cleared on any reset, only by the CCP special event triggers.

T1CON register is reset to 00h on a Power-on Reset or a Brown-out Reset. In any other reset, the register is unaffected.

Timer3 is the default time base for the CCP1 and CCP2 modules. The timer can be disabled as the time base for either CCP1, CCP2, or both, and Timer3 can be substituted. This is achieved by setting control bits in the Timer3 control register.

This is explained in Section 16.8 - Timer3 and CCPx Enable.

When Timer3 is disabled as a the time base for a CCP, the reset on compare for that particular CCP will have no effect on Timer3.

16.9 Timer3 Prescaler

The prescaler counter is cleared on writes to the TMR3H or TMR3L registers.

16.9.1 Timer3 Prescaler 16-bit Read/Write Mode

Writes to TMR3H do not clear the Timer3 prescaler in 16-bit read/write mode, because a write to TMR3H only modifies the Timer3 latch and does not change the contents of Timer3. The prescaler is only cleared on writes to TMR3L.
Section 16. Timer3

16.10 16-bit Mode Timer Reads/Writes

Timer3 has modes that allow the 16-bit time register to be read as two 8-bit registers or one 16-bit register depending on the state of the RD16 bit.

When the RD16 control bit is set, the address for TMR3H is mapped to a buffer register for the high byte of Timer3. A read from TMR3L will load the contents of the high byte of Timer3 into the Timer3 high byte buffer. This provides the user with the ability to accurately read all 16 bits of Timer3 without having to determine whether a read of the high byte followed by a read of the low byte is valid due to a rollover between reads.

16.10.1 16-bit Mode Timer Write

A write to the high byte of Timer3 must also take place through the TMR3H buffer register. Timer3 high byte is updated with the contents of TMR3H when a write occurs to TMR3L. This allows a user to write all 16 bits to both the high and low bytes of Timer3 at once (See Figure 16-3).

The high byte of Timer3 is not directly readable or writable in this mode. All reads and writes must take place through the Timer3 high byte buffer register.

16.10.2 16-bit Read/Modify Write

Read modify write instructions like **BSF** or **BCF** will read the contents of a register, make the appropriate changes, and place the result back into the register. In the case of Timer3 when configured in 16-bit Read/Write mode, the read portion of a read-modify-write instruction of TMR3L will not update the contents of the TMR3H buffer. The TMR3H buffer will remain unchanged. When the write of TMR3L portion of the instruction takes place, the contents of TMR3H will be placed into the high byte of Timer3.

**Figure 16-3: Timer3 Block Diagram Configured in 16-bit Read/Write Mode**

- Data Bus<7:0>
- TMR3L
- TMR3H
- Write TMR3L
- Read TMR3L
- Set TMR3IF flag bit on Overflow
- Timer3 High Byte
- TMR3L
- CLR
- To Timer1 Clock Input
- T11p(1)
- CCP Special Event Trigger
- T3CCPx
- Clock Input
- Synchronized
- Prescaler 1, 2, 4, 8
- T3SYNC
- T3CKPS1:T3CKPS0
- SLEEP input
- Synchronized
- TMR3ON on/off
- 0
- 1

**Note 1:** Signal coming from TMR1 oscillator (see Figure 14-2).
16.11 Typical Application

The external oscillator clock input feature is typically used in applications where real-time needs to be kept, but it is also desirable to have the lowest possible power consumption. The Timer1 oscillator allows the device to be placed in sleep, while the timer continues to increment. When Timer3 overflows the interrupt wakes up the device so that the appropriate registers can be updated.

**Figure 16-4: Timer3 Application**

In this example, a 32kHz crystal is used as the time base for the Real Time Clock. If the clock needs to be updated at 1 second intervals, then the Timer1 must be loaded with a value to allow the Timer1 to overflow at the desired rate. In the case of a 1 second Timer1 overflow, the TMR1H register should be loaded with a value of 80k after each overflow.

**Note:** The TMR3L register should never be modified, since an external clock is asynchronous to the system clock. Writes to the TRM3L register may corrupt the real time counter value causing inaccuracies.
16.12 Sleep Operation

When Timer3 is configured for asynchronous operation, the TMR3 registers will continue to increment for each timer clock (or prescale multiple of clocks). When the TMR3 register overflows, the TMR3IF bit will get set, and if enabled, generate an interrupt that will wake the processor from sleep mode.

The Timer1 oscillator will add a delta current, due to the operation of this circuitry. That is, the power-down current will no longer only be the leakage current of the device, but also the active current of the Timer1 oscillator and other Timer1 circuitry.

16.13 Timer3 Prescaler

The prescaler counter is cleared on writes to the TMR3H or TMR3L registers.

Table 16-3: Registers Associated with Timer3 as a Timer/Counter

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE</td>
<td>PEIE</td>
<td>T0IE</td>
<td>INTE</td>
<td>RBIE</td>
<td>T0IF</td>
<td>INTF</td>
<td>RBIF</td>
<td>0000 000x</td>
<td>0000 000u</td>
</tr>
<tr>
<td>PIR</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>TBIF</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>PIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>T0IF</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>PR</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>TMR3IP(1)</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>TMR3L</td>
<td>Holding register for the Least Significant Byte of the 16-bit TMR3 register</td>
<td>xxxxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
</tr>
<tr>
<td>TMR3H</td>
<td>Holding register for the Most Significant Byte of the 16-bit TMR3 register</td>
<td>xxxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
<td>xxxx</td>
</tr>
<tr>
<td>T3CON</td>
<td>RD16</td>
<td></td>
<td>T3CKPS1</td>
<td>T3CKPS0</td>
<td>T3SCEN</td>
<td>T3SYNC</td>
<td>TMR3CS</td>
<td>TMR3N</td>
<td>--00</td>
<td>0000</td>
</tr>
</tbody>
</table>

Legend:  x = unknown,  u = unchanged,  -- = unimplemented read as '0'.

Shaded cells are not used by the Timer1 module.

Note 1: The placement of this bit is device dependent.
16.14 Initialization

Since Timer3 has a software programmable clock source, there are three examples to show the initialization of each mode. Example 16-4 shows the initialization for the internal clock source, Example 16-5 shows the initialization for the external clock source, and Example 16-6 shows the initialization of the external oscillator mode.

Example 16-4: Timer3 Initialization (Internal Clock Source)

```
CLRF T3CON ; Stop Timer3, Internal Clock Source, 
 ; T1 oscillator disabled, 
 ; prescaler = 1:1
CLRF TMR3H ; Clear Timer3 High byte register
CLRF TMR3L ; Clear Timer3 Low byte register
CLRF INTCON ; Disable interrupts
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts Flags
MOVLW 0x30 ; Internal Clock source 
 ; with 1:8 prescaler
MOVWF T3CON ; Timer3 is stopped and
 ; T1 osc is disabled
BSF T3CON, TMR3ON ; Timer3 starts to increment
; The Timer3 interrupt is disabled, do polling on the overflow bit
T3_OVFL_WAIT
BTFSS PIR1, TMR3IF
GOTO T3_OVFL_WAIT
; Timer has overflowed
BCF PIR1, TMR3IF
```

Example 16-5: Timer3 Initialization (External Clock Source)

```
CLRF T3CON ; Stop Timer3, Internal Clock Source, 
 ; T1 oscillator disabled, 
 ; prescaler = 1:1
CLRF TMR3H ; Clear Timer3 High byte register
CLRF TMR3L ; Clear Timer3 Low byte register
CLRF INTCON ; Disable interrupts
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts Flags
MOVLW 0x32 ; External Clock source 
 ; Clock source is synchronized to device 
 ; Timer3 is stopped and 
 ; T1 osc is disabled
MOVWF T3CON ; Timer3 starts to increment
; The Timer3 interrupt is disabled, do polling on the overflow bit
T3_OVFL_WAIT
BTFSS PIR1, TMR3IF
GOTO T3_OVFL_WAIT
; Timer has overflowed
BCF PIR1, TMR3IF
```
Section 16. Timer3

Example 16-6: Timer3 Initialization (External Oscillator Clock Source)

```
CLRF T3CON ; Stop Timer3, Internal Clock Source, 
; T1 oscillator disabled, 
; prescaler = 1:1
CLRF TMRR3H ; Clear Timer3 High byte register
CLRF TMRR3L ; Clear Timer3 Low byte register
CLRF INTCON ; Disable interrupts
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts Flags
MOVWF T3CON ; External Clock source 
; with oscillator 
MOVWF T3CON ; circuitry, 1:8 prescaler, 
; Clock source is 
; asynchronous to device 
; Timer3 is stopped
BSF T3CON, TMR3ON ; Timer3 starts to increment
; The Timer3 interrupt is disabled, do polling on the overflow bit
T3_OVFL_WAIT
BTFSS PIR1, TMR3IF
GOTO T3_OVFL_WAIT
; Timer has overflowed
BCF PIR1, TMR3IF
```
PIC18C Reference Manual

16.15 Design Tips

Question 1: **Timer3 does not seem to be keeping accurate time.**

Answer 1:

There are a few reasons that this could occur:

1. You should never write to Timer3, where that could cause the loss of time. In most cases, that means you should not write to the TMR3L register, but if the conditions are ok, you may write to the TMR3H register. Normally you write to the TMR3H register if you want the Timer3 overflow interrupt to be sooner than the full 16-bit time-out.

2. You should ensure your layout uses good PCB layout techniques so that noise does not couple onto the Timer1/Timer3 oscillator lines.
16.16 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhance family (that is they may be written for the Baseline, the Midrange, or High-end families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to Timer1 are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using Timer1 in Asynchronous Clock Mode</td>
<td>AN580</td>
</tr>
<tr>
<td>Low Power Real Time Clock</td>
<td>AN582</td>
</tr>
<tr>
<td>Yet another Clock using the PIC16C92X</td>
<td>AN649</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/qa/codeex/
16.17 Revision History

Revision A

This is the initial released revision of the Timer3 module description.
Section 17. Compare/Capture/PWM (CCP)

HIGHLIGHTS
This section of the manual contains the following major topics:

17.1 Introduction ............................................................................................................... 17-2
17.2 CCP Control Register ............................................................................................. 17-3
17.3 Capture Mode .......................................................................................................... 17-4
17.4 Compare Mode ........................................................................................................ 17-7
17.5 PWM Mode ............................................................................................................ 17-10
17.6 Initialization ........................................................................................................... 17-15
17.7 Design Tips .......................................................................................................... 17-17
17.8 Related Application Notes .................................................................................... 17-19
17.9 Revision History .................................................................................................. 17-20
PIC18C Reference Manual

17.1 Introduction

Each CCP (Capture/Compare/PWM) module has three 8-bit registers. These are:
- An 8-bit control register (CCPxCON)
- A 16-bit register (CCPRxH:CCPRxL) that operates as:
  - a 16-bit capture register
  - a 16-bit compare register
  - a 10-bit PWM master/slave duty cycle register

Multiple CCP modules may exist on a single device. The CCP modules are identical in operation, with the exception of the operation of the special event trigger.

Throughout this section, we use generic names for the CCP registers. These generic names are shown in Table 17-1.

Table 17-1: Specific to Generic CCP Nomenclature

<table>
<thead>
<tr>
<th>Generic Name</th>
<th>CCP1</th>
<th>CCP2</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>CCPxCON</td>
<td>CCP1CON</td>
<td>CCP2CON</td>
<td>CCP Control Register</td>
</tr>
<tr>
<td>CCPRxH</td>
<td>CCPR1H</td>
<td>CCPR2H</td>
<td>CCP high byte</td>
</tr>
<tr>
<td>CCPRxL</td>
<td>CCPR1L</td>
<td>CCPR2L</td>
<td>CCP low byte</td>
</tr>
<tr>
<td>CCPx</td>
<td>CCP1</td>
<td>CCP2</td>
<td>CCP pin</td>
</tr>
</tbody>
</table>

17.1.1 Timer Resources

Table 17-2 shows the resources of the CCP modules, in each of its modes. Table 17-3 shows the interactions between the CCP modules, where CCPx is one CCP module and CCPy is another CCP module.

Table 17-2: CCP Mode - Timer Resource

<table>
<thead>
<tr>
<th>CCP Mode</th>
<th>Timer Resource</th>
</tr>
</thead>
<tbody>
<tr>
<td>Capture</td>
<td>Timer1 or Timer3</td>
</tr>
<tr>
<td>Compare</td>
<td>Timer1 or Timer3</td>
</tr>
<tr>
<td>PWM</td>
<td>Timer2</td>
</tr>
</tbody>
</table>

Table 17-3: Interaction of Two CCP Modules

<table>
<thead>
<tr>
<th>CCPx Mode</th>
<th>CCPy Mode</th>
<th>Interaction</th>
</tr>
</thead>
<tbody>
<tr>
<td>Capture</td>
<td>Capture</td>
<td>TMR1 or TMR3 time-base. Time base can be different for each CCP.</td>
</tr>
<tr>
<td>Capture</td>
<td>Compare</td>
<td>The compare could be configured for the special event trigger, which clears either TMR1 or TMR3 depending upon which time base is used.</td>
</tr>
<tr>
<td>Compare</td>
<td>Compare</td>
<td>The compare(s) could be configured for the special event trigger, which clears TMR1 or TMR3 depending upon which time base is used.</td>
</tr>
<tr>
<td>PWM</td>
<td>PWM</td>
<td>The PWMs will have the same frequency, and update rate (TMR2 interrupt).</td>
</tr>
<tr>
<td>PWM</td>
<td>Capture</td>
<td>None</td>
</tr>
<tr>
<td>PWM</td>
<td>Compare</td>
<td>None</td>
</tr>
</tbody>
</table>
Section 17. CCP

17.2 CCP Control Register

Register 17-1 shows the CCP Control Register. This register selects the mode of operation of the CCP module, as well as contains the 2-LSb of the PWM Duty Cycle.

Register 17-1: CCPxCON Register

<table>
<thead>
<tr>
<th>bit 7-6</th>
<th>bit 5-4</th>
<th>bit 3-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>DCxB&lt;1:0&gt;</td>
<td>CCPxM&lt;3:0&gt;</td>
<td></td>
</tr>
</tbody>
</table>

Unimplemented: Read as '0'

Capture Mode:

Unused

Compare Mode:

Unused

PWM Mode:

These bits are the two LSbs (bit1 and bit0) of the 10-bit PWM duty cycle. The upper eight bits (DCx<9:2>) of the duty cycle are found in CCPRxL.

Legend

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as '0'

-n = Value at POR reset  '1' = bit is set  '0' = bit is cleared  x = bit is unknown
17.3 Capture Mode

In Capture mode, CCPRxH:CCPRxL captures the 16-bit value of the TMR1 or TMR3 register when an event occurs on the CCPx pin. An event is defined as:

- Every falling edge
- Every rising edge
- Every 4th rising edge
- Every 16th rising edge

An event is selected by control bits CCPxM3:CCPxM0 (CCPxCON<3:0>). When a capture is made, the interrupt request flag bit, CCPxIF, is set. The CCPxIF bit must be cleared in software. If another capture occurs before the value in register CCPRx is read, the old captured value will be lost.

**Note:** The dedicated time base (Timer1 or Timer3) must be running in Timer mode or Synchronized Counter mode for the CCP module to use the capture feature. In Asynchronous Counter mode, the capture operation may not work.

When the Capture mode is changed, a false capture interrupt may be generated. The user should keep bit CCPxIE clear to avoid false interrupts and should clear flag bit CCPxIF following any such change in operating mode.

*Figure 17-1* shows that a capture does not modify (clear) the 16-bit timer register. This is so the timer (Timer1 or Timer3) can also be used as the time-base for other operations.

The time between two captures can easily be computed as the difference between the value of the 2nd capture and that of the 1st capture. When the timer overflows, the timer interrupt bit, TMRxIF will be set. If enabled, an interrupt will occur, allowing the time-base to be extended to greater than 16 bits.
Section 17. CCP

17.3.1 CCP Pin Configuration

In Capture mode, the CCPx pin should be configured as an input by setting its corresponding TRIS bit. The prescaler can be used to get a very fine average resolution on a constant input frequency. For example, if we have a stable input frequency and we set the prescaler to 1:16, then the total error for those 16 periods is \( T_{CY} \). This gives an effective resolution of \( T_{CY}/16 \), which at 40 MHz is 6.25 ns. This technique is only valid where the input frequency is “stable” over the 16 samples. Without using the prescaler (1:1), each sample would have a resolution of \( T_{CY} \).

Note: If the CCPx pin is configured as an output, a write to the port can cause a capture condition.

Figure 17-1: Capture Mode Operation Block Diagram
17.3.2 Changing Between Capture Modes

When the Capture mode is changed, a capture interrupt may be generated. The user should keep the CCPxIE bit clear to disable these interrupts and should clear the CCPxIF flag bit following any such change in operating mode.

17.3.2.1 CCP Prescaler

There are four prescaler settings, specified by the CCPxM3:CCPxM0 bits. Whenever the CCP module is turned off, or the CCP module is not in capture mode, the prescaler counter is cleared. This means that any reset will clear the prescaler counter.

Switching from one capture prescale setting to another may generate an interrupt. Also, the prescaler counter will not be cleared, therefore the first capture may be from a non-zero prescaler. Example 17-1 shows the recommended method for switching between capture prescale settings. This example uses CCP1 and clears the prescaler counter so not to generate an unintended interrupt.

Example 17-1: Changing Between Capture Prescalers

```
CLRF CCP1CON ; Turn CCP module off
MOVLW NEW_CAPT_PS ; Load the W reg with the new prescaler
MOVWF CCP1CON ; mode value and CCP ON
```

To clear the Capture prescaler count, the CCP module must be configured into any non-capture CCP mode (Compare, PWM, or CCP off modes).

17.3.3 Sleep Operation

When the device is placed in SLEEP, the timer will not increment (since it is in synchronous mode), but the prescaler will continue to count events (not synchronized). When a specified capture event occurs, the CCPxIF bit will be set, but the capture register will not be updated. If the CCP interrupt is enabled, the device will wake-up from SLEEP. The value in the 16-bit TMR1 register is not transferred to the 16-bit capture register. Effectively, this allows the CCP pin to be used as another external interrupt.

17.3.4 Effects of a Reset

The CCP module is off, and the value in the capture prescaler is cleared.
Section 17. CCP

17.4 Compare Mode

In Compare mode, the 16-bit CCPRx register value is constantly compared against the TMR1 or TMR3 register pair value. When a match occurs, the CCPx pin is:

- Driven High
- Driven Low
- Toggle output (Low to High or High to Low)
- Not affected (remains unchanged) and configured as I/O pin

The action on the pin is based on the value of control bits CCPxM3:CCPxM0 (CCPxCON3:CCPxCON0). At the same time, interrupt flag bit CCPxIF is set.

Note: The dedicated time base (Timer1 or Timer3) must be running in Timer mode or Synchronized Counter mode if the CCP module is using the compare feature. In Asynchronous Counter mode, the compare operation may not work.

Figure 17-2: Compare Mode Operation Block Diagram
17.4.1 CCP Pin Operation in Compare Mode

The user must configure the CCPx pin as an output by clearing the appropriate TRIS bit.

| Note: | Clearing the CCPxCON register will force the CCPx compare output latch to the default low level. This is not the Port I/O data latch. |

Selecting the compare output mode, forces the state of the CCP pin to the state that is opposite of the match state. So if the Compare mode is selected to force the output pin low on match, then the output will be forced high until the match occurs (or the mode is changed). In the compare toggle mode, the CCPx pin output is initially forced to the low state.

17.4.2 Software Interrupt Mode

When Generate Software Interrupt mode is chosen, the CCPx pin is not affected. Only a CCP interrupt is generated (if enabled).

17.4.3 Special Event Trigger

In this mode, an internal hardware trigger is generated that may be used to initiate an action.

The special event trigger output of CCPx resets the assigned timer register pair (TMR1 or TMR3 depending upon the state of the T3CCPx bits). This allows the CCPRxH:CCPRxL registers to effectively be a 16-bit programmable period register for the timer (Timer1 or Timer3).

For some devices, the special trigger output of the CCP module resets the timer (TMR1 or TMR3) register pair (depending upon the state of the T3CCPx bits), and starts an A/D conversion (if the A/D module is enabled).

| Note: | The special event trigger will not set the Timers interrupt flag bit, TMRxIF. |

17.4.4 Sleep Operation

When the device is placed in SLEEP, the timer will not increment (since it is in Synchronous mode), and the state of the module will not change. If the CCP pin is driving a value, it will continue to drive that value. When the device wakes-up, it will continue from this state.

17.4.5 Effects of a Reset

The CCP module is off.
### Section 17. CCP

#### Table 17-4: Registers Associated with Capture, Compare, Timer1 and Timer3

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE/</td>
<td>PEIE/</td>
<td>PEIE/</td>
<td>TMR0IE</td>
<td>INT0IE</td>
<td>RBIE/</td>
<td>INT0IF</td>
<td>RBIF/</td>
<td>0000 000x</td>
<td>0000 000u</td>
</tr>
<tr>
<td></td>
<td>GIEH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIR1</td>
<td>PSPIF</td>
<td>ADIF</td>
<td>RCIF</td>
<td>TXIF</td>
<td>SSPIF</td>
<td>CCP1IF</td>
<td>TMR2IF</td>
<td>TMR1IF</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIE1</td>
<td>PSPIE</td>
<td>ADIE</td>
<td>RIE</td>
<td>TXIE</td>
<td>SSPIE</td>
<td>CCP1E</td>
<td>TMR2IE</td>
<td>TMR1IE</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>IPR1</td>
<td>PSPIP</td>
<td>ADIP</td>
<td>RCIP</td>
<td>TXIP</td>
<td>SSPIP</td>
<td>CCP1P</td>
<td>TMR2IP</td>
<td>TMR1IP</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TRISC</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR1L</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR1H</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T1CON</td>
<td>RD16</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CCP1L</td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CCP2L</td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR2L</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR3L</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T3CON</td>
<td>RD16</td>
<td>T3CCP</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Legend: x = unknown, u = unchanged, - = unimplemented read as '0'.</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Shaded cells are not used by Capture, Compare, Timer1 and Timer3.</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** The PSPIF, PSPIE and PSPIP bits are reserved on the PIC18C2x2 devices. Always maintain these bits clear.
17.5 PWM Mode

In Pulse Width Modulation (PWM) mode, the CCPx pin produces up to a 10-bit resolution PWM output. Since the CCPx pin is multiplexed with the PORT data latch, the corresponding TRIS bit must be cleared to make the CCPx pin an output.

**Note:** Clearing the CCPxCON register will force the CCPx PWM output latch to the default low level. This is not the Port I/O data latch.

Figure 17-3 shows a simplified block diagram of one CCP module in PWM mode. Depending on the device there can be more than one CCP module connected to Timer2. Each CCP module can support one Pulse Width Modulation (PWM) output signal. This PWM signal can attain a resolution of up to 10-bits, from the 8-bit Timer2 module. Two extra bits are used to extend Timer2 to 10 bits (see Section 17.5.1). A PWM output waveform is shown in Figure 17-4.

For a step-by-step procedure on how to set up the CCP module for PWM operation, see Section 17.5.4.

**Figure 17-3: Simplified PWM Block Diagram**

**Figure 17-4: PWM Output Waveform**

- DutyCycle = DCxB9:DCxB0
- Period = PR2 + 1

1. Timer2 is cleared and new duty cycle value is loaded from the Duty Cycle latch into the Duty Cycle Slave register
2. Timer2 value equals to value in Duty Cycle Latch register, CCP Pin is driven low
3. Timer2 overflow, value from Duty Cycle Latch is loaded into Slave Register, CCP Pin driven high

**Note 1:** For 10-bit time base generation see Section 17.5.1.
Section 17. CCP

17.5.1 10-bit Time Base Generation

The PWM output has up to 10-bits of resolution. This is achieved by creating a 10-bit PWM Time base (TB9:TB0). Figure 17-5 shows the block diagram of this 10-bit PWM Time base. When TSClk is the clock source to the 10-bit counter, the counter increments on each TSClk. If a prescaler is selected, the 10-bit counter increments every prescale would by TSClk.

Figure 17-5: 10-bit Time Base Block Diagram

17.5.2 PWM Period

The PWM period is specified by writing to the PR2 register. The PWM period can be calculated using Equation 17-1.

Equation 17-1: Calculation for PWM Period

\[
T_{PWM \ period} = \left( \frac{PR2 + 1}{4} \right) \cdot T_{SCLK} \cdot (TMR2 \ prescale \ value)
\]

Where

- PR2 = Value in PR2 Register
- TSClk = Oscillator Clock

When TMR2 is equal to PR2, the following three events occur on the next increment cycle:
- TMR2 is cleared
- The CCPx pin is set (exception: if PWM duty cycle = 0%, the CCPx pin will not be set)
- The PWM duty cycle is latched from CCPRxL into CCPRxH

Note: The Timer2 postscaler is not used in the determination of the PWM frequency. The postscaler could be used to generate TMR2 interrupts at a different frequency than the PWM output.
The PWM duty cycle is specified by writing to the CCPRxL register and to the DCxB1:DCxB0 (CCPxCON<5:4>) bits, if 10-bit resolution is desired. The CCPRxL contains the eight MSbs and CCPxCON<5:4> contains the two LSbs. This 10-bit value is represented by DCxB9:DCxB0. Equation 17-2 is used to calculate the PWM duty cycle.

Equation 17-2: Equation for calculating the PWM Duty Cycle

\[
\text{PWM Duty Cycle} = (\text{DCxB}<9:0> \text{ bits value}) \times \text{TSLCK} \times (\text{TMR2 prescale value})
\]

Where PWM Duty Cycle = PWM Duty Cycle Time

TSLCK = Oscillator Clock

The DCxB<9:0> bits can be written to at any time, but the duty cycle value is not latched into CCPRxH until after a match between PR2 and TMR2 occurs (which is the end of the current period). In PWM mode, CCPRxH is a read-only register.

The CCPRxH register and a 2-bit internal latch are used to double buffer the PWM duty cycle. This double buffering is essential for glitchless PWM operation.

When CCPRxH and a 2-bit latch match the value of TMR2 concatenated with the internal 2-bit Q clock (or two bits of the TMR2 prescaler), the CCPx pin is cleared. This is the end of the duty cycle. Equation 17-3 is used to calculate the maximum PWM resolution in bits for a given PWM frequency.

Equation 17-3: Calculation for Maximum PWM Resolution

\[
\text{Maximum PWM Resolution (bits)} = \log \left( \frac{\text{Fosc}}{\text{FPWM}} \right) \div \log(2) \text{ bits}
\]

Note: If the PWM duty cycle value is longer than the PWM period, the CCPx pin will not be cleared. This allows a duty cycle of 100%.

17.5.3.1 Minimum Resolution

The minimum resolution (in time) of each bit of the PWM duty cycle depends on the prescaler of Timer2. Table 17-5 shows the selections for the minimum resolution time.

<table>
<thead>
<tr>
<th>Prescaler Value</th>
<th>T2CKPS1:T2CKPS0</th>
<th>Minimum Resolution (Time)</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>0 0</td>
<td>TSLCK</td>
</tr>
<tr>
<td>4</td>
<td>0 1</td>
<td>TCY</td>
</tr>
<tr>
<td>16</td>
<td>1 x</td>
<td>4 TCY</td>
</tr>
</tbody>
</table>
17.5.3.2 Example Calculation for PWM Period and Duty Cycle

This section shows an example calculation for the PWM Period and Duty Cycle. Furthermore, example PWM frequencies based upon different oscillator frequencies are given.

Example 17-2: PWM Period and Duty Cycle Calculation

Desired PWM frequency is 78.125 kHz.
Fosc = 20 MHz
TMR2 prescale = 1

\[
\frac{1}{78.125 \text{ kHz}} = \left(\frac{\text{PR2} + 1}{2^{16}}\right) \cdot 4 \cdot 1/20 \text{ MHz} \cdot 1
\]

12.8 ms = \left(\frac{\text{PR2} + 1}{2^{16}}\right) \cdot 4 \cdot 50 \text{ ns} \cdot 1

PR2 = 63

Find the maximum resolution of the duty cycle that can be used with a 78.125 kHz frequency and 20 MHz oscillator:

\[
\frac{1}{78.125 \text{ kHz}} = 2^{\text{PWM Resolution}} \cdot 1/20 \text{ MHz} \cdot 1
\]

12.8 ms = 2^{\text{PWM Resolution}} \cdot 50 \text{ ns} \cdot 1

256 = 2^{\text{PWM Resolution}}

\[
\log(256) = (\text{PWM Resolution}) \cdot \log(2)
\]

PWM Resolution = 8.0

At most, an 8-bit resolution duty cycle can be obtained from a 78.125 kHz frequency and a 20 MHz oscillator (i.e., 0 ≤ DCxB9:DCxB0 ≤ 255). Any value greater than 255 will result in a 100% duty cycle.

In order to achieve higher resolution, the PWM frequency must be decreased. In order to achieve higher PWM frequency, the resolution must be decreased.

Table 17-6 lists example PWM frequencies and resolutions for Fosc = 20 MHz. Table 17-7 lists example PWM frequencies and resolutions for Fosc = 40 MHz. The TMR2 prescaler and PR2 values are also shown.

### Table 17-6: Example PWM Frequencies and Bit Resolutions at 20 MHz

<table>
<thead>
<tr>
<th>PWM Frequency</th>
<th>1.22 kHz</th>
<th>4.88 kHz</th>
<th>19.53 kHz</th>
<th>78.12 kHz</th>
<th>156.3 kHz</th>
<th>208.3 kHz</th>
</tr>
</thead>
<tbody>
<tr>
<td>Timer Prescaler (1, 4, 16)</td>
<td>16</td>
<td>4</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>PR2 Value</td>
<td>0xFF</td>
<td>0xFF</td>
<td>0xFF</td>
<td>0x3F</td>
<td>0x1F</td>
<td>0x17</td>
</tr>
<tr>
<td>Maximum Resolution (bits)</td>
<td>10</td>
<td>10</td>
<td>10</td>
<td>8</td>
<td>7</td>
<td>5.5</td>
</tr>
</tbody>
</table>

### Table 17-7: Example PWM Frequencies and Bit Resolutions at 40 MHz

<table>
<thead>
<tr>
<th>PWM Frequency</th>
<th>2.44 kHz</th>
<th>9.76 kHz</th>
<th>39.06 kHz</th>
<th>78.12 kHz</th>
<th>208.3 kHz</th>
<th>416.6 kHz</th>
</tr>
</thead>
<tbody>
<tr>
<td>Timer Prescaler (1, 4, 16)</td>
<td>16</td>
<td>4</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>PR2 Value</td>
<td>0xFF</td>
<td>0xFF</td>
<td>0xFF</td>
<td>0x3F</td>
<td>0x1F</td>
<td>0x17</td>
</tr>
<tr>
<td>Maximum Resolution (bits)</td>
<td>10</td>
<td>10</td>
<td>10</td>
<td>8</td>
<td>7</td>
<td>5.5</td>
</tr>
</tbody>
</table>
17.5.4 Set-up for PWM Operation

The following steps configure the CCP module for PWM operation:

1. Establish the PWM period by writing to the PR2 register.
2. Establish the PWM duty cycle by writing to the DCxB9:DCxB0 bits.
3. Make the CCPx pin an output by clearing the appropriate TRIS bit.
4. Establish the TMR2 prescale value and enable Timer2 by writing to T2CON.
5. Configure the CCP module for PWM operation.

17.5.5 Sleep Operation

When the device is placed in sleep, Timer2 will not increment, and the state of the module will not change. If the CCP pin is driving a value, it will continue to drive that value. When the device wakes-up, it will continue from this state.

17.5.6 Effects of a Reset

The CCP module is off.

Table 17-8: Registers Associated with PWM and Timer2

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000 000x</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PIE1</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PR1</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TRISC</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111</td>
<td>1111 1111</td>
</tr>
<tr>
<td>TMR2</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td>0000</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PR2</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td>1111</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>T2CON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>-00 0000</td>
</tr>
<tr>
<td>CCP1L</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP1H</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP1CON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP2L</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP2H</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP3CON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>DC1B1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>DC1B0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP1M3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP1M2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP1M1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP1M0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP2M3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP2M2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP2M1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
<tr>
<td>CCP2M0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxxx xxxxx</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged, — = unimplemented read as ‘0’.
Shaded cells are not used by PWM and Timer2.

Note 1: The PSPIF, PSPIE and PSPIP bits are reserved on the PIC18C2X2 devices. Always maintain these bits clear.
Section 17. CCP

17.6 Initialization

The CCP module has three modes of operation. Example 17-3 shows the initialization of capture mode, Example 17-4 shows the initialization of compare mode and Example 17-5 shows the initialization of PWM mode.

Example 17-3: Capture Initialization

```
CLRF CCP1CON ; CCP Module is off
CLRF TMR1H ; Clear Timer1 High byte
CLRF TMR1L ; Clear Timer1 Low byte
CLRF INTCON ; Disable interrupts and clear TOIF
BSF TRISC, CCP1 ; Make CCP pin input
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts flags
MOVLW 0x06 ; Capture mode, every 4th rising edge
MOVF CCP1CON ;
BSF T1CON, TMR1ON ; Timer1 starts to increment
;
; The CCP1 interrupt is disabled,
; do polling on the CCP Interrupt flag bit
;
Capture_Event
BTFSS PIR1, CCP1IF
GOTO Capture_Event
;
; Capture has occurred
;
BCF PIR1, CCP1IF ; This needs to be done before
; next compare
```
Example 17-4: Compare Initialization

```
CLRF CCPICON ; CCP Module is off
CLRF TMR1H ; Clear Timer1 High byte
CLRF TMR1L ; Clear Timer1 Low byte
CLRF INTCON ; Disable interrupts and clear TOIF
MOVLW 0x80 ; Load 0x80 (Example Value)
    ; into W-Register
MOVWF CCPRIH ; Load value to compare into CCPRIH
MOVWF CCPRIL ; Load value to compare into CCPRIL
BCF TRISC, CCP1 ; Make CCP pin output if controlling
    ; state of pin
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts Flags
MOVLW 0x08 ; Compare mode, set CCP1 pin on match
MOVWF CCP1CON ;
    ; The CCP1 interrupt is disabled, do polling on the CCP Interrupt flag bit
    ; Compare_Event
    BTFSS PIR1, CCP1IF
    GOTO Compare_Event
    ; Compare has occurred
    BCF PIR1, CCP1IF ; This needs to be done before
    ; next compare
```

Example 17-5: PWM Initialization

```
CLRF CCPICON ; CCP Module is off
CLRF TMR2 ; Clear Timer2
MOVLW 0x7F ;
MOVWF PR2 ;
MOVLW 0x1F ; Duty Cycle is 25% of PWM Period
MOVWF CCPRIL ;
CLRF INTCON ; Disable interrupts and clear TOIF
BCF TRISC, PWM1 ; Make pin output
CLRF PIE1 ; Disable peripheral interrupts
CLRF PIR1 ; Clear peripheral interrupts Flags
MOVLW 0x2C ; PWM mode, 2 LSbs of
    ; Duty cycle = 10
MOVWF CCP1CON ;
    ; The CCP1 interrupt is disabled, do polling on the TMR2 Interrupt flag bit
    ; PWM_Period_Match
    BTFSS PIR1, TMR2IF
    GOTO PWM_Period_Match
    ; Update this PWM period and the following PWM Duty cycle
    BCF PIR1, TMR2IF
```
17.7 Design Tips

Question 1: What timers can I use for the capture and compare modes?
Answer 1:
The capture and compare modes are designed around Timer1 and Timer3, so no other timer can be used for these functions. This also means that if multiple CCP modules are being used for a capture or compare function, they can share the same timer.

Question 2: What timers can I use with the PWM mode?
Answer 2:
The PWM mode is designed around Timer2, so no other timer can be used for this function. It is the only timer with a period register associated with it. If multiple CCP modules are doing PWM, they will share the same timer and have the same PWM period and frequency.

Question 3: Can I use one CCP module to do capture (or compare) AND PWM at the same time, since they use different timers as their reference?
Answer 3:
The timers may be different, but other logic functions are shared. However, you can switch from one mode to the other. For a device with 2 CCP modules, you can also have CCP1 set up for PWM and CCP2 set up for capture or compare (or vice versa) since they are two independent modules.

Question 4: How does a reset affect the CCP module?
Answer 4:
Any reset will turn the CCP module off. See the “Reset” section to see reset values.

Question 5: I am setting up the CCP1CON module for “Compare Mode, trigger special event” (1011) that resets TMR1. When a compare match occurs, will I have both the TMR1 and the CCP1 interrupts pending (TMR1IF is set, CCP1IF is set)?
Answer 5:
The CCP1IF flag will be set on the match condition. TMR1IF is set when Timer1 overflows, and the special trigger reset of Timer1 is not considered an overflow. However, if both the CCPR1L and CCPR1H registers are set at FFh, then an overflow occurs at the same time as the match, which will then set both CCP1IF and TMR1IF.

Question 6: How do I use Timer2 as a general purpose timer, with an interrupt flag on rollover?
Answer 6:
Timer2 always resets to zero when it equals PR2 and flag bit TMR2IF always gets set at this time. By putting FFh into PR2, you will get an interrupt on overflow at FFh. Quite often it is desirable to have an event occur at a periodic rate, perhaps an interrupt driven event. Normally an initial value would be placed into the timer so that the overflow will occur at the desired time. This value would have to be placed back into the timer every time it overflowed to make the interrupts occur at the same desired rate. The benefit of Timer2 is that a value can be written to PR2 that will cause it to reset at your desired time interval. This means you do not have the housekeeping chore of reloading the timer every time it overflows, since PR2 maintains its value.
Question 7: I am using a CCP module in PWM mode. The duty cycle being outputted is almost always 100%, even when my program writes a value like 7Fh to the duty cycle register, which should be 50%. What am I doing wrong?

Answer 7:
1. The value in CCPRxL is higher than PR2. This happens quite often when a user desires a fast PWM output frequency and writes a small value in the PR2. In this case, if a value of 7Eh were written to PR2, then a value 7Fh in CCPRxL will result in 100% duty cycle.
2. If the TRIS bit corresponding to the CCP output pin you are using is configured as an input, the PWM output cannot drive the pin. In this case, the pin would float and duty cycle may appear to be 0%, 100% or some other floating value.

Question 8: I want to determine a signal frequency using the CCP module in capture mode to find the period. I am currently resetting Timer1 on the first edge, then using the value in the capture register on the second edge as the time period. The problem is that my code to clear the timer does not occur until almost twelve instructions after the first capture edge (interrupt latency plus saving of registers in interrupt), so I cannot measure very fast frequencies. Is there a better way to do this?

Answer 8: You do not need to zero the counter to find the difference between two pulse edges. Just take the first captured value and put it into another set of registers. Then when the second capture event occurs, subtract the first event from the second. Assuming that your pulse edges are not so far apart that the counter can wrap around past the last capture value, the answer will always be correct. This is illustrated by the following example:

1. First captured value is FFFEh. Store this value in two registers.
2. The second capture value is 0001h (the counter has incremented three times).
3. 0001h - FFFEh = 0003, which is the same as if you had cleared Timer1 to zero and let it count to 3. (Theoretically, except that there was a delay getting to the code that clears Timer1, so actual values would differ).

The interrupt overhead is now less important because the values are captured automatically. For even faster inputs, do not enable interrupts and just test the flag bit in a loop. If you must also capture very long time periods, such that the timer can wrap around past the previous capture value, then consider using an auto-scaling technique that starts with a large prescale, and shorten the prescale as you converge on the exact frequency.
Section 17. CCP

17.8 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the CCP modules are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using the CCP Modules</td>
<td>AN594</td>
</tr>
<tr>
<td>Implementing Ultrasonic Ranging</td>
<td>AN597</td>
</tr>
<tr>
<td>Air Flow Control Using Fuzzy Logic</td>
<td>AN600</td>
</tr>
<tr>
<td>Adaptive Differential Pulse Code Modulation</td>
<td>AN643</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
17.9 Revision History

Revision A

This is the initial released revision of the Enhanced MCU CCP module description.
Section 18. ECCP

Please check the Microchip web site for Revision B of the ECCP Section.
18.1 Revision History

Revision A

This is the initial released revision of the Enhanced MCU ECCP module description.
Section 19. Synchronous Serial Port (SSP)

HIGHLIGHTS

This section of the manual contains the following major topics:

19.1 Introduction ............................................................................................................... 19-2
19.2 Control Registers ......................................................................................................... 19-4
19.3 SPI Mode .................................................................................................................... . 19-8
19.4 SSP I2C Operation .................................................................................................... 19-18
19.5 Initialization ............................................................................................................. ... 19-28
19.6 Design Tips ................................................................................................................ 19-30
19.7 Related Application Notes.......................................................................................... 19-31
19.8 Revision History......................................................................................................... 19-32
19.1 Introduction

The Synchronous Serial Port (SSP) module is a serial interface useful for communicating with other peripherals or microcontroller devices. These peripheral devices may be serial EEPROMs, shift registers, display drivers, A/D converters, etc. The SSP module can operate in one of two modes:

- Serial Peripheral Interface (SPI™)
- Inter-Integrated Circuit (I²C™)
  - Slave mode
  - I/O slope control, and Start and Stop bit detection to ease software implementation of Master and Multi-master modes
Section 19. SSP

Section 19.2 forced to next page for formatting purposes.
19.2 Control Registers

Register 19-1 shows the SSPSTAT register while Register 19-2 shows the SSPCON register.

**Register 19-1: SSPSTAT: Synchronous Serial Port Status Register**

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>SMP</th>
<th>Bit 6</th>
<th>CKE</th>
<th>Bit 5</th>
<th>D/A</th>
<th>Bit 4</th>
<th>P</th>
<th>Bit 3</th>
<th>S</th>
<th>Bit 2</th>
<th>R/W</th>
<th>Bit 1</th>
<th>UA</th>
<th>Bit 0</th>
<th>BF</th>
</tr>
</thead>
<tbody>
<tr>
<td>SMP: SPI data input sample phase</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SPI Master Mode</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Input data sampled at end of data output time</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Input data sampled at middle of data output time</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SPI Slave Mode</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SMP must be cleared when SPI is used in slave mode</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>I2C Mode</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>This bit must be maintained clear.</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CKE: SPI Clock Edge Select (Figure 19-3, Figure 19-4, and Figure 19-5)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SPI Mode</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CKP = 0 (SSPCON&lt;4&gt;)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Data transmitted on rising edge of SCK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Data transmitted on falling edge of SCK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CKP = 1 (SSPCON&lt;4&gt;)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Data transmitted on falling edge of SCK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Data transmitted on rising edge of SCK</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>I2C Mode</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>This bit must be maintained clear.</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D/A: Data/Address bit (I2C mode only)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Indicates that the last byte received or transmitted was data</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Indicates that the last byte received or transmitted was address</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>P: Stop bit</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>(I2C mode only. This bit is cleared when the SSP module is disabled)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Indicates that a stop bit has been detected last (this bit is ‘0’ on RESET)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Stop bit was not detected last</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>S: Start bit</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>(I2C mode only. This bit is cleared when the SSP module is disabled)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Indicates that a start bit has been detected last (this bit is ‘0’ on RESET)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Start bit was not detected last</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>R/W: Read/Write bit information (I2C mode only)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>This bit holds the R/W bit information following the last address match. This bit is only valid from the address match to the next start bit, stop bit, or not ACK bit.</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Read</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Write</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>UA: Update Address (10-bit I2C mode only)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Indicates that the user needs to update the address in the SSPADD register</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Address does not need to be updated</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
Section 19. SSP

bit 0  **BF**: Buffer Full Status bit

**Receive** (SPI and I²C modes)
- 1 = Receive complete, SSPBUF is full
- 0 = Receive not complete, SSPBUF is empty

**Transmit** (I²C mode only)
- 1 = Transmit in progress, SSPBUF is full
- 0 = Transmit complete, SSPBUF is empty

**Legend**
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- _n_ = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown
Register 19-2: SSPCON: Synchronous Serial Port Control Register

<table>
<thead>
<tr>
<th>bit 7</th>
<th>bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>WCOL</td>
<td>SSPOV</td>
</tr>
<tr>
<td>SSPE</td>
<td>CKP</td>
</tr>
<tr>
<td>SSPM3</td>
<td>SSPM2</td>
</tr>
<tr>
<td>SSPM1</td>
<td>SSPM0</td>
</tr>
</tbody>
</table>

### bit 7: WCOL (Write Collision Detect bit)

- **1**: The SSPBUF register was written to while the previous word was being transmitted. (must be cleared in software)
- **0**: No collision

### bit 6: SSPOV (Receive Overflow Indicator bit)

- **In SPI mode:**
  - **1**: A new byte was received while the SSPBUF register was still holding the previous data. In case of overflow, the data in SSPSR is lost and the SSPBUF is no longer updated. Overflow can only occur in slave mode. The user must read the SSPBUF, even if only transmitting data, to avoid setting overflow. In master mode the overflow bit is not set since each new reception (and transmission) is initiated by writing to the SSPBUF register.
  - **0**: No overflow
- **In I2C mode:**
  - **1**: A byte was received while the SSPBUF register was still holding the previous byte. SSPO is a “don’t care” in transmit mode. SSPOV must be cleared in software in either mode.
  - **0**: No overflow

### bit 5: SSPE (Synchronous Serial Port Enable bit)

- **In both modes, when enabled, these pins must be properly configured as input or output.**
  - **In SPI mode:**
    - **1**: Enables serial port and configures SCK, SDO, SDI, and SS as the source of the serial port pins
    - **0**: Disables serial port and configures these pins as I/O port pins
  - **In I2C mode:**
    - **1**: Enables the serial port and configures the SDA and SCL pins as the source of the serial port pins
    - **0**: Disables serial port and configures these pins as I/O port pins

### bit 4: CKP (Clock Polarity Select bit)

- **In SPI mode:**
  - **1**: Idle state for clock is a high level
  - **0**: Idle state for clock is a low level
- **In I2C mode:**
  - SCK release control
  - **1**: Enable clock
  - **0**: Holds clock low (clock stretch) (Used to ensure data setup time)
### Section 19. SSP

**bit 3:0** \textbf{SSPM3:SSPM0}: Synchronous Serial Port Mode Select bits

<table>
<thead>
<tr>
<th>Value</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0000</td>
<td>SPI master mode, clock = FOSC/4</td>
</tr>
<tr>
<td>0001</td>
<td>SPI master mode, clock = FOSC/16</td>
</tr>
<tr>
<td>0010</td>
<td>SPI master mode, clock = FOSC/64</td>
</tr>
<tr>
<td>0011</td>
<td>SPI master mode, clock = TMR2 output/2</td>
</tr>
<tr>
<td>0100</td>
<td>SPI slave mode, clock = SCK pin. SS pin control enabled. SS can be used as I/O pin</td>
</tr>
<tr>
<td>0101</td>
<td>SPI slave mode, clock = SCK pin. SS pin control disabled. SS can be used as I/O pin</td>
</tr>
<tr>
<td>0110</td>
<td>I²C slave mode, 7-bit address</td>
</tr>
<tr>
<td>0111</td>
<td>I²C slave mode, 10-bit address</td>
</tr>
<tr>
<td>1000</td>
<td>Reserved</td>
</tr>
<tr>
<td>1001</td>
<td>Reserved</td>
</tr>
<tr>
<td>1010</td>
<td>Reserved</td>
</tr>
<tr>
<td>1011</td>
<td>I²C firmware controlled master mode (slave idle)</td>
</tr>
<tr>
<td>1100</td>
<td>Reserved</td>
</tr>
<tr>
<td>1101</td>
<td>Reserved</td>
</tr>
<tr>
<td>1110</td>
<td>I²C slave mode, 7-bit address with start and stop bit interrupts enabled</td>
</tr>
<tr>
<td>1111</td>
<td>I²C slave mode, 10-bit address with start and stop bit interrupts enabled</td>
</tr>
</tbody>
</table>

**Legend**

- \textbf{R} = Readable bit
- \textbf{W} = Writable bit
- \textbf{U} = Unimplemented bit, read as '0'
- \textbf{-n} = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- \textbf{x} = bit is unknown

Microwire is a trademark of National Semiconductor.

© 2000 Microchip Technology Inc.
19.3 SPI Mode

The SPI mode allows 8-bits of data to be synchronously transmitted and received simultaneously. All four modes of SPI are supported, as well as Microwire™ (sample edge) when the SPI is in the master mode.

To accomplish communication, typically three pins are used:
- Serial Data Out (SDO)
- Serial Data In (SDI)
- Serial Clock (SCK)

Additionally a fourth pin may be used when in a slave mode of operation:
- Slave Select (SS)

19.3.1 Operation

When initializing the SPI, several options need to be specified. This is done by programming the appropriate control bits in the SSPCON register (SSPCON<5:0>) and SSPSTAT<7:6>. These control bits allow the following to be specified:
- Master Mode (SCK is the clock output)
- Slave Mode (SCK is the clock input)
- Clock Polarity (Idle state of SCK)
- Clock edge (output data on rising/falling edge of SCK)
- Data Input Sample Phase
- Clock Rate (Master mode only)
- Slave Select Mode (Slave mode only)

Figure 19-1 shows the block diagram of the SSP module, when in SPI mode.

Figure 19-1: SSP Block Diagram (SPI Mode)
The SSP consists of a transmit/receive Shift Register (SSPSR) and a buffer register (SSPBUF). The SSPSR shifts the data in and out of the device, MSb first. The SSPBUF holds the data that was written to the SSPSR, until the received data is ready. Once the 8-bits of data have been received, that byte is moved to the SSPBUF register. Then the buffer full detect bit, BF (SSPSTAT<0>), and interrupt flag bit, SSPIF, are set. This double buffering of the received data (SSPBUF) allows the next byte to start reception before reading the data that was just received. Any write to the SSPBUF register during transmission/reception of data will be ignored, and the write collision detect bit, WCOL (SSPCON<7>), will be set. User software must clear the WCOL bit so that it can be determined if the following write(s) to the SSPBUF register completed successfully. When the application software is expecting to receive valid data, the SSPBUF should be read before the next byte of data to transfer is written to the SSPBUF. Buffer full bit, BF (SSPSTAT<0>), indicates when SSPBUF has been loaded with the received data (transmission is complete). When the SSPBUF is read, the BF bit is cleared. This data may be irrelevant if the SPI is only a transmitter. Generally the SSP Interrupt is used to determine when the transmission/reception has completed. The SSPBUF must be read and/or written. If the interrupt method is not going to be used, then software polling can be done to ensure that a write collision does not occur. Example 19-1 shows the loading of the SSPBUF (SSPSR) for data transmission. The shaded instruction is only required if the received data is meaningful (some SPI applications are transmit only).

Example 19-1: Loading the SSPBUF (SSPSR) Register

```assembly
LOOP BTFSS SSPSTAT, BF ; Has data been received
    ; (transmit complete)?
    GOTO LOOP ; No
    MOVF SSPBUF, W ; W reg = contents of SSPBUF
    MOVWF RXDATA ; Save in user RAM, ; if data is meaningful
    MOVFF TXDATA, SSPBUF ; contents of TXDATA ; is the new data to transmit
```

The SSPSR is not directly readable or writable, and can only be accessed from addressing the SSPBUF register. Additionally, the SSP status register (SSPSTAT) indicates the various status conditions.
19.3.2 Enabling SPI I/O

To enable the serial port, the SSP Enable bit, SSPEN (SSPCON<5>), must be set. To reset or reconfigure SPI mode, clear the SSPEN bit which re-initializes the SSPCON register, and then set the SSPEN bit. This configures the SDI, SDO, SCK, and SS pins as serial port pins. For the pins to behave as the serial port function, they must have their data direction bits (in the TRIS register) appropriately programmed. That is:

- SDI must have the TRIS bit set
- SDO must have the TRIS bit cleared
- SCK (Master mode) must have the TRIS bit cleared
- SCK (Slave mode) must have the TRIS bit set
- SS must have the TRIS bit set

Any serial port function that is not desired may be overridden by programming the corresponding data direction (TRIS) register to the opposite value. An example would be in master mode where you are only sending data (to a display driver), then both SDI and SS could be used as general purpose outputs by clearing their corresponding TRIS register bits.
Section 19. SSP

19.3.3 Typical Connection

Figure 19-2 shows a typical connection between two microcontrollers. The master controller (Processor 1) initiates the data transfer by sending the SCK signal. Data is shifted out of both shift registers on their programmed clock edge, and latched on the edge of the clock specified by the SMP bit. Both processors should be programmed to same Clock Polarity (CKP), so both controllers would send and receive data at the same time. Whether the data is meaningful (or dummy data) depends on the application software. This leads to three scenarios for data transmission:

- Master sends data — Slave sends dummy data
- Master sends data — Slave sends data
- Master sends dummy data — Slave sends data

Figure 19-2: SPI Master/Slave Connection
19.3.4 Master Operation

The master can initiate the data transfer at any time because it controls the SCK. The master determines when the slave (Processor 2) is to broadcast data by the software protocol.

In master mode the data is transmitted/received as soon as the SSPBUF register is written to. If the SPI is only going to receive, the SDO output could be disabled (programmed as an input). The SSPSR register will continue to shift in the signal present on the SDI pin at the programmed clock rate. As each byte is received, it will be loaded into the SSPBUF register as if a normal received byte (interrupts and status bits appropriately set). This could be useful in receiver applications as a "line activity monitor" mode.

The clock polarity is selected by appropriately programming bit CKP (SSPCON<4>). This would give waveforms for SPI communication as shown in Figure 19-3, Figure 19-4, and Figure 19-5 where the MSb is transmitted first. In master mode, the SPI clock rate (bit rate) is user programmable to be one of the following:

- Fosc/4 (or Tcy)
- Fosc/16 (or 4 • Tcy)
- Fosc/64 (or 16 • Tcy)
- Timer2 output/2

This allows a maximum data rate of 5 Mbps (at 20 MHz).

Figure 19-3: SPI Mode Waveform, Master Mode
19.3.5 Slave Operation

In slave mode, the data is transmitted and received as the external clock pulses appear on SCK. When the last bit is latched, the interrupt flag bit SSPIF is set.

The clock polarity is selected by appropriately programming bit CKP (SSPCON). This then would give waveforms for SPI communication as shown in Figure 19-3, Figure 19-4, and Figure 19-5 where the MSb is transmitted first. When in slave mode the external clock must meet the minimum high and low times.

In sleep mode, the slave can transmit and receive data. When a byte is received, the device will wake-up from sleep, if the interrupt is enabled.

Figure 19-4: SPI Mode Waveform (Slave Mode With CKE = 0)
19.3.6 Slave Select Mode

When in slave select mode, the SS pin allows multi-drop for multiple slaves with a single master. The SPI must be in slave mode (SSPCON<3:0> = 04h) and the TRIS bit, for the SS pin, must be set for the slave select mode to be enabled. When the SS pin is low, transmission and reception are enabled and the SDO pin is driven. When the SS pin goes high, the SDO pin is no longer driven, even if in the middle of a transmitted byte, and becomes a floating output. External pull-up/ pull-down resistors may be desirable, depending on the application.

When the SPI is in Slave Mode with SS pin control enabled, (SSPCON<3:0> = 0100) the SPI module will reset if the SS pin is set to VDD. If the SPI is used in Slave Mode with the CKE bit is set, then the SS pin control must be enabled.

When the SPI module resets, the bit counter is forced to 0. This can be done by either by forcing the SS pin to a high level or clearing the SSPEN bit (Figure 19-6).

To emulate two-wire communication, the SDO pin can be connected to the SDI pin. When the SPI needs to operate as a receiver the SDO pin can be configured as an input. This disables transmissions from the SDO. The SDI can always be left as an input (SDI function) since it cannot create a bus conflict.
Section 19. SSP

Figure 19-6: Slave Synchronization Waveform

- SCK (CKP = 0, CKE = 0)
- SCK (CKP = 1, CKE = 0)
- Write to SSPBUF
- SDO
- SDI (SMP = 0)
- Input Sample (SMP = 0)
- SSPF Interrupt Flag
- SSPSR to SSPBUF

Notes:
- SCK: Clock signal
- SDI: Data input
- SDO: Data output
- SSPF: SSP Flag
- SSPSR: SSP Status Register
- SSPBUF: SSP Buffer Register
19.3.7 Sleep Operation

In master mode all module clocks are halted, and the transmission/reception will remain in that state until the device wakes from sleep. After the device returns to normal mode, the module will continue to transmit/receive data.

In slave mode, the SPI transmit/receive shift register operates asynchronously to the device. This allows the device to be placed in sleep mode, and data to be shifted into the SPI transmit/receive shift register. When all 8-bits have been received, the SSP interrupt flag bit will be set and if enabled will wake the device from sleep.
### Section 19. SSP

#### 19.3.8 Effects of a Reset

A reset disables the SSP module and terminates the current transfer.

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE/GIEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x 0000 000u</td>
</tr>
<tr>
<td>PIR</td>
<td></td>
<td>SPIF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>IPR</td>
<td></td>
<td>SPIF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>SSPBUF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>xxxx xxxx uuuu uuuu</td>
</tr>
<tr>
<td>SSPCON</td>
<td>WCOL</td>
<td>SSPV</td>
<td>SSPEN</td>
<td>CKP</td>
<td>SSPM3</td>
<td>SSPM2</td>
<td>SSPM1</td>
<td>SSPM0</td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
<tr>
<td>TRISA</td>
<td>0</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>--11 1111 --11 1111</td>
<td></td>
</tr>
<tr>
<td>TRISC</td>
<td>1111 1111 1111 1111</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>1111 1111 1111 1111</td>
<td></td>
</tr>
<tr>
<td>SSPSTAT</td>
<td>SMP</td>
<td>CKE</td>
<td>D/A</td>
<td>P</td>
<td>S</td>
<td>R/W</td>
<td>UA</td>
<td>BF</td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
</tbody>
</table>

**Legend:**
- x = unknown,
- u = unchanged,
- - = unimplemented read as '0'. Shaded cells are not used by the SSP in SPI mode.

**Note 1:** The position of this bit is device dependent.
19.4 SSP I\(^2\)C Operation

The SSP module in I\(^2\)C mode fully implements all slave functions, except general call support, and provides interrupts on start and stop bits in hardware to facilitate software implementations of the master functions. The SSP module implements the standard mode specifications as well as 7-bit and 10-bit addressing. The "Appendix" section gives an overview of the I\(^2\)C bus specification.

Two pins are used for data transfer. These are the SCL pin, which is the clock, and the SDA pin, which is the data. The user must configure these pins as inputs through the TRIS bits. The SSP module functions are enabled by setting SSP Enable bit, SSPEN (SSPCON).

A "glitch" filter is on the SCL and SDA pins when the pin is an input. This filter operates in both the 100 KHz and 400 KHz modes. In the 100 KHz mode, when these pins are an output, there is a slew rate control of the pin that is independent of device frequency.

Figure 19-7: SSP Block Diagram (I\(^2\)C Mode)
Section 19. SSP

The SSP module has five registers for I^2C operation. They are:
- SSP Control Register (SSPCON)
- SSP Status Register (SSPSTAT)
- Serial Receive/Transmit Buffer (SSPBUF)
- SSP Shift Register (SSPSR) - Not directly accessible
- SSP Address Register (SSPADD)

The SSPCON register allows control of the I^2C operation. Four mode selection bits (SSPCON<3:0>) allow one of the following I^2C modes to be selected:
- I^2C Slave mode (7-bit address)
- I^2C Slave mode (10-bit address)
- I^2C Firmware controlled Multi-Master mode (start and stop bit interrupts enabled)
- I^2C Firmware controlled Master mode, slave is idle

Before selecting any I^2C mode, the SCL and SDA pins must be programmed to inputs by setting the appropriate TRIS bits. Selecting an I^2C mode by setting the SSPEN bit enables the SCL and SDA pins to be used as the clock and data lines in I^2C mode.

The SSPSTAT register gives the status of the data transfer. This information includes detection of a START or STOP bit, specifies if the received byte was data or address, if the next byte is the completion of 10-bit address, and if this will be a read or write data transfer.

The SSPBUF is the register to which transfer data is written to or read from. The SSPSR register shifts the data in or out of the device. In receive operations, the SSPBUF and SSPSR create a doubled buffered receiver. This allows reception of the next byte to begin before reading the last byte of received data. When the complete byte is received, it is transferred to the SSPBUF register and flag bit SSPIF is set. If another complete byte is received before the SSPBUF register is read, a receiver overflow has occurred and the SSPOV bit (SSPCON<6>) is set and the byte in the SSPSR is lost.

The SSPADD register holds the slave address. In 10-bit mode, the user needs to write the high byte of the address (1111 0 A9 A8 0). Following the high byte address match, the low byte of the address needs to be loaded (A7:A0).
19.4.1 Slave Mode

In slave mode, the SCL and SDA pins must be configured as inputs (TRIS set). The SSP module will override the input state with the output data when required (slave-transmitter).

When an address is matched or the data transfer after an address match is received, the hardware automatically will generate the acknowledge (ACK) pulse, and then load the SSPBUF register with the received value currently in the SSPSR register.

There are certain conditions that will cause the SSP module not to give this ACK pulse. These are if either (or both):

a) The buffer full bit, BF (SSPSTAT<0>), was set before the message completed.
b) The overflow bit, SSPOV (SSPCON<6>), was set before the message completed.

In this case, the SSPSR register value is not loaded into the SSPBUF, but the SSPIF and SSPOV bits are set. Table 19-2 shows what happens when a data transfer byte is received, given the status of bits BF and SSPOV. The shaded cells show the condition where user software did not properly clear the overflow condition. The BF flag bit is cleared by reading the SSPBUF register while bit SSPOV is cleared through software.

The SCL clock input must have a minimum high and low time for proper operation. The high and low times of the I²C specification as well as the requirement of the SSP module is shown in the Device Data Sheet electrical specifications parameters 100 and 101.
Section 19. SSP

19.4.1.1 Addressing

Once the SSP module has been enabled, it waits for a START condition to occur. Following the START condition, the 8-bits are shifted into the SSPSR register. All incoming bits are sampled with the rising edge of the clock (SCL) line. The value of register SSPSR<7:1> is compared to the value of the SSPADD register. The address is compared on the falling edge of the eighth clock (SCL) pulse. If the addresses match, and the BF and SSPOV bits are clear, the following events occur:

a) The SSPSR register value is loaded into the SSPBUF register on the falling edge of the eighth SCL pulse.
b) The buffer full bit, BF, is set on the falling edge of the eighth SCL pulse.
c) An ACK pulse is generated.
d) The SSP interrupt flag bit, SSPIF, is set (and an interrupt is generated if enabled) - on the falling edge of the ninth SCL pulse.

In 10-bit address mode, two address bytes need to be received by the slave. The five Most Significant bits (MSbs) of the first address byte specify if this is a 10-bit address. The R/W bit (SSPSTAT) must specify a write so the slave device will receive the second address byte. For a 10-bit address the first byte would equal '1111 0 A9 A8 C', where A9 and A8 are the two MSbs of the address. The sequence of events for a 10-bit address is as follows, with steps 7-9 for slave-transmitter:

1. Receive first (high) byte of Address (the SSPIF, BF, and UA (SSPSTAT) bits are set).
2. Update the SSPADD register with second (low) byte of Address (clears the UA bit and releases the SCL line).
3. Read the SSPBUF register (clears the BF bit) and clear the SSPIF flag bit.
4. Receive second (low) byte of Address (the SSPIF, BF, and UA bits are set).
5. Update the SSPADD register with the high byte of Address. This will clear the UA bit and releases SCL line.
6. Read the SSPBUF register (clears the BF bit) and clear the SSPIF flag bit.
7. Receive repeated START condition.
8. Receive first (high) byte of Address (the SSPIF and BF bits are set).
9. Read the SSPBUF register (clears the BF bit) and clear the SSPIF flag bit.

Table 19-2: Data Transfer Received Byte Actions

<table>
<thead>
<tr>
<th>Status Bits as Data Transfer is Received</th>
<th>SSPSR → SSPBUF</th>
<th>Generate ACK Pulse</th>
<th>Set bit SSPIF (SSP Interrupt occurs if enabled)</th>
</tr>
</thead>
<tbody>
<tr>
<td>BF</td>
<td>SSPOV</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>No</td>
<td>No</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>No</td>
<td>No</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>Yes</td>
<td>No</td>
</tr>
</tbody>
</table>

Note: Following the RESTART condition (step 7) in 10-bit mode, the user only needs to match the first 7-bit address. The user does not update the SSPADD for the second half of the address.

Note: Shaded cells show the conditions where the user software did not properly clear the overflow condition.
19.4.1.2 Reception

When the R/W bit of the address byte is clear and an address match occurs, the R/W bit of the SSPSTAT register is cleared. The received address is loaded into the SSPBUF register.

When the address byte overflow condition exists, then no acknowledge (ACK) pulse is given. An overflow condition is defined as either the BF bit (SSPSTAT) is set or the SSPOV bit (SSPCON) is set. When a byte is received with these conditions, and attempts to move from the SSPSR register to the SSPBUF register, no acknowledge pulse is given.

An SSP interrupt is generated for each data transfer byte. The SSPIF flag bit must be cleared in software. The SSPSTAT register is used to determine the status of the receive byte.

Figure 19-8: I²C Waveforms for Reception (7-bit Address)
Figure 19-9: I²C Waveforms for Reception (10-bit Address)

- **SDA**: Serial Data Line
- **SCL**: Serial Clock Line
- **SSPIF**: SSP Interrupt Flag
- **BF**: SSP receive buffer flag
- **S**: Start bit
- **R**: Read bit
- **W**: Write bit
- **ACK**: Acknowledge bit
- **D** (D0-D7): Receive Data Byte
- **UA**: SSP interrupt after receiving the address

Event Description:
- **Clear of software**: SSPDBF is written with contents of SSPSR.
- **Receive First Byte of Address**: SSPBUF is written with contents of SSPSR.
- **Receive Second Byte of Address**: SSPBUF is written with contents of SSPSR.
- **SSPBUR**: SSP interrupt after receiving the address
- **Clear in software**: SSPDBF is written with contents of SSPSR.
- **Clock is held low until update of SSPADD has taken place**: SSPADD is updated.
- **Clear in software**: SSPDBF is written with contents of SSPSR.
- **Read of SSPBUF after UA**: SSP interrupt after receiving the address

- **UA**: SSP interrupt after receiving the address
- **SSPADD**: SSP interrupt after receiving the address

- **Clear in software**: SSPDBF is written with contents of SSPSR.
- **Dummy read of SSPBUF to clear BF flag**: SSPBUF is written with contents of SSPSR.
- **Receive Data Byte**: SSPBUF is written with contents of SSPSR.
- **Receive First Byte of Address**: SSPBUF is written with contents of SSPSR.
- **Receive Second Byte of Address**: SSPBUF is written with contents of SSPSR.

Note: The diagram illustrates the timing and sequence of events during I²C reception for a 10-bit address.
19.4.1.3 Transmission

When the R/W bit of the incoming address byte is set and an address match occurs, the R/W bit of the SSPSTAT register is set. The received address is loaded into the SSPBUF register. The ACK pulse will be sent on the ninth bit, and the SCL pin is held low. The transmit data must be loaded into the SSPBUF register, which also loads the SSPSR register. Then the SCL pin should be enabled by setting the CKP bit (SSPCON<4>). The master must monitor the SCL pin prior to asserting another clock pulse. The slave devices may be holding off the master by stretching the clock. The eight data bits are shifted out on the falling edge of the SCL input. This ensures that the SDA signal is valid during the SCL high time (Figure 19-10).

An SSP interrupt is generated for each data transfer byte. The SSPIF flag bit must be cleared in software, and the SSPSTAT register is used to determine the status of the byte transfer. The SSPIF flag bit is set on the falling edge of the ninth clock pulse.

As a slave-transmitter, the ACK pulse from the master-receiver is latched on the rising edge of the ninth SCL input pulse. If the SDA line was high (not ACK), then the data transfer is complete. When the not ACK is latched by the slave, the slave logic is reset and the slave then monitors for another occurrence of the START bit. If the SDA line was low (ACK), the transmit data must be loaded into the SSPBUF register, which also loads the SSPSR register. Then the SCL pin should be enabled by setting the CKP bit.

Figure 19-10: I2C Waveforms for Transmission (7-bit Address)
Section 19. SSP

Figure 19-11: I²C Waveforms for Transmission (10-bit Address)

- Master sends NACK
- UA (SSPSTAT<1>) is set indicating that SSPADD needs to be updated.
- SSPBUF is written with contents of SSPSR.
- SSPADD is updated with new contents.
- UA is cleared in software when SSPADD is updated.
- Clock is held low until update of SSPADD has taken place.
- UA is cleared in software when SSPADD is updated.
- Bus Master terminates the transfer.
19.4.1.4 Clock Arbitration

Clock arbitration has the SCL pin to inhibit the master device from sending the next clock pulse. The SSP module in I2C slave mode will hold the SCL pin low when the CPU needs to respond to the SSP interrupt (SSPIF bit is set and the CKP bit is cleared). The data that needs to be transmitted will need to be written to the SSPBUF register, and then the CKP bit will need to be set to allow the master to generate the required clocks.

19.4.2 Master Mode (Firmware)

Master mode of operation is supported by interrupt generation on the detection of the START and STOP conditions. The STOP (P) and START (S) bits are cleared from a reset or when the SSP module is disabled. Control of the I2C bus may be taken when the P bit is set, or the bus is idle with both the S and P bits clear.

In master mode the SCL and SDA lines are manipulated by clearing the corresponding TRIS bit(s). The output level is always low, irrespective of the value(s) in the PORT register. So when transmitting data, a ‘1’ data bit must have it’s TRIS bit set (input) and a ‘0’ data bit must have it’s TRIS bit cleared (output). The same scenario is true for the SCL line with the TRIS bit.

The following events will cause SSP Interrupt Flag bit, SSPIF, to be set (SSP Interrupt if enabled):
- START condition
- STOP condition
- Data transfer byte transmitted/received

Master mode of operation can be done with either the slave mode idle (SSPM3:SSPM0 = 1011) or with the slave active (SSPM3:SSP0 = 1110 or 1111). When the slave modes are enabled, the software needs to differentiate the source(s) of the interrupt.

19.4.3 Multi-Master Mode (Firmware)

In multi-Master mode, the interrupt generation on the detection of the START and STOP conditions allows the determination of when the bus is free. The STOP (P) and START (S) bits are cleared from a reset or when the SSP module is disabled. Control of the I2C bus may be taken when the P bit (SSPSTAT<4>) is set, or the bus is idle with both the S and P bits clear. When the bus is busy, enabling the SSP Interrupt will generate the interrupt when the STOP condition occurs.

In Multi-Master operation, the SDA line must be monitored to see if the signal level is the expected output level. This check only needs to be done when a high level is output. If a high level is expected and a low level is present, the device needs to release the SDA and SCL lines (set the TRIS bits). There are two stages where this arbitration can be lost, they are:
- Address transfer
- Data transfer

When the slave logic is enabled, the slave continues to receive. If arbitration was lost during the address transfer stage, communication to the device may be in progress. If addressed an ACK pulse will be generated. If arbitration was lost during the data transfer stage, the device will need to retransfer the data at a later time.
Section 19. SSP

19.4.4 Sleep Operation

While in sleep mode, the I²C module can receive addresses or data, and when an address match or complete byte transfer occurs wake the processor from sleep (if the SSP interrupt is enabled).

19.4.5 Effect of a Reset

A reset disables the SSP module and terminates the current transfer.

Table 19-3: Registers Associated with I²C Operation

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE</td>
<td>PEIE</td>
<td>TMR0IE</td>
<td>INT0IE</td>
<td>RBIE</td>
<td>TMR0IF</td>
<td>INT0IF</td>
<td>RBIF</td>
<td>0000 000x</td>
<td>0000 000u</td>
</tr>
<tr>
<td>PIR</td>
<td>SSIIF</td>
<td>0</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>IPR</td>
<td>SSSIP</td>
<td>0</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SSPBUF</td>
<td>Synchronous Serial Port Receive Buffer/Transmit Register</td>
<td>xxxx xxxx</td>
<td>uuuu</td>
<td>uuuu</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SSPADD</td>
<td>Synchronous Serial Port (I²C mode) Address Register</td>
<td>0000 0000</td>
<td>0000</td>
<td>0000</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SSPSTAT</td>
<td>SMP</td>
<td>CKE</td>
<td>D/A</td>
<td>P</td>
<td>S</td>
<td>R/W</td>
<td>UA</td>
<td>BF</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
</tbody>
</table>

Legend:  
- x = unknown, u = unchanged, - = unimplemented read as '0'.  
- Shaded cells are not used by SSP in I²C mode.  

Note 1: The positions of these bits are device dependent.
19.5 Initialization

Example 19-2: SPI Master Mode Initialization

```
CLRF SSPSTAT ; SMP = 0, CKE = 0, and clear status bits
BSF SSPSTAT, CKE ; CKE = 1
MOVLW 0x31 ; Set up SPI port, Master mode, CLK/16,
MOVWF SSPCON ; Data xmit on falling edge (CKE=1 & CKP=1)
; Data sampled in middle (SMP=0 & Master mode)
BSF PIE, SSPIE ; Enable SSP interrupt
BSF INTCON, GIE ; Enable, enabled interrupts
MOVLW DataByte ; Data to be Transmitted
; Could move data from RAM location
MOVWF SSPBUF ; Start Transmission
```
19.5.1 SSP Module / Basic SSP Module Compatibility

When upgrading from the Mid-Range family’s basic SSP module, the SSPSTAT register contains two additional control bits. These bits are used only in SPI mode and are:

- SMP, SPI data input sample phase
- CKE, SPI Clock Edge Select

To be compatible with the SPI of the basic SSP module, these bits must be appropriately configured. If these bits are not at the states shown in Table 19-4, improper SPI communication may occur.

Table 19-4: New Bit States for Compatibility

<table>
<thead>
<tr>
<th>Mid-Range Family’s Basic SSP Module</th>
<th>SSP Module</th>
</tr>
</thead>
<tbody>
<tr>
<td>CKP</td>
<td>CKP</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>
19.6 Design Tips

Question 1: Using SPI mode, I do not seem able to talk to an SPI device.

Answer 1:
Ensure that you are using the correct SPI mode for that device. This SPI supports all four SPI modes so you should be able to get it to function. Check the clock polarity and the clock phase. These settings should match what the SPI is interfacing to.

Question 2: Using I2C mode, I do not seem able to make the master mode work.

Answer 2:
This SSP module does not have master mode fully automated in hardware, see Application Note AN578 for software which uses the SSP module to implement master mode. If you require a fully automated hardware implementation of I2C Master Mode, please refer to the Microchip Line Card for devices that have the Master SSP module.

Question 3: Using I2C mode, I write data to the SSPBUF register, but the data did not transmit.

Answer 3:
Ensure that you set the CKP bit to release the I2C clock.
Section 19. SSP

19.7 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is they may be written for the Base-Line, Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the SSP Module are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Use of the SSP Module in the I²C Multi-Master Environment.</td>
<td>AN578</td>
</tr>
<tr>
<td>Using Microchip 93 Series Serial EEPROMs with Microcontroller SPI Ports</td>
<td>AN613</td>
</tr>
<tr>
<td>Software implementation of I²C Bus Master</td>
<td>AN554</td>
</tr>
<tr>
<td>Use of the SSP module in the Multi-master Environment</td>
<td>AN578</td>
</tr>
<tr>
<td>Interfacing PIC16C64/74 to Microchip SPI Serial EEPROM</td>
<td>AN647</td>
</tr>
<tr>
<td>Interfacing a Microchip PIC16C92x to Microchip SPI Serial EEPROM</td>
<td>AN668</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
19.8 Revision History

Revision A
This is the initial released revision of the SSP module description.
Section 20. Master SSP

HIGHLIGHTS

This section of the manual contains the following major topics:

- 20.1 Introduction .......................................................... 20-2
- 20.2 Control Registers .................................................. 20-4
- 20.3 SPI Mode ............................................................... 20-9
- 20.4 MSSP I2C Operation .............................................. 20-17
- 20.5 Design Tips ......................................................... 20-55
- 20.6 Related Application Notes ................................. 20-56
- 20.7 Revision History .................................................. 20-57
20.1 Introduction

The Master Synchronous Serial Port (MSSP) module is a serial interface useful for communicating with other peripheral or microcontroller devices. These peripheral devices may be serial EEPROMs, Shift Registers, display drivers, A/D converters, etc. The MSSP module can operate in one of two modes:

- Serial Peripheral Interface (SPI)
- Inter-Integrated Circuit (I^2C)
  - Full Master Mode
  - Slave Mode (with general address call)

The I^2C interface supports the following modes in hardware:

- Master Mode
- Multi-Master Mode
- Slave Mode

Figure 20-1 shows a block diagram for the SPI Mode, while Figure 20-2 and Figure 20-3 show the block diagrams for the two different I^2C Modes of operation.
Section 20. Master SSP

Figure 20-2: I^2C Slave Mode Block Diagram

Figure 20-3: I^2C Master Mode Block Diagram
20.2 Control Registers

The Master SSP (MSSP) module has three registers that control the operation and indicate the status of the module. These are the SSPSTAT register (Register 20-1), the SSPCON1 register (Register 20-2), and the SSPCON2 register (Register 20-3).

Register 20-1: SSPSTAT: MSSP Status Register

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>SMP</td>
<td>CKE</td>
<td>D/A</td>
<td>P</td>
<td>S</td>
<td>R/W</td>
<td>UA</td>
<td>BF</td>
</tr>
</tbody>
</table>

bit 7  **SMP**: Sample bit
SPI Master Mode
1 = Input data sampled at end of data output time
0 = Input data sampled at middle of data output time
SPI Slave Mode
SMP must be cleared when SPI is used in Slave Mode
In I^2^C Master or Slave Mode:
1 = Slew rate control disabled for standard speed mode (100 kHz and 1 MHz)
0 = Slew rate control enabled for high speed mode (400 kHz)

bit 6  **CKE**: SPI Clock Edge Select
CKP = 0
1 = Data transmitted on rising edge of SCK
0 = Data transmitted on falling edge of SCK
CKP = 1
1 = Data transmitted on falling edge of SCK
0 = Data transmitted on rising edge of SCK

bit 5  **D/A**: Data/Address bit (I^2^C Mode only)
1 = Indicates that the last byte received or transmitted was data
0 = Indicates that the last byte received or transmitted was address

bit 4  **P**: Stop bit
(I^2^C Mode only. This bit is cleared when the MSSP module is disabled, SSPEN is cleared)
1 = Indicates that a Stop bit has been detected last (this bit is '0' on RESET)
0 = Stop bit was not detected last

bit 3  **S**: Start bit
(I^2^C Mode only. This bit is cleared when the MSSP module is disabled, SSPEN is cleared)
1 = Indicates that a Start bit has been detected last (this bit is '0' on RESET)
0 = Start bit was not detected last

bit 2  **R/W**: Read/Write bit information (I^2^C Mode only)
This bit holds the R/W bit information following the last address match. This bit is only valid from the address match to the next Start bit, Stop bit, or not ACK bit.
In I^2^C Slave Mode:
1 = Read
0 = Write
In I^2^C Master Mode:
1 = Transmit is in progress
0 = Transmit is not in progress.
Or'ing this bit with SEN, RSEN, PEN, RCEN, or ACKEN will indicate if the MSSP is in idle Mode.

bit 1  **UA**: Update Address (10-bit I^2^C mode only)
1 = Indicates that the user needs to update the address in the SSPADD Register
0 = Address does not need to be updated
Section 20. Master SSP

bit 0  BF: Buffer Full Status bit

Receive (SPI and I2C Modes)
  1 = Receive complete, SSPBUF is full
  0 = Receive not complete, SSPBUF is empty

Transmit (I2C Mode only)
  1 = Data transmit in progress (does not include the ACK and Stop bits), SSPBUF is full
  0 = Data transmit complete (does not include the ACK and Stop bits), SSPBUF is empty

Legend
R = Readable bit  W = Writable bit  U = Unimplemented bit, read as ‘0’
- n = Value at POR reset  ‘1’ = bit is set  ‘0’ = bit is cleared  x = bit is unknown
### Register 20-2: SSPCON1: MSSP Control Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Name</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>WCOL</td>
<td>Write Collision Detect bit</td>
</tr>
<tr>
<td></td>
<td>Master Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = A write to the SSPBUF Register was attempted while the I²C conditions were not valid for a transmission to be started</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = No collision</td>
<td></td>
</tr>
<tr>
<td></td>
<td>Slave Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = The SSPBUF Register is written while it is still transmitting the previous word (must be cleared in software)</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = No collision</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td>SSPOV</td>
<td>Receive Overflow Indicator bit</td>
</tr>
<tr>
<td></td>
<td>In SPI Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = A new byte is received while the SSPBUF Register is still holding the previous data. In case of overflow, the data in SSPSR is lost. Overflow can only occur in Slave Mode. In Slave Mode, the user must read the SSPBUF, even if only transmitting data, to avoid setting overflow. In Master Mode the overflow bit is not set since each new reception (and transmission) is initiated by writing to the SSPBUF Register. (Must be cleared in software)</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = No overflow</td>
<td></td>
</tr>
<tr>
<td></td>
<td>In I²C Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = A byte is received while the SSPBUF Register is still holding the previous byte. SSPOV is a &quot;don’t care&quot; in transmit mode. (Must be cleared in software)</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = No overflow</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td>SSPEN</td>
<td>Synchronous Serial Port Enable bit</td>
</tr>
<tr>
<td></td>
<td>In both modes, when enabled, the I/O pins must be properly configured as input or output.</td>
<td></td>
</tr>
<tr>
<td></td>
<td>In SPI Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = Enables serial port and configures SCK, SDO, SDI, and SS as the source of the serial port pins</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = Disables serial port and configures these pins as I/O port pins</td>
<td></td>
</tr>
<tr>
<td></td>
<td>In I²C Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = Enables the serial port and configures the SDA and SCL pins as the source of the serial port pins</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = Disables serial port and configures these pins as I/O port pins</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td>CKP</td>
<td>Clock Polarity Select bit</td>
</tr>
<tr>
<td></td>
<td>In SPI Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = Idle state for clock is a high level</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = Idle state for clock is a low level</td>
<td></td>
</tr>
<tr>
<td></td>
<td>In I²C Slave Mode:</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SCK release control</td>
<td></td>
</tr>
<tr>
<td></td>
<td>1 = Enable clock</td>
<td></td>
</tr>
<tr>
<td></td>
<td>0 = Holds clock low (clock stretch) (Used to ensure data setup time)</td>
<td></td>
</tr>
<tr>
<td></td>
<td>In I²C Master Mode</td>
<td></td>
</tr>
<tr>
<td></td>
<td>Unused in this mode</td>
<td></td>
</tr>
</tbody>
</table>
## Section 20. Master SSP

### bit 3 - 0  **SSPM3:SSPM0**: Synchronous Serial Port Mode Select bits

<table>
<thead>
<tr>
<th>Value</th>
<th>Mode Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0000</td>
<td>SPI Master Mode, clock = Fosc/4</td>
</tr>
<tr>
<td>0001</td>
<td>SPI Master Mode, clock = Fosc/16</td>
</tr>
<tr>
<td>0010</td>
<td>SPI Master Mode, clock = Fosc/64</td>
</tr>
<tr>
<td>0011</td>
<td>SPI Master Mode, clock = TMR2 output/2</td>
</tr>
<tr>
<td>0100</td>
<td>SPI Slave Mode, clock = SCK pin. SS pin control enabled.</td>
</tr>
<tr>
<td>0101</td>
<td>SPI Slave Mode, clock = SCK pin. SS pin control disabled. SS can be used as I/O pin</td>
</tr>
<tr>
<td>0110</td>
<td>I²C Slave Mode, 7-bit address</td>
</tr>
<tr>
<td>0111</td>
<td>I²C Slave Mode, 10-bit address</td>
</tr>
<tr>
<td>1000</td>
<td>I²C Master Mode, clock = Fosc / (4 * (SSPADD+1))</td>
</tr>
<tr>
<td>1001</td>
<td>Reserved</td>
</tr>
<tr>
<td>1010</td>
<td>Reserved</td>
</tr>
<tr>
<td>1011</td>
<td>I²C firmware controlled master mode (Slave idle)</td>
</tr>
<tr>
<td>1100</td>
<td>Reserved</td>
</tr>
<tr>
<td>1101</td>
<td>Reserved</td>
</tr>
<tr>
<td>1110</td>
<td>I²C Slave Mode, 7-bit address with Start and Stop bit interrupts enabled</td>
</tr>
<tr>
<td>1111</td>
<td>I²C Slave Mode, 10-bit address with Start and Stop bit interrupts enabled</td>
</tr>
</tbody>
</table>

### Legend

- **R** = Readable bit  
- **W** = Writable bit  
- **U** = Unimplemented bit, read as '0'  
- **n** = Value at POR reset  
- **'1'** = bit is set  
- **'0'** = bit is cleared  
- **x** = bit is unknown
## Register 20-3: SSPCON2: MSSP Control Register2

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>GCEN</td>
<td>ACKSTAT</td>
</tr>
<tr>
<td>ACKDT</td>
<td>ACKEN</td>
</tr>
<tr>
<td>PEN</td>
<td>RSEN</td>
</tr>
<tr>
<td>SEN</td>
<td></td>
</tr>
</tbody>
</table>

### bit 7 GCEN: General Call Enable bit (In I²C Slave Mode only)
- **1**: Enable interrupt when a general call address (0000h) is received in the SSPSR
- **0**: General call address disabled

### bit 6 ACKSTAT: Acknowledge Status bit (In I²C Master Mode only)
- **In Master Transmit Mode**: 1 = Acknowledge was not received from slave
- **0**: Acknowledge was received from slave

### bit 5 ACKDT: Acknowledge Data bit (In I²C Master Mode only)
- **In Master Receive Mode**: Value that will be transmitted when the user initiates an Acknowledge sequence at the end of a receive.
  - **1**: Not Acknowledge
  - **0**: Acknowledge

### bit 4 ACKEN: Acknowledge Sequence Enable bit (In I²C Master Mode only)
- **In Master Receive Mode**: 1 = Initiate Acknowledge sequence on SDA and SCL pins, and transmit ACKDT data bit. Automatically cleared by hardware.
  - **0**: Acknowledge sequence idle

### bit 3 RCEN: Receive Enable bit (In I²C Master Mode only)
- **1**: Enables Receive mode for I²C
  - **0**: Receive idle

### bit 2 PEN: Stop condition enable bit (In I²C Master Mode only)
- SCK release control
  - **1**: Initiate Stop condition on SDA and SCL pins. Automatically cleared by hardware.
  - **0**: Stop condition idle

### bit 1 RSEN: Repeated Start condition enabled bit (In I²C Master Mode only)
- **1**: Initiate Repeated Start condition on SDA and SCL pins. Automatically cleared by hardware.
  - **0**: Repeated Start condition idle.

### bit 0 SEN: Start condition enabled bit (In I²C Master Mode only)
- **1**: Initiate Start condition on SDA and SCL pins. Automatically cleared by hardware.
  - **0**: Start condition idle

### Note:
For the ACKEN, RCEN, PEN, RSEN, SEN bits: If the I²C module is not in the idle mode, the bit may not be set (no spooling) and the SSPBUF may not be written (writes to the SSPBUF are disabled).

### Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **n** = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown
Section 20. Master SSP

20.3 SPI Mode

The SPI mode allows 8 bits of data to be synchronously transmitted and received simultaneously. All four modes of SPI are supported. To accomplish communication, typically three pins are used:

- Serial Data Out (SDO)
- Serial Data In (SDI)
- Serial Clock (SCK)

Additionally a fourth pin may be used when in a Slave Mode of operation:

- Slave Select (SS)

20.3.1 Operation

When initializing the SPI, several options need to be specified. This is done by programming the appropriate control bits (SSPCON1<5:0>) and SSPSTAT<7:6>. These control bits allow the following to be specified:

- Master Mode (SCK is the clock output)
- Slave Mode (SCK is the clock input)
- Clock Polarity (Idle state of SCK)
- Data input sample phase (middle or end of data output time)
- Clock edge (output data on rising/falling edge of SCK)
- Clock Rate (Master Mode only)
- Slave Select Mode (Slave Mode only)

Figure 20-4 shows the block diagram of the MSSP module, when in SPI mode.

Figure 20-4: MSSP Block Diagram (SPI Mode)
The MSSP consists of a transmit/receive Shift Register (SSPSR) and a Buffer Register (SSPBUF). The SSPSR shifts the data in and out of the device, MSb first. The SSPBUF holds the data that was written to the SSPSR, until the received data is ready. Once the 8 bits of data have been received, that byte is moved to the SSPBUF Register. Then the buffer full detect bit, BF (SSPSTAT register), and the interrupt flag bit, SSPIF, are set. This double buffering of the received data (SSPBUF) allows the next byte to start reception before reading the data that was just received. Any write to the SSPBUF Register during transmission/reception of data will be ignored, and the write collision detect bit, WCOL (SSPCON1 register), will be set. User software must clear the WCOL bit so that it can be determined if the following write(s) to the SSPBUF Register completed successfully.

When the application software is expecting to receive valid data, the SSPBUF should be read before the next byte of data to transfer is written to the SSPBUF. Buffer full bit, BF (SSPSTAT register), indicates when SSPBUF has been loaded with the received data (transmission is complete). When the SSPBUF is read, the BF bit is cleared. This data may be irrelevant if the SPI is only a transmitter. Generally the MSSP Interrupt is used to determine when the transmission/reception has completed. The SSPBUF must be read and/or written. If the interrupt method is not going to be used, then software polling can be done to ensure that a write collision does not occur. Example 20-1 shows the loading of the SSPBUF (SSPSR) for data transmission.

**Example 20-1: Loading the SSPBUF (SSPSR) Register**

```
LOOP BTFSS SSPSTAT, BF ; Has data been received (transmit complete)?
   GOTO LOOP ; No
   MOVF SSPBUF, W ; WREG reg = contents of SSPBUF
   MOVWF RXDATA ; Save in user RAM, if data is meaningful
   MOVF TXDATA, W ; W reg = contents of TXDATA
   MOVWF SSPBUF ; New data to xmit
```

The SSPSR is not directly readable or writable, and can only be accessed by addressing the SSPBUF Register. Additionally, the MSSP Status Register (SSPSTAT) indicates the various status conditions.
Section 20. Master SSP

20.3.2 Enabling SPI I/O

To enable the serial port, SSP Enable bit, SSPEN, must be set. To reset or reconfigure SPI mode, clear the SSPEN bit, re-initialize the SSPCON Registers, and then set the SSPEN bit. This configures the SDI, SDO, SCK, and SS pins as serial port pins. For the pins to behave as the serial port function, some must have their data direction bits (in the TRIS Register) appropriately programmed. That is:

- SDI is automatically controlled by the SPI module
- SDO must have the TRIS bit cleared
- SCK (Master Mode) must have the TRIS bit cleared
- SCK (Slave Mode) must have the TRIS bit set
- SS must have the TRIS bit set

Any serial port function that is not desired may be overridden by programming the corresponding data direction (TRIS) Register to the opposite value.

20.3.3 Typical Connection

Figure 20-5 shows a typical connection between two microcontrollers. The master controller (Processor 1) initiates the data transfer by sending the SCK signal. Data is shifted out of both Shift Registers on their programmed clock edge, and latched on the opposite edge of the clock. Both processors should be programmed to same Clock Polarity (CKP), then both controllers would send and receive data at the same time. Whether the data is meaningful (or dummy data) depends on the application software. This leads to three scenarios for data transmission:

- Master sends data — Slave sends dummy data
- Master sends data — Slave sends data
- Master sends dummy data — Slave sends data
20.3.4 SPI Master Mode

The master can initiate the data transfer at any time because it controls the SCK. The master determines when the slave (Processor 2, Figure 20-5) is to broadcast data by the software protocol.

In Master Mode, the data is transmitted/received as soon as the SSPBUF Register is written to. If the SPI is only going to receive, the SDO output could be disabled (programmed as an input). The SSPSR Register will continue to shift in the signal present on the SDI pin at the programmed clock rate. As each byte is received, it will be loaded into the SSPBUF Register (interrupts and status bits appropriately set). This could be useful in receiver applications as a “line activity monitor” mode.

The clock polarity is selected by appropriately programming the CKP bit. This gives waveforms for SPI communication as shown in Figure 20-6, Figure 20-8, and Figure 20-9 where the Msb is transmitted first. In Master Mode, the SPI clock rate (bit rate) is user programmable to be one of the following:

- \( \text{Fosc}/4 \) (or \( \text{Tcy} \))
- \( \text{Fosc}/16 \) (or \( 4 \times \text{Tcy} \))
- \( \text{Fosc}/64 \) (or \( 16 \times \text{Tcy} \))
- Timer2 output/2

This allows a maximum data rate (at 40 MHz) of 10.00 Mbps.

Figure 20-6 shows the waveforms for Master Mode. When the CKE bit is set, the SDO data is valid before there is a clock edge on SCK. The change of the input sample is shown based on the state of the SMP bit. The time when the SSPBUF is loaded with the received data is shown.

Figure 20-6: SPI Mode Waveform (Master Mode)
Section 20. Master SSP

20.3.5 SPI Slave Mode

In Slave Mode, the data is transmitted and received as the external clock pulses appear on SCK. When the last bit is latched, the SSPIF interrupt flag bit is set.

While in Slave Mode, the external clock is supplied by the external clock source on the SCK pin. This external clock must meet the minimum high and low times as specified in the electrical specifications.

While in sleep mode, the slave can transmit/receive data. When a byte is received, the device will wake-up from sleep.

20.3.6 Slave Select Synchronization

The SS pin is a Slave Select pin, and functions similar to a chip select pin. The SPI must be in Slave Mode with SS pin control enabled (SSPCON<3:0> = 04h). The pin must be configured as an input by setting the corresponding TRIS bit. When the SS pin is low, transmission and reception are enabled and the SDO pin is driven. When the SS pin goes high, the SDO pin is no longer driven, even if in the middle of a transmitted byte, and becomes a floating output. External pull-up/pull-down resistors may be desirable, depending on the application.

If the TRIS bit is cleared, making the pin an output, and the pin outputs a high, the SPI receive logic (slave mode) will be in reset. It will remain in reset until either the pin outputs a low, or the pin’s TRIS bit is set and external circuits pull the pin low.

Note 1: When the SPI is in Slave Mode with SS pin control enabled, (SSPCON<3:0> = 0100) the SPI module will reset if the SS pin is set to VDD.

Note 2: If the SPI is used in Slave Mode with CKE set, then the SS pin control must be enabled.

When the SPI module resets, the bit counter is forced to 0. This can be done by either by forcing the SS pin to a high level or clearing the SSPEN bit.

To emulate two-wire communication, the SDO pin can be connected to the SDI pin. When the SPI needs to operate as a receiver, the SDO pin can be configured as an input. This disables transmissions from the SDO. The SDI can always be left as an input (SDI function) since it cannot create a bus conflict.
Figure 20-7: Slave Synchronization Waveform

Figure 20-8: SPI Slave Mode Waveform (CKE = 0)
Section 20. Master SSP

Figure 20-9: SPI Slave Mode Waveform (CKE = 1)

- SS required
- SCK (KP = 0)
- SCK (CKP = 1)
- Write to SSPBUF
- SDO
- SDI (SMP = 0)
- Input Sample (SMP = 0)
- SSPIF Interrupt Flag
- SSPSR to SSPBUFF

Next Q4 cycle after Q2Ø
20.3.7 Sleep Operation

In Master Mode, when the SLEEP instruction is executed, all module clocks are halted. The trans-
mission/reception that is in progress will remain in the current state until the device wakes from
sleep. After the device returns to normal mode, the module will continue to transmit/receive data.
In Slave Mode, the SPI transmit/receive Shift Register operates asynchronously to the device.
This allows the device to be placed in sleep mode, and data to be shifted into the SPI trans-
mitt/receive Shift Register. When all 8 bits have been received, the MSSP interrupt flag bit will be
set. If the SSPIF is enabled, it will wake the device from sleep.

20.3.8 Effects of a Reset

A reset disables the MSSP module and terminates the current transfer.

20.3.9 Bus Mode Compatibility

Table 20-1 shows the compatibility between the standard SPI modes and the states of the CKP
and CKE control bits.

<table>
<thead>
<tr>
<th>Table 20-1: SPI Bus Modes</th>
</tr>
</thead>
<tbody>
<tr>
<td>Standard SPI Mode Terminology</td>
</tr>
<tr>
<td>-------------------------------</td>
</tr>
<tr>
<td>0, 0</td>
</tr>
<tr>
<td>0, 1</td>
</tr>
<tr>
<td>1, 0</td>
</tr>
<tr>
<td>1, 1</td>
</tr>
</tbody>
</table>

There is also a SMP bit that controls when the data is sampled.

<table>
<thead>
<tr>
<th>Table 20-2: Registers Associated with SPI Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>Name</td>
</tr>
<tr>
<td>------</td>
</tr>
<tr>
<td>INTCON</td>
</tr>
<tr>
<td>PIR1</td>
</tr>
<tr>
<td>PIE1</td>
</tr>
<tr>
<td>IPR1</td>
</tr>
<tr>
<td>TRISC</td>
</tr>
<tr>
<td>SSPBUF</td>
</tr>
<tr>
<td>SSPCON</td>
</tr>
<tr>
<td>TRISA</td>
</tr>
<tr>
<td>SSPSTAT</td>
</tr>
</tbody>
</table>

Legend: x = unknown, u = unchanged, - = unimplemented read as '0'.

Note 1: The PSPIF, PSPIE and PSPIP bits are reserved on the PIC18C2X2 devices. Always maintain these bits clear.
20.4 MSSP I^2C Operation

The MSSP module in I^2C mode fully implements all master and slave functions (including general call support) and provides interrupts on Start and Stop bits in hardware to determine a free bus (multi-master function). The MSSP module implements the standard mode specifications as well as 7-bit and 10-bit addressing. Appendix A gives an overview of the I^2C bus specification.

A "glitch" filter is on the SCL and SDA pins when the pin is an input. This filter operates in both the 100 kHz and 400 kHz modes. In the 100 kHz mode, when these pins are an output, there is a slew rate control of the pin that is independent of device frequency.

Figure 20-10: I^2C Slave Mode Block Diagram

Figure 20-11: I^2C Master Mode Block Diagram
Two pins are used for data transfer. These are the SCL pin, which is the clock, and the SDA pin, which is the data. The SDA and SCL pins must be configured as inputs in the corresponding TRIS registers when the \( \text{I}^{2}\text{C} \) mode is enabled. The MSSP module functions are enabled by setting the MSSP Enable bit, SSPEN (SSPCON register). The MSSP module has six registers for \( \text{I}^{2}\text{C} \) operation. They are the:

- MSSP Control Register 1 (SSPCON1)
- MSSP Control Register 2 (SSPCON2)
- MSSP Status Register (SSPSTA)
- Serial Receive/Transmit Buffer (SSPBUF)
- MSSP Shift Register (SSPSR) - Not directly accessible
- MSSP Address Register (SSPADD)

The SSPCON1 Register allows control of the \( \text{I}^{2}\text{C} \) operation. Four mode selection bits (SSPCON1<3:0>) allow one of the following \( \text{I}^{2}\text{C} \) modes to be selected:

- \( \text{I}^{2}\text{C} \) Slave Mode (7-bit address)
- \( \text{I}^{2}\text{C} \) Slave Mode (10-bit address)
- \( \text{I}^{2}\text{C} \) Master Mode, clock = OSC/4 (SSPADD +1)
- \( \text{I}^{2}\text{C} \) Slave Mode (7-bit address), with Start and Stop bit interrupts enabled
- \( \text{I}^{2}\text{C} \) Slave Mode (10-bit address), with Start and Stop bit interrupts enabled
- \( \text{I}^{2}\text{C} \) Firmware controlled master operation, slave is idle

Before selecting any \( \text{I}^{2}\text{C} \) mode, the SCL and SDA pins must be programmed to inputs by setting the appropriate TRIS bits. Selecting an \( \text{I}^{2}\text{C} \) mode, by setting the SSPEN bit, enables the SCL and SDA pins to be used as the clock and data lines in \( \text{I}^{2}\text{C} \) mode.

The SSPSTAT Register gives the status of the data transfer. This information includes detection of a Start or Stop bit, specifies if the received byte was data or address, if the next byte is the completion of 10-bit address, and if this will be a read or write data transfer.

The SSPBUF is the register to which transfer data is written to or read from. The SSPSR Register shifts the data in or out of the device. In receive operations, the SSPBUF and SSPSR create a double buffered receiver. This allows reception of the next byte to begin before reading the current byte of received data. When the complete byte is received, it is transferred to the SSPBUF Register and the SSPIF bit is set. If another complete byte is received before the SSPBUF Register is read, a receiver overflow has occurred and the SSPOV bit (SSPCON1 register) is set and the byte in the SSPSR is lost.

The SSPADD Register holds the slave address. In 10-bit mode, the user needs to write the high byte of the address (1111 0 A9 A8 0). Following the high byte address match, the low byte of the address needs to be loaded (A7:A0).
Section 20. Master SSP

20.4.1 Slave Mode

In Slave Mode, the SCL and SDA pins must be configured as inputs. The MSSP module will override the input state with the output data when required (slave-transmitter).

When an address is matched or the data transfer after an address match is received, the hardware automatically generates the acknowledge (ACK) pulse, and loads the SSPBUF Register with the received value currently in the SSPSR Register.

There are certain conditions that will cause the MSSP module not to give this ACK pulse. These are if either (or both):

a) The buffer full bit, BF (SSPSTAT register), was set before the transfer was received.
b) The overflow bit, SSPOV (SSPCON1 register), was set before the transfer was received.

If the BF bit is set, the SSPSR Register value is not loaded into the SSPBUF, but the SSPIF and SSPOV bits are set. Table 20-3 shows what happens when a data transfer byte is received, given the status of the BF and SSPOV bits. The shaded cells show the condition where user software did not properly clear the overflow condition. The BF bit is cleared by reading the SSPBUF register while bit SSPOV is cleared through software.

The SCL clock input must have a minimum high and low time for proper operation. The high and low times of the I2C specification as well as the requirement of the MSSP module is shown in timing parameters 100 and 101 of the “Electrical Specifications” section.
20.4.1.1 Addressing

Once the MSSP module has been enabled, it waits for a Start condition to occur. Following the Start condition, the 8 bits are shifted into the SSPSR Register. All incoming bits are sampled with the rising edge of the clock (SCL) line. The value of register SSPSR<7:1> is compared to the value of the SSPADD Register (bits 7:1). The address is compared on the falling edge of the eighth clock (SCL) pulse. If the addresses match, and the BF and SSPOV bits are clear, the following events occur:

a) The SSPSR Register value is loaded into the SSPBUF Register on the falling edge of the eighth SCL pulse.

b) The buffer full bit, BF, is set on the falling edge of the eighth SCL pulse.

c) An ACK pulse is generated.

d) MSSP interrupt flag bit, SSPIF, is set (interrupt is generated if enabled) - on the falling edge of the ninth SCL pulse.

In 10-bit address mode, two address bytes need to be received by the slave. The five Most Significant bits (MSbs) of the first address byte specify if this is a 10-bit address. The R/W bit (SSPSTAT<2>) must specify a write so the slave device will receive the second address byte. For a 10-bit address the first byte would equal ‘11110 A9 A8 0’, where A9 and A8 are the two MSbs of the address. The sequence of events for a 10-bit address is as follows (with steps 7-9 for a slave-transmitter):

1. Receive first (high) byte of the address (the SSPIF, BF, and UA (SSPSTAT register) bits are set).
2. Update the SSPADD Register with second (low) byte of the address (clears the UA bit and releases the SCL line).
3. Read the SSPBUF Register (clears the BF bit) and clear flag bit SSPIF.
4. Receive second (low) byte of the address (the SSPIF, BF, and UA bits are set).
5. Update the SSPADD Register with the first (high) byte of the address. This will clear the UA bit and release the SCL line.
6. Read the SSPBUF Register (clears the BF bit) and clear the SSPIF flag bit.
7. Receive repeated Start condition.
8. Receive first (high) byte of the address (the SSPIF and BF bits are set).
9. Read the SSPBUF Register (clears the BF bit) and clear the SSPIF flag bit.

Note: Following the Repeated Start condition (step 7) in 10-bit mode, the user only needs to match the first 7-bit address. The user does not update the SSPADD for the second half of the address.

Table 20-3: Data Transfer Received Byte Actions

<table>
<thead>
<tr>
<th>Status Bits as Data Transfer is Received</th>
<th>SSPSR → SSPBUF</th>
<th>Generate ACK Pulse</th>
<th>Set bit SSPIF (SSP Interrupt occurs if enabled)</th>
</tr>
</thead>
<tbody>
<tr>
<td>BF</td>
<td>SSPOV</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>No</td>
<td>No</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>No</td>
<td>No</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>Yes</td>
<td>No</td>
</tr>
</tbody>
</table>

Note: Shaded cells show the conditions where the user software did not properly clear the overflow condition.
Section 20. Master SSP

20.4.1.2 Slave Reception

When the R/W bit of the address byte is clear and an address match occurs, the R/W bit of the SSPSTAT Register is cleared. The received address is loaded into the SSPBUF Register.

When the address byte overflow condition exists, then no acknowledge (ACK) pulse is given. An overflow condition is defined as either the BF bit (SSPSTAT register) is set or the SSPOV bit (SSPCON1 register) is set.

An MSSP interrupt is generated for each data transfer byte. The SSPIF flag bit must be cleared in software. The SSPSTAT Register is used to determine the status of the received byte.

Note: The SSPBUF will be loaded if the SSPOV bit is set and the BF flag bit is cleared. If a read of the SSPBUF was performed, but the user did not clear the state of the SSPOV bit before the next receive occurred. The ACK is not sent and the SSPBUF is updated.
20.4.1.3 Slave Transmission

When the R/W bit of the incoming address byte is set and an address match occurs, the R/W bit of the SSPSTAT Register is set. The received address is loaded into the SSPBUF Register. The ACK pulse will be sent on the ninth bit, and the SCL pin is held low. The transmit data must be loaded into the SSPBUF Register, which also loads the SSPSR Register and sets the BF bit. Then the SCL pin should be enabled by setting the CKP bit (SSPCON1 register). The master should monitor the SCL pin prior to asserting another clock pulse. The slave devices may be holding off the master by stretching the clock. The eight data bits are shifted out on the falling edge of the SCL input. This ensures that the SDA signal is valid during the SCL high time (Figure 20-13). When all eight bits have been shifted out, the BF bit will be cleared.

An MSSP interrupt is generated for each data transfer byte. The SSPIF flag bit must be cleared in software, and the SSPSTAT Register is used to determine the status of the byte transfer. The SSPIF flag bit is set on the falling edge of the ninth clock pulse.

As a slave-transmitter, the ACK pulse from the master-receiver is latched on the rising edge of the ninth SCL input pulse. If the SDA line was high (not ACK), then the data transfer is complete. When the not ACK is latched by the slave, the slave logic is reset and the slave then monitors for another occurrence of the Start bit. If the SDA line was low (ACK), the transmit data must be loaded into the SSPBUF Register, which also loads the SSPSR Register and sets the BF bit. Then the SCL pin should be enabled by setting the CKP bit.

---

Figure 20-12: I2C Slave Mode Waveforms for Reception (7-bit Address)

- **SDA**: Data line
- **SCL**: Clock line
- **SSPIF**: MSSP interrupt flag
- **BF**: Buffer full flag
- **SSPOV**: SSP overflow flag

---

Figure 20-13: I2C Slave Mode Waveforms for Transmission (7-bit Address)

- **SDA**: Data line
- **SCL**: Clock line
- **SSPIF**: MSSP interrupt flag
- **BF**: Buffer full flag
- **CKP**: Clock polarity bit

---
Figure 20-14: I2C Slave Mode Waveform (Transmission 10-bit Address)

- **SDA**: Serial Data Line
- **SCL**: Serial Clock Line
- **SSPIF** (PIR1<3>): Received First Byte of Address
- **BF (SSPSTA<T<0>):**: Received First Byte of Address
- **UA (SSPSTAT<1>):**: Receive Second Byte of Address

**Clock is held low until update of SSPADD has taken place**

- **Master sends NACK**: Transmits NACK
- **Receive First Byte of Address**: Master sends R/W = 1, receives address
- **Receive Second Byte of Address**: Master sends R/W = 0, receives address
- **SSPBUF is written with contents of SSPSR**: Update SSPADD
- **SSPBUF is written with contents of SSPSR**: Clear BF flag

**Cleared in software**

- **Cleared in software**
- **Cleared in software**
- **Cleared by hardware when SSPADD is updated**
- **Cleared by hardware when SSPADD is updated**

**Write of SSPBUF initiates transmit**

- **Bus Master terminates transfer**
- **Clock to be released**
- **Dummy read of SSPBUF to clear BF flag**
- **Dummy read of SSPBUF to clear BF flag**

- **Master sends NACK**: Transmits NACK
- **Transmit is complete**: CKP has to be set for Bus Master to terminate transfer
Figure 20-15: I2C Slave Mode Waveform (Reception 10-bit Address)

- **Receive First Byte of Address**
  - Clock is held low until update of SSPADD has taken place
  - R/W = 0
  - ACK
  - Receive Second Byte of Address
  - Cleared in software

- **Receive Data Byte**
  - R/W = 1
  - ACK

- **Bus Master terminates transfer**

**SDA**
- 1 1 1 1 0 A 9 A 8 A 7 A 6 A 5 A 4 A 3 A 2 A 1 A 0 D 7 D 6 D 5 D 4 D 3 D 2 D 1 D 0

- **Receive First Byte of Address**
  - SSPBUF is written with contents of SSPSR
  - UA (SSPSTAT<1>)
  - UA is set indicating that the SSPADD needs to be updated

- **Receive Second Byte of Address**
  - SSPBUF is written with contents of SSPSR
  - UA (SSPSTAT<1>)
  - UA is set indicating that SSPADD needs to be updated

- **Receive Data Byte**
  - SSPBUF is written with contents of SSPSR
  - UA (SSPSTAT<1>)
  - UA is set indicating that SSPADD needs to be updated

- **Bus Master terminates transfer**

**SCL**
- 1 2 3 4 5 6 7 8 9 1

**SSPIF**
- (PIR1<3>)
- (SSPSTAT<0>)
- SSPBUF is written with contents of SSPSR

**BF**
- SSPBUF is written with contents of SSPSR
- UA (SSPSTAT<1>)
- UA is set indicating that SSPADD needs to be updated

**R/W**
- ACK
- ACC
- Read of SSPBUF clears BF flag

**Dummy read of SSPBUF to clear BF flag**
- Cleared by hardware when SSPADD is updated with low byte of address
- UA is set indicating that SSPADD needs to be updated

**Dummy read of SSPBUF to clear BF flag**
- Cleared by hardware when SSPADD is updated with high byte of address
Section 20. Master SSP

20.4.2 General Call Address Support

The addressing procedure for the I²C bus is such that the first byte after the Start condition usually determines which device will be the slave addressed by the master. The exception is the general call address, which can address all devices. When this address is used, all devices should respond with an acknowledge.

The general call address is one of eight addresses reserved for specific purposes by the I²C protocol. It consists of all 0’s with R/W = 0.

The general call address is recognized when the General Call Enable bit (GCEN) is set. Following a Start bit detect, 8 bits are shifted into the SSPSR and the address is compared against the SSPADD, and is also compared to the general call address, fixed in hardware.

If the general call address matches, the SSPSR is transferred to the SSPBUF, the BF flag bit is set (during the eighth bit), and on the falling edge of the ninth bit (the ACK bit) the SSPIF interrupt flag bit is set.

When the interrupt is serviced, the source for the interrupt can be checked by reading the contents of the SSPBUF to determine if the address was device specific or a general call address.

In 10-bit address mode, SSPADD must be updated for the second half of the address to match and the UA bit to be set.

If the general call address is sampled when the GCEN bit is set, then the second half of the address is not necessary. The UA bit will not be set, and the slave (configured in 10-bit address mode) will begin receiving data after the acknowledge (Figure 20-16).

Figure 20-16: Slave Mode General Call Address Sequence (7 or 10-bit Address Mode)
20.4.3 Sleep Operation

While in sleep mode, the I\textsuperscript{2}C module can receive addresses or data. When an address match or complete byte transfer occurs, the processor will wake-up from sleep (if the MSSP interrupt is enabled).

20.4.4 Effect of a Reset

A reset disables the MSSP module and terminates the current transfer.

Table 20-4: Registers Associated with I\textsuperscript{2}C Operation

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>IIE/GIEH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>PEIE/GIEL</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>TMROIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>INT0IE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>RBIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>TMROIF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>INT0IF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>RBIF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIR</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPIF, BCLIF \textsuperscript{(1)}</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPIE, BCLIF \textsuperscript{(1)}</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SSPADD</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>Synchronous Serial Port (I\textsuperscript{2}C mode)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>Address Register (Slave Mode)/Baud Rate Generator (Master Mode)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SSPBUF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>Synchronous Serial Port Receive Buffer/Transmit Register</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SSPCON1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>WCOL</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPOV</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>CKP</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPM3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPM2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPM1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SSPM0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td>SSPCON2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>GREN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ACKSTAT</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ACKDT</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ACKEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>RCEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>PEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>RSEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>SEN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td>SSPSTAT</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>SMP</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>CKE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>D/A</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>P</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>S</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>R/W</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>UA</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
<tr>
<td></td>
<td>BF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>\textsuperscript{3}x</td>
<td></td>
</tr>
</tbody>
</table>

Legend: \textsuperscript{x} = unknown, \textsuperscript{u} = unchanged, \textsuperscript{=} = unimplemented read as ‘0’.

Shaded cells are not used by the MSSP in I\textsuperscript{2}C mode.

Note 1: The position of these bits is device dependent.
Section 20. Master SSP

20.4.5 Master Mode

Master Mode of operation is supported by interrupt generation on the detection of the Start and Stop conditions. The Stop (P) and Start (S) bits are cleared when a reset occurs or when the MSSP module is disabled. Control of the I^{2}C bus may be taken when the P bit is set, or the bus is idle with both the S and P bits clear.

In Master Mode, the SCL and SDA lines are manipulated by the MSSP hardware.

The following events will cause SSP Interrupt Flag bit, SSPIF, to be set (SSP Interrupt if enabled):

- Start condition
- Stop condition
- Data transfer byte transmitted/received
- Acknowledge Transmit
- Repeated Start

Figure 20-17: MSSP Block Diagram (I^{2}C Master Mode)
20.4.6 Multi-Master Mode

In Multi-Master Mode, the interrupt generation on the detection of the Start and Stop conditions allows the determination of when the bus is free. The Stop (P) and Start (S) bits are cleared from a reset or when the MSSP module is disabled. Control of the I²C bus may be taken when the P bit (SSPSTAT register) is set, or the bus is idle with both the S and P bits clear. When the bus is busy, enabling the MSSP Interrupt will generate the interrupt when the Stop condition occurs.

In multi-master operation, the SDA line must be monitored, for arbitration, to see if the signal level is the expected output level. This check is performed in hardware, with the result placed in the BCLIF bit.

The states where arbitration can be lost are:
- Address transfer
- Data transfer
- A Start condition
- A Repeated Start condition
- An Acknowledge condition

20.4.7 I²C Master Mode Support

Master Mode is enabled by setting and clearing the appropriate SSPM bits in SSPCON1 and by setting the SSPEN bit. Once Master Mode is enabled, the user has six options.

1. Assert a Start condition on SDA and SCL.
2. Assert a Repeated Start condition on SDA and SCL.
3. Write to the SSPBUF Register initiating transmission of data/address.
4. Generate a Stop condition on SDA and SCL.
5. Configure the I²C port to receive data.
6. Generate an acknowledge condition at the end of a received byte of data.

Note: The MSSP Module when configured in I²C Master Mode does not allow queueing of events. For instance: The user is not allowed to initiate a Start condition, and immediately write the SSPBUF Register to imitate transmission before the Start condition is complete. In this case the SSPBUF will not be written to, and the WCOL bit will be set, indicating that this write to the SSPBUF did not occur.
Section 20. Master SSP

20.4.7.1 I²C Master Mode Operation

The master device generates all of the serial clock pulses and the Start and Stop conditions. A transfer is ended with a Stop condition or with a Repeated Start condition. Since the Repeated Start condition is also the beginning of the next serial transfer, the I²C bus will not be released.

In master transmitter mode, serial data is output through SDA, while SCL outputs the serial clock. The first byte transmitted contains the slave address of the receiving device (7 bits), and the Read/Write (R/W) bit. In this case, the R/W bit will be logic '0'. Serial data is transmitted 8 bits at a time. After each byte is transmitted, an Acknowledge bit is received. Start and Stop conditions are output to indicate the beginning and the end of a serial transfer.

In master receive mode, the first byte transmitted contains the slave address of the transmitting device (7 bits), and the R/W bit. In this case, the R/W bit will be logic '1'. Thus, the first byte transmitted is a 7-bit slave address followed by a '1' to indicate receive bit. Serial data is received via the SDA pin, while the SCL pin outputs the serial clock. Serial data is received 8 bits at a time. After each byte is received, an Acknowledge bit is transmitted. Start and Stop conditions indicate the beginning and end of transmission.

The baud rate generator used for SPI mode operation is now used to set the SCL clock frequency for either 100 kHz, 400 kHz, or 1 MHz I²C operation. The baud rate generator reload value is contained in the lower 7 bits of the SSPADD Register. The baud rate generator will automatically begin counting on a write to the SSPBUF. Once the given operation is complete (i.e., transmission of the last data bit is followed by ACK), the internal clock will automatically stop counting and the SCL pin will remain in its last state.

A typical transmit sequence would go as follows:

a) The user generates a Start condition by setting the Start enable bit, SEN (SSPCON2 register).
b) SSPIF is set. The MSSP module will wait the required start time before any other operation takes place.
c) The user loads the SSPBUF with the address to transmit.
d) Address is shifted out the SDA pin until all 8 bits are transmitted.
e) The MSSP module shifts in the ACK bit from the slave device, and writes its value into the SSPCON2 Register (SSPCON2 register).
f) The MSSP module generates an interrupt at the end of the ninth clock cycle by setting the SSPIF bit.
g) The user loads the SSPBUF with eight bits of data.
h) DATA is shifted out the SDA pin until all 8 bits are transmitted.
i) The MSSP module shifts in the ACK bit from the slave device, and writes its value into the SSPCON2 Register (SSPCON2 register).
j) The MSSP module generates an interrupt at the end of the ninth clock cycle by setting the SSPIF bit.
k) The user generates a Stop condition by setting the Stop enable bit, PEN (SSPCON2 register).
l) Interrupt is generated once the Stop condition is complete.
20.4.8 Baud Rate Generator

In I2C Master Mode, the reload value for the BRG is located in the lower 7 bits of the SSPADD Register (Figure 20-18). When the BRG is loaded with this value, the BRG counts down to 0 and stops until another reload has taken place. The BRG count is decremented twice per instruction cycle (TCY) on the Q2 and Q4 clocks. In I2C Master Mode, the BRG is reloaded automatically. If clock arbitration is taking place for instance, the BRG will be reloaded when the SCL pin is sampled high (Figure 20-19).

Figure 20-18: Baud Rate Generator Block Diagram

Figure 20-19: Baud Rate Generator Timing With Clock Arbitration
Section 20. Master SSP

20.4.9 \( \text{I} \text{C Master Mode Start Condition Timing} \)

To initiate a Start condition, the user sets the Start condition enable bit, SEN (SSPCON2 register). If the SDA and SCL pins are sampled high, the baud rate generator is re-loaded with the contents of SSPADD<6:0>, and starts its count. If the SCL and SDA pins are both sampled high when the baud rate generator times out (TBRG), the SDA pin is driven low. The action of the SDA pin being driven low while the SCL pin is high in the Start condition, and causes the S bit (SSPSTAT register) to be set. Following this, the baud rate generator is reloaded with the contents of SSPADD<6:0> and resumes its count. When the baud rate generator times out (TBRG) the SEN bit (SSPCON2 register) will be automatically cleared by hardware, the baud rate generator is suspended leaving the SDA line held low, and the Start condition is complete.

**Note:** If at the beginning of Start condition, the SDA and SCL pins are already sampled low, or if during the Start condition the SCL pin is sampled low before the SDA pin is driven low, a bus collision occurs. The Bus Collision Interrupt Flag, BCLIF, is set, the Start condition is aborted, and the I\( \text{C} \) module is reset into its idle state.

20.4.9.1 WCOL Status Flag

If the user writes the SSPBUF when an Start sequence is in progress, then WCOL is set and the contents of the buffer are unchanged (the write doesn’t occur).

**Note:** Because queueing of events is not allowed, writing to the lower 5 bits of SSPCON2 is disabled until the Start condition is complete.

Figure 20-20: First Start Bit Timing

<table>
<thead>
<tr>
<th>Event</th>
<th>Timing</th>
</tr>
</thead>
<tbody>
<tr>
<td>Write to SEN Bit occurs here</td>
<td>SDA = 1, SCL = 1</td>
</tr>
<tr>
<td>Set S bit (SSPSTAT&lt;3&gt;)</td>
<td>TBRG</td>
</tr>
<tr>
<td>SDA = 1, SCL = 1</td>
<td>S</td>
</tr>
<tr>
<td>At completion of Start Bit, hardware clears SEN Bit and sets SSPIF Bit</td>
<td>TBRG</td>
</tr>
<tr>
<td>Write to SSPBUF occurs here</td>
<td>SBRG</td>
</tr>
<tr>
<td>1st Bit</td>
<td></td>
</tr>
<tr>
<td>2nd Bit</td>
<td></td>
</tr>
</tbody>
</table>
Figure 20-21: Start Condition Flowchart

- Bus collision detected
  - Set BCLIF Bit
  - Release SCL Bit
  - Clear SEN Bit
- SEN (SSPCON2<0> = 1)
- SDA = 1?
  - Yes: Load BRG with SSP ADD<6:0>
  - No: Set S Bit
- SCL = 1?
  - Yes: Force SDA = 0, Load BRG with SSP ADD<6:0>, Set S Bit
  - No: SDA = 0?
    - Yes: BRG Rollover?
      - Yes: Reset BRG
      - No: SCL = 0?
        - Yes: Force SDA = 0, Load BRG with SSP ADD<6:0>, Set S Bit
        - No: BRG rollover?
          - Yes: Reset BRG
          - No: SCL = 0?
            - Yes: Force SCL = 0, Start condition Done, Clear SEN bit, Set SSPIF bit
            - No: Reset BRG

Idle Mode
- SSPEN = 1
- SSPCON1<3:0> = 1000
20.4.10  I²C Master Mode Repeated Start Condition Timing

A Repeated Start condition occurs when the RSEN bit (SSPCON2 register) is programmed high and the I²C logic module is in the idle state. When the RSEN bit is set, the SCL pin is asserted low. When the SCL pin is sampled low, the baud rate generator is loaded with the contents of SSPADD<5:0>, and begins counting. The SDA pin is released (brought high) for one baud rate generator count (TBRG). When the baud rate generator times out, if SDA is sampled high, the SCL pin will be de-asserted (brought high). When the SCL pin is sampled high, the baud rate generator is re-loaded with the contents of SSPADD<6:0> and begins counting. SDA and SCL must be sampled high for one TBRG. This action is then followed by assertion of the SDA pin (SDA = 0) for one TBRG while SCL is high. Following this, the RSEN bit (SSPCON2 register) will be automatically cleared and the baud rate generator is not reloaded, leaving the SDA pin held low. As soon as a Start condition is detected on the SDA and SCL pins, the S bit (SSPSTAT register) will be set. The SSPIF bit will not be set until the baud rate generator has timed-out.

| Note 1: | If RSEN is programmed while any other event is in progress, it will not take effect. |
| Note 2: | A bus collision during the Repeated Start condition occurs if: |
|        | • SDA is sampled low when SCL goes from low to high. |
|        | • SCL goes low before SDA is asserted low. This may indicate that another master is attempting to transmit a data "1". |

Immediately following the SSPIF bit getting set, the user may write the SSPBUF with the 7-bit address in 7-bit mode, or the default first address in 10-bit mode. After the first eight bits are transmitted and an ACK is received, the user may then transmit an additional eight bits of address (10-bit mode) or eight bits of data (7-bit mode).
20.4.10.1 WCOL Status Flag

If the user writes the SSPBUF when a Repeated Start sequence is in progress, then WCOL is set and the contents of the buffer are unchanged (the write doesn’t occur).

**Note:** Because queueing of events is not allowed, writing of the lower 5 bits of SSPCON2 is disabled until the Repeated Start condition is complete.

Figure 20-22: Repeat Start Condition Waveform
Figure 20-23: Repeated Start Condition Flowchart (part 1 of 2)

Start

B

Idle Mode, SSPEN = 1, SSPCON1<3:0> = 1000

RSEN = 1

Force SCL = 0

SCL = 0?

No

Yes

Release SDA pin, Load BRG with SSPADD<6:0>

BRG rollover?

No

Yes

Release SCL pin

SCL = 1?

No

Yes

Bus Collision, Set BCLIF, Release SDA pin, Clear RSEN

SDA = 1?

No

Yes

Load BRG with SSPADD<6:0>

A

C
Figure 20-24: Repeated Start Condition Flowchart (part 2 of 2)

- Force SDA = 0
- Load BRG with SSPADD<6:0>

Yes

- BRG rollover?
  - Yes: Reset BRG
  - No: Force SDA = 0, Load BRG with SSPADD<6:0>

- SDA = 0?
  - Yes: Set S
  - No: SCL = 1?
    - Yes: Force SDA = 0, Load BRG with SSPADD<6:0>
    - No: SCL = '0'?
      - Yes: Reset BRG
      - No: BRG rollover?
        - Yes: Force SCL = 0, Repeated Start condition done, Clear RSEN, Set SSPIF
        - No: SCL = 0?
          - Yes: Reset BRG
          - No: SDA = 0?
Section 20. Master SSP

20.4.11 I²C Master Mode Transmission

Transmission of a data byte, a 7-bit address, or the other half of a 10-bit address is accomplished by simply writing a value to SSPBUF Register. This action will set the buffer full flag bit, BF, and allow the baud rate generator to begin counting and start the next transmission. Each bit of address/data will be shifted out onto the SDA pin after the falling edge of SCL is asserted (see data hold time specification parameter 106 in the “Electrical Specifications” section). SCL is held low for one baud rate generator roll over count (TBRG). Data should be valid before SCL is released high (see data setup time specification parameter 107 in the “Electrical Specifications” section). When the SCL pin is released high, it is held that way for TBRG, the data on the SDA pin must remain stable for that duration and some hold time after the next falling edge of SCL. After the eighth bit is shifted out (the falling edge of the eighth clock), the BF bit is cleared and the master releases the SDA pin. This allows the slave device being addressed to respond with an ACK bit during the ninth bit time, if an address match occurs or if data was received properly. The status of ACK is written into the ACKDT bit on the falling edge of the ninth clock. If the master receives an acknowledge, the acknowledge status bit, ACKSTAT, is cleared. If not, the bit is set. After the ninth clock, the SSPIF bit is set and the master clock (baud rate generator) is suspended until the next data byte is loaded into the SSPBUF, leaving the SCL pin low and the SDA pin unchanged (Figure 20-26).

After the write to the SSPBUF, each bit of address will be shifted out on the falling edge of SCL until all seven address bits and the R/W bit are completed. On the falling edge of the eighth clock, the master will de-assert the SDA pin allowing the slave to respond with an acknowledge. On the falling edge of the ninth clock, the master will sample the SDA pin to see if the address was recognized by a slave. The status of the ACK bit is loaded into the ACKSTAT status bit (SSPCON2 register). Following the falling edge of the ninth clock transmission of the address, the SSPIF is set, the BF flag is cleared, and the baud rate generator is turned off until another write to the SSPBUF takes place, holding SCL low and allowing SDA to float.

20.4.11.1 BF Status Flag

In transmit mode, the BF bit (SSPSTAT register) is set when the CPU writes to SSPBUF and is cleared when all 8 bits are shifted out.

20.4.11.2 WCOL Status Flag

If the user writes the SSPBUF when a transmit is already in progress (i.e. SSPSR is still shifting out a data byte), then WCOL is set and the contents of the buffer are unchanged (the write doesn’t occur).

WCOL must be cleared in software.

20.4.11.3 ACKSTAT Status Flag

In transmit mode, the ACKSTAT bit (SSPCON2 register) is cleared when the slave has sent an acknowledge (ACK = 0), and is set when the slave does not acknowledge (ACK = 1). A slave sends an acknowledge when it has recognized its address (including a general call), or when the slave has properly received its data.
Figure 20-25: Master Transmit Flowchart

Idle Mode
Write SSPBUF
Num_Clocks = 0, BF = 1
Force SCL = 0

Num_Clocks = 8?
Yes
Load BRG with SSPADD<6:0>, start BRG count, SDA = Current Data bit
BRG rollover?
No
Yes
Stop BRG, Force SCL = 1

SCL = 1?
No
Yes
SDA = Data bit?
No

(Successive Clock Arbitration)
Load BRG with SSPADD<6:0>, count SCL high time

BRG rollover?
No
Yes
Num_Clocks = Num_Clocks + 1

SDA = Data bit?
No

Release SDA so slave can drive ACK, Force BF = 0
Load BRG with SSPADD<6:0>, start BRG count
BRG rollover?
No
Yes
Force SCL = 1, Stop BRG

SCL = 1?
No
Yes
SDA = Data bit?
No

Bus collision detected
Set BCLIF, hold prescale off, Clear Transmit enable

Clock Arbitration

Load BRG with SSPADD<6:0>, count high time

Rollover?
No
Yes
Force SCL = 0, SSPBUF

Set SSPIF
Figure 20-26: I2C Master Mode Waveform (Transmission, 7 or 10-bit Address)

- SDI
- SCL
- SSPIF
- BF (SSPSTA<0>)
- SEN
- A7 A6 A5 A4 A3 A2 A1
- ACK
- Transmitting data or second half
- R/W
- (0x0) = Transmit Address to Slave
- Start transmit
- SCL high
- while CPU responds to SSPIF
- SEN = 0
- SSPBUF written with 7-bit address and R/W (0x0)
- Start condition begins
- From Slave, clear ACKST (SSPSTA<6>)
- Clear in software
- SSPBUF written
- From SSP interrupt
- ACKST in SSPCON2 = 1
- From CPU, clear SEN (SSPSTA<4>)
- SEN = 1
- Start condition begins
- From Slave, address
- SSPBUF written with 10-bit data or second half
- R/W
- (0x0) = Transmit Address to Slave
- SSPBUF written in software
- Cleared in software
- SSPBUF is written in software
- From SSP interrupt
- Cleared in software
- PEN
- R/W
- (0x0) = 0 D7 D6 D5 D4 D3 D2 D1 D0
- (0x0) = 1 D7 D6 D5 D4 D3 D2 D1 D0
20.4.12 I²C Master Mode Reception

Master Mode reception is enabled by programming the receive enable bit, RCEN (SSPCON2 register).

**Note:** The MSSP module must be in an idle state before the RCEN bit is set, or the RCEN bit will be disregarded.

The baud rate generator begins counting, and on each rollover, the state of the SCL pin changes (high to low/low to high) and data is shifted into the SSPSR. After the falling edge of the eighth clock, the receive enable flag is automatically cleared, the contents of the SSPSR are loaded into the SSPBUF, the BF flag bit is set, the SSPIF flag bit is set, and the baud rate generator is suspended from counting, holding SCL low. The MSSP is now in idle state, awaiting the next command. When the buffer is read by the CPU, the BF flag bit is automatically cleared. The user can then send an acknowledge bit at the end of reception by setting the acknowledge sequence enable bit, ACKEN (SSPCON2 register).

20.4.12.1 BF Status Flag

In receive mode, the BF bit is set when an address or data byte is loaded into SSPBUF from SSPSR. It is cleared when the SSPBUF Register is read.

20.4.12.2 SSPOV Status Flag

In receive mode, the SSPOV bit is set when 8 bits are received into the SSPSR, and the BF flag bit is already set from a previous reception.

20.4.12.3 WCOL Status Flag

If the user writes the SSPBUF when a receive is already in progress (i.e., SSPSR is still shifting in a data byte), then the WCOL bit is set and the contents of the buffer are unchanged (the write doesn’t occur).

The MSSP module must be in an idle state before the RCEN bit is set, or the RCEN bit will be disregarded.
Section 20. Master SSP

Figure 20-27: Master Receiver Flowchart

1. **Idle Mode**
   - RCEN = 1
   - Num_Clocks = 0, Release SDA

2. **Force SCL=0, Load BRG w/ SSPADD<6:0>, start count**

3. **BRG rollover?**
   - Yes: Release SCL
   - No: (Clock Arbitration)

4. **SCL = 1?**
   - Yes: Sample SDA pin, Shift data into SSPSR
   - No: Load BRG with SSPADD<6:0>, start count

5. **BRG rollover?**
   - Yes: SCL = 5?
     - Yes: Num_Clocks = Num_Clocks + 1
     - No: Force SCL = 0, Set SSPIF bit, Set BF bit, Move contents of SSPSR into SSPBUF, Clear RCEN
   - No: Num_Clocks = Num_Clocks + 1
     - Yes: Force SCL = 0, Set SSPIF bit, Set BF bit, Move contents of SSPSR into SSPBUF, Clear RCEN

6. **Num_Clocks = 8?**
   - Yes: Force SCL = 0, Set SSPIF bit, Set BF bit, Move contents of SSPSR into SSPBUF, Clear RCEN
   - No: Continue with previous steps
Figure 20-28: I²C Master Mode Waveform (Reception 7-Bit Address)

- **D0D1D2D3D4D5D6D7**
- **A7 A6 A5 A4 A3 A2 A1**
- **SDA**
- **SCL**

- **Bus Master terminates transfer**
- **ACK**
- **R/W = 1**

- **Write to SSPCON2<0>**
  - (SEN = 1) ACK from Slave
- **Begin Start condition**

- **Write to SSPBUF occurs here**
- **Data shifted in on falling edge of CLK**

- **Cleared in software**
- **Clears in software**
- **SSPOV**
- **ACKEN**

- **Master configured as a receiver by programming**
- **SDA = ACKDT (SSPCON2<5>) = 0**
- **SSPCON2<3>, (RCEN = 1)**
- **SSPBUF is still full**
- **SSPOV is set because SSPBUF is still full**
- **Set SSPIF interrupt at end of acknowledge sequence**
- **Set P bit (SSPSTAT<4>) and SSPIF**

- **SDA = 0, SCL = 1**
  - while CPU responds to SSPIF
  - BF (SSPSTAT<0>)

- **Last bit is shifted into SSPSR and contents are unloaded into SSPBUF**
- **Set SSPIF interrupt at end of acknowledge sequence**
- **Set SSPIF at end of receive**

- **Set ACKEN, start acknowledge sequence**
- **RCEN cleared automatically**
- **ACK from Master**
- **SDA = ACKDT = 0**
- **RCEN = 1 start next receive**
- **PEN bit = 1 written here**

- **BUS Master terminates transfer**
- **Set SSPIF interrupt at end of acknowledge sequence**
- **Set SSPIF at end of receive**

- **Write to SSPCON2<4> (SEN = 0)**
- **Start transmit**
- **Write to SSPBUF occurs here**
- **RCEN cleared automatically**

- **ACK is not sent**
- **SDA = 0, SCL = 1**
- **(SSPSTAT<0>)**

- **Cleared in software**
- **Cleared in software**
- **SSPBUF is still full**
- **Set SSPIF interrupt at end of receive**

- **Set P bit (SSPSTAT<4>) and SSPIF**
Section 20. Master SSP

20.4.13 Acknowledge Sequence Timing

An acknowledge sequence is enabled by setting the acknowledge sequence enable bit, ACKEN (SSPCON2 register). When this bit is set, the SCL pin is pulled low and the contents of the acknowledge data bit is presented on the SDA pin. If the user wishes to generate an acknowledge, then the ACKDT bit should be cleared. If not, the user should set the ACKDT bit before starting an acknowledge sequence. The baud rate generator then counts for one rollover period (TBRG), and the SCL pin is de-asserted (pulled high). When the SCL pin is sampled high (clock arbitration), the baud rate generator counts for TBRG. The SCL pin is then pulled low. Following this, the ACKEN bit is automatically cleared, the baud rate generator is turned off, and the MSSP module then goes into idle mode (Figure 20-29).

20.4.13.1 WCOL Status Flag

If the user writes the SSPBUF when an acknowledge sequence is in progress, then WCOL is set and the contents of the buffer are unchanged (the write doesn’t occur).

Figure 20-29: Acknowledge Sequence Waveform

Note: TBRG = one baud rate generator period.
Figure 20-30: Acknowledge Flowchart

1. Idle Mode
   - Set ACKEN
   - Force SCL = 0
     - SCL = 0?
       - Yes: Drive ACKDT bit (SSPCON2<5>) onto SDA pin, Load BRG with SSPADD<6:0>, start count
       - No: BRG rollover?
         - Yes: Force SCL = 0
         - No: SCL = 0?
           - Yes: Set ACKEN, Set SSPIF
           - No: BRG rollover?
             - Yes: Force SCL = 0
             - No: SCL = 0?
               - Yes: Bus collision detected, Set BCLIIF, Release SCL, Clear ACKEN
               - No: SCL = 1?
                 - Yes: Load BRG with SSPADD<6:0>, start count
                 - No: (Clock Arbitration)

2. Acknowledge
   - No ACKDT = 1?
     - Yes: SCL = 1?
       - Yes: Reset BRG, Clear ACKEN, Set SSPIF
       - No: SCL = 1?
         - Yes: Load BRG with SSPADD<6:0>, start count
         - No: SCL = 0?
           - Yes: Force SCL = 0
           - No: Release SCL, Ye s

3. BRG rollover?
   - Yes: Force SCL = 0
   - No: SCL = 0?
20.4.14 Stop Condition Timing

A Stop bit is asserted on the SDA pin at the end of a receive/transmit by setting the Stop sequence enable bit, PEN (SSPCON2 register). At the end of a receive/transmit, the SCL pin is held low after the falling edge of the ninth clock. When the PEN bit is set, the master will assert the SDA line low. When the SDA line is sampled low, the baud rate generator is reloaded and counts down to 0. When the baud rate generator times out, the SCL pin will be brought high, and one TBRG (baud rate generator rollover count) later, the SDA pin will be de-asserted. When the SDA pin is sampled high while the SCL pin is high, the P bit (SSPSTAT register) is set. A TBRG later, the PEN bit is cleared and the SSPIF bit is set (Figure 20-31).

Whenever the firmware decides to take control of the bus, it will first determine if the bus is busy by checking the S and P bits in the SSPSTAT Register. If the bus is busy, then the CPU can be interrupted (notified) when a Stop bit is detected (i.e., bus is free).

20.4.14.1 WCOL Status Flag

If the user writes the SSPBUF when a Stop sequence is in progress, then the WCOL bit is set and the contents of the buffer are unchanged (the write doesn’t occur).

![Figure 20-31: Stop Condition Receive or Transmit Mode](image)

Note: TBRG = one baud rate generator period.
Figure 20-32: Stop Condition Flowchart

Idle Mode, SSPEN = 1, SSPCON<3:0> = 0100

PEN = 1
Force SDA = 0 SCL doesn’t change
SDA = 0?
No
Yes
Start BRG
BRG rollover?
No
Yes
Release SDA, Start BRG
BRG rollover?
No
Yes
Bus Collision detected, Set BCLIF, Clear PEN
P bit set?
No
Yes
SDA going from 0 to 1 while SCL = 1, Set SSPIF, Stop condition done PEN cleared

Start BRG
BRG rollover?
No
Yes
PEN cleared

De-assert SCL, SCL = 1
(Clock Arbitration)
SCL = 1?
No
Yes

Release SDA, Start BRG
BRG rollover?
No
Yes
Bus Collision detected, Set BCLIF, Clear PEN
P bit set?
No
Yes
SDA going from 0 to 1 while SCL = 1, Set SSPIF, Stop condition done PEN cleared

PEN = 1
Section 20. Master SSP

20.4.15 Clock Arbitration

Clock arbitration occurs when the master, during any receive, transmit, or Repeated Start/Stop condition de-asserts the SCL pin (SCL allowed to float high). When the SCL pin is allowed to float high, the baud rate generator (BRG) is suspended from counting until the SCL pin is actually sampled high. When the SCL pin is sampled high, the baud rate generator is reloaded with the contents of SSPADD<6:0> and begins counting. This ensures that the SCL high time will always be at least one BRG rollover count in the event that the clock is held low by an external device (Figure 20-33).

Figure 20-33: Clock Arbitration Timing in Master Transmit Mode

20.4.15.1 Sleep Operation

While in sleep mode, the I2C module can receive addresses or data. When an address match or complete byte transfer occurs, the processor will wake-up from sleep (if the MSSP interrupt is enabled).

20.4.15.2 Effect of a Reset

A reset disables the MSSP module and terminates the current transfer.
20.4.16 Multi-Master Communication, Bus Collision, and Bus Arbitration

Multi-Master Mode support is achieved by bus arbitration. When the master outputs address/data bits onto the SDA pin, arbitration takes place when the master outputs a '1' on SDA by letting SDA float high and another master asserts a '0'. When the SCL pin floats high, data should be stable. If the expected data on SDA is a '1' and the data sampled on the SDA pin = '0', then a bus collision has taken place. The master will set the Bus Collision Interrupt Flag, BCLIF and reset the I2C port to its idle state. (Figure 20-34).

If a transmit was in progress when the bus collision occurred, the transmission is halted, the BF flag is cleared, the SDA and SCL pins are de-asserted, and the SSPBUF can be written to. When the user services the bus collision interrupt service routine, and if the I2C bus is free, the user can resume communication by asserting a Start condition.

If a Start, Repeated Start, Stop, or Acknowledge condition was in progress when the bus collision occurred, the condition is aborted, the SDA and SCL lines are de-asserted, and the respective control bits in the SSPCON2 Register are cleared. When the user services the bus collision interrupt service routine, and if the I2C bus is free, the user can resume communication by asserting a Start condition.

The Master will continue to monitor the SDA and SCL pins, and if a Stop condition occurs, the SSPIF bit will be set.

A write to the SSPBUF will start the transmission of data at the first data bit, regardless of where the transmitter left off when bus collision occurred.

In multi-Master Mode, the interrupt generation on the detection of Start and Stop conditions allows the determination of when the bus is free. Control of the I2C bus can be taken when the P bit is set in the SSPSTAT Register, or the bus is idle and the S and P bits are cleared.

Figure 20-34: Bus Collision Timing for Transmit and Acknowledge

Data changes while SCL = 0

SDA line pulled low by another source

While the SCL pin is high data doesn’t match what is driven by the Master.

Bus collision has occurred

Set bus collision interrupt (BCLIF)
20.4.16.1 Bus Collision During a Start Condition

During a Start condition, a bus collision occurs if:
- SDA or SCL pins are sampled low at the beginning of the Start condition (Figure 20-35).
- SCL pins are sampled low before the SDA pin is asserted low (Figure 20-36).

During a Start condition both the SDA and the SCL pins are monitored.

If one of the following conditions exists:
- the SDA pin is already low
- or the SCL pin is already low,

Then, the following actions occur:
- the Start condition is aborted,
- the BCLIF bit is set,
- the MSSP module is reset to its idle state (Figure 20-35).

The Start condition begins with the SDA and SCL pins de-asserted. When the SDA pin is sampled high, the baud rate generator is loaded from SSPADD<6:0> and counts down to 0. If the SCL pin is sampled low while SDA is high, a bus collision occurs, because it is assumed that another master is attempting to drive a data '1' during the Start condition.

If the SDA pin is sampled low during this count, the BRG is reset and the SDA line is asserted early (Figure 20-37). If however a '1' is sampled on the SDA pin, the SDA pin is asserted low at the end of the BRG count. The baud rate generator is then reloaded and counts down to 0. During this time, if the SCL pins is sampled as '0', a bus collision does not occur. At the end of the BRG count the SCL pin is asserted low.

Note: The reason that bus collision is not a factor during a Start condition is that no two bus masters can assert a Start condition at the exact same time. Therefore, one master will always assert SDA before the other. This condition does not cause a bus collision because the two masters must be allowed to arbitrate the first address following the Start condition, and if the address is the same, arbitration must be allowed to continue into the data portion, Repeated Start, or Stop conditions.

Figure 20-35: Bus Collision During Start Condition (SDA only)
Figure 20-36: Bus Collision During Start Condition (SCL = 0)

- SDA = 0, SCL = 1
- TBRG
- Set SEN, enable start sequence if SDA = 1, SCL = 1
- SCL = 0 before SDA = 0. Bus collision occurs, Set BCLIF
- Interrupt cleared in software

Figure 20-37: BRG Reset Due to SDA Arbitration During Start Condition

- SDA = 0, SCL = 1
- SDA pulled low by other Master. Reset BRG and assert SDA
- SCL
- SCL pulled low after BRG Timeout
- SEN
- Set SEN, enable START sequence if SDA = 1, SCL = 1
- BCLIF
- Intercepts cleared in software
- S
- SSPIF
- Interrupts cleared in software

Interrupt cleared in software

Set S

Set SSPIF

Set SSPIF

Set S

Set SSPIF

Set S

Set SSPIF

Set S

Set SSPIF
20.4.16.2 Bus Collision During a Repeated Start Condition

During a Repeated Start condition, a bus collision occurs if:

a) A low level is sampled on SDA when SCL goes from low level to high level.
b) SCL goes low before SDA is asserted low, indicating that another master is attempting to transmit a data '1'.

When the user de-asserts SDA and the pin is allowed to float high, the BRG is loaded with SSPADD<6:0> and counts down to 0. The SCL pin is then de-asserted, and when sampled high, the SDA pin is sampled.

If SDA is low, a bus collision has occurred (i.e., another master, is attempting to transmit a data '0'). If the SDA pin is sampled high, then the BRG is reloaded and begins counting. If the SDA pin goes from high to low before the BRG times out, no bus collision occurs because no two masters can assert SDA at exactly the same time, (Figure 20-38).

If the SCL pin goes from high to low before the BRG times out and the SDA pin has not already been asserted, then a bus collision occurs. In this case, another master is attempting to transmit a data '1' during the Repeated Start condition, (Figure 20-39).

If at the end of the BRG time-out, both the SCL and SDA pins are still high, the SDA pin is driven low and the BRG is reloaded and begins counting. At the end of the count, regardless of the status of the SCL pin, the SCL pin is driven low and the Repeated Start condition is complete.

Figure 20-38: Bus Collision During a Repeated Start Condition (Case 1)

Figure 20-39: Bus Collision During Repeated Start Condition (Case 2)
20.4.16.3 Bus Collision During a Stop Condition

Bus collision occurs during a Stop condition if:

a) After the SDA pin has been de-asserted and allowed to float high, SDA is sampled low after the BRG has timed out.

b) After the SCL pin is de-asserted, SCL is sampled low before SDA goes high.

The Stop condition begins with SDA asserted low. When SDA is sampled low, the SCL pin is allowed to float. When the pin is sampled high (clock arbitration), the baud rate generator is loaded with SSPADD<6:0> and counts down to 0. After the BRG times out, SDA is sampled. If SDA is sampled low, a bus collision has occurred. This is due to another master attempting to drive a data '0' (Figure 20-40). If the SCL pin is sampled low before SDA is allowed to float high, a bus collision occurs. This is another case of another master attempting to drive a data '0' (Figure 20-41).

**Figure 20-40: Bus Collision During a Stop Condition (Case 1)**

- SDA asserted low
- SDA sampled low after $T_{BRG}$, Set BCLIF

**Figure 20-41: Bus Collision During a Stop Condition (Case 2)**

- Assert SDA
- SCL goes low before SDA goes high
- Set BCLIF

---

*DS39520A-page 20-52 © 2000 Microchip Technology Inc.*
20.4.17 Connection Considerations for I^2C Bus

For standard-mode I^2C bus devices, the values of resistors Rp and Rs in Figure 20-42 depend on the following parameters:

- Supply voltage
- Bus capacitance
- Number of connected devices (input current + leakage current)

The supply voltage limits the minimum value of resistor Rp due to the specified minimum sink current of 3 mA at VOLMAX = 0.4V for the specified output stages. For example, with a supply voltage of $V_{DD} = 5V \pm 10\%$ and $V_{OLMAX} = 0.4V$ at 3 mA, $R_{P\,MIN} = (5.5 - 0.4)/0.003 = 1700 \Omega$. $V_{DD}$ as a function of $R_{P}$ is shown in Figure 20-42. The desired noise margin of 0.1VDD for the low level.

This limits the maximum value of Rs. Series resistors are optional, and used to improve ESD susceptibility.

The bus capacitance is the total capacitance of wire, connections, and pins. This capacitance limits the maximum value of Rp due to the specified rise time (Figure 20-42).

The SMP bit is the slew rate control enabled bit. This bit is in the SSPSTAT Register, and controls the slew rate of the I/O pins when in I^2C mode (master or slave).

**Figure 20-42: Sample Device Configuration for I^2C Bus**

Note: I^2C devices with input levels related to VDD must have one common supply line to which the pull up resistor is also connected.
20.4.18 Initialization

Example 20-2: SPI Master Mode Initialization

```assembly
CLRF STATUS ; Bank 0
CLRF SSPSTAT ; SMP = 0, CKE = 0, and
               ; clear status bits
BSF SSPSTAT, CKE ; CKE = 1
MOVLW 0x31 ; Set up SPI port, Master Mode, CLK/16,
MOVWF SSPCON ; Data xmit on falling edge
               ; (CKE=1 & CKP=1)
               ; Data sampled in middle
               ; (SMP=0 & Master Mode)
BSF PIE, SSPIE ; Enable SSP interrupt
BSF INTCON, GIE ; Enable, enabled interrupts
MOVLW DataByte ; Data to be Transmitted
                 ; Could move data from RAM location
MOVWF SSPBUF ; Start Transmission
```

20.4.19 Master SSP Module / Basic SSP Module Compatibility

When changing from the SPI in the Mid-range Family Basic SSP module, the SSPSTAT Register contains two additional control bits. These bits are:

- SMP, SPI data input sample phase
- CKE, SPI Clock Edge Select

To be compatible with the SPI of the Master SSP module, these bits must be appropriately configured. If these bits are not at the states shown in Table 20-5, improper SPI communication may occur.

Table 20-5: New bit States for Compatibility

<table>
<thead>
<tr>
<th>Basic SSP Module</th>
<th>Master SSP Module</th>
</tr>
</thead>
<tbody>
<tr>
<td>CKP</td>
<td>CKP</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>
Section 20. Master SSP

20.5 Design Tips

Question 1: Using SPI mode, I do not seem able to talk to an SPI device.
Answer 1:
Ensure that you are using the correct SPI mode for that device. This SPI supports all 4 SPI modes so you should be able to get it to function. Check the clock polarity and the clock phase.

Question 2: Using I2C mode, I write data to the SSPBUF Register, but the data did not transmit.
Answer 2:
Ensure that you set the CKP bit to release the I2C clock.
20.6 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Baseline, the Midrange, or High-end families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the Master SSP modules are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Use of the SSP Module in the I ²C Multi-Master Environment.</td>
<td>AN578</td>
</tr>
<tr>
<td>Using Microchip 93 Series Serial EEPROMs with Microcontroller SPI Ports</td>
<td>AN613</td>
</tr>
<tr>
<td>Interfacing PIC16C64/74 to Microchip SPI Serial EEPROM</td>
<td>AN647</td>
</tr>
<tr>
<td>Interfacing a Microchip PIC16C92x to Microchip SPI Serial EEPROM</td>
<td>AN668</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faq/codeex/
20.7 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Master SSP module description.
Section 21. Addressable USART

HIGHLIGHTS

This section of the manual contains the following major topics:

21.1 Introduction ............................................................................................................... 21-2
21.2 Control Registers ...................................................................................................... 21-3
21.3 USART Baud Rate Generator (BRG) ...................................................................... 21-5
21.4 USART Asynchronous Mode ................................................................................... 21-9
21.5 USART Synchronous Master Mode ........................................................................ 21-18
21.6 USART Synchronous Slave Mode .......................................................................... 21-23
21.7 Initialization ............................................................................................................ 21-25
21.8 Design Tips ............................................................................................................. 21-26
21.9 Related Application Notes ..................................................................................... 21-27
21.10 Revision History .................................................................................................... 21-28
21.1 Introduction

The Addressable Universal Synchronous Asynchronous Receiver Transmitter (Addressable USART) module is one of the serial I/O modules available in the PIC18CXXX family (another is the MSSP module). The Addressable USART can be configured as a full duplex asynchronous system that can communicate with peripheral devices, such as CRT terminals and personal computers, or it can be configured as a half duplex synchronous system that can communicate with peripheral devices, such as A/D or D/A integrated circuits, Serial EEPROMs, etc.

The Addressable USART can be configured in the following modes:

- Asynchronous (full duplex)
- Synchronous - Master (half duplex)
- Synchronous - Slave (half duplex)

The SPEN bit (RCSTA register) and the TRIS bits, for the USART’s pins, need to be set in order to configure the TX/CK and RX/DT pins for the Addressable USART.
Section 21. Addressable USART

21.2 Control Registers

Register 21-1: TXSTA: Transmit Status and Control Register

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>CSRC</td>
<td>TX9</td>
</tr>
<tr>
<td>TXEN</td>
<td>SYNC</td>
</tr>
<tr>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BRGH</td>
<td>TRMT</td>
</tr>
<tr>
<td>TX9D</td>
<td></td>
</tr>
</tbody>
</table>

Legend

- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as '0'
- n = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown

bit 7 CSRC: Clock Source Select bit
  - When SYNC = 0 (Asynchronous mode)
    - Don’t care
  - When SYNC = 1 (Synchronous mode)
    - 1 = Master mode (Clock generated internally from BRG)
    - 0 = Slave mode (Clock from external source)

bit 6 TX9: 9-bit Transmit Enable bit
  - 1 = Selects 9-bit transmission
  - 0 = Selects 8-bit transmission

bit 5 TXEN: Transmit Enable bit
  - 1 = Transmit enabled
  - 0 = Transmit disabled

Note: The Receive Enable (SREN/CREN) bit overrides Transmit Enable (TXEN) bit in SYNC mode.

bit 4 SYNC: Addressable USART Mode Select bit
  - 1 = Synchronous mode
  - 0 = Asynchronous mode

bit 3 Unimplemented: Read as '0'

bit 2 BRGH: High Baud Rate Select bit
  - When SYNC = 0 (Asynchronous mode)
    - 1 = High speed
    - 0 = Low speed
  - When SYNC = 1 (Synchronous mode)
    - Unused in this mode

bit 1 TRMT: Transmit Shift Register Status bit
  - 1 = TSR empty
  - 0 = TSR full

bit 0 TX9D: 9th bit of transmit data.

This bit can be used as an address/data bit or a parity bit.
<table>
<thead>
<tr>
<th>bit 7</th>
<th>SPEN: Serial Port Enable bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Serial port enabled (Configures RX/DT and TX/CK pins as serial port pins)</td>
</tr>
<tr>
<td>0</td>
<td>Serial port disabled</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>bit 6</th>
<th>RX9: 9-bit Receive Enable bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Selects 9-bit reception</td>
</tr>
<tr>
<td>0</td>
<td>Selects 8-bit reception</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>bit 5</th>
<th>SREN: Single Receive Enable bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Enables single receive</td>
</tr>
<tr>
<td>0</td>
<td>Disables single receive</td>
</tr>
</tbody>
</table>

  - When SYNC = 0 (Asynchronous mode) - master
  - Enables single receive
  - Don’t care
  - This bit is cleared after reception of one byte is complete.

  - When SYNC = 1 (Synchronous mode) - slave
  - Unused in this mode

<table>
<thead>
<tr>
<th>bit 4</th>
<th>CREN: Continuous Receive Enable bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Enables continuous receive</td>
</tr>
<tr>
<td>0</td>
<td>Disables continuous receive</td>
</tr>
</tbody>
</table>

  - When SYNC = 0 (Asynchronous mode) |
  - Enables continuous receive (CREN overrides SREN)
  - Don’t care

  - When SYNC = 1 (Synchronous mode) |
  - Unused in this mode

<table>
<thead>
<tr>
<th>bit 3</th>
<th>ADDEN: Address Detect Enable bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Enables address detection, enable interrupt and loads of the receive buffer when RSR&lt;8&gt; is set</td>
</tr>
<tr>
<td>0</td>
<td>Disables address detection, all bytes are received, and ninth bit can be used as parity bit</td>
</tr>
</tbody>
</table>

  - When SYNC = 0 (Asynchronous mode) with RX9 = 1 (9-bit receive enabled) |
  - Enables address detection, enable interrupt and loads of the receive buffer when RSR<8> is set
  - Don’t care

  - When SYNC = 0 (Asynchronous mode) with RX9 = 0 (9-bit receive disabled) |
  - Don’t care

<table>
<thead>
<tr>
<th>bit 2</th>
<th>FERR: Framing Error bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Framing error (Can be updated by reading RCREG register and receive next valid byte)</td>
</tr>
<tr>
<td>0</td>
<td>No framing error</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>bit 1</th>
<th>OERR: Overrun Error bit</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Overrun error (Can be cleared by clearing bit CREN)</td>
</tr>
<tr>
<td>0</td>
<td>No overrun error</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>bit 0</th>
<th>RX9D: 9th bit of received data. Can be address/data bit or a parity bit.</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Ninth received bit was ‘1’</td>
</tr>
<tr>
<td>0</td>
<td>Ninth received bit was ‘0’</td>
</tr>
</tbody>
</table>

Legend:
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- -n = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown
21.3 **USART Baud Rate Generator (BRG)**

The BRG supports both the Asynchronous and Synchronous modes of the USART. It is a dedicated 8-bit baud rate generator. The SPBRG register controls the period of a free running 8-bit timer. In Asynchronous mode, the BRGH bit (TXSTA<2>) also controls the baud rate. In Synchronous mode, the BRGH bit is ignored. Table 21-1 shows the formula for computation of the baud rate for different USART modes that only apply in master mode (internal clock).

Given the desired baud rate and Fosc, the nearest integer value for the SPBRG register can be calculated using the formula in Table 21-1, where X equals the value in the SPBRG register (0 to 255). From this, the error in baud rate can be determined.

Table 21-1: Baud Rate Formula

<table>
<thead>
<tr>
<th>SYNC</th>
<th>BRGH = 0 (Low Speed)</th>
<th>BRGH = 1 (High Speed)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>(Asynchronous) Baud Rate = ( Fosc / (64(X+1)) )</td>
<td>Baud Rate = ( Fosc / (16(X+1)) )</td>
</tr>
<tr>
<td>1</td>
<td>(Synchronous) Baud Rate = ( Fosc / (4(X+1)) )</td>
<td>NA</td>
</tr>
</tbody>
</table>

X = value in SPBRG (0 to 255)

Example 21-1 shows the calculation of the baud rate error for the following conditions:

Fosc = 16 MHz
Desired Baud Rate = 9600
BRGH = 0
SYNC = 0

Example 21-1: Calculating Baud Rate Error

Desired Baud Rate \( = Fosc / (64 \times (X + 1)) \)

Solving for X:

\[
\begin{align*}
X &= \left( \frac{Fosc \times (\text{Desired Baud Rate})}{64} \right) - 1 \\
X &= \left( \frac{16000000 \times 9600}{64} \right) - 1 \\
X &= 25.042 - 25 \\
X &= 25
\end{align*}
\]

Calculated Baud Rate = \( 16000000 / (64 \times (25 + 1)) \)
= 9615

Error = \( \frac{(\text{Calculated Baud Rate} - \text{Desired Baud Rate})}{\text{Desired Baud Rate}} \)
= \( \frac{(9615 - 9600)}{9600} \)
= 0.16%

It may be advantageous to use the high baud rate (BRGH = 1) even for slower baud clocks. This is because the \( \frac{Fosc}{(16X + 1)} \) equation can reduce the baud rate error in some cases.

Writing a new value to the SPBRG register causes the BRG timer to be reset (or cleared). This ensures the BRG does not wait for a timer overflow before outputting the new baud rate.

### 21.3.1 SAMPLING

The data on the RX/DT pin is sampled three times by a majority detect circuit to determine if a high or a low level is present at the RX pin. See Section 21.4.4 for additional information.

Table 21-2: Registers Associated with Baud Rate Generator

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>TXSTA</td>
<td>CSRC</td>
<td>TX9</td>
<td>TXEN</td>
<td>SYNC</td>
<td>—</td>
<td>BRGH</td>
<td>TRMT</td>
<td>TX0D</td>
<td>0000 010</td>
<td>0000 010</td>
</tr>
<tr>
<td>RCSTA</td>
<td>SPEN</td>
<td>RX9</td>
<td>SREN</td>
<td>CREN</td>
<td>ADDEN</td>
<td>FERR</td>
<td>OERR</td>
<td>RX0D</td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>SPBRG Baud Rate Generator Register</td>
<td>0000 000</td>
<td>0000 000</td>
<td>0000 000</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
### Table 21-3: Baud Rates for Synchronous Mode

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 40 MHz</th>
<th>SPBRG value (decimal)</th>
<th>33 MHz</th>
<th>SPBRG value (decimal)</th>
<th>25 MHz</th>
<th>SPBRG value (decimal)</th>
<th>20 MHz</th>
<th>SPBRG value (decimal)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0.3</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>1.2</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>2.4</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>9.6</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>19.2</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>76.8</td>
<td>76.9 ±0.16</td>
<td>129</td>
<td>77.10</td>
<td>-0.39</td>
<td>106</td>
<td>77.16</td>
<td>+0.47</td>
<td>80</td>
</tr>
<tr>
<td>96</td>
<td>96.15 ±0.16</td>
<td>103</td>
<td>95.93</td>
<td>-0.07</td>
<td>85</td>
<td>96.15</td>
<td>+0.16</td>
<td>64</td>
</tr>
<tr>
<td>300</td>
<td>303.03 ±0.01</td>
<td>32</td>
<td>294.64</td>
<td>-1.79</td>
<td>27</td>
<td>297.62</td>
<td>-0.79</td>
<td>20</td>
</tr>
<tr>
<td>500</td>
<td>500 ±0</td>
<td>19</td>
<td>485.30</td>
<td>-2.94</td>
<td>16</td>
<td>486.77</td>
<td>-3.85</td>
<td>12</td>
</tr>
<tr>
<td>HIGH</td>
<td>10000 - 0</td>
<td>0</td>
<td>8250</td>
<td>-</td>
<td>0</td>
<td>6250</td>
<td>-</td>
<td>0</td>
</tr>
<tr>
<td>LOW</td>
<td>39.06 - 255</td>
<td>32.23</td>
<td>32.23</td>
<td>-</td>
<td>255</td>
<td>24.41</td>
<td>-</td>
<td>255</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 16 MHz</th>
<th>SPBRG value (decimal)</th>
<th>10 MHz</th>
<th>SPBRG value (decimal)</th>
<th>7.15909 MHz</th>
<th>SPBRG value (decimal)</th>
<th>5.0608 MHz</th>
<th>SPBRG value (decimal)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0.3</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>1.2</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>2.4</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>9.6</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>19.2</td>
<td>19.2 ±0.16</td>
<td>207</td>
<td>19.23</td>
<td>+0.16</td>
<td>129</td>
<td>19.24</td>
<td>+0.23</td>
<td>92</td>
</tr>
<tr>
<td>76.8</td>
<td>76.9 ±0.16</td>
<td>51</td>
<td>75.76</td>
<td>-1.36</td>
<td>32</td>
<td>77.82</td>
<td>+1.32</td>
<td>22</td>
</tr>
<tr>
<td>96</td>
<td>95.24 ±0.79</td>
<td>41</td>
<td>96.15</td>
<td>+0.16</td>
<td>25</td>
<td>94.20</td>
<td>-1.88</td>
<td>18</td>
</tr>
<tr>
<td>300</td>
<td>307.70 ±2.56</td>
<td>12</td>
<td>312.50</td>
<td>-4.17</td>
<td>7</td>
<td>298.35</td>
<td>-0.57</td>
<td>5</td>
</tr>
<tr>
<td>500</td>
<td>500 ±0</td>
<td>7</td>
<td>500</td>
<td>-</td>
<td>4</td>
<td>447.44</td>
<td>-10.51</td>
<td>3</td>
</tr>
<tr>
<td>HIGH</td>
<td>4000 - 0</td>
<td>0</td>
<td>2500</td>
<td>-</td>
<td>0</td>
<td>1789.80</td>
<td>-</td>
<td>0</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 4 MHz</th>
<th>SPBRG value (decimal)</th>
<th>3.579545 MHz</th>
<th>SPBRG value (decimal)</th>
<th>1 MHz</th>
<th>SPBRG value (decimal)</th>
<th>32.768 kHz</th>
<th>SPBRG value (decimal)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0.3</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>0.30</td>
<td>+1.14</td>
</tr>
<tr>
<td>1.2</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>1.17</td>
<td>-2.48</td>
</tr>
<tr>
<td>2.4</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>2.73</td>
<td>+13.78</td>
</tr>
<tr>
<td>9.6</td>
<td>9.62 ±0.16</td>
<td>103</td>
<td>9.62</td>
<td>+0.23</td>
<td>92</td>
<td>9.62</td>
<td>+0.16</td>
<td>25</td>
</tr>
<tr>
<td>19.2</td>
<td>19.2 ±0.16</td>
<td>51</td>
<td>19.04</td>
<td>-0.83</td>
<td>46</td>
<td>19.23</td>
<td>+0.16</td>
<td>12</td>
</tr>
<tr>
<td>76.8</td>
<td>76.9 ±0.16</td>
<td>12</td>
<td>74.57</td>
<td>-2.90</td>
<td>11</td>
<td>83.33</td>
<td>+8.51</td>
<td>2</td>
</tr>
<tr>
<td>96</td>
<td>1000.0 ±4.17</td>
<td>9</td>
<td>99.43</td>
<td>+3.57</td>
<td>8</td>
<td>83.33</td>
<td>-13.19</td>
<td>2</td>
</tr>
<tr>
<td>300</td>
<td>333.33 ±11.1</td>
<td>2</td>
<td>298.30</td>
<td>-0.57</td>
<td>2</td>
<td>250</td>
<td>-16.67</td>
<td>0</td>
</tr>
<tr>
<td>500</td>
<td>500 ±0</td>
<td>1</td>
<td>447.44</td>
<td>-10.51</td>
<td>1</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
</tr>
<tr>
<td>HIGH</td>
<td>10000 - 0</td>
<td>0</td>
<td>894.89</td>
<td>-</td>
<td>0</td>
<td>250</td>
<td>-</td>
<td>0</td>
</tr>
<tr>
<td>LOW</td>
<td>3.91 - 255</td>
<td>3.50</td>
<td>3.50</td>
<td>-</td>
<td>255</td>
<td>0.98</td>
<td>-</td>
<td>255</td>
</tr>
</tbody>
</table>
## Section 21. Addressable USART

### Table 21-4: Baud Rates for Asynchronous Mode (BRGH = 0)

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 40 MHz</th>
<th>SPBRG value (decimal)</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
</tr>
</thead>
<tbody>
<tr>
<td>0.3</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>1.2</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>2.4</td>
<td>9.62</td>
<td>+0.16 64</td>
<td>9.55</td>
<td>-0.54 53</td>
<td>9.53</td>
<td>-0.76 40</td>
<td>9.47</td>
<td>-1.36 32</td>
<td>9.47</td>
<td>-1.36 32</td>
</tr>
<tr>
<td>19.2</td>
<td>18.94</td>
<td>-1.36 32</td>
<td>19.10</td>
<td>-0.54 26</td>
<td>19.53</td>
<td>+1.73 19</td>
<td>19.53</td>
<td>+1.73 15</td>
<td>19.53</td>
<td>+1.73 15</td>
</tr>
<tr>
<td>76.8</td>
<td>78.13</td>
<td>+1.73 7</td>
<td>73.66</td>
<td>-4.09 6</td>
<td>78.13</td>
<td>+1.73 4</td>
<td>78.13</td>
<td>+1.73 3</td>
<td>78.13</td>
<td>+1.73 3</td>
</tr>
<tr>
<td>96</td>
<td>89.29</td>
<td>-6.99 6</td>
<td>103.13</td>
<td>+7.42 4</td>
<td>97.66</td>
<td>+1.73 3</td>
<td>104.17</td>
<td>+8.51 2</td>
<td>104.17</td>
<td>+8.51 2</td>
</tr>
<tr>
<td>300</td>
<td>312.50</td>
<td>+4.17 1</td>
<td>257.81</td>
<td>-14.06 1</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>500</td>
<td>625</td>
<td>+25.00 0</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>HIGH</td>
<td>625</td>
<td>-</td>
<td>515.63</td>
<td>-</td>
<td>390.63</td>
<td>0</td>
<td>312.50</td>
<td>0</td>
<td>312.50</td>
<td>0</td>
</tr>
<tr>
<td>LOW</td>
<td>2.44</td>
<td>-</td>
<td>2.01</td>
<td>-</td>
<td>255</td>
<td>1.53</td>
<td>255</td>
<td>1.22</td>
<td>255</td>
<td>1.22</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 16 MHz</th>
<th>SPBRG value (decimal)</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
</tr>
</thead>
<tbody>
<tr>
<td>0.3</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>1.2</td>
<td>1.20</td>
<td>+0.16 207</td>
<td>1.20</td>
<td>+0.16 129</td>
<td>1.20</td>
<td>+0.23 92</td>
<td>1.20</td>
<td>0 65</td>
<td>1.20</td>
<td>0 65</td>
</tr>
<tr>
<td>2.4</td>
<td>2.40</td>
<td>+0.16 103</td>
<td>2.40</td>
<td>+0.16 64</td>
<td>2.38</td>
<td>-0.83 46</td>
<td>2.40</td>
<td>0 32</td>
<td>2.40</td>
<td>0 32</td>
</tr>
<tr>
<td>19.2</td>
<td>19.23</td>
<td>-0.16 12</td>
<td>19.53</td>
<td>+1.73 7</td>
<td>18.64</td>
<td>-2.90 5</td>
<td>19.80</td>
<td>+3.13 3</td>
<td>19.80</td>
<td>+3.13 3</td>
</tr>
<tr>
<td>76.8</td>
<td>83.33</td>
<td>-8.51 2</td>
<td>78.13</td>
<td>+1.73 1</td>
<td>111.86</td>
<td>+4.65 0</td>
<td>79.20</td>
<td>+3.13 0</td>
<td>79.20</td>
<td>+3.13 0</td>
</tr>
<tr>
<td>96</td>
<td>83.33</td>
<td>-8.51 2</td>
<td>78.13</td>
<td>-18.62 1</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>300</td>
<td>250</td>
<td>-16.67 0</td>
<td>156.25</td>
<td>-47.92 0</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>500</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>HIGH</td>
<td>250</td>
<td>-</td>
<td>156.25</td>
<td>-</td>
<td>111.86</td>
<td>0</td>
<td>79.20</td>
<td>0</td>
<td>79.20</td>
<td>0</td>
</tr>
<tr>
<td>LOW</td>
<td>0.98</td>
<td>-</td>
<td>0.61</td>
<td>-</td>
<td>255</td>
<td>0.44</td>
<td>255</td>
<td>0.31</td>
<td>255</td>
<td>0.31</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 4 MHz</th>
<th>SPBRG value (decimal)</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
<th>KBAUD</th>
<th>% ERROR</th>
</tr>
</thead>
<tbody>
<tr>
<td>0.3</td>
<td>0.30</td>
<td>-0.16 207</td>
<td>0.30</td>
<td>+0.23 185</td>
<td>0.30</td>
<td>+0.16 51</td>
<td>0.26</td>
<td>-14.67 1</td>
<td>0.26</td>
<td>-14.67 1</td>
</tr>
<tr>
<td>1.2</td>
<td>1.20</td>
<td>+1.67 51</td>
<td>1.19</td>
<td>-0.83 46</td>
<td>1.20</td>
<td>+0.16 12</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>2.4</td>
<td>2.40</td>
<td>+1.67 25</td>
<td>2.43</td>
<td>-1.32 22</td>
<td>2.23</td>
<td>-6.99 6</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>9.6</td>
<td>5.93</td>
<td>-6.99 6</td>
<td>9.32</td>
<td>-2.90 5</td>
<td>7.81</td>
<td>-18.62 1</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>19.2</td>
<td>20.83</td>
<td>+8.51 2</td>
<td>18.64</td>
<td>-2.90 2</td>
<td>15.63</td>
<td>-18.62 0</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>76.8</td>
<td>62.50</td>
<td>-18.62 0</td>
<td>55.93</td>
<td>-27.17 0</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>96</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>300</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>500</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
<td>NA</td>
<td>-</td>
</tr>
<tr>
<td>HIGH</td>
<td>62.50</td>
<td>-</td>
<td>55.93</td>
<td>-</td>
<td>15.63</td>
<td>0</td>
<td>0.51</td>
<td>0</td>
<td>0.51</td>
<td>0</td>
</tr>
<tr>
<td>LOW</td>
<td>0.24</td>
<td>-</td>
<td>0.22</td>
<td>-</td>
<td>255</td>
<td>0.86</td>
<td>255</td>
<td>0.002</td>
<td>255</td>
<td>0.002</td>
</tr>
</tbody>
</table>
### Table 21-5: Baud Rates for Asynchronous Mode (BRGH = 1)

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 40 MHz</th>
<th>33 MHz</th>
<th>25 MHz</th>
<th>20 MHz</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>SPBRG value</td>
<td>SPBRG value</td>
<td>SPBRG value</td>
<td>SPBRG value</td>
</tr>
<tr>
<td></td>
<td>(decimal)</td>
<td>(decimal)</td>
<td>(decimal)</td>
<td>(decimal)</td>
</tr>
<tr>
<td></td>
<td>% ERROR</td>
<td>% ERROR</td>
<td>% ERROR</td>
<td>% ERROR</td>
</tr>
<tr>
<td>0.3</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
</tr>
<tr>
<td>1.2</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
</tr>
<tr>
<td>2.4</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
</tr>
<tr>
<td>9.6</td>
<td>NA</td>
<td>9.60 - 0.07</td>
<td>9.59 - 0.15</td>
<td>9.62 + 0.16</td>
</tr>
<tr>
<td>19.2</td>
<td>19.28 + 0.39</td>
<td>19.30 + 0.47</td>
<td>19.23 + 0.16</td>
<td>19.23 + 0.16</td>
</tr>
<tr>
<td>76.8</td>
<td>76.39 - 0.54</td>
<td>78.13 + 1.73</td>
<td>78.13 + 1.73</td>
<td>78.13 + 1.73</td>
</tr>
<tr>
<td>96</td>
<td>98.21 - 2.31</td>
<td>97.66 + 1.73</td>
<td>96.15 + 0.16</td>
<td>96.15 + 0.16</td>
</tr>
<tr>
<td>300</td>
<td>294.64 - 1.79</td>
<td>312.50 + 4.17</td>
<td>312.50 + 4.17</td>
<td>312.50 + 4.17</td>
</tr>
<tr>
<td>500</td>
<td>515.63 + 3.13</td>
<td>520.83 + 4.17</td>
<td>416.67 - 1.67</td>
<td>416.67 - 1.67</td>
</tr>
<tr>
<td>HIGH</td>
<td>2062.50 - 0</td>
<td>1562.50 - 0</td>
<td>1250 - 0</td>
<td>1250 - 0</td>
</tr>
<tr>
<td>LOW</td>
<td>9.77 - 255</td>
<td>8.06 - 255</td>
<td>6.10 - 255</td>
<td>4.88 - 255</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 16 MHz</th>
<th>10 MHz</th>
<th>7.15909 MHz</th>
<th>5.0688 MHz</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>SPBRG value</td>
<td>SPBRG value</td>
<td>SPBRG value</td>
<td>SPBRG value</td>
</tr>
<tr>
<td></td>
<td>(decimal)</td>
<td>(decimal)</td>
<td>(decimal)</td>
<td>(decimal)</td>
</tr>
<tr>
<td></td>
<td>% ERROR</td>
<td>% ERROR</td>
<td>% ERROR</td>
<td>% ERROR</td>
</tr>
<tr>
<td>0.3</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
</tr>
<tr>
<td>1.2</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
</tr>
<tr>
<td>2.4</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
<td>NA</td>
</tr>
<tr>
<td>9.6</td>
<td>9.62 + 0.16</td>
<td>9.62 + 0.16</td>
<td>9.62 + 0.16</td>
<td>9.62 + 0.16</td>
</tr>
<tr>
<td>19.2</td>
<td>19.28 + 0.39</td>
<td>19.30 + 0.47</td>
<td>19.23 + 0.16</td>
<td>19.23 + 0.16</td>
</tr>
<tr>
<td>76.8</td>
<td>76.39 - 0.54</td>
<td>78.13 + 1.73</td>
<td>78.13 + 1.73</td>
<td>78.13 + 1.73</td>
</tr>
<tr>
<td>96</td>
<td>98.21 - 2.31</td>
<td>97.66 + 1.73</td>
<td>96.15 + 0.16</td>
<td>96.15 + 0.16</td>
</tr>
<tr>
<td>300</td>
<td>294.64 - 1.79</td>
<td>312.50 + 4.17</td>
<td>312.50 + 4.17</td>
<td>312.50 + 4.17</td>
</tr>
<tr>
<td>500</td>
<td>515.63 + 3.13</td>
<td>520.83 + 4.17</td>
<td>416.67 - 1.67</td>
<td>416.67 - 1.67</td>
</tr>
<tr>
<td>HIGH</td>
<td>2062.50 - 0</td>
<td>1562.50 - 0</td>
<td>1250 - 0</td>
<td>1250 - 0</td>
</tr>
<tr>
<td>LOW</td>
<td>9.77 - 255</td>
<td>8.06 - 255</td>
<td>6.10 - 255</td>
<td>4.88 - 255</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>BAUD RATE (Kbps)</th>
<th>FOSC = 4 MHz</th>
<th>3.57954 MHz</th>
<th>1 MHz</th>
<th>32.768 kHz</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>SPBRG value</td>
<td>SPBRG value</td>
<td>SPBRG value</td>
<td>SPBRG value</td>
</tr>
<tr>
<td></td>
<td>(decimal)</td>
<td>(decimal)</td>
<td>(decimal)</td>
<td>(decimal)</td>
</tr>
<tr>
<td></td>
<td>% ERROR</td>
<td>% ERROR</td>
<td>% ERROR</td>
<td>% ERROR</td>
</tr>
<tr>
<td>0.3</td>
<td>NA</td>
<td>NA</td>
<td>0.30 + 0.16</td>
<td>0.29 - 2.48</td>
</tr>
<tr>
<td>1.2</td>
<td>1.20 + 0.16</td>
<td>1.20 + 0.16</td>
<td>1.20 + 0.16</td>
<td>1.02 - 14.67</td>
</tr>
<tr>
<td>2.4</td>
<td>2.40 + 0.16</td>
<td>2.40 + 0.16</td>
<td>2.40 + 0.16</td>
<td>2.05 - 14.67</td>
</tr>
<tr>
<td>9.6</td>
<td>9.62 + 0.16</td>
<td>9.62 + 0.16</td>
<td>9.62 + 0.16</td>
<td>9.62 + 0.16</td>
</tr>
<tr>
<td>19.2</td>
<td>19.28 + 0.39</td>
<td>19.28 + 0.39</td>
<td>19.28 + 0.39</td>
<td>19.28 + 0.39</td>
</tr>
<tr>
<td>76.8</td>
<td>76.39 - 0.54</td>
<td>78.13 + 1.73</td>
<td>78.13 + 1.73</td>
<td>78.13 + 1.73</td>
</tr>
<tr>
<td>96</td>
<td>98.21 - 2.31</td>
<td>97.66 + 1.73</td>
<td>96.15 + 0.16</td>
<td>96.15 + 0.16</td>
</tr>
<tr>
<td>300</td>
<td>294.64 - 1.79</td>
<td>312.50 + 4.17</td>
<td>312.50 + 4.17</td>
<td>312.50 + 4.17</td>
</tr>
<tr>
<td>500</td>
<td>515.63 + 3.13</td>
<td>520.83 + 4.17</td>
<td>416.67 - 1.67</td>
<td>416.67 - 1.67</td>
</tr>
<tr>
<td>HIGH</td>
<td>2062.50 - 0</td>
<td>1562.50 - 0</td>
<td>1250 - 0</td>
<td>1250 - 0</td>
</tr>
<tr>
<td>LOW</td>
<td>9.77 - 255</td>
<td>8.06 - 255</td>
<td>6.10 - 255</td>
<td>4.88 - 255</td>
</tr>
</tbody>
</table>
Section 21. Addressable USART

21.4 USART Asynchronous Mode

In this mode, the USART uses standard nonreturn-to-zero (NRZ) format (one start bit, eight or nine data bits and one stop bit). The most common data format is 8 bits. An on-chip dedicated 8-bit baud rate generator can be used to derive standard baud rate frequencies from the oscillator. The USART transmits and receives the LSB first. The USART’s transmitter and receiver are functionally independent, but use the same data format and baud rate. The baud rate generator produces a clock either \( x16 \) or \( x64 \) of the bit shift rate, depending on the BRGH bit (TXSTA register). Parity is not supported by the hardware, but can be implemented in software (stored as the ninth data bit). Asynchronous mode is stopped during SLEEP.

Asynchronous mode is selected by clearing the SYNC bit (TXSTA register).

The USART Asynchronous module consists of the following important elements:

- Baud Rate Generator
- Sampling Circuit
- Asynchronous Transmitter
- Asynchronous Receiver

21.4.1 USART Asynchronous Transmitter

The USART transmitter block diagram is shown in Figure 21-1. The heart of the transmitter is the Transmit Shift Register (TSR). The shift register obtains its data from the transmit buffer, TXREG. The TXREG register is loaded with data in software. The TSR register is not loaded until the STOP bit has been transmitted from the previous load. As soon as the STOP bit is transmitted, the TSR is loaded with new data from the TXREG register (if available). Once the TXREG register transfers the data to the TSR register (occurs in one Tcy), the TXREG register is empty and the TXIF flag bit is set. This interrupt can be enabled/disabled by setting/clearing the TXIE enable bit. The TXIF flag bit will be set, regardless of the state of the TXIE enable bit and cannot be cleared in software. It will reset only when new data is loaded into the TXREG register. While the TXIF flag bit indicated the status of the TXREG register, the TRMT bit (TXSTA register) shows the status of the TSR register. The TRMT status bit is a read only bit, which is set when the TSR register is empty. No interrupt logic is tied to this bit, so the user has to poll this bit in order to determine if the TSR register is empty.

Transmission is enabled by setting the TXEN enable bit (TXSTA register). The actual transmission will not occur until the TXREG register has been loaded with data and the Baud Rate Generator (BRG) has produced a shift clock (Figure 21-1). The transmission can also be started by first loading the TXREG register and then setting the TXEN enable bit. Normally when transmission is first started, the TSR register is empty, so a transfer to the TXREG register will result in an immediate transfer to TSR, resulting in an empty TXREG. A back-to-back transfer is thus possible (Figure 21-3). Clearing the TXEN enable bit during a transmission will cause the transmission to be aborted and will reset the transmitter. As a result, the TX/CK pin will revert to hi-impedance.

In order to select 9-bit transmission the TX9 bit (TXSTA register) should be set and the ninth bit should be written to the TX9D bit (TXSTA register). The ninth bit must be written before writing the 8-bit data to the TXREG register. This is because a data write to the TXREG register can result in an immediate transfer of the data to the TSR register (if the TSR is empty). In such a case, an incorrect ninth data bit may be loaded in the TSR register.

Note 1: The TSR register is not mapped in data memory, so it is not available to the user.

2: When the TXEN bit is set, the TXIF flag bit will also be set since the transmit buffer is not yet full (can move transmit data to the TXREG register).
Steps to follow when setting up an Asynchronous Transmission:

1. Initialize the SPBRG register for the appropriate baud rate. If a high speed baud rate is desired, set the BRGH bit. (Subsection 21.3 “USART Baud Rate Generator (BRG)”)
2. Enable the asynchronous serial port by clearing the SYNC bit and setting the SPEN bit.
3. If interrupts are desired, then set the TXIE, GIE/GIEH and PEIE/GIEL bits. Specify the interrupt priority if required.
4. If 9-bit transmission is desired, then set the TX9 bit (can be used as address/data bit).
5. Enable the transmission by setting the TXEN bit, which will also set the TXIF bit.
6. If 9-bit transmission is selected, the ninth bit should be loaded in the TX9D bit.
7. Load data to the TXREG register (starts transmission).

Figure 21-2: Asynchronous Transmission (8- or 9-bit Data)
Section 21. Addressable USART

Figure 21-3: Asynchronous Transmission (Back to Back)

Table 21-6: Registers Associated with Asynchronous Transmission

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x 0000 000u</td>
<td></td>
</tr>
<tr>
<td>PIRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 0</td>
<td></td>
</tr>
<tr>
<td>PIEx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 0</td>
<td></td>
</tr>
<tr>
<td>IPRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0 0</td>
<td></td>
</tr>
<tr>
<td>RCSTA</td>
<td>SPEN</td>
<td>RX9</td>
<td>SREN</td>
<td>GREN</td>
<td>ADDEN</td>
<td>FERR</td>
<td>OERR</td>
<td>RX9D</td>
<td>0000 000x 0000 000x</td>
<td></td>
</tr>
<tr>
<td>TXREG</td>
<td>TX7</td>
<td>TX6</td>
<td>TX5</td>
<td>TX4</td>
<td>TX3</td>
<td>TX2</td>
<td>TX1</td>
<td>TX0</td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
<tr>
<td>TXSTA</td>
<td>CSRC</td>
<td>TX9</td>
<td>TXEN</td>
<td>—</td>
<td>BRGH</td>
<td>TRMT</td>
<td>TX9D</td>
<td>—</td>
<td>0000 -010 0000 -010</td>
<td></td>
</tr>
<tr>
<td>SPBRG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
</tbody>
</table>

Legend: x = unknown, - = unimplemented locations read as 0.

Note 1: The PSPIF, PSPIE and PSPIP bits are reserved on the PIC18C2X2 devices. Always maintain these bits clear.

2: The position of this bit is device dependent.
21.4.2 USART Asynchronous Receiver

The receiver block diagram is shown in Figure 21-4. The data is received on the RX/D0 pin and drives the data recovery block. The data recovery block is actually a high speed shifter operating at x16 times the baud rate, whereas the main receive serial shifter operates at the bit rate or at FOSC. This mode would typically be used in RS-232 systems.

The USART module has a special provision for multi-processor communication. When the RX9 bit is set in the RCSTA register, 9-bits are received and the ninth bit is placed in the RX9D status bit of the RSTA register. The port can be programmed such that when the stop bit is received, the serial port interrupt will only be activated if the RX9D bit is set. This feature is enabled by setting the ADDEN bit in the RCSTA register and can be used in a multi-processor system in the following manner.

To transmit a block of data in a multi-processor system, the master processor must first send an address byte that identifies the target slave. An address byte is identified by the RX9D bit being a ‘1’ (instead of a ‘0’ for a data byte). If the ADDEN bit is set in the slave’s RCSTA register, all data bytes will be ignored. However, if the ninth received bit is equal to a ‘1’, indicating that the received byte is an address, the slave will be interrupted and the contents of the Receive Shift Register (RSR) will be transferred into the receive buffer. This allows the slave to be interrupted only by addresses, so that the slave can examine the received byte to see if it is addressed. The addressed slave will then clear its ADDEN bit and prepare to receive data bytes from the master.

When the ADDEN bit is set, all data bytes are ignored. Following the STOP bit, the data will not be loaded into the receive buffer and no interrupt will occur. If another byte is shifted into the RSR register, the previous data byte will be lost.

The ADDEN bit will only take effect when the receiver is configured in 9-bit mode.

Once Asynchronous mode is selected, reception is enabled by setting the CREN bit.

The heart of the receiver is the Receive (serial) Shift Register (RSR). After sampling the RX/TX pin for the STOP bit, the received data in the RSR is transferred to the RCREG register (if it is empty). If the transfer is complete, the RCIF flag bit is set. The actual interrupt can be enabled/disabled by setting/clearing the RCIE enable bit. The RCIF flag bit is a read only bit that is cleared when the RCREG register has been read and is empty.

The RCREG is a double-buffered register (i.e., it is a two-deep FIFO). It is possible for two bytes of data to be received and transferred to the RCREG FIFO and a third byte begin shifting to the RSR register. On the detection of the STOP bit of the third byte, if the RCREG register is still full, overrun error bit, OERR, will be set. The word in the RSR will be lost. The RCREG register can be read twice to retrieve the two bytes in the FIFO. The OERR bit has to be cleared in software. This is done by resetting the receive logic (the CREN bit is cleared and then set). If the OERR bit is set, transfers from the RSR register to the RCREG register are inhibited, so it is essential to clear the OERR bit if it is set. Framing error bit, FERR, is set if a stop bit is detected as a low level. The FERR bit and the 9th receive bit are buffered the same way as the receive data. Reading the RCREG will load the RX9D and FERR bits with new values. Therefore, it is essential for the user to read the RCSTA register before reading the next RCREG register, in order not to lose the old (previous) information in the FERR and RX9D bits.

Figure 21-4 shows a block diagram for the receive of the Addressable USART.
Section 21. Addressable USART

Figure 21-4: Addressable USART Receive Block Diagram
21.4.2.1 Asynchronous Receptions (no Address Detect)

Steps to follow when setting up an Asynchronous Reception:

1. Initialize the SPBRG register for the appropriate baud rate. If a high speed baud rate is desired, set bit BRGH. (Subsection 21.3 “USART Baud Rate Generator (BRG)”).
2. Enable the asynchronous serial port by clearing the SYNC bit and setting the SPEN bit.
3. If interrupts are desired, then set the RCIE bit and configure the RCIP, GIE/GIEH and PEIE/GIEL bits, appropriately.
4. If 9-bit reception is desired, then set the RX9 bit.
5. Enable the reception by setting the CREN bit.
6. The RCIF flag bit will be set when reception is complete. An interrupt will be generated depending on the configuration of the RCIE, RCIP, GIE/GIEH and PEIE/GIEL bits.
7. Read the RCSTA register to get the ninth bit (if enabled) and determine if any error occurred during reception.
8. Read the 8-bit received data by reading the RCREG register.
9. If any error occurred, clear the error by clearing the CREN bit.

Figure 21-5: Asynchronous Reception (8- or 9-bit Data)

Note: This timing diagram shows three words appearing on the RX input. The RCREG (receive buffer) is read after the third word, causing the OERR (overrun) bit to be set.
Section 21. Addressable USART

21.4.3 Setting up 9-bit mode with Address Detect

Address detect mode allows an Addressable USART node to ignore all data on the bus until a new address byte is present. This reduces the interrupt overhead since not every byte will generate an interrupt (only bytes that are directed to that node).

21.4.3.1 Transmit

Steps to follow when setting up an Asynchronous Transmission:
1. Initialize the SPBRG register for the appropriate baud rate. If a high speed baud rate is desired, set the BRGH bit. (Subsection 21.3 “USART Baud Rate Generator (BRG)”)
2. Enable the asynchronous serial port by clearing the SYNC bit and setting the SPEN bit.
3. If interrupts are desired, then set the TXIE, TXIP, GIE/GIEH and PEIE/GIEL bits.
4. If 9-bit transmission is desired, then set the TX9 bit (can be used as address/data bit).
5. Enable the transmission by setting the TXEN bit, which will also set the TXIF bit.
6. If 9-bit transmission is selected, set the TX9D bit for address, clear the TX9D bit for data, set the TX9D bit for address and clear the TX9D bit for data.
7. Load data to the TXREG register (starts transmission).

21.4.3.2 Receive

Steps to follow when setting up an Asynchronous Reception with Address Detect enabled:
1. Initialize the SPBRG register for the appropriate baud rate. If a high speed baud rate is desired, set bit BRGH.
2. Enable the asynchronous serial port by clearing bit SYNC and setting bit SPEN.
3. If interrupts are desired, then set the RCIE bit and configure the RCIP, GIE/GIEH and PEIE/GIEL bits, appropriately.
4. Set bit RX9 to enable 9-bit reception.
5. Set ADDEN to enable address detect.
6. Enable the reception by setting enable bit CREN.
7. The RCIF flag bit will be set when reception is complete. An interrupt will be generated depending on the configuration of the RCIE, RCIP, GIE/GIEH and PEIE/GIEL bits.
8. Read the RCSTA register to get the ninth bit and determine if any error occurred during reception.
9. Read the 8-bit received data by reading the RCREG register, to determine if the device is being addressed.
10. If any error occurred, clear the error by clearing enable bit CREN.
11. If the device has been addressed, clear the ADDEN bit to allow data bytes and address bytes to be read into the receive buffer, and interrupt the CPU.
Figure 21-6: USART Receive Block Diagram

Figure 21-7: Asynchronous Reception with Address Detect

Note: This timing diagram shows a data byte followed by an address byte. The data byte is not read into the RCREG (receive buffer) because ADDEN = 0.

Figure 21-8: Asynchronous Reception with Address Byte First

Note: This timing diagram shows an address byte followed by a data byte. The data byte is not read into the RCREG (receive buffer) because ADDEN was not updated and still = 0.
Section 21. Addressable USART

21.4.4 Sampling

The data on the RX/DT pin is sampled three times by a majority detect circuit to determine if a high or a low level is present at the RX pin. Figure 21-9 shows the waveform for the sampling circuit. The sampling operates the same regardless of the state of the BRGH bit, only the source of the x16 clock is different.

Figure 21-9: RX Pin Sampling Scheme, BRGH = 0 or BRGH = 1

Table 21-7: Registers Associated with Asynchronous Reception

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>GIE/GIEH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>PIRx</td>
<td></td>
<td>RCIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>PEx</td>
<td></td>
<td></td>
<td>RCIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>IPx</td>
<td></td>
<td></td>
<td></td>
<td>RCIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>RCSTA</td>
<td>SPEN</td>
<td>RX9</td>
<td>SREN</td>
<td>CREN</td>
<td>ADDEN</td>
<td>FERR</td>
<td>OERR</td>
<td>RX9D</td>
<td>0000 000x</td>
<td>0000 000x</td>
</tr>
<tr>
<td>RCREG</td>
<td>RX7</td>
<td>RX6</td>
<td>RX5</td>
<td>RX4</td>
<td>RX3</td>
<td>RX2</td>
<td>RX1</td>
<td>RX0</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TXSTA</td>
<td>CSRC</td>
<td>TX9</td>
<td>TXEN</td>
<td>SYNC</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 -010</td>
<td>0000 -010</td>
</tr>
<tr>
<td>SPBRG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
</tbody>
</table>

Legend: x = unknown, - = unimplemented locations read as '0'.
Shaded cells are not used for Asynchronous Reception.

Note 1: The position of this bit is device dependent.
21.5 USART Synchronous Master Mode

In Synchronous Master mode, the data is transmitted in a half-duplex manner, (i.e., transmission and reception do not occur at the same time). When transmitting data, the reception is inhibited and vice versa. Synchronous mode is entered by setting the SYNC bit. In addition, the SPEN enable bit is set in order to configure the TX/CK and RX/DT I/O pins to CK (clock) and DT (data) lines respectively. The Master mode indicates that the processor transmits the master clock on the CK line. The Master mode is entered by setting the CSRC bit.

21.5.1 USART Synchronous Master Transmission

The USART transmitter block diagram is shown in Figure 21-1. The heart of the transmitter is the Transmit Shift Register (TSR). The shift register obtains its data from the read/write transmit buffer register TXREG. The TXREG register is loaded with data in software. The TSR register is not loaded until the last bit has been transmitted from the previous load. As soon as the last bit is transmitted, the TSR is loaded with new data from the TXREG (if available). Once the TXREG register transfers the data to the TSR register (occurs in one Tcycle), the TXREG is empty and the TXIF interrupt flag bit is set. The interrupt can be enabled/disabled by setting/clearing the TXIE enable bit. The TXIF flag bit will be set regardless of the state of the TXIE enable bit and cannot be cleared in software. It will reset only when new data is loaded into the TXREG register. While the TXIF flag bit indicates the status of the TXREG register, the TRMT bit shows the status of the TSR register. The TRMT bit is a read only bit that is set when the TSR is empty. No interrupt logic is tied to this bit, so the user has to poll this bit in order to determine if the TSR register is empty. The TSR is not mapped in data memory, so it is not available to the user.

Transmission is enabled by setting the TXEN bit. The actual transmission will not occur until the TXREG register has been loaded with data. The first data bit will be shifted out on the next available rising edge of the clock on the CK line. Data out is stable at the falling edge of the synchronous clock (Figure 21-10). The transmission can also be started by first loading the TXREG register and then setting the TXEN bit. This is advantageous when slow baud rates are selected, since the BRG is kept in RESET when the TXEN, CREN and SREN bits are clear. Setting the TXEN bit will start the BRG, creating a shift clock immediately. Normally, when transmission is first started, the TSR register is empty, so a transfer to the TXREG register will result in an immediate transfer to TSR, resulting in an empty TXREG. Back-to-back transfers are possible.

Clearing the TXEN bit during a transmission will cause the transmission to be aborted and will reset the transmitter. The DT and CK pins will revert to hi-impedance. If either of the CREN or SREN bits are set during a transmission, the transmission is aborted and the DT pin reverts to a hi-impedance state (for a reception). The CK pin will remain an output if the CSRC bit is set (internal clock). The transmitter logic is not reset although it is disconnected from the pins. In order to reset the transmitter, the user has to clear the TXEN bit. If the SREN bit is set (to interrupt an on-going transmission and receive a single word), then after the single word is received, the SREN bit will be cleared and the serial port will revert back to transmitting, since the TXEN bit is still set. The DT line will immediately switch from hi-impedance receive mode to transmit and start driving. To avoid this, the TXEN bit should be cleared.

In order to select 9-bit transmission, the TX9 bit should be set and the ninth bit should be written to the TX9D bit. The ninth bit must be written before writing the 8-bit data to the TXREG register. This is because a data write to the TXREG can result in an immediate transfer of the data to the TSR register (if the TSR is empty). If the TSR was empty and the TXREG was written before writing the “new” value to the TX9D bit, the “present” value of the TX9D bit is loaded.
Section 21. Addressable USART

Steps to follow when setting up a Synchronous Master Transmission:
1. Initialize the SPBRG register for the appropriate baud rate. (Subsection 21.3 “USART Baud Rate Generator (BRG)”)
2. Enable the synchronous master serial port by setting the SYNC, SPEN and CSRC bits.
3. If interrupts are desired, then set the TXIE bit and configure the RCIP, GIE/GIEH and PEIE/GIEL bits, appropriately.
4. If 9-bit transmission is desired, then set the TX9 bit.
5. Enable the transmission by setting the TXEN bit.
6. If 9-bit transmission is selected, the ninth bit should be loaded in the TX9D bit.
7. Start transmission by loading data to the TXREG register.

Table 21-8: Registers Associated with Synchronous Master Transmission

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCN</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>GIE/GIEH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>PEIE/GIEL</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TMR0IE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>INTOE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>RBIE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TMR0IF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>INTOIF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>RBIF</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>PIRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TXIF (1)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>PIEx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>IPRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TXIE (1)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>IPEx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>RCSTA</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TPx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TXREG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 00000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>TXSTA</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 00000x</td>
<td>0000 0000u</td>
</tr>
<tr>
<td>SPBRG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 00000x</td>
<td>0000 0000u</td>
</tr>
</tbody>
</table>

Legend:  x = unknown, — = unimplemented, read as ‘0’.
Shaded cells are not used for Synchronous Master Transmission.

Note 1: The position of this bit is device dependent.
**Figure 21-10: Synchronous Transmission**

- **RX/DT pin**
- **TX/CK pin**
- **Write to TXREG reg**
- **TXIF bit** (Interrupt flag)
- **TRMT bit**
- **TXEN bit**

**Note:** Sync Master mode; SPBRG = '0'. Continuous transmission of two 8-bit words.

**Figure 21-11: Synchronous Transmission (Through TXEN)**

- **RX/DT pin**
- **TX/CK pin**
- **Write to TXREG reg**
- **TXIF bit**
- **TRMT bit**
- **TXEN bit**
Section 21. Addressable USART

21.5.1.1 USART Synchronous Master Reception

Once Synchronous mode is selected, reception is enabled by setting either the SREN or CREN bits. Data is sampled on the RX/DT pin on the falling edge of the clock. If the SREN bit is set, then only a single word is received. If the CREN bit is set, the reception is continuous until the CREN bit is cleared. If both bits are set, then the CREN bit takes precedence. After clocking the last serial data bit, the received data in the Receive Shift Register (RSR) is transferred to the RCREG register (if it is empty). When the transfer is complete, the RCIF interrupt flag bit is set. The actual interrupt can be enabled/disabled by setting/clearing the RCIE enable bit. The RCIF flag bit is a read only bit that is cleared by the hardware. In this case, it is cleared when the RCREG register has been read and is empty. The RCREG is a double buffered register (i.e., it is a two-deep FIFO). It is possible for two bytes of data to be received and transferred to the RCREG FIFO and a third byte to begin shifting into the RSR register. On the clocking of the last bit of the third byte, if the RCREG register is still full, then the overrun error bit, OERR, is set and the word in the RSR is lost. The RCREG register can be read twice to retrieve the two bytes in the FIFO. The OERR bit has to be cleared in software (by clearing the CREN bit). If the OERR bit is set, transfers from the RSR to the RCREG are inhibited, so it is essential to clear the OERR bit if it is set. The 9th receive bit is buffered the same way as the receive data. Reading the RCREG register will load the RX9D bit with a new value; therefore, it is essential for the user to read the RCSTA register before reading RCREG in order to not lose the old (previous) information in the RX9D bit.

Steps to follow when setting up a Synchronous Master Reception:

1. Initialize the SPBRG register for the appropriate baud rate. (Subsection 21.3 “USART Baud Rate Generator (BRG)”)
2. Enable the synchronous master serial port by setting the SYNC, SPEN and CSRC bits.
3. Ensure that the CREN and SREN bits are clear.
4. If interrupts are desired, then set the RCIE bit and configure the RCIP, GIE/GIEH and PEIE/GIEL bits, appropriately.
5. If 9-bit reception is desired, then set the RX9 bit.
6. If a single reception is required, set the SREN bit. For continuous reception, set the CREN bit.
7. The RCIF bit will be set when reception is complete and an interrupt will be generated if the RCIE bit is set.
8. Read the RCSTA register to get the ninth bit (if enabled) and determine if any error occurred during reception.
9. Read the 8-bit received data by reading the RCREG register.
10. If any error occurred, clear the error by clearing the CREN bit.

Table 21-9: Registers Associated with Synchronous Master Reception

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on all other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE/</td>
<td>PEIE/</td>
<td>TMR0IE</td>
<td>INT0IE</td>
<td>RBIE/</td>
<td>TMR0IF</td>
<td>INT0IF</td>
<td>RBIF/</td>
<td>0000 000x 0000 000x</td>
<td>0000 0000 0000 0000</td>
</tr>
<tr>
<td>PIRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>RCIF (1)</td>
<td></td>
</tr>
<tr>
<td>PIEx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000</td>
</tr>
<tr>
<td>IPRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>RCIP (1)</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 000x</td>
<td>0000 0000</td>
</tr>
<tr>
<td>RCSTA</td>
<td>SPEN</td>
<td>RX9</td>
<td>SREN</td>
<td>CREN</td>
<td>ADDEN</td>
<td>FERR</td>
<td>OERR</td>
<td>RX9D</td>
<td>0000 000x</td>
<td>0000 0000</td>
</tr>
<tr>
<td>RCREG</td>
<td>RX7</td>
<td>RX6</td>
<td>RX5</td>
<td>RX4</td>
<td>RX3</td>
<td>RX2</td>
<td>RX1</td>
<td>RX0</td>
<td>0000 000x</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TXSTA</td>
<td>CSRC</td>
<td>TX9</td>
<td>TXEN</td>
<td>SYNC</td>
<td>—</td>
<td>BRGH</td>
<td>TRMT</td>
<td>TX9D</td>
<td>0000 000–010</td>
<td>0000 0000–0010</td>
</tr>
<tr>
<td>SPBRG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
</tbody>
</table>

Legend: x = unknown, – = unimplemented read as ‘0’.

Shaded cells are not used for Synchronous Master Reception.

Note 1: The position of this bit is device dependent.
Figure 21-12: Synchronous Reception (Master Mode, SREN)

Note: Timing diagram demonstrates SYNC Master mode with SREN = '1' and BRG = '0'.
Section 21. Addressable USART

21.6 USART Synchronous Slave Mode

Synchronous slave mode differs from the Master mode in the fact that the shift clock is supplied externally at the TX/CK pin (instead of being supplied internally in Master mode). This allows the device to transfer or receive data while in SLEEP mode. Slave mode is entered by clearing the CSRC bit (TXSTA<7>).

21.6.1 USART Synchronous Slave Transmit

The operation of the Synchronous Master and Slave modes are identical, except in the case of the SLEEP mode.

If two words are written to the TXREG and then the SLEEP instruction is executed, the following will occur:

a) The first word will immediately transfer to the TSR register and transmit.

b) The second word will remain in TXREG register.

c) The TXIF flag bit will not be set.

d) When the first word has been shifted out of TSR, the TXREG register will transfer the second word to the TSR and the TXIF flag bit will now be set.

e) If the TXIE enable bit is set, the interrupt will wake the chip from SLEEP and if the global interrupt is enabled, the program will branch to the interrupt vector.

Steps to follow when setting up a Synchronous Slave Transmission:

1. Enable the synchronous slave serial port by setting the SYNC and SPEN bits and clearing the CSRC bit.

2. Clear the CREN and SREN bits.

3. If interrupts are desired, then set the TXIE enable bit and configure the RCIP, GIE/GIEH and PEIE/GIEL bits, appropriately.

4. If 9-bit transmission is desired, then set the TX9 bit.

5. Enable the transmission by setting the TXEN enable bit.

6. If 9-bit transmission is selected, the ninth bit should be loaded into the TX9D bit.

7. Start transmission by loading data to the TXREG register.

Table 21-10: Registers Associated with Synchronous Slave Transmission

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on:</th>
<th>Value on all other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCON</td>
<td>GIE/</td>
<td>PEIE/</td>
<td>TMRIIE</td>
<td>INTOIE</td>
<td>RBIE</td>
<td>TMROIF</td>
<td>INTOIF</td>
<td>RBIF</td>
<td>0000 0000x</td>
<td>0000 0000x</td>
</tr>
<tr>
<td></td>
<td>GIEH</td>
<td>GIEL</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PIRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>TXIF (1)</td>
<td>00 0</td>
</tr>
<tr>
<td>PEx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>TXIE (1)</td>
<td>0 0</td>
</tr>
<tr>
<td>IPRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>TXIP (1)</td>
<td>0 0</td>
</tr>
<tr>
<td>RCSTA</td>
<td>SPEN</td>
<td>RX9</td>
<td>SREN</td>
<td>CREN</td>
<td>ADDEN</td>
<td>FERR</td>
<td>OERR</td>
<td>RX9D</td>
<td>0000 0000x</td>
<td>0000 0000x</td>
</tr>
<tr>
<td>TXREG</td>
<td>TX7</td>
<td>TX6</td>
<td>TX5</td>
<td>TX4</td>
<td>TX3</td>
<td>TX2</td>
<td>TX1</td>
<td>TX0</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TXSTA</td>
<td>CSRC</td>
<td>TX9</td>
<td>TXEN</td>
<td>SYNC</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>SPBRG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
</tbody>
</table>

Legend: x = unknown, – = unimplemented read as '0'.

Shaded cells are not used for Synchronous Slave Transmission.

Note 1: The position of this bit is device dependent.
21.6.2 USART Synchronous Slave Reception

The operation of the Synchronous Master and Slave modes is identical, except in the case of the SLEEP mode. Also, bit SREN is a "don't care" in Slave mode.

If receive is enabled, by setting the CREN bit prior to the SLEEP instruction, then a word may be received during SLEEP. On completely receiving the word, the RSR register will transfer the data to the RCREG register and if the RCIE enable bit is set, the interrupt generated will wake the chip from SLEEP. If the global interrupt is enabled, the program will branch to the appropriate interrupt vector.

Steps to follow when setting up a Synchronous Slave Reception:

1. Enable the synchronous master serial port by setting the SYNC and SPEN bits and clearing the CSRC bit.
2. If interrupts are desired, then set the RCIE enable bit and configure the RCIP, GIE/GIEH and PEIE/GIEL bits, appropriately.
3. If 9-bit reception is desired, then set the RX9 bit.
4. To enable reception, set the CREN enable bit.
5. The RCIF bit will be set when reception is complete and an interrupt will be generated, if the RCIE bit is set.
6. Read the RCSTA register to get the ninth bit (if enabled) and determine if any error occurred during reception.
7. Read the 8-bit received data by reading the RCREG register.
8. If any error occurred, clear the error by clearing the CREN bit.

Table 21-11: Registers Associated with Synchronous Slave Reception

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on: POR, BOR</th>
<th>Value on all other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>INTCN</td>
<td>GIE/</td>
<td>GIEH</td>
<td>PEIE/</td>
<td>GIEL</td>
<td>TMR0IE</td>
<td>INT0IE</td>
<td>RBIE</td>
<td>TMR0IF</td>
<td>RBIF</td>
<td>0000 000x 0000 000x</td>
</tr>
<tr>
<td>PIRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>RCIF (1)</td>
<td></td>
</tr>
<tr>
<td>PIELx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>IPRx</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>RCIP (1)</td>
<td>0</td>
</tr>
<tr>
<td>RCSTA</td>
<td>SPEN</td>
<td>RX9</td>
<td>SREN</td>
<td>CREN</td>
<td>ADDEN</td>
<td>FERR</td>
<td>OERR</td>
<td>RX9D</td>
<td>0000 000x 0000 000x</td>
<td></td>
</tr>
<tr>
<td>RCREG</td>
<td>RX7</td>
<td>RX6</td>
<td>RX5</td>
<td>RX4</td>
<td>RX3</td>
<td>RX2</td>
<td>RX1</td>
<td>RX0</td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
<tr>
<td>TXSTA</td>
<td>CSR</td>
<td>TX9</td>
<td>TXEN</td>
<td>SYNC</td>
<td></td>
<td>BRGH</td>
<td>TRMT</td>
<td>TX9D</td>
<td>0000 -010 0000 -010</td>
<td></td>
</tr>
<tr>
<td>SPBRG</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
</tbody>
</table>

Legend: x = unknown, - = unimplemented read as '0'.
Shaded cells are not used for Synchronous Slave Reception.

Note 1: The position of this bit is device dependent.
Section 21. Addressable USART

21.7 Initialization

Example 21-2 is an initialization routine for Asynchronous Transmitter/Receiver mode. Example 21-3 is for the Synchronous mode. In both examples, the data is 8 bits, and the value to load into the SPBRG register is dependent on the desired baud rate and the device frequency. Example 21-4 is an initialization of the Addressable USART in 9-bit address detect mode.

Example 21-2: Asynchronous Transmitter/Receiver

```
MOVLW baudrate ; Set Baudrate
MOVWF SPBRG
MOVLW 0x20 ; 8-bit transmit, transmitter enabled,
MOVWF TXSTA ; asynchronous mode, low speed mode
CLRF PIR1 ; Clear all interrupt flags
; including AUSART TX & RX
BSF PIE1,TXIE ; Enable transmit interrupts
BSF PIE1,RCIE ; Enable receive interrupts
MOVLW 0x90 ; 8-bit receive, receiver enabled,
MOVWF RCSTA ; serial port enabled
```

Example 21-3: Synchronous Transmitter/Receiver

```
MOVLW baudrate ; Set Baudrate
MOVWF SPBRG
MOVLW 0xB0 ; Synchronous Master, 8-bit transmit,
MOVWF TXSTA ; transmitter enabled, low speed mode
CLRF PIR1 ; Clear all interrupt flags
; including AUSART TX & RX
BSF PIE1,TXIE ; Enable transmit interrupts
BSF PIE1,RCIE ; Enable receive interrupts
MOVLW 0x90 ; 8-bit receive, receiver enabled,
MOVWF RCSTA ; continuous receive, serial port enabled
```

Example 21-4: Asynchronous 9-bit Transmitter/Receiver (Address Detect Enabled)

```
MOVLW baudrate ; Set Baudrate
MOVWF SPBRG
MOVLW 0x60 ; 9-bit transmit, transmitter enabled,
MOVWF TXSTA ; asynchronous mode, low speed mode
CLRF PIR1 ; Clear all interrupt flags
; including AUSART TX & RX
BSF PIE1,TXIE ; Enable transmit interrupts
BSF PIE1,RCIE ; Enable receive interrupts
MOVLW 0xD8 ; 9-bit, Address Detect Enable
MOVWF RCSTA ; serial port enabled
```
21.8 Design Tips

Question 1: Using the Asynchronous mode I am getting a lot of transmission errors.
Answer 1:
The most common reasons are
1. You have incorrectly calculated the value to load into the SPBRG register.
2. The sum of the baud errors for the transmitter and receiver is too high.

Question 2: The PICmicro device is not receiving the data transmitted even though there are good levels on the Addressable USART pins.
Answer 2:
Ensure that the Address Detect Enable bit is at the desired setting.
### Section 21. Addressable USART

#### 21.9 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to this section are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Serial Port Utilities</td>
<td>AN547</td>
</tr>
<tr>
<td>Servo Control of a DC Brush Motor</td>
<td>AN532</td>
</tr>
<tr>
<td>Brush-DC Servomotor Implementation using PIC17C756A</td>
<td>AN718</td>
</tr>
</tbody>
</table>

**Note:** Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
21.10 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Addressable USART module description.
Section 22. CAN

HIGHLIGHTS

This section of the manual contains the following major topics:

22.1 Introduction ............................................................................................................... 22-2
22.2 Control Registers for the CAN Module ................................................................. 22-3
22.3 CAN Overview ........................................................................................................... 22-26
22.4 CAN Bus Features ................................................................................................... 22-32
22.5 CAN Module Implementation .................................................................................. 22-33
22.6 Frame Types ............................................................................................................. 22-37
22.7 Modes of Operation ............................................................................................... 22-44
22.8 CAN Bus Initialization ............................................................................................. 22-48
22.9 Message Reception ................................................................................................ 22-49
22.10 Transmission .......................................................................................................... 22-60
22.11 Error Detection ..................................................................................................... 22-69
22.12 Baud Rate Setting ................................................................................................. 22-71
22.13 Interrupts ............................................................................................................... 22-75
22.14 Timestamping ....................................................................................................... 22-77
22.15 CAN Module I/O ................................................................................................... 22-77
22.16 Design Tips ............................................................................................................ 22-78
22.17 Related Application Notes .................................................................................... 22-79
22.18 Revision History ................................................................................................... 22-80
22.1 Introduction

The Controller Area Network (CAN) module is a serial interface useful for communicating with other peripherals or microcontroller devices. This interface/protocol was designed to allow communications within noisy environments. Figure 22-1 shows an example CAN Bus network.

Figure 22-1: Example CAN Bus Network
22.2 Control Registers for the CAN Module

There are many registers associated with the CAN module. Descriptions of these registers are grouped into sections. These sections are:

- Control and Status Registers
- Transmit Buffer Registers
- Receive Buffer Registers
- Baud Rate Control Registers
- Interrupt Status and Control Registers

22.2.1 CAN Control and Status Registers

This section shows the CAN Control and Status registers.

Register 22-1: CANCON: CAN Control Register

<table>
<thead>
<tr>
<th>R/W-1</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>REQOP2</td>
<td>REQOP1</td>
<td>REQOP0</td>
<td>ABAT</td>
<td>WIN2</td>
<td>WIN1</td>
<td>WIN0</td>
</tr>
</tbody>
</table>

bit 7 - 5 

REQP2-REQOP0: Request CAN Operation mode bits

1xx = Request Configuration mode
011 = Request Listen Only mode
010 = Request Loopback mode
001 = Request Disable mode
000 = Request Normal mode

bit 4 

ABAT: Abort All Pending Transmissions bit

1 = Abort All Pending Transmissions (in all transmit buffers)
0 = Transmissions proceeding as normal, or all Transmissions aborted

Note: This bit will automatically be cleared when all transmissions are aborted.

bit 3 - 1 

WIN2:WIN0: Window Address bits

This selects which of the CAN buffers to switch into the access bank area. This allows access to the buffer registers from any data memory bank. After a frame has caused an interrupt, the iCODE2:iCODE0 bits can be copied to the WIN2:WIN0 bits to select the correct buffer.

111 = Receive Buffer 0
110 = Receive Buffer 0
101 = Receive Buffer 1
100 = Transmit Buffer 0
011 = Transmit Buffer 1
010 = Transmit Buffer 2
001 = Receive Buffer 0
000 = Receive Buffer 0

bit 0 

Unimplemented: Read as ‘0’

Legend

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as ‘0’
- n = Value at POR reset  ‘1’ = bit is set  ‘0’ = bit is cleared  x = bit is unknown
Register 22-2: CANSTAT: CAN Status Register

```
<table>
<thead>
<tr>
<th>R-1</th>
<th>R-0</th>
<th>R-0</th>
<th>U-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>U-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>OPMODE2</td>
<td>OPMODE1</td>
<td>OPMODE0</td>
<td>—</td>
<td>ICODE2</td>
<td>ICODE1</td>
<td>ICODE0</td>
<td>—</td>
</tr>
</tbody>
</table>
```

Bit 7

**OPMODE2:OPMODE0:** Operation Mode Status bits

- **111** = Reserved
- **110** = Reserved
- **101** = Reserved
- **100** = Configuration mode
- **011** = Listen Only mode
- **010** = Loopback mode
- **001** = Disable mode
- **000** = Normal mode

**Note:** Before the device goes into SLEEP mode, select Disable Mode.

Bit 4

**Unimplemented:** Read as '0'

Bit 3-1

**ICODE2:ICODE0:** Interrupt Code bits

When an interrupt occurs, a prioritized coded Interrupt value will be present in the ICODE2:ICODE0 bits. These codes indicate the source of the interrupt. The ICODE2:ICODE0 bits can be copied to the WIN2:WIN0 bits to select the correct buffer to map into the Access Bank area.

- **111** = Wake-up on Interrupt
- **110** = RXB0 Interrupt
- **101** = RXB1 Interrupt
- **100** = TXB0 Interrupt
- **011** = TXB1 Interrupt
- **010** = TXB2 Interrupt
- **001** = Error Interrupt
- **000** = No Interrupt

Bit 0

**Unimplemented:** Read as '0'

___

**Legend**

<table>
<thead>
<tr>
<th>R</th>
<th>W</th>
<th>U</th>
<th>n</th>
</tr>
</thead>
<tbody>
<tr>
<td>Readable bit</td>
<td>Writable bit</td>
<td>Unimplemented bit, read as ‘0’</td>
<td>Value at POR reset</td>
</tr>
</tbody>
</table>

- **’1’** = bit is set
- **’0’** = bit is cleared
- **x** = bit is unknown

---

PIC18C Reference Manual
Section 22. CAN

Register 22-3:  COMSTAT: Communication Status Register

<table>
<thead>
<tr>
<th>R/C-0</th>
<th>R/C-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>R-0</th>
<th>RX0OVFL</th>
<th>RX1OVFL</th>
<th>TXBO</th>
<th>TXBP</th>
<th>RXBP</th>
<th>TXWARN</th>
<th>RXWARN</th>
<th>EWARN</th>
</tr>
</thead>
<tbody>
<tr>
<td>bit 7</td>
<td>bit 6</td>
<td>bit 5</td>
<td>bit 4</td>
<td>bit 3</td>
<td>bit 2</td>
<td>bit 1</td>
<td>bit 0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7  RX0OVFL: Receive Buffer 0 Overflow bit
1 = Receive Buffer 0 Overflowed
0 = Receive Buffer 0 has not overflowed.

bit 6  RX1OVFL: Receive Buffer 1 Overflow bit
1 = Receive Buffer 1 Overflowed
0 = Receive Buffer 1 has not overflowed

bit 5  TXB0: Transmitter Bus Off bit
1 = Transmit Error Counter > 255
0 = Transmit Error Counter ≤ 255

bit 4  TXBP: Transmitter Bus Passive bit
1 = Transmission Error Counter > 127
0 = Transmission Error Counter ≤ 127

bit 3  RXBP: Receiver Bus Passive bit
1 = Receive Error Counter > 127
0 = Receive Error Counter ≤ 127

bit 2  TXWARN: Transmitter Warning bit
1 = Transmit Error Counter > 95
0 = Transmit Error Counter ≤ 95

bit 1  RXWARN: Receiver Warning bit
1 = Receive Error Counter > 95
0 = Receive Error Counter ≤ 95

bit 0  EWARN: Error Warning bit
This bit is a flag of the RXWARN and TXWARN bits
1 = The RXWARN or the TXWARN bits are set
0 = Neither the RXWARN or the TXWARN bits are set

Legend

R = Readable bit  C = Clearable bit  U = Unimplemented bit, read as ‘0’
- n = Value at POR reset  ’1’ = bit is set  ’0’ = bit is cleared  x = bit is unknown
22.2.2 CAN Transmit Buffer Registers

This section describes the CAN Transmit Buffer Register and the associated Transmit Buffer Control Registers.

Register 22-4: TXB0CON: Transmit Buffer 0 Control Register
TXB1CON: Transmit Buffer 1 Control Register
TXB2CON: Transmit Buffer 2 Control Register

<table>
<thead>
<tr>
<th>bit 7</th>
<th>bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>TXABT</td>
<td>TXPRI0</td>
</tr>
<tr>
<td>TXLABR</td>
<td>TXPRI1</td>
</tr>
<tr>
<td>TXERR</td>
<td>TXREQ</td>
</tr>
</tbody>
</table>

Legend:
- U = Unimplemented bit, read as '0'
- R = Readable bit
- W = Writable bit
- n = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown

bit 7: Unimplemented: Read as '0'
bit 6: TXABT: Transmission Aborted Status bit
   1 = Message was aborted
   0 = Message completed transmission successfully
bit 5: TXLABR: Transmission Lost Arbitration Status bit
   1 = Message lost arbitration while being sent
   0 = Message did not lose arbitration while being sent
bit 4: TXERR: Transmission Error detected Status bit
   1 = A bus error occurred while the message was being sent
   0 = A bus error did not occur while the message was being sent
bit 3: TXREQ: Transmit Request Status bit
   1 = Requests sending a message. Clears the TXABT, TXLABR, and TXERR bits.
   0 = Automatically cleared when the message is successfully sent

Note: Clearing this bit in software, while the bit is set will request a message abort.

bit 2: Unimplemented: Read as '0'
bit 1:0: TXPRI1:TXPRI0: Transmit Priority bits
   11 = Priority Level 3 (Highest Priority)
   10 = Priority Level 2
   01 = Priority Level 1
   00 = Priority Level 0 (Lowest Priority)
Section 22. CAN

Register 22-5: TXB0SIDH: Transmit Buffer 0 Standard Identifier High Byte Register
TXB1SIDH: Transmit Buffer 1 Standard Identifier High Byte Register
TXB2SIDH: Transmit Buffer 2 Standard Identifier High Byte Register

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>SID10</td>
<td>SID9</td>
<td>SID8</td>
<td>SID7</td>
<td>SID6</td>
<td>SID5</td>
<td>SID4</td>
<td>SID3</td>
</tr>
</tbody>
</table>

Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown

Register 22-6: TXB0SIDL: Transmit Buffer 0 Standard Identifier Low Byte Register
TXB1SIDL: Transmit Buffer 1 Standard Identifier Low Byte Register
TXB2SIDL: Transmit Buffer 2 Standard Identifier Low Byte Register

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>SID2</td>
<td>SID1</td>
<td>SID0</td>
<td>—</td>
<td>EXIDEN</td>
<td>—</td>
<td>EID17</td>
<td>EID16</td>
</tr>
</tbody>
</table>

Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown

**bit 7-0 SID10/SID3:** Standard Identifier bits

- **bit 7-5 SID2/SID0:** Standard Identifier bits
- **bit 4 Unimplemented:** Read as ‘0’
- **bit 3 EXIDEN:** Extended Identifier Enable bit
  - 1 = Message will transmit Extended ID
  - 0 = Message will transmit Standard ID
- **bit 2 Unimplemented:** Read as ‘0’
- **bit 1-0 EID17/EID16:** Extended Identifier bits
Register 22-7: TXB0EIDH: Transmit Buffer 0 Extended Identifier High Byte Register
TXB1EIDH: Transmit Buffer 1 Extended Identifier High Byte Register
TXB2EIDH: Transmit Buffer 2 Extended Identifier High Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
</tr>
</thead>
<tbody>
<tr>
<td>EID15</td>
<td>EID14</td>
<td>EID13</td>
<td>EID12</td>
<td>EID11</td>
<td>EID10</td>
<td>EID0</td>
<td>EID8</td>
</tr>
</tbody>
</table>

bit 7-0 \textbf{EID15:EID8}: Extended Identifier bits EID15 to EID8

Legend:
- \textbf{R} = Readable bit
- \textbf{W} = Writable bit
- \textbf{U} = Unimplemented bit, read as ‘0’
- -\textbf{n} = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- \textbf{x} = bit is unknown

Register 22-8: TXB0EIDL: Transmit Buffer 0 Extended Identifier Low Byte Register
TXB1EIDL: Transmit Buffer 1 Extended Identifier Low Byte Register
TXB2EIDL: Transmit Buffer 2 Extended Identifier Low Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
</tr>
</thead>
<tbody>
<tr>
<td>EID7</td>
<td>EID6</td>
<td>EID5</td>
<td>EID4</td>
<td>EID3</td>
<td>EID2</td>
<td>EID1</td>
<td>EID0</td>
</tr>
</tbody>
</table>

bit 7-0 \textbf{EID7:EID0}: Extended Identifier bits EID7 to EID0

Legend:
- \textbf{R} = Readable bit
- \textbf{W} = Writable bit
- \textbf{U} = Unimplemented bit, read as ‘0’
- -\textbf{n} = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- \textbf{x} = bit is unknown

Register 22-9: TXBnDm: Transmit Buffer 0 Data Field Byte m Register
TXB1Dm: Transmit Buffer 1 Data Field Byte m Register
TXB2Dm: Transmit Buffer 2 Data Field Byte m Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
</tr>
</thead>
<tbody>
<tr>
<td>TXBnDm7</td>
<td>TXBnDm6</td>
<td>TXBnDm5</td>
<td>TXBnDm4</td>
<td>TXBnDm3</td>
<td>TXBnDm2</td>
<td>TXBnDm1</td>
<td>TXBnDm0</td>
</tr>
</tbody>
</table>

bit 7-0 \textbf{TXBnDm7:TXBnDm0}: Transmit Buffer n Data Field Byte m bits (where 0 < n < 3 and 0 < m < 8)

Each Transmit Buffer has an array of registers. For example, Transmit buffer 0 has 7 registers: TXB0D1 to TXB0D7.
Section 22. CAN

Register 22-10: TXB0DLC: Transmit Buffer 0 Data Length Code Register
TXB1DLC: Transmit Buffer 1 Data Length Code Register
TXB2DLC: Transmit Buffer 2 Data Length Code Register

<table>
<thead>
<tr>
<th>U-0</th>
<th>R/W-x</th>
<th>U-0</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7  Unimplemented: Read as '0'

bit 6  TXRTR: Transmission Frame Remote Transmission Request bit
1 = Transmitted Message will have TXRTR bit set
0 = Transmitted Message will have TXRTR bit cleared.

bit 5-4  Unimplemented: Read as '0'

bit 3-0  DLC3:DLC0: Data Length Code bits
1111 = Reserved
1110 = Reserved
1101 = Reserved
1100 = Reserved
1011 = Reserved
1010 = Reserved
1001 = Reserved
1000 = Data Length = 8 bytes
0111 = Data Length = 7 bytes
0110 = Data Length = 6 bytes
0101 = Data Length = 5 bytes
0100 = Data Length = 4 bytes
0011 = Data Length = 3 bytes
0010 = Data Length = 2 bytes
0001 = Data Length = 1 bytes
0000 = Data Length = 0 bytes

Legend
R = Readable bit
W = Writable bit
U = Unimplemented bit, read as '0'
- n = Value at POR reset
'1' = bit is set
'0' = bit is cleared
x = bit is unknown

Register 22-11: TXERRCNT: Transmit Error Count Register

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>TEC7</td>
<td>TEC6</td>
<td>TEC5</td>
<td>TEC4</td>
<td>TEC3</td>
<td>TEC2</td>
<td>TEC1</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7-0  TEC7:TEC0: Transmit Error Counter bits
This register contains a value which is derived from the rate at which errors occur. When the error count overflows, the bus off state occurs. When the bus has an occurrence of 11 consecutive recessive bits, the counter value is cleared.

Legend
R = Readable bit
W = Writable bit
U = Unimplemented bit, read as '0'
- n = Value at POR reset
'1' = bit is set
'0' = bit is cleared
x = bit is unknown
22.2.3 CAN Receive Buffer Registers

This section shows the Receive buffer registers with their associated control registers.

Register 22-12: RXB0CON: Receive Buffer 0 Control Register

<table>
<thead>
<tr>
<th>R/C-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>RXFUL</td>
<td>RXM1</td>
<td>RXM0</td>
<td>---</td>
<td>RXRTRRO</td>
<td>JTOFF</td>
<td>FILHIT0</td>
</tr>
</tbody>
</table>

bit 7  
RXFUL: Receive Full status bit
1 = Receive buffer contains a valid received message
0 = Receive buffer is open to receive a new message

Note: This bit is set by the CAN module and should be cleared by software after the buffer is read.

bit 6-5  
RXM1:RXM0: Receive Buffer Mode bits
11 = Receive all messages (including those with errors)
10 = Receive only valid messages with extended identifier
01 = Receive only valid messages with standard identifier
00 = Receive all valid messages

bit 4  
Unimplemented: Read as ‘0’

bit 3  
RXRTRRO: Receive Remote Transfer Request Read Only bit
1 = Remote Transfer Request
0 = No Remote Transfer Request

bit 2  
RX0DBEN: Receive Buffer 0 Double Buffer Enable bit
1 = Receive Buffer 0 overflow will write to Receive Buffer 1
0 = No Receive Buffer 0 overflow to Receive Buffer 1

bit 1  
JTOFF: Jump Table offset bit
1 = Allows Jump Table offset between 6 and 7
0 = Allows Jump Table offset between 1 and 0

bit 0  
FILHIT0: Filter Hit bit
This bit indicates which acceptance filter enabled the message reception into receive buffer 0.
1 = Acceptance Filter 1 (RXF1)
0 = Acceptance Filter 0 (RXF0)

Legend:

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as ‘0’
C = Clearable bit
- n = Value at POR reset  ‘1’ = bit is set  ‘0’ = bit is cleared  x = bit is unknown
Register 22-13: RXB1CON: Receive Buffer 1 Control Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Name</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>RXFUL</td>
<td>Receive Full status bit&lt;br&gt;1 = Receive buffer contains a valid received message&lt;br&gt;0 = Receive buffer is open to receive a new message&lt;br&gt;Note: This bit is set by the CAN module and should be cleared by software after the buffer is read.</td>
</tr>
<tr>
<td>6-5</td>
<td>RXM1:RXM0</td>
<td>Receive Buffer Mode bits&lt;br&gt;11 = Receive all messages (including those with errors)&lt;br&gt;10 = Receive only valid messages with extended identifier&lt;br&gt;01 = Receive only valid messages with standard identifier&lt;br&gt;00 = Receive all valid messages</td>
</tr>
<tr>
<td>4</td>
<td></td>
<td>Unimplemented: Read as '0'</td>
</tr>
<tr>
<td>3</td>
<td>RXRTRRO</td>
<td>Receive Remote Transfer Request bit (read only)&lt;br&gt;1 = Remote Transfer Request&lt;br&gt;0 = No Remote Transfer Request</td>
</tr>
<tr>
<td>2-0</td>
<td>FILHIT2:FILHIT0</td>
<td>Filter Hit bits&lt;br&gt;These bits indicate which acceptance filter enabled the last message reception into Receive Buffer 1&lt;br&gt;111 = Reserved&lt;br&gt;110 = Reserved&lt;br&gt;101 = Acceptance Filter 5 (RXF5)&lt;br&gt;100 = Acceptance Filter 4 (RXF4)&lt;br&gt;011 = Acceptance Filter 3 (RXF3)&lt;br&gt;010 = Acceptance Filter 2 (RXF2)&lt;br&gt;001 = Acceptance Filter 1 (RXF1) only possible when RX0DBEN bit is set&lt;br&gt;000 = Acceptance Filter 0 (RXF0) only possible when RX0DBEN bit is set</td>
</tr>
</tbody>
</table>

Legend<br>R = Readable bit<br>W = Writable bit<br>U = Unimplemented bit, read as '0'<br>C = Clearable bit<br>- n = Value at POR reset<br>'1' = bit is set<br>'0' = bit is cleared<br>x = bit is unknown
Register 22-14: RXB0SIDH: Receive Buffer 0 Standard Identifier High Byte Register
RXB1SIDH: Receive Buffer 1 Standard Identifier High Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>SID10</th>
<th>SID9</th>
<th>SID8</th>
<th>SID7</th>
<th>SID6</th>
<th>SID5</th>
<th>SID4</th>
<th>SID3</th>
</tr>
</thead>
</table>

Legend

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as '0'
- n = Value at POR reset  '1' = bit is set  '0' = bit is cleared  x = bit is unknown

Register 22-15: RXB0SIDL: Receive Buffer 0 Standard Identifier Low Byte Register
RXB1SIDL: Receive Buffer 1 Standard Identifier Low Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>SID2</th>
<th>SID1</th>
<th>SID0</th>
<th>SRR</th>
<th>EXID</th>
<th>—</th>
<th>EID17</th>
<th>EID16</th>
</tr>
</thead>
</table>

Legend

R = Readable bit  W = Writable bit  U = Unimplemented bit, read as '0'
- n = Value at POR reset  '1' = bit is set  '0' = bit is cleared  x = bit is unknown
### Section 22. CAN

#### Register 22-16: RXB0EIDH: Receive Buffer 0 Extended Identifier High Byte Register

<table>
<thead>
<tr>
<th>Bit 7-0</th>
<th>EID15: EID8: Extended Identifier bits EID15 to EID8</th>
</tr>
</thead>
</table>

#### Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **- n** = Value at POR reset
  - '1' = bit is set
  - '0' = bit is cleared
- **x** = bit is unknown

#### Register 22-17: RXB0EIDL: Transmit Buffer 0 Extended Identifier Low Byte Register

<table>
<thead>
<tr>
<th>Bit 7-0</th>
<th>EID7: EID0: Extended Identifier bits EID7 to EID0</th>
</tr>
</thead>
</table>

#### Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **- n** = Value at POR reset
  - '1' = bit is set
  - '0' = bit is cleared
- **x** = bit is unknown
Register 22-18: RXB0DLC: Receive Buffer 0 Data Length Code Register
RXB1DLC: Receive Buffer 1 Data Length Code Register

<table>
<thead>
<tr>
<th>U-x</th>
<th>R/W-x</th>
<th>R-x</th>
<th>R-x</th>
<th>R-x</th>
<th>R-x</th>
<th>R-x</th>
<th>R-x</th>
<th>R-x</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>RXRTR</td>
<td>RB1</td>
<td>RB0</td>
<td>DLC3</td>
<td>DLC2</td>
<td>DLC1</td>
<td>DLC0</td>
<td></td>
</tr>
</tbody>
</table>

bit 7
- **Unimplemented**: Read as '0'

bit 6
- **RXRTR**: Receiver Remote Transmission Request bit
  - 1 = Remote Transfer Request
  - 0 = No Remote Transfer Request

bit 5
- **RB1**: Reserved bit 1
  - Reserved by CAN Specification and read as '0'

bit 4
- **RB0**: Reserved bit 0
  - Reserved by CAN Specification and read as '0'

bit 3-0
- **DLC3:DLC0**: Data Length Code bits
  - 1111 = Invalid
  - 1110 = Invalid
  - 1101 = Invalid
  - 1100 = Invalid
  - 1011 = Invalid
  - 1010 = Invalid
  - 1001 = Invalid
  - 1000 = Data Length = 8 bytes
  - 0111 = Data Length = 7 bytes
  - 0110 = Data Length = 6 bytes
  - 0101 = Data Length = 5 bytes
  - 0100 = Data Length = 4 bytes
  - 0011 = Data Length = 3 bytes
  - 0010 = Data Length = 2 bytes
  - 0001 = Data Length = 1 byte
  - 0000 = Data Length = 0 byte

Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- - n = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown
Section 22. CAN

Register 22-19: RXB0Dm: Receive Buffer 0 Data Field Byte m Register
RXB1Dm: Receive Buffer 1 Data Field Byte m Register

R/W-x R/W-x R/W-x R/W-x R/W-x R/W-x R/W-x R/W-x
RXBnDm7 RXBnDm6 RXBnDm5 RXBnDm4 RXBnDm3 RXBnDm2 RXBnDm1 RXBnDm0

bit 7-0 RXBnDm7:RXBnDm0: Receive Buffer n Data Field Byte m bits (where 0 < n < 1 and 0 < m < 7)
Each Receive Buffer has an array of registers. For example Receive buffer 0 has 7 registers: RXB0D1 to RXB0D7.

Legend
R = Readable bit W = Writable bit U = Unimplemented bit, read as ‘0’
- n = Value at POR reset ‘1’ = bit is set ‘0’ = bit is cleared x = bit is unknown

Register 22-20: RXERRCNT: Receive Error Count Register

R-0 R-0 R-0 R-0 R-0 R-0 R-0 R-0
REC7 REC6 REC5 REC4 REC3 REC2 REC1 REC0

bit 7-0 REC7:REC0: Receive Error Counter bits
This register contains the number of errors that occurred for the Reception of this buffers message.
When the error count overflows, the bus off state occurs. When the bus has 256 occurrences of 11 consecutive recessive bits, the counter value is cleared.

Legend
R = Readable bit W = Writable bit U = Unimplemented bit, read as ‘0’
- n = Value at POR reset ‘1’ = bit is set ‘0’ = bit is cleared x = bit is unknown
### 22.2.4 Message Acceptance Filters

This subsection describes the Message Acceptance filters.

#### Register 22-21: RXF0SIDH: Receive Acceptance Filter 0 Std. Identifier Filter High Byte
- RXF1SIDH: Receive Acceptance Filter 1 Std. Identifier Filter High Byte
- RXF2SIDH: Receive Acceptance Filter 2 Std. Identifier Filter High Byte
- RXF3SIDH: Receive Acceptance Filter 3 Std. Identifier Filter High Byte
- RXF4SIDH: Receive Acceptance Filter 4 Std. Identifier Filter High Byte
- RXF5SIDH: Receive Acceptance Filter 5 Std. Identifier Filter High Byte

| Bit 7-0 | SID10:SID3: Standard Identifier Filter bits SID10 to SID3
|--------|-------------------------------------------------|

---

### Register 22-22: RXF0SIDL: Receive Acceptance Filter 0 Std. Identifier Filter Low Byte
- RXF1SIDL: Receive Acceptance Filter 1 Std. Identifier Filter Low Byte
- RXF2SIDL: Receive Acceptance Filter 2 Std. Identifier Filter Low Byte
- RXF3SIDL: Receive Acceptance Filter 3 Std. Identifier Filter Low Byte
- RXF4SIDL: Receive Acceptance Filter 4 Std. Identifier Filter Low Byte
- RXF5SIDL: Receive Acceptance Filter 5 Std. Identifier Filter Low Byte

| Bit 7-0 | SID2:SID0: Standard Identifier Filter bits SID2 to SID0
|--------|-------------------------------------------------|

---

| Bit 4 | Unimplemented: Read as '0'
|------|-------------------------------------------------|
| Bit 3 | EXIDEN: Extended Identifier Filter Enable bit
- 1 = Message will transmit Extended ID
- 0 = Message will not transmit Extended ID.
| Bit 2 | Unimplemented: Read as '0'
| Bit 1-0 | EID17:EID16: Extended Identifier Filter bits EID17 to EID16

---

Legend:
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **- n** = Value at POR reset
- **'1'** = bit is set
- **'0'** = bit is cleared
- **x** = bit is unknown
Section 22. CAN

Register 22-23: RXF0EIDH: Receive Acceptance Filter 0 Extended Identifier High Byte
RXF1EIDH: Receive Acceptance Filter 1 Extended Identifier High Byte
RXF2EIDH: Receive Acceptance Filter 2 Extended Identifier High Byte
RXF3EIDH: Receive Acceptance Filter 3 Extended Identifier High Byte
RXF4EIDH: Receive Acceptance Filter 4 Extended Identifier High Byte
RXF5EIDH: Receive Acceptance Filter 5 Extended Identifier High Byte

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>EID15</th>
<th>EID14</th>
<th>EID13</th>
<th>EID12</th>
<th>EID11</th>
<th>EID10</th>
<th>EID9</th>
<th>EID8</th>
</tr>
</thead>
</table>

**Legend**

R = Readable bit  
W = Writable bit  
U = Unimplemented bit, read as ‘0’

- n = Value at POR reset  
  ‘1’ = bit is set  
  ‘0’ = bit is cleared  
  x = bit is unknown

Register 22-24: RXB0EIDL: Receive Buffer 0 Extended Identifier Low Byte Register
RXB1EIDL: Receive Buffer 1 Extended Identifier Low Byte Register
RXB2EIDL: Receive Buffer 2 Extended Identifier Low Byte Register
RXB3EIDL: Receive Buffer 3 Extended Identifier Low Byte Register
RXB4EIDL: Receive Buffer 4 Extended Identifier Low Byte Register
RXB5EIDL: Receive Buffer 5 Extended Identifier Low Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>EID7</th>
<th>EID6</th>
<th>EID5</th>
<th>EID4</th>
<th>EID3</th>
<th>EID2</th>
<th>EID1</th>
<th>EID0</th>
</tr>
</thead>
</table>

**Legend**

R = Readable bit  
W = Writable bit  
U = Unimplemented bit, read as ‘0’

- n = Value at POR reset  
  ‘1’ = bit is set  
  ‘0’ = bit is cleared  
  x = bit is unknown
### Register 22-25: RXM0SIDH: Receive Acceptance Mask 0 Std. Identifier Mask High Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>SID10</th>
<th>SID9</th>
<th>SID8</th>
<th>SID7</th>
<th>SID6</th>
<th>SID5</th>
<th>SID4</th>
<th>SID3</th>
</tr>
</thead>
<tbody>
<tr>
<td>bit 7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7-0 **SID10:SID3:** Standard Identifier Mask bits SID10 to SID3

### Register 22-26: RXM0SIDL: Receive Acceptance Mask 0 Std. Identifier Mask Low Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>SID2</th>
<th>SID1</th>
<th>SID0</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>—</th>
<th>EID17</th>
<th>EID16</th>
</tr>
</thead>
<tbody>
<tr>
<td>bit 7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7-5 **SID2:SID0:** Standard Identifier Mask bits SID2 to SID0

bit 4-2 **Unimplemented:** Read as ‘0’

bit 1-0 **EID17:EID16:** Extended Identifier Mask bits EID17 to EID16

Legend

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **n** = Value at POR reset
- **’1’** = bit is set
- **’0’** = bit is cleared
- **x** = bit is unknown
Section 22. CAN

Register 22-27: RXM0EIDH: Receive Acceptance Mask 0 Extended Identifier Mask High Byte Register
RXM1EIDH: Receive Acceptance Mask 1 Extended Identifier Mask High Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
</tr>
</thead>
<tbody>
<tr>
<td>EID15</td>
<td>EID14</td>
<td>EID13</td>
<td>EID12</td>
<td>EID11</td>
<td>EID10</td>
<td>EID9</td>
<td>EID8</td>
</tr>
</tbody>
</table>

bit 7 bit 0

EID15:EID8: Extended Identifier Mask bits EID15 to EID8

Legend
R = Readable bit   W = Writable bit   U = Unimplemented bit, read as ‘0’
- n = Value at POR reset   ‘1’ = bit is set   ‘0’ = bit is cleared   x = bit is unknown

Register 22-28: RXM0EIDL: Receive Acceptance Mask 0 Extended Identifier Mask Low Byte Register
RXM1EIDL: Receive Acceptance Mask 1 Extended Identifier Mask Low Byte Register

<table>
<thead>
<tr>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
<th>R/W-x</th>
</tr>
</thead>
<tbody>
<tr>
<td>EID7</td>
<td>EID6</td>
<td>EID5</td>
<td>EID4</td>
<td>EID3</td>
<td>EID2</td>
<td>EID1</td>
<td>EID0</td>
</tr>
</tbody>
</table>

bit 7 bit 0

EID7:EID0: Extended Identifier Mask bits EID7 to EID0

Legend
R = Readable bit   W = Writable bit   U = Unimplemented bit, read as ‘0’
- n = Value at POR reset   ‘1’ = bit is set   ‘0’ = bit is cleared   x = bit is unknown
### 22.2.5 CAN Baud Rate Registers

This subsection describes the CAN baud rate registers.

#### Register 22-29: BRGCON1: Baud Rate Control Register 1

<table>
<thead>
<tr>
<th>Bit 7-6</th>
<th>SJW1: SJW0: Synchronized Jump Width bits</th>
</tr>
</thead>
<tbody>
<tr>
<td>11</td>
<td>Synchronization Jump Width Time = 4 x Tq</td>
</tr>
<tr>
<td>10</td>
<td>Synchronization Jump Width Time = 3 x Tq</td>
</tr>
<tr>
<td>01</td>
<td>Synchronization Jump Width Time = 2 x Tq</td>
</tr>
<tr>
<td>00</td>
<td>Synchronization Jump Width Time = 1 x Tq</td>
</tr>
</tbody>
</table>

#### Bit 5-0: BRP5:BRP0: Baud Rate Prescaler bits

<table>
<thead>
<tr>
<th>Value</th>
<th>Baud Rate Prescaler</th>
</tr>
</thead>
<tbody>
<tr>
<td>11111</td>
<td>Tq = (2 x 64)/Fosc</td>
</tr>
<tr>
<td>11110</td>
<td>Tq = (2 x 63)/Fosc</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>00001</td>
<td>Tq = (2 x 2)/Fosc</td>
</tr>
<tr>
<td>00000</td>
<td>Tq = (2 x 1)/Fosc</td>
</tr>
</tbody>
</table>

**Legend**
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- -n = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown

**Note:** This register is only accessible in configuration mode (see Section 22.7.1).
### Section 22. CAN

**Register 22-30: BRGCON2: Baud Rate Control Register 2**

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5-3</th>
<th>Bit 2-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>SEG2PHTS: Phase Segment 2 Time Select bit</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Freely programmable</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Maximum of PHEG1 or Information Processing Time (IPT), whichever is greater</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SAM: Sample of the CAN bus line bit</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 = Bus line is sampled three times at the sample point</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 = Bus line is sampled once at the sample point</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SEG1PH2:SEG1PH0: Phase segment 1 bits</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>111 = Phase segment 1 Time = 8 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>110 = Phase segment 1 Time = 7 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>101 = Phase segment 1 Time = 6 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>100 = Phase segment 1 Time = 5 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>011 = Phase segment 1 Time = 4 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>010 = Phase segment 1 Time = 3 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>001 = Phase segment 1 Time = 2 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>000 = Phase segment 1 Time = 1 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PRSEG2:PRSEG0: Propagation Time Select bits</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>111 = Propagation Time = 8 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>110 = Propagation Time = 7 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>101 = Propagation Time = 6 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>100 = Propagation Time = 5 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>011 = Propagation Time = 4 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>010 = Propagation Time = 3 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>001 = Propagation Time = 2 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>000 = Propagation Time = 1 x TQ</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

**Legend**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **n** = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown

**Note:** This register is only accessible in configuration mode (see Section 22.7.1).
## Register 22-31: BRGCON3: Baud Rate Control Register 3

<table>
<thead>
<tr>
<th>bit 7</th>
<th>bit 6</th>
<th>bit 5-3</th>
<th>bit 2-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>—</td>
<td>—</td>
<td>—</td>
<td>WAKFIL</td>
</tr>
<tr>
<td>U-0</td>
<td>R/W-0</td>
<td>U-0</td>
<td>—</td>
</tr>
<tr>
<td>U-0</td>
<td>U-0</td>
<td>U-0</td>
<td>SEG2PH2</td>
</tr>
<tr>
<td>R/W-0</td>
<td>R/W-0</td>
<td>R/W-0</td>
<td>SEG2PH1</td>
</tr>
<tr>
<td>R/W-0</td>
<td>R/W-0</td>
<td>R/W-0</td>
<td>SEG2PH0</td>
</tr>
</tbody>
</table>

**bit 7 Unimplemented**: Read as '0'

**bit 6 WAKFIL**: Selects CAN Bus Line Filter for Wake-up bit
- 1 = Use CAN bus line filter for wake-up
- 0 = CAN bus line filter is not used for wake-up

**bit 5-3 Unimplemented**: Read as '0'

**bit 2-0 SEG2PH2:SEG2PH0**: Phase Segment 2 Time Select bits
- 111 = Phase Segment 2 Time = 8 x TQ
- 110 = Phase Segment 2 Time = 7 x TQ
- 101 = Phase Segment 2 Time = 6 x TQ
- 100 = Phase Segment 2 Time = 5 x TQ
- 011 = Phase Segment 2 Time = 4 x TQ
- 010 = Phase Segment 2 Time = 3 x TQ
- 001 = Phase Segment 2 Time = 2 x TQ
- 000 = Phase Segment 2 Time = 1 x TQ

**Note**: Ignored if SEG2PHTS bit is clear.

---

**Legend**
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as '0'
- - n = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown
## Section 22. CAN

### 22.2.6 CAN Module I/O Control Register

This subsection describes the CAN Module I/O Control register.

**Register 22-32: CIOCON: CAN I/O Control Register**

<table>
<thead>
<tr>
<th>Bit 7</th>
<th>TX1SRC: TX1 Pin Data Source</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>TX1 pin will output the CAN clock</td>
</tr>
<tr>
<td>0</td>
<td>TX1 pin will output TXD</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Bit 6</th>
<th>TX1EN: TX1 Pin Enable</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>TX1 pin will output TXD or CAN clock</td>
</tr>
<tr>
<td>0</td>
<td>TX1 pin will have digital I/O function</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Bit 5</th>
<th>ENDRHI: Enable Drive High</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>TX0, TX1 pins will drive Vdd when recessive</td>
</tr>
<tr>
<td>0</td>
<td>TX0, TX1 pins will tri-state when recessive</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Bit 4</th>
<th>CANCAP: CAN Message Receive Capture Enable</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>Enable CAN capture</td>
</tr>
<tr>
<td>0</td>
<td>Disable CAN capture</td>
</tr>
</tbody>
</table>

| Bit 3-0 | Unimplemented: Read as '0' |

---

**Legend**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **n** = Value at POR reset
- **'1'** = bit is set
- **'0'** = bit is cleared
- **x** = bit is unknown
### 22.2.7 CAN Interrupt Registers

This subsection documents the CAN Registers which are associated to Interrupts.

**Register 22-33: PIR3: Peripheral Interrupt Flag Register**

<table>
<thead>
<tr>
<th>Bit</th>
<th>Function</th>
<th>Value</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>IRXIF: CAN Invalid Received message Interrupt Flag bit</td>
<td>1</td>
<td>An invalid message has occurred on the CAN bus</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>No invalid message on CAN bus</td>
</tr>
<tr>
<td>6</td>
<td>WAKIF: CAN Bus Activity Wake-up Interrupt Flag bit</td>
<td></td>
<td>Activity on CAN bus has occurred</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>No activity on CAN bus</td>
</tr>
<tr>
<td>5</td>
<td>ERRIF: CAN bus Error Interrupt Flag bit</td>
<td>1</td>
<td>An error has occurred in the CAN module (multiple sources)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>No CAN module errors</td>
</tr>
<tr>
<td>4</td>
<td>TXB2IF: CAN Transmit Buffer 2 Interrupt Flag bit</td>
<td>1</td>
<td>Transmit Buffer 2 has completed transmission of a message, and may be re-loaded</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>Transmit Buffer 2 has not completed transmission of a message</td>
</tr>
<tr>
<td>3</td>
<td>TXB1IF: CAN Transmit Buffer 1 Interrupt Flag bit</td>
<td>1</td>
<td>Transmit Buffer 1 has completed transmission of a message, and may be re-loaded</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>Transmit Buffer 1 has not completed transmission of a message</td>
</tr>
<tr>
<td>2</td>
<td>TXB0IF: CAN Transmit Buffer 0 Interrupt Flag bit</td>
<td>1</td>
<td>Transmit Buffer 0 has completed transmission of a message, and may be re-loaded</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>Transmit Buffer 0 has not completed transmission of a message</td>
</tr>
<tr>
<td>1</td>
<td>RXB1IF: CAN Receive Buffer 1 Interrupt Flag bit</td>
<td>1</td>
<td>Receive Buffer 1 has received a new message</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>Receive Buffer 1 has not received a new message</td>
</tr>
<tr>
<td>0</td>
<td>RXB0IF: CAN Receive Buffer 0 Interrupt Flag bit</td>
<td>1</td>
<td>Receive Buffer 0 has received a new message</td>
</tr>
<tr>
<td></td>
<td></td>
<td>0</td>
<td>Receive Buffer 0 has not received a new message</td>
</tr>
</tbody>
</table>

**Legend**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **-n** = Value at POR reset
- **'1'** = bit is set
- **'0'** = bit is cleared
- **x** = bit is unknown
### Register 22-34: PIE3: Peripheral Interrupt Enable Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>Name</th>
<th>Description</th>
<th>Value at POR</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>IRXIE</td>
<td>CAN Invalid Received Message Interrupt Enable</td>
<td>1</td>
<td>Enable invalid message received interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable invalid message received interrupt</td>
</tr>
<tr>
<td>6</td>
<td>WAKIE</td>
<td>CAN Bus Activity Wake-up Interrupt Enable</td>
<td>1</td>
<td>Enable Bus Activity Wake-up Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable Bus Activity Wake-up Interrupt</td>
</tr>
<tr>
<td>5</td>
<td>ERRIE</td>
<td>CAN Bus Error Interrupt Enable</td>
<td>1</td>
<td>Enable CAN bus Error Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable CAN bus Error Interrupt</td>
</tr>
<tr>
<td>4</td>
<td>TXB2IE</td>
<td>CAN Transmit Buffer 2 Interrupt Enable</td>
<td>1</td>
<td>Enable Transmit Buffer 2 Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable Transmit Buffer 2 Interrupt</td>
</tr>
<tr>
<td>3</td>
<td>TXB1IE</td>
<td>CAN Transmit Buffer 1 Interrupt Enable</td>
<td>1</td>
<td>Enable Transmit Buffer 1 Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable Transmit Buffer 1 Interrupt</td>
</tr>
<tr>
<td>2</td>
<td>TXB0IE</td>
<td>CAN Transmit Buffer 0 Interrupt Enable</td>
<td>1</td>
<td>Enable Transmit Buffer 0 Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable Transmit Buffer 0 Interrupt</td>
</tr>
<tr>
<td>1</td>
<td>RXB1IE</td>
<td>CAN Receive Buffer 1 Interrupt Enable</td>
<td>1</td>
<td>Enable Receive Buffer 1 Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable Receive Buffer 1 Interrupt</td>
</tr>
<tr>
<td>0</td>
<td>RXB0IE</td>
<td>CAN Receive Buffer 0 Interrupt Enable</td>
<td>1</td>
<td>Enable Receive Buffer 0 Interrupt</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>Disable Receive Buffer 0 Interrupt</td>
</tr>
</tbody>
</table>

**Legend**

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **n** = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown
Register 22-35: IPR3: Peripheral Interrupt Priority Register

<p>| | | | | | | | |</p>
<table>
<thead>
<tr>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>IRXIP</td>
<td>WAKIP</td>
<td>ERRIP</td>
<td>TXB2IP</td>
<td>TXB1IP</td>
<td>TXB0IP</td>
<td>RXB1IP</td>
</tr>
<tr>
<td>bit 7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>High Priority</td>
<td>1</td>
<td>High Priority</td>
<td>1</td>
<td>High Priority</td>
<td>1</td>
<td>High Priority</td>
</tr>
<tr>
<td>0</td>
<td>Low Priority</td>
<td>0</td>
<td>Low Priority</td>
<td>0</td>
<td>Low Priority</td>
<td>0</td>
<td>Low Priority</td>
</tr>
<tr>
<td>bit 6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>bit 0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Legend:
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- n = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown
### Section 22. CAN

#### Table 22-1: CAN Controller Register Map

<table>
<thead>
<tr>
<th>Address</th>
<th>Register Name</th>
<th>Address</th>
<th>Register Name</th>
<th>Address</th>
<th>Register Name</th>
<th>Address</th>
<th>Register Name</th>
<th>Address</th>
<th>Register Name</th>
</tr>
</thead>
<tbody>
<tr>
<td>F7FH</td>
<td>—</td>
<td>F5FH</td>
<td>—</td>
<td>F3FH</td>
<td>—</td>
<td>F1FH</td>
<td>RXM1EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F7EH</td>
<td>—</td>
<td>F5EH</td>
<td>CANSTATRO1</td>
<td>F3EH</td>
<td>CANSTATRO3</td>
<td>F1EH</td>
<td>RXM1EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F7DH</td>
<td>—</td>
<td>F5DH</td>
<td>RXB1D7</td>
<td>F3DH</td>
<td>TXB1D7</td>
<td>F1DH</td>
<td>RXM1SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F7CH</td>
<td>—</td>
<td>F5CH</td>
<td>RXB1D6</td>
<td>F3CH</td>
<td>TXB1D6</td>
<td>F1CH</td>
<td>RXM1SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F7BH</td>
<td>—</td>
<td>F5BH</td>
<td>RXB1D5</td>
<td>F3BH</td>
<td>TXB1D5</td>
<td>F1BH</td>
<td>RXM0EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F7AH</td>
<td>—</td>
<td>F5AH</td>
<td>RXB1D4</td>
<td>F3AH</td>
<td>TXB1D4</td>
<td>F1AH</td>
<td>RXM0EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F79H</td>
<td>—</td>
<td>F59H</td>
<td>RXB1D3</td>
<td>F39H</td>
<td>TXB1D3</td>
<td>F19H</td>
<td>RXM0SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F78H</td>
<td>—</td>
<td>F58H</td>
<td>RXB1D2</td>
<td>F38H</td>
<td>TXB1D2</td>
<td>F18H</td>
<td>RXM0SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F77H</td>
<td>—</td>
<td>F57H</td>
<td>RXB1D1</td>
<td>F37H</td>
<td>TXB1D1</td>
<td>F17H</td>
<td>RXF5EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F76H</td>
<td>TXERRCNT</td>
<td>F56H</td>
<td>RXB1D0</td>
<td>F36H</td>
<td>TXB1D0</td>
<td>F16H</td>
<td>RXF5EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F75H</td>
<td>RXERRCNT</td>
<td>F55H</td>
<td>RXB1DLC</td>
<td>F35H</td>
<td>TXB1DLC</td>
<td>F15H</td>
<td>RXF5SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F74H</td>
<td>CIOCON</td>
<td>F54H</td>
<td>RXB1EIDH</td>
<td>F34H</td>
<td>TXB1EIDH</td>
<td>F14H</td>
<td>RXF5SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F73H</td>
<td>CIOCON</td>
<td>F53H</td>
<td>RXB1EIDL</td>
<td>F33H</td>
<td>TXB1EIDL</td>
<td>F13H</td>
<td>RXF4EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F72H</td>
<td>BRGCON3</td>
<td>F52H</td>
<td>RXB1SIDL</td>
<td>F32H</td>
<td>TXB1SIDL</td>
<td>F12H</td>
<td>RXF4EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F71H</td>
<td>BRGCON2</td>
<td>F51H</td>
<td>RXB1SIDH</td>
<td>F31H</td>
<td>TXB1SIDH</td>
<td>F11H</td>
<td>RXF4SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F70H</td>
<td>BRGCON1</td>
<td>F50H</td>
<td>RXB1CON</td>
<td>F30H</td>
<td>TXB1CON</td>
<td>F10H</td>
<td>RXF4SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F6FH</td>
<td>CANCON</td>
<td>F4FH</td>
<td>—</td>
<td>F2FH</td>
<td>—</td>
<td>F0FH</td>
<td>RXF3EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F6EH</td>
<td>CANSTAT</td>
<td>F4EH</td>
<td>CANSTATRO2</td>
<td>F2EH</td>
<td>CANSTATRO4</td>
<td>F0EH</td>
<td>RXF3EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F6DH</td>
<td>RXB0D7</td>
<td>F4DH</td>
<td>TXB0D7</td>
<td>F2DH</td>
<td>TXB2D7</td>
<td>F0DH</td>
<td>RXF3SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F6CH</td>
<td>RXB0D6</td>
<td>F4CH</td>
<td>TXB0D6</td>
<td>F2CH</td>
<td>TXB2D6</td>
<td>F0CH</td>
<td>RXF3SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F6BH</td>
<td>RXB0D5</td>
<td>F4BH</td>
<td>TXB0D5</td>
<td>F2BH</td>
<td>TXB2D5</td>
<td>F0BH</td>
<td>RXF2EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F6AH</td>
<td>RXB0D4</td>
<td>F4AH</td>
<td>TXB0D4</td>
<td>F2AH</td>
<td>TXB2D4</td>
<td>F0AH</td>
<td>RXF2EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F69H</td>
<td>RXB0D3</td>
<td>F49H</td>
<td>TXB0D3</td>
<td>F29H</td>
<td>TXB2D3</td>
<td>F09H</td>
<td>RXF2SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F68H</td>
<td>RXB0D2</td>
<td>F48H</td>
<td>TXB0D2</td>
<td>F28H</td>
<td>TXB2D2</td>
<td>F08H</td>
<td>RXF2SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F67H</td>
<td>RXB0D1</td>
<td>F47H</td>
<td>TXB0D1</td>
<td>F27H</td>
<td>TXB2D1</td>
<td>F07H</td>
<td>RXE1IDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F66H</td>
<td>RXB0D0</td>
<td>F46H</td>
<td>TXB0D0</td>
<td>F26H</td>
<td>TXB2D0</td>
<td>F06H</td>
<td>RXE1IDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F65H</td>
<td>RXB0DLC</td>
<td>F45H</td>
<td>TXB0DLC</td>
<td>F25H</td>
<td>TXB2DLC</td>
<td>F05H</td>
<td>RXF1SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F64H</td>
<td>RXB0EIDL</td>
<td>F44H</td>
<td>TXB0EIDL</td>
<td>F24H</td>
<td>TXB2EIDL</td>
<td>F04H</td>
<td>RXF1SIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F63H</td>
<td>RXB0EIDH</td>
<td>F43H</td>
<td>TXB0EIDH</td>
<td>F23H</td>
<td>TXB2EIDH</td>
<td>F03H</td>
<td>RXF0EIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F62H</td>
<td>RXB0SIDL</td>
<td>F42H</td>
<td>TXB0SIDL</td>
<td>F22H</td>
<td>TXB2SIDL</td>
<td>F02H</td>
<td>RXF0EIDH</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F61H</td>
<td>RXB0SIDH</td>
<td>F41H</td>
<td>TXB0SIDH</td>
<td>F21H</td>
<td>TXB2SIDH</td>
<td>F01H</td>
<td>RXF0SIDL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F60H</td>
<td>RXB0CON</td>
<td>F40H</td>
<td>TXB0CON</td>
<td>F20H</td>
<td>TXB2CON</td>
<td>F00H</td>
<td>RXF0SIDH</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

**Note:** The shaded addresses indicate the registers that are in the access RAM.
22.3 CAN Overview

The Controller Area Network (CAN) is a serial communications protocol which efficiently supports distributed real-time control with a very high level of robustness. The Protocol is fully defined by Robert Bosch GmbH, in the CAN Specification V2.0B from 1991.

Its domain of application ranges from high speed networks to low cost multiplex wiring. In automotive electronics; (engine control units, sensors, anti-skid-systems, etc.) are connected using CAN with bit rates up to 1 Mbit/sec. The CAN Network allows a cost effective replacement of the wiring harnesses in the automobile. The robustness of the bus in noisy environments and the ability to detect and recover from fault conditions makes the bus suitable for industrial control applications such as DeviceNet, SDS and other fieldbus protocols.

CAN is an asynchronous serial bus system with one logical bus line. It has an open, linear bus structure with equal bus nodes. A CAN bus consists of two or more nodes. The number of nodes on the bus may be changed dynamically without disturbing the communication of other nodes. This allows easy connection and disconnection of bus nodes (e.g. for addition of system function, error recovery or bus monitoring).

The bus logic corresponds to a "wired-AND" mechanism, "Recessive" bits (mostly, but not necessarily equivalent to the logic level "1") are overwritten by "Dominant" bits (mostly logic level "0"). As long as no bus node is sending a dominant bit, the bus line is in the recessive state, but a dominant bit from any bus node generates the dominant bus state. Therefore, for the CAN bus line, a medium must be chosen that is able to transmit the two possible bit states (dominant and recessive). One of the most common and cheapest ways is to use a twisted wire pair. The bus lines are then called "CANH" and "CANL", and may be connected directly to the nodes or via a connector. There's no standard defined by CAN regarding the connector to be used. The twisted wire pair is terminated by terminating resistors at each end of the bus line. The maximum bus speed is 1 Mbit, which can be achieved with a bus length of up to 40 meters. For bus lengths longer than 40 meters the bus speed must be reduced (a 1000 m bus can be realized with a 40 Kbit bus speed). For a bus length above 1000 meters special drivers should be used. At least 20 nodes may be connected without additional equipment. Due to the differential nature of transmission, CAN is insensitive to EMI because both bus lines are affected in the same way which leaves the differential signal unaffected. The bus lines can also be shielded to reduce the electromagnetic emission of the bus itself, especially at high baud rates.

The binary data is coded corresponding to the NRZ code (Non-Return-to-Zero; low level = dominant state; high level = recessive state). To ensure clock synchronization of all bus nodes, bit-stuffing is used. This means that during the transmission of a message a maximum of five consecutive bits may have the same polarity. Whenever five consecutive bits of the same polarity have been transmitted, the transmitter will insert one additional bit of the opposite polarity into the bit stream before transmitting further bits. The receiver also checks the number of bits with the same polarity and removes the stuff bits from the bit stream (destuffing).

In the CAN protocol it is not bus nodes that are addressed, but the address information is contained in the messages that are transmitted. This is done via an identifier (part of each message) which identifies the message content (e.g. engine speed, oil temperature etc..). The identifier additionally indicates the priority of the message. The lower the binary value of the identifier, the higher the priority of the message.

For bus arbitration, Carrier Sense Multiple Access/Collision Detection (CSMA/CD) with Non-Destructive Arbitration (NDA) is used. If bus node A wants to transmit a message across the network, it first checks that the bus is in the idle state ("Carrier Sense") i.e., no node is currently transmitting. If this is the case (and no other node wishes to start a transmission at the same moment) node A becomes the bus master and sends its message. All other nodes switch to receive mode during the first transmitted bit (Start Of Frame bit). After correct reception of the message (which is acknowledged by each node) each bus node checks the message identifier and stores the message, if required. Otherwise, the message is discarded.

If two or more bus nodes start their transmission at the same time ("Multiple Access"), collision of the messages is avoided by bitwise arbitration ("Collision Detection/Non-Destructive Arbitration" together with the "Wired-AND" mechanism, "dominant" bits override "recessive" bits). Each node sends the bits of its message identifier (MSb first) and monitors the bus level. A node that sends a recessive identifier bit but reads back a dominant one loses bus arbitration and switches to receive mode. This condition occurs when the message identifier of a competing node has a
Section 22. CAN

lower binary value (dominant state = logic 0) and therefore, the competing node is sending a message with a higher priority. In this way, the bus node with the highest priority message wins arbitration without losing time by having to repeat the message. All other nodes automatically try to repeat their transmission once the bus returns to the idle state. It is not permitted for different nodes to send messages with the same identifier as arbitration could fail leading to collisions and errors later in the message.

The original CAN specifications (Versions 1.0, 1.2 and 2.0A) defined the message identifier as having a length of 11 bits giving a possible 2048 message identifiers. The specification has since been updated (to version 2.0B) to remove this limitation. CAN specification Version 2.0B allows message identifier lengths of 11 and/or 29 bits to be used (an identifier length of 29 bits allows over 536 Million message identifiers). Version 2.0B CAN is also referred to as "Extended CAN"; and Versions 1.0, 1.2 and 2.0A) are referred to as "Standard CAN".

22.3.1 Standard CAN vs. Extended CAN

Those Data Frames and Remote Frames, which only contain the 11 bit identifier, are called Standard Frames according to CAN specification V2.0A. With these frames, 2048 different messages can be identified (identifiers 0-2047). However, the 16 messages with the lowest priority (2032-2047) are reserved. Extended Frames according to CAN specification V2.0B have a 29 bit identifier. As already mentioned, this 29 bit identifier is made up of the 11 bit identifier ("Standard ID") and the 18 bit Extended identifier ("Extended ID").

CAN modules specified by CAN V2.0A are only able to transmit and receive Standard Frames according to the Standard CAN protocol. Messages using the 29 bit identifier cause errors. If a device is specified by CAN V2.0B, there is one more distinction. Modules named "Part B Passive" can only transmit and receive Standard Frames but tolerate Extended Frames without generating Error Frames. "Part B Active" devices are able to transmit and receive both Standard and Extended Frames.

22.3.2 Basic CAN vs. Full CAN

There is one more CAN characteristic concerning the interface between the CAN module and the host CPU, dividing CAN chips into "Basic CAN" and "Full CAN" devices. This distinction is not related to Standard vs. Extended CAN, which makes it possible to use both Basic and Full CAN devices in the same network.

In the Basic CAN devices, only basic functions of the protocol are implemented in hardware, e.g. the generation and the check of the bit stream. The decision, if a received message has to be stored or not (acceptance filtering) and the whole message management has to be done by software, i.e., by the host CPU. In addition, the CAN chip typically provides only one transmit buffer and one or two receive buffers. So the host CPU load is quite high using Basic CAN modules, and these devices can only be used at low baud rates and low bus loads with only a few different messages. The advantages of Basic CAN are the small chip size leading to low costs of these devices.

Full CAN devices implement the whole bus protocol in hardware including the acceptance filtering and the message management. They contain several so called message objects which handle the identifier, the data, the direction (receive or transmit) and the information Standard CAN/Extended CAN. During the initialization of the device, the host CPU defines which messages are to be sent and which are to be received. The host CPU is informed by interrupt if the identifier of a received message matches with one of the programmed (receive-) message objects. In this way, the CPU load is reduced. Using Full CAN devices, high baud rates and high bus loads with many messages can be handled. These chips are more expensive than the Basic CAN devices, though.

Many Full CAN chips provide a "Basic-CAN Feature". One of the messages objects can be programmed in so that every message is stored there that does not match with one of the other message objects. This can be very helpful in a number of applications.
22.3.3 ISO Model

The ISO/OSI Reference Model is used to define the layers of protocol of a communication system as shown in Figure 22-2. At the highest end, the applications need to communicate between each other. At the lowest end, some physical medium is used to provide electrical signaling.

The higher levels of the protocol are run by software. Within the CAN bus specification, there is no definition of the type of message or the contents or meaning of the messages transferred. These definitions are made in systems such as Volcano, the Volvo automotive CAN specification; J1939, the U.S. heavy truck multiplex wiring specification; and Allen-Bradly DeviceNet and Honeywell SDS, examples of industrial protocols.

The CAN bus module definition encompasses two levels of the overall protocol.

- The Data Link Layer
  - The Logical Link Control (LLC) sub layer
  - The Medium Access Control (MAC) sub layer

- The Physical Layer
  - The Physical Signaling (PLS) sub layer

The LLC sub layer is concerned with Message Filtering, Overload Notification and Error Recovery Management. The scope of the LLC sub layer is:

- To provide services for data transfer and for remote data request,
- To decide which messages received by the LLC sub layer are actually to be accepted,
- To provide means for error recovery management and overload notifications.

The MAC sub layer represents the kernel of the CAN protocol. The MAC sub layer defines the transfer protocol, i.e., controlling the Framing, Performing Arbitration, Error Checking, Error Signalling and Fault Confinement. It presents messages received from the LLC sub layer and accepts messages to be transmitted to the LLC sub layer. Within the MAC sub layer it is decided whether the bus is free for starting a new transmission or whether a reception is just starting. The MAC sub layer is supervised by a management entity called Fault Confinement which is a self-checking mechanism for distinguishing short disturbances from permanent failures. Also, some general features of the bit timing are regarded as part of the MAC sub layer.

The physical layer defines the actual transfer of the bits between the different nodes with respect to all electrical properties. The PLS sub layer defines how signals are actually transmitted and therefore deals with the description of Bit Timing, Bit Encoding, and Synchronization.

The lower levels of the protocol are implemented in driver/receiver chips and the actual interface such as twisted pair wiring or optical fiber etc. Within one network, the physical layer has to be the same for all nodes. The Driver/Receiver Characteristics of the Physical Layer are not defined so as to allow transmission medium and signal level implementations to be optimized for their application. The most common example is defined in ISO11898 Road Vehicles multiplex wiring specification.
Section 22. CAN

Figure 22-2: CAN Bus in ISO/OSI Reference Model

OSI REFERENCE LAYERS

Application
Presentation
Session
Transport
Network

Data Link Layer

LLC (Logical Link Control)
Acceptance Filtering
Overload Notification
Recovery Management

MAC (Medium Access Control)
Data Encapsulation/Decapsulation
Frame Coding (stuffing, destuffing)
Medium Access Management
Error Detection
Error Signalling
Acknowledgment
Serialization/Deserialization

Physical Layer

PLS (Physical Signalling)
Bit Encoding/Decoding
Bit Timing
Synchronization

PMA (Physical Medium Attachment)
Driver/Receiver Characteristics

MDI (Medium Dependent Interface)
Connectors

Supervisor

Fault confinement (MAC-LME)

Bus Failure management (PLS-LME)

Shaded Regions Implemented by the CAN Module

Has to be Implemented in PICmicro Firmware

Connector

CAN Transceiver Chip

Connector
22.4 CAN Bus Features

The CAN module is a communication controller implementing the CAN 2.0A/B protocol as defined in the BOSCH specification. The module will support CAN 1.2, CAN 2.0A, CAN 2.0B Passive, and CAN 2.0B Active versions of the protocol. The module implementation is a Full CAN system.

The module features are as follows:

- Implementation of the CAN protocol CAN 1.2, CAN 2.0A and CAN 2.0B
- Standard and extended data frames
- Data length from 0 - 8 bytes
- Programmable bit rate up to 1 Mbit/sec
- Support for remote frames
- Double buffered receiver with two prioritized received message storage buffers
- 6 full (standard/extended identifier) acceptance filters, 2 associated with the high priority receive buffer, and 4 associated with the low priority receive buffer
- 2 full acceptance filter masks, one each associated with the high and low priority receive buffers
- Three transmit buffers with application specified prioritization and abort capability
- Programmable wake-up functionality with integrated low-pass filter
- Programmable loop-back mode and programmable state clocking supports self-test operation
- Signaling via interrupt capabilities for all CAN receiver and transmitter error states
- Programmable clock source
- Programmable link to timer module for time-stamping and network synchronization
- Low power SLEEP mode
22.5 CAN Module Implementation

This subsection will discuss the implementation of the CAN module and the supported frame formats.

22.5.1 Overview of the Module

The CAN bus module consists of a Protocol Engine and message buffering and control. The Protocol Engine can best be understood by defining the types of data frames to be transmitted and received by the module. These blocks are shown in Figure 22-3.

Figure 22-3: CAN Buffers and Protocol Engine Block Diagram
22.5.1.1 Typical Connection

Figure 22-4 shows a typical connection between multiple CAN nodes with CAN bus terminators.

Figure 22-4: CAN Bus Connection with CAN Bus Terminators

22.5.2 CAN Protocol Engine

The CAN protocol engine combines several functional blocks, shown in Figure 22-5. These units and the functions they provide are described below.

The heart of the engine is the Protocol Finite State Machine (FSM). This state machine sequences through the messages on a bit by bit basis, changing states of the machine as various fields of various frame types are transmitted or received. The framing messages in Section 22.6 show the states associated with each bit. The FSM is a sequencer controlling the sequential data stream between the TX/RX Shift Register, the CRC Register, and the bus line. The FSM also controls the Error Management Logic (EML) and the parallel data stream between the TX/RX Shift Register and the buffers such that the processes of reception arbitration, transmission, and error signaling are performed according to the CAN protocol. Note that the automatic retransmission of messages on the bus line is handled by the FSM.

The data interface to the engine consists of byte wide transmit and receive data. Rather than assembling and shifting an entire frame, the frames are broken into bytes. A receive or transmit address from the Protocol FSM signifies which byte of the frame is current. For transmission, the appropriate byte from the transmit buffer is selected and presented to the engine, which then uses an 8 bit shift register to serialize the data. For reception, an 8 bit shift register assembles a byte which is then loaded into the appropriate byte in the message assembly buffer.

The Cyclic Redundancy Check Register generates the Cyclic Redundancy Check (CRC) code to be transmitted over the data bytes and checks the CRC code of incoming messages.

The Error Management Logic (EML) is responsible for the fault confinement of the CAN device. Its counters, the Receive Error Counter and the Transmit Error Counter, are incremented and decremented by commands from the Bit Stream Processor. According to the values of the error counters, the CAN controller is set into the states error active, error passive or bus off.

The Bit Timing Logic (BTL) monitors the bus line input and handles the bus line related bit timing according to the CAN protocol. The BTL synchronizes on a recessive to dominant busline transition at Start of Frame (hard synchronization) and on any further recessive to dominant bus line transition, if the CAN controller itself does not transmit a dominant bit (resynchronization). The BTL also provides programmable time segments to compensate for the propagation delay time and for phase shifts and in defining the position of the Sample Point in the bit time. The programming of the BTL depends on the baud rate and on external physical delay times.
Section 22. CAN

Figure 22-5: CAN Protocol Engine Block Diagram

- CANRX
- Bit Timing Logic (BTL)
- Sample <2:0>
- Majority Decision
- StuffReg <5:0>
- BusMon
- CRC <14:0>
- Comparator
- Receive Shift
- Transmit Shift
- RXERRCNT
- TXERRCNT
- Protocol FSM
- Transmit Logic
- Receive Error Counter
- Transmit Error Counter

Interface to Standard Buffer
22.5.3 CAN Module Functionality

The CAN protocol engine handles all functions for receiving and transmitting messages on the CAN bus. Messages are transmitted by first loading the appropriate data registers. Status and errors can be checked by reading the appropriate registers. Any message detected on the CAN bus is checked for errors and then matched against filters to see if it should be received and stored in one of the 2 receive registers.

The CAN Module supports the following Frame types:

- Standard Data Frame
- Extended Data Frame
- Remote Frame
- Error Frame
- Overload Frame
- Interframe Space

Section 22.6 describes the Frames and their formats.
Section 22. CAN

22.6 Frame Types

This chapter describes the CAN Frame types supported by the CAN module.

22.6.1 Standard Data Frame

A Standard Data Frame is generated by a node when the node wishes to transmit data. The Standard CAN Data Frame is shown in Figure 22-6. In common with all other frames, the frame begins with a Start Of Frame bit (SOF - dominant state) for hard synchronization of all nodes. The SOF is followed by the Arbitration Field consisting of 12 bits, the 11 bit Identifier (reflecting the contents and priority of the message) and the RTR bit (Remote Transmission Request bit). The RTR bit is used to distinguish a data Frame (RTR - dominant) from a Remote Frame.

The next field is the Control Field, consisting of 6 bits. The first bit of this field is called the Identifier Extension (IDE) bit and is at dominant state to specify that the frame is a Standard Frame. The following bit is reserved, RB0, and defined as a dominant bit. The remaining 4 bits of the Control Field are the Data Length Code (DLC) and specify the number of bytes of data contained in the message.

The data being sent follows in the Data Field which is of the length defined by the DLC above (0 - 8 bytes).

The Cyclic Redundancy Field (CRC) follows and is used to detect possible transmission errors. The CRC Field consists of a 15 bit CRC sequence, completed by the End of Frame (EOF) field, which consists of seven recessive bits (no bit-stuffing).

The final field is the Acknowledge Field. During the ACK Slot bit the transmitting node sends out a recessive bit. Any node that has received an error free frame acknowledges the correct reception of the frame by sending back a dominant bit (regardless of whether the node is configured to accept that specific message or not). The recessive Acknowledge Delimiter completes the Acknowledge Slot and may not be overwritten by a dominant bit.

22.6.1.1 Extended Data Frame

In the Extended CAN Data Frame, shown in Figure 22-7, the Start of Frame bit (SOF) is followed by the Arbitration Field consisting of 38 bits. The first 11 bits are the 11 most significant bits of the 29 bit identifier ("Base-ID"). These 11 bits are followed by the Substitute Remote Request bit (SRR), which is transmitted as recessive. The SRR is followed by the IDE bit which is recessive to denote that the frame is an Extended CAN frame. It should be noted from this, that if arbitration remains unresolved after transmission of the first 11 bits of the identifier, and one of the nodes involved in arbitration is sending a Standard CAN frame (11 bit identifier), then the Standard CAN frame will win arbitration due to the assertion of a dominant IDE bit. Also, the SRR bit in an Extended CAN frame must be recessive to allow the assertion of a dominant RTR bit by a node that is sending a Standard CAN Remote Frame. The SRR and IDE bits are followed by the remaining 18 bits of the identifier ("ID-Extension") and the Remote Transmission Request bit.

To enable standard and extended frames to be sent across a shared network, it is necessary to split the 29 bit extended message Identifier into 11 bit (most significant) and 18 bit (least significant) sections. This split ensures that the Identifier Extension bit (IDE) can remain at the same bit position in both standard and extended frames.

The next field is the Control Field, consisting of 6 bits. The first 2 bits of this field are reserved and are at dominant state. The remaining 4 bits of the Control Field are the Data Length Code (DLC) and specify the number of data bytes.

The remaining portion of the frame (Data Field, CRC Field, Acknowledge Field, End Of Frame and Intermission) is constructed in the same way as for a Standard Data Frame.
22.6.1.2 Remote Frame

Normally data transmission is performed on an autonomous basis with the data source node (e.g. a sensor sending out a Data Frame). It is possible, however, for a destination node to request the data from the source. For this purpose, the destination node sends a "Remote Frame" with an identifier that matches the identifier of the required Data Frame. The appropriate data source node will then send a Data Frame as a response to this Remote request.

There are 2 differences between a Remote Frame and a Data Frame, shown in Figure 22-8. First, the RTR bit is at the recessive state and second there is no Data Field. In the very unlikely event of a Data Frame and a Remote Frame with the same identifier being transmitted at the same time, the Data Frame wins arbitration due to the dominant RTR bit following the identifier. In this way, the node that transmitted the Remote Frame receives the desired data immediately.

22.6.1.3 The Error Frame

An Error Frame is generated by any node that detects a bus error. An error frame, shown in Figure 22-9, consists of 2 fields, an Error Flag field followed by an Error Delimiter field. The Error Delimiter consists of 8 recessive bits and allows the bus nodes to restart bus communications cleanly after an error. There are two forms of Error Flag fields. The form of the Error Flag field depends on the error status of the node that detects the error.

If an error-active node detects a bus error then the node interrupts transmission of the current message by generating an active error flag. The active error flag is composed of six consecutive dominant bits. This bit sequence actively violates the bit-stuffing rule. All other stations recognize the resulting bit-stuffing error and in turn generate Error Frames themselves, called Error Echo Flags. The Error Flag field therefore consists of between six and twelve consecutive dominant bits (generated by one or more nodes). The Error Delimiter field completes the Error Frame. After completion of the Error Frame, bus activity retains to normal and the interrupted node attempts to resend the aborted message.

If an error passive node detects a bus error then the node transmits an error passive flag followed, again, by the Error Delimiter field. The error passive flag consists of six consecutive recessive bits. From this it follows that, unless the bus error is detected by the bus master node or other error active receiver, that is actually transmitting, the transmission of an Error Frame by an error passive node will not affect any other node on the network. If the bus master node generates an error passive flag then this may cause other nodes to generate error frames due to the resulting bit-stuffing violation. After transmission of an Error Frame, an error passive node must wait for 6 consecutive recessive bits on the bus before attempting to rejoin bus communications.

22.6.1.4 The Overload Frame

An Overload Frame, shown in Figure 22-10, has the same format as an Active Error Frame. An Overload Frame, however can only be generated during Interframe Space. This way, an Overload Frame can be differentiated from an Error Frame (an Error Frame is sent during the transmission of a message). The Overload Frame consists of 2 fields, an Overload Flag followed by an Overload Delimiter. The Overload Flag consists of six dominant bits followed by Overload Flags generated by other nodes (as for active error flag, again giving a maximum of twelve dominant bits). The Overload Delimiter consists of eight recessive bits. An Overload Frame can be generated by a node as a result of 2 conditions. First, the node detects a dominant bit during Interframe Space which is an illegal condition. Second, due to internal conditions, the node is not yet able to start reception of the next message. A node may generate a maximum of 2 sequential Overload Frames to delay the start of the next message.

22.6.1.5 The Interframe Space

Interframe Space separates a proceeding frame (of whatever type) from a following Data or Remote Frame. Interframe space is composed of at least 3 recessive bits, called the intermission. This is provided to allow nodes time for internal processing before the start of the next message frame. After the intermission, the bus line remains in the recessive state (Bus idle) until the next transmission starts.
Figure 22-6: Standard Data Frame
Figure 22-7: Extended Data Format
Figure 22-8: Remote Data Frame
Figure 22-9: Error Frame
Section 22. CAN

Figure 22-10: Overload Frame
22.7 Modes of Operation

The CAN Module can operate in one of several operation modes selected by the user. These modes include:

- Initialization Mode
- Disable Mode
- Normal Operation Mode
- Listen Only Mode
- Loop Back Mode
- Error Recognition Mode (selected through CANRXM bits)

Modes are requested by setting the REQOP2:REQOP0 bits except the Error Recognition Mode, which is requested through the CANRXM bits. Entry into a mode is acknowledged by monitoring the OPMODE bits. The module will not change the mode and the OPMODE2:OPMODE0 bits until a change in mode is acceptable, generally during bus idle time which is defined as at least 11 consecutive recessive bits.

22.7.1 Initialization Mode

In the initialization mode, the module will not transmit or receive. The error counters are cleared and the interrupt flags remain unchanged. The programmer will have access to configuration registers that are access restricted in other modes. The CAN bus configuration mode is explained in Section 22.8.
Section 22. CAN

22.7.2 Disable Mode

In Disable Mode, the module will not transmit or receive. The module has the ability to set the WAKIF bit due to bus activity, however any pending interrupts will remain and the error counters will retain their value.

If the REQOP2:REQOP0 bits = 001, the module will enter the module disable mode. This mode is similar to disabling other peripheral modules by turning off the module enables. This causes the module internal clock to stop unless the module is active (i.e., receiving or transmitting a message). If the module is active, the module will wait for 11 recessive bits on the CAN bus, detect that condition as an idle bus then accept the module disable command. When the OPMODE2:OPMODE0 bits = 001, that indicates whether the module successfully went into module disable mode (see Figure 22-11).

The WAKIF interrupt is the only module interrupt that is still active in the module disable mode. If the WAKIE is set, the processor will receive an interrupt whenever the CAN bus detects a dominant state, as occurs with a Start of Frame (SOF).

The I/O pins will revert to normal I/O function when the module is in the module disable mode.

Figure 22-11: Entering and Exiting Module Disable Mode

- Processor writes REQOP2:REQOP0 while module receiving/transmitting message. Module continues with CAN message.
- Module detects 11 recessive bits. Module acknowledges disable mode and sets OPMODE2:OPMODE0 bits. Module disables.
- CAN bus message will set WAKIF bit. If WAKIE = ‘1’, processor will vector to the interrupt address. CAN message ignored.
- Processor writes REQOP2:REQOP0 during CAN bus activity. Module waits for 11 recessive bits before accepting activate.
- Module detects 11 recessive bits. Module acknowledges normal mode and sets OPMODE2:OPMODE0 bits. Module activates.
22.7.2.1 SLEEP Mode

A CPU SLEEP instruction will stop the crystal oscillator and shut down all system clocks. The user is responsible to take care that the module is not active when the CPU goes into SLEEP mode. The pins will revert into normal I/O function, dependent on the value in the TRIS register. The recommended procedure is to bring the module into disabled mode before the CPU SLEEP instruction is executed.

Figure 22-12 illustrates the sequence of events when a CAN message is received during execution of the SLEEP instruction.

**Figure 22-12:** SLEEP Interrupted By Message

- Processor requests and receives module disable mode. Wake up interrupt enabled.
- CAN bus activity sets WAKIF flag. If GIE = ‘1’ processor will vector to interrupt address, bypassing SLEEP instruction.
- Processor attempts to execute SLEEP instruction. Since WAKIF = ‘1’, WAKIE = ‘1’ and GIE = ‘0’ processor will execute NOP in place of SLEEP instruction. CAN message ignored.
- Processor requests and receives module normal mode. CAN activity resumes.
22.7.2.2  Wake-up from SLEEP

Figure 22-13 depicts how the CAN module will execute the SLEEP instruction and how the module wakes up on bus activity. Upon a Wake-up from SLEEP the WAKIF flag is set.

The module will monitor the RX line for activity while the MCU is in SLEEP mode.

If the module is in CPU SLEEP mode and the WAKIE wake-up interrupt enable is set, the module will generate an interrupt, bringing up the CPU. Due to the delays in starting up the oscillator and CPU, the message activity that caused the wake-up will be lost.

If the module is in CPU SLEEP mode and the WAKIE is not set, no interrupt will be generated and the CPU and the module will continue to sleep.

If the CAN module is in disable mode, the module will wake-up and, depending on the condition of the WAKIE bit, may generate an interrupt. It is expected that the module will correctly receive the message that caused the wake-up from SLEEP mode.

The module can be programmed to apply a low-pass filter function to the RxCAN input line while the module or the CPU is in SLEEP mode. This feature can be used to protect the module from Wake-up due to short glitches on the CAN bus lines. Such glitches can result from electromagnetic inference within noisy environments. The WAKFIL bit enables or disables the filter.

Figure 22-13: Processor SLEEP and CAN Bus Wake-up Interrupt

1. Processor requests and receives module disable mode. Wake-up interrupt enabled.
2. Processor requests normal operating mode. Module waits for 11 recessive bits before accepting CAN bus activity. CAN message lost.
3. SOF of message wakes up processor. Oscillator start time begins. CAN message lost. WAKIF bit set.
4. Processor completes oscillator start time. Processor resumes program or interrupt, based on GIE bits. Processor requests normal operating mode. Module waits for 11 recessive bits before accepting CAN bus activity. CAN message lost.
5. Module detects 11 recessive bits. Module will begin to receive messages and transmit any pending messages.
22.7.3 Normal Operation Mode

Normal operating mode is selected when REQOP2:REQOP0 = 000. In this mode, the module is activated, the I/O pins will assume the CAN bus functions. The module will transmit and receive CAN bus messages as described in subsequent paragraphs.

22.7.4 Listen Only Mode

Listen only mode and loopback modes are special cases of normal operation mode to allow system debug. If the listen only mode is activated, the module on the CAN bus is passive. The transmitter buffers revert to Port I/O function. The receive pins remain input. For the receiver, no error flags or acknowledge signals are sent. The error counters are deactivated in this state. The listen only mode can be used for detecting the baud rate on the CAN bus. To use this, it is necessary that there are at least two further nodes that communicate with each other. The baud rate can be detected empirically by testing different values. This mode is also useful as a bus monitor without influencing the data traffic.

22.7.5 Error Recognition Mode

The module can be set to ignore all errors and receive any message. The error recognition mode is activated by setting the RXM1:RXM0 bits in the RXBnCON registers to 11. In this mode the data which is in the message assembly buffer until the error time is copied in the receive buffer and can be read via the CPU interface. In addition the data which was on the internal sampling of the CAN bus at the error time and the state vector of the protocol state machine and the bit counter CntCan are stored in registers and can be read.

22.7.6 Loop Back Mode

If the loopback mode is activated, the module will connect the internal transmit signal to the internal receive signal at the module boundary. The transmit and receive pins revert to their Port I/O function.

The transmitter will receive an acknowledge for its sent messages. Special hardware will generate an acknowledge for the transmitter.

22.8 CAN Bus Initialization

After a RESET the CAN module is in the configuration mode (OPMODE2 is set). The error counters are cleared and all registers contain the reset values. It should be ensured that the initialization is performed before REQOP2 bit is cleared.

22.8.1 Initialization

The CAN module has to be initialized before the activation. This is only possible if the module is in the configuration mode. The configuration mode is requested by setting REQOP2 bit. Only when the status bit OPMODE2 has a high level, the initialization can be performed. Afterwards the configuration registers and the acceptance mask registers and the acceptance filter registers can be written. The module is activated by setting the control bits CFGREQ to zero.

The module will protect the user from accidentally violating the CAN protocol through programming errors. All registers which control the configuration of the module can not be modified while the module is on-line. The CAN module will not be allowed to enter the configuration mode while a transmission is taking place. The CONFIG mode serves as a lock to protect the following registers.

- All Module Control Registers
- Configuration Registers
- Bus Timing Registers
- Identifier Acceptance Filter Registers
- Identifier Acceptance Mask Registers
Section 22. CAN

22.9 Message Reception

This chapter describes the message reception.

22.9.1 Receive Buffers

The CAN bus module has 3 receive buffers. However, one of the receive buffers is always committed to monitoring the bus for incoming messages. This buffer is called the message assembly buffer, MAB. So there are 2 receive buffers visible, RXB0 and RXB1, that can essentially instantaneously receive a complete message from the protocol engine. The CPU can be operating on one while the other is available for reception or holding a previously received message.

The MAB holds the destuffed bit stream from the bus line to allow parallel access to the whole data or Remote Frame for the acceptance match test and the parallel transfer of the frame to the receive buffers. The MAB will assemble all messages received. These messages will be transferred to the RXBn buffers only if the acceptance filter criterion are met. When a message is received, the RXFUL bit will be set. This bit can only be set by the module when a message is received. The bit is cleared by the CPU when it has completed processing the message in the buffer. This bit provides a positive lockout to ensure that the CPU has finished with the message buffer. If the RXnIE bit is set, an interrupt will be generated when a message is received.

There are 2 programmable acceptance filter masks associated with the receive buffers, one for each buffer.

When the message is received, the FILHIT2:FILHIT0 bits (RXBnCON register) indicate the acceptance criterion for the message. The number of the acceptance filter that enabled the reception will be indicated as well as a status bit that indicates that the received message is a remote transfer request.
22.9.1.1 Receive Buffer Priority

To provide flexibility, there are several acceptance filters corresponding to each receive buffer. There is also an implied priority to the receive buffers. RXB0 is the higher priority buffer and has 2 message acceptance filters associated with it. RXB1 is the lower priority buffer and has 4 acceptance filters associated with it. The lower number of possible acceptance filters makes the match on RXB0 more restrictive and implies the higher criticality associated with that buffer. Additionally, if the RXB0 contains a valid message, and another valid message is received, the RXB0 can be setup such that it will not overrun and the new message for RXB0 will be placed into RXB1. Figure 22-14 shows a block diagram of the receive buffer, while Figure 22-15 shows a flow chart for receive operation.
**Section 22. CAN**

**Figure 22-15: Receive Flowchart**

- **Start**
  - Detect Start of Message?
    - Yes
      - Begin Loading Message into Message Assembly Buffer (MAB)
    - No
  - Valid Message Received?
    - Yes
      - Yes, meets criteria for RXBO
      - Yes, meets criteria for RXB1
      - No
    - No
      - Generate Error Frame
      - Yes
      - Generate Interrupt
      - No

The RXRDY bit determines if the receive register is empty and able to accept a new message.

The RX0DBEN bit determines if RXB0 can roll over into RXB1 if it is full.

- Is RXRDY=0?
  - Yes
    - Move message into RXB1
    - Set RXRDY = 1
    - Set FILHIT<0> according to which filter criteria was met
  - No
    - Is RX0DBEN=1?
      - Yes
        - Generate Overrun Error: Set RXIOVFL
      - No
        - Move message into RXB1
        - Set RXRDY = 1
        - Set FILHIT<2:0> according to which filter criteria was met
        - Is RXRDY=0?
          - Yes
            - Move message into RXB1
            - Set RXRDY = 1
            - Set FILHIT<2:0> according to which filter criteria was met
          - No
            - Go to Start
        - Is RXIE=1?
          - Yes
            - Generate Overrun Error: Set RX1OVFL
            - No
              - Go to Start
  - No
    - Generate Interrupt
    - Yes
      - Set CANSTAT<3:0> according to which receive buffer the message was loaded into
      - No
        - Does RXIE=1?
22.9.2 Message Acceptance Filters

The message acceptance filters and masks are used to determine if a message in the message assembly buffer should be loaded into either of the receive buffers. Once a valid message has been received into the MAB, the identifier fields of the message are compared to the filter values. If there is a match, that message will be loaded into the appropriate receive buffer. The filter masks are used to determine which bits in the identifiers are examined with the filters. A truth-table is shown in Table 22-2 that indicates how each bit in the identifier is compared to the masks and filters to determine if the message should be loaded into a receive buffer. The mask bit essentially determines which bits to apply the filter to. If any mask bit is set to a zero, then that bit will automatically be accepted regardless of the filter bit.

<table>
<thead>
<tr>
<th>Mask Bit n</th>
<th>Filter Bit n</th>
<th>Message Identifier bit</th>
<th>Accept or reject bit n</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>x</td>
<td>x</td>
<td>Accept</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>Accept</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>Reject</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>0</td>
<td>Reject</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
<td>Accept</td>
</tr>
</tbody>
</table>

Legend: x = 0 don’t care.

The acceptance filter looks at incoming messages for the EXIDEN bit to determine how to compare the identifiers. If the EXIDEN bit is clear, the message is a standard frame, and only filters with the EXIDEN bit clear are compared. If the EXIDEN bit is set, the message is an extended frame, and only filters with the EXIDEN bit set are compared. Configuring the RXM1:RXM0 bits to 01 or 10 can override the EXIDEN bit.

As shown in the Receive Buffers Block Diagram, Figure 22-14, RXF0 and RXF1 filters with RXM0 mask are associated with RXB0. The filters RXF2, RXF3, RXF4, and RXF5 and the mask RXM1 are associated with RXB1. When a filter matches and a message is loaded into the receive buffer, the number of the filter that enabled the message reception is coded into a portion of the RXBn-CON register. The RXB1CON register contains the FILHIT2:FILHIT0 bits. They are coded as shown in Table 22-3.

<table>
<thead>
<tr>
<th>FILHIT2:FILHIT0</th>
<th>Acceptance Filter</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>000 (1)</td>
<td>RXF0</td>
<td>Only if RX0DBEN = 1</td>
</tr>
<tr>
<td>001 (1)</td>
<td>RXF1</td>
<td>Only if RX0DBEN = 1</td>
</tr>
<tr>
<td>010</td>
<td>RXF2</td>
<td>—</td>
</tr>
<tr>
<td>011</td>
<td>RXF3</td>
<td>—</td>
</tr>
<tr>
<td>100</td>
<td>RXF4</td>
<td>—</td>
</tr>
<tr>
<td>101</td>
<td>RXF5</td>
<td>—</td>
</tr>
</tbody>
</table>

Note 1: Is only valid if the RX0DBEN bit is set.
Section 22. CAN

The coding of the RX0DBEN bit enables these 3 bits to be used similarly to the FILHIT bits and to distinguish a hit on filter RXF0 and RXF1 in either RXB0 or overrun into RXB1.

111 = Acceptance Filter 1 (RXF1)
110 = Acceptance Filter 0 (RXF0)
001 = Acceptance Filter 1 (RXF1)
000 = Acceptance Filter 0 (RXF0)

If the RX0DBEN bit is clear, there are 6 codes corresponding to the 6 filters. If the RX0DBEN bit is set, there are 6 codes corresponding to the 6 filters plus 2 additional codes corresponding to RXF0 and RXF1 filters overrun to RXB1.

If more than 1 acceptance filter matches, the FILHIT bits will encode the lowest binary value of the filters that matched. In other words, if filter 2 and filter 4 match, FILHIT will code the value for 2. This essentially prioritizes the acceptance filters with lower numbers having priority.

Figure 22-16 shows a block diagram of the message acceptance filters.
22.9.3 Receiver Overrun

An overrun condition occurs when the MAB has assembled a valid received message, the message is accepted through the acceptance filters, however, the receive buffer associated with the filter has not been designated as clear of the previous message.

The overrun error flag, RXnOVR and the EERIF bit will be set and the message in the MAB will be discarded. While in the overrun situation, the module will stay synchronized with the CAN bus and is able to transmit messages but will discard all incoming messages destined for the overrun buffer.

If the RX0DBEN bit is clear, RXB1 and RXB0 operate independently. When this is the case, a message intended for RXB0 will not be diverted into RXB1 if RXB0 contains an unread message and the RX0OVFL bit will be set.

If the RX0DBEN bit is set, the overrun for RXB0 is handled differently. If a valid message is received for RXB0 and RXFUL = 1 indicates that RXB0 is full and RXFUL = 0 indicates that RXB1 is empty, the message for RXB0 will be loaded into RXB1. An overrun error will not be generated for RXB0. If a valid message is received for RXB0 and RXFUL = 1 and RXFUL = 1 indicating that both RXB0 and RXB1 are full the message will be lost and an overrun will be indicated for RXB1.

If the RX0DBEN bit is set, there are 6 codes corresponding to the 6 filters. If the RX0DBEN bit is set, there are 6 codes corresponding to the 6 filters plus 2 additional codes corresponding to RXF0 and RXF1 filters overrun to RXB1. These codes are given in Table 22-4.

<table>
<thead>
<tr>
<th>Message Matches Filter 0 or 1</th>
<th>Message Matches Filter 2,3,4,5</th>
<th>RXFUL0 Bit</th>
<th>RXFUL1 Bit</th>
<th>RX0DBEN Bit</th>
<th>Action</th>
<th>Action</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>X</td>
<td>X</td>
<td>X</td>
<td>None</td>
<td>No message received</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>X</td>
<td>0</td>
<td>X</td>
<td>MAB → RXB1</td>
<td>Message for RXB1, RXB1 available</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>X</td>
<td>1</td>
<td>X</td>
<td>MAB discarded RX1OVFL = 1</td>
<td>Message for RXB1, RXB1 full</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>X</td>
<td>X</td>
<td>MAB → RXB0</td>
<td>Message for RXB0, RXB0 available</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>X</td>
<td>0</td>
<td>MAB discarded RX0OVFL = 1</td>
<td>Message for RXB0, RXB0 full, RX0DBEN not enabled</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>1</td>
<td>MAB → RXB1</td>
<td>Message for RXB0, RXB0 full, RX0DBEN enabled, RXB1 available</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>MAB discarded RX1OVFL = 1</td>
<td>Message for RXB0, RXB0 full, RX0DBEN enabled, RXB1 full</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>0</td>
<td>X</td>
<td>X</td>
<td>MAB → RXB0</td>
<td>Message for RXB0 and RXB1, RXB0 available</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
<td>X</td>
<td>0</td>
<td>MAB discarded RX0OVFL = 1</td>
<td>Message for RXB0 and RXB1, RXB0 full, RX0DBEN not enabled</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>1</td>
<td>MAB → RXB1</td>
<td>Message for RXB0 and RXB1, RXB0 full, RX0DBEN enabled, RXB1 available</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>MAB discarded RX1OVFL = 1</td>
<td>Message for RXB0 and RXB1, RXB0 full, RX0DBEN enabled, RXB1 full</td>
</tr>
</tbody>
</table>

Legend: X = Don't care.
Section 22. CAN

22.9.4 Effects of a RESET

Upon any RESET the CAN module has to be initialized. All registers are set according to the reset values. The content of a received message is lost. The initialization is discussed in Section 22.8.

22.9.5 Baud Rate Setting

All nodes on any particular CAN bus must have the same nominal bit rate. The Baud Rate is set once during the initialization mode of the CAN module. After that the baud Rate is not changed again. Section 22.12 explains the setting of the Baud Rate.

22.9.6 Receive Errors

The CAN module will detect the following receive errors:

- Cyclic Redundancy Check (CRC) Error
- Bit Stuffing Error
- Invalid message receive error.

These receive errors do not generate an interrupt. However, the receive error counter is incremented by one in case one of these errors occur. The RXWARN bit indicates that the Receive Error Counter has reached the CPU Warning limit of 96 and an interrupt is generated.

22.9.6.1 Cyclic Redundancy Check (CRC) Error

With the Cyclic Redundancy Check, the transmitter calculates special check bits for the bit sequence from the start of a frame until the end of the Data Field. This CRC sequence is transmitted in the CRC Field. The receiving node also calculates the CRC sequence using the same formula and performs a comparison to the received sequence. If a mismatch is detected, a CRC error has occurred and an Error Frame is generated. The message is repeated. The receive error interrupt counter is incremented by one. An Interrupt will only be generated if the error counter passes a threshold value.

22.9.6.2 Bit Stuffing Error

If in between Start of Frame and CRC Delimiter 6 consecutive bits with the same polarity are detected, the bit-stuffing rule has been violated. A Bit-Stuffing error occurs and an Error Frame is generated. The message is repeated. No Interrupt will be generated upon this event.

22.9.6.3 Invalid Message Received Error

If any type of error occurs during reception of a message, an error will be indicated by the IXRIF bit. This bit can be used (optionally with an interrupt) for autobaud detection with the device in listen-only mode. This error is not an indicator that any action needs to occur, but an indicator that an error has occurred on the CAN bus.
22.9.6.4 Rules for Modifying the Receive Error Counter

The Receive Error Counter is modified according to the following rules:

- When the receiver detects an error, the Receive Error Counter is incremented by 1, except when the detected error was a Bit Error during the transmission of an Active Error Flag or an Overload Flag.
- When the receiver detects a “dominant” bit as the first bit after sending an Error Flag, the Receive Error Counter will be incremented by 8.
- If a Receiver detects a Bit Error while sending an Active Error Flag or an Overload Flag, the Receive Error Counter is incremented by 8.
- Any Node tolerates up to 7 consecutive “dominant” bits after sending an Active Error Flag, Passive Error Flag or an Overload Flag. After detecting the 14th consecutive “dominant” bit (in case of an Active Error Flag or an Overload flag) or after detecting the 8th consecutive “dominant” following a passive error flag, and after each sequence of additional eight consecutive “dominant” bits, every Transmitter increases its Transmission Error Counter and every Receiver increases its Receive Error Counter by 8.
- After a successful reception of a message (reception without error up to the ACK slot and the successful sending of the ACK bit), the Receive Error Counter is decreased by one, if the Receive Error Counter was between 1 and 127. If the Receive Error Counter was 0 it will stay 0. If the Receive Error Counter was greater than 127, it will change to a value between 119 and 127.

22.9.7 Receive Interrupts

Several Interrupts are linked to the message reception. The receive interrupts can be broken up into two separate groups:

- Receive Error Interrupts
- Receive interrupts

22.9.7.1 Receive Interrupt

A message has been successfully received and loaded into one of the receive buffers. This interrupt is activated immediately after receiving the End of Frame (EOF) field. Reading the RXnIF flag will indicate which receive buffer caused the interrupt. Figure 22-17 depicts when the receive buffer interrupt flag RXnIF will be set.

22.9.7.2 Wake-up Interrupt

The Wake-up Interrupt sequences are described in Section 22.7.2.2.
Section 22. CAN

Figure 22-17: Receive Buffer Interrupt Flag
22.9.7.3 Receive Error Interrupts

A receive error interrupt will be indicated by the ERRIF bit. This bit shows that an error condition occurred. The source of the error can be determined by checking the bits in the Communication Status Register COMSTAT. The bits in this register are related to receive and transmit errors. The following subsequences will show which flags are linked to the receive errors.

22.9.7.3.1 Invalid Message Received Interrupt

If any type of error occurred during reception of the last message, an error will be indicated by the IXRIF bit. The specific error that occurred is unknown. This bit can be used (optionally with an interrupt) for autobaud detection with the device in listen only mode. This error is not an indicator that any action needs to occur, but an indicator that an error has occurred on the CAN bus.

22.9.7.3.2 Receiver Overrun Interrupt

The RXnOVR bit indicates that an Overrun condition occurred. An overrun condition occurs when the Message Assembly Buffer (MAB) has assembled a valid received message, the message is accepted through the acceptance filters, however, the receive buffer associated with the filter is not clear of the previous message. The overflow error interrupt will be set and the message is discarded. While in the overrun situation, the module will stay synchronized with the CAN bus and is able to transmit and receive messages.

22.9.7.4 Receiver Warning Interrupt

The RXWARN bit indicates that the Receive Error Counter has reached the CPU Warning limit of 96. When RXWARN transitions from a 0 to a 1, it will cause the Error Interrupt Flag ERRIF to become set. This bit cannot be manually cleared, as it should remain an indicator that the Receive Error Counter has reached the CPU Warning limit of 96. The RXWARN bit will become clear automatically if the Receive Error Counter becomes less than or equal to 95. The ERRIF bit can be manually cleared allowing the interrupt service routine to be exited without affecting the RXWARN bit.

22.9.7.5 Receiver Error Passive

The RXBP bit indicates that the Receive Error Counter has exceeded the Error Passive limit of 127 and the module has gone to Error State Passive. When the RXWARN transitions from a 0 to a 1, it will cause the Error Interrupt Flag ERRIF to become set. The RXBP bit cannot be manually cleared, as it should remain an indicator that the Bus is in Error State Passive. The RXBP bit will become clear automatically if the Receive Error Counter becomes less than or equal to 127. The ERRIF bit can be manually cleared allowing the interrupt service routine to be exited without affecting the RXBP bit.

22.9.8 Receive Modes

The RXM1:RXM0 bits will set special receive modes. Normally, these bits are set to 00 to enable reception of all valid messages as accepted by the acceptance filters. In this case, the determination of whether or not to receive standard or extended messages is determined by the EXIDEN bit in the Acceptance Filter Registers. If the RXM1:RXM0 bits are set to 01 or 10, the receiver will accept only messages with standard or extended identifiers respectively. If an acceptance filter has the EXIDEN bit such that it does not correspond with the RXM1:RXM0 mode, that acceptance filter is rendered useless. These 2 modes of RXM1:RXM0 bits can be used in systems where it is known that only standard or extended messages will be on the bus. If the RXM1:RXM0 bits are set to 11, the buffer will receive all messages regardless of the values of the acceptance filters. Also, if a message has an error before the End of Frame, that portion of the message assembled in the Message Assembly Buffer (MAB) before the error frame will be loaded into the buffer. This mode may have some value in debugging a CAN system.
22.9.8.1 Listen Only Mode

If the receive only mode is activated, the module on the CAN bus is passive. That means that no error flags or acknowledge signals are sent. The error counters are deactivated in this state. The receive only mode can be used for detecting the baud rate on the CAN bus. For this it is necessary that there are at least two further nodes, which communicate with each other. The baud rate can be detected empirically by testing different values. This mode is also useful as a bus monitor without influencing the data traffic.

22.9.8.2 Error Recognition Mode

The module can be set to ignore all errors and receive any message. The error recognition mode is activated by configuring the RXM1:RXM0 bits (RXBnCON registers) = ‘11’. In this mode the data which is in the message assembly buffer until the error time is copied in the receive buffer and can be read via the CPU interface.
22.10 Transmission

This subsection describes how the CAN module is used for receiving CAN messages.

22.10.1 Real Time Communication and Transmit Message Buffering

For an application to effectively transmit messages in real time, the CAN nodes must be able to dominate and hold the bus assuming that messages are of a high enough priority to win arbitration on the bus. If a node only has 1 transmission buffer, it must transmit a message, then release the bus while the CPU reloads the buffer. If a node has two transmission buffers, one buffer could be transmitting while the second buffer is being reloaded. However, the CPU would need to maintain tight tracking of the bus activity to ensure that the second buffer is reloaded before the first message completes.

Typical applications require three transmit message buffers. Having three buffers, one buffer can be transmitting, the second buffer can be ready to transmit as soon as the first is complete, and the third can be reloaded by the CPU. This eases the burden of the software to maintain synchronization with the bus (see Figure 22-18).

Additionally, having three buffers allows some degree of prioritizing of the outgoing messages. For example, the application software may have a message enqueued in the second buffer while it is working on the third buffer. The application may require that the message going into the third buffer is of higher importance than the one already enqueued. If only 2 buffers are available, the enqueued message would have to be deleted and replaced with the third. The process of deleting the message may mean losing control of the bus. With 3 buffers, both the second and the third message can be enqueued, and the module can be instructed that the third message is of higher priority than the second. The third message will be the next one sent followed by the second.

22.10.2 Transmit Message Buffers

The CAN module has three transmit buffers. Each of the three buffers occupies 14 bytes of data. Eight of the bytes are the maximum 8 bytes of the transmitted message. Five bytes hold the standard and extended identifiers and other message arbitration information.

The last byte is a control byte associated with each message. The information in this byte determines the conditions under which the message will be transmitted and indicates status of the transmission of the message.

The TXBnIF bit will be set and the TXREQ bit will be clear, indicating that the message buffer has completed a transmission. The CPU will then load the message buffer with the contents of the message to be sent. At a minimum, the standard identifier register TXBnSIDH and TXBnSIDL must be loaded. If data bytes are present in the message, the TXBnDm registers are loaded. If the message is to use extended identifiers, the TXBnEIDm registers are loaded and the EXIDEN bit is set.

Prior to sending the message, the user must initialize the TXIE bit to enable or disable an interrupt when the message is sent. The user must also initialize the transmit priority. Figure 22-18 shows a block diagram of the transmit buffers.

Figure 22-18: Transmit Buffers
22.10.3 Transmit Message Priority

Transmit priority is a prioritization within each node of the pending transmittable messages. Prior to sending the SOF (Start of Frame), the priorities of all buffers ready for transmission are compared. The transmit buffer with the highest priority will be sent first. For example, if transmit buffer 0 has a higher priority setting than transmit buffer 1, buffer 0 will be sent first. If two buffers have the same priority setting, the buffer with the highest address will be sent. For example, if transmit buffer 1 has the same priority setting as transmit buffer 0, buffer 1 will be sent first. There are 4 levels of transmit priority. If TXPRI1:TXPRI0 for a particular message buffer is set to 11, that buffer has the highest priority. If TXPRI1:TXPRI0 for a particular message buffer is set to 10 or 01, that buffer has an intermediate priority. If TXPRI1:TXPRI0 for a particular message buffer is 00, that buffer has the lowest priority.

22.10.4 Message Transmission

To initiate transmitting the message, the TXREQ bit must be set. The CAN bus module resolves any timing conflicts between setting of the TXREQ bit and the SOF time, ensuring that if the priority was changed, it is resolved correctly before SOF. When TXREQ is set the TXABT, TXLARB and TXERR flag bits will be cleared.

Setting TXREQ bit does not actually start a message transmission, it flags a message buffer as enqueued for transmission. Transmission will start when the module detects an available bus for SOF. The module will then begin transmission on the message which has been determined to have the highest priority.

If the transmission completes successfully on the first try, the TXREQ bit will clear and an interrupt will be generated if TXIE was set.

If the message fails to transmit, one of the other condition flags will be set, the TXREQ bit will remain set indicating that the message is still pending for transmission. If the message tried to transmit but encountered an error condition, the TXERR bit will be set. In this case, the error condition can also cause an interrupt. If the message tried to transmit but lost arbitration, the TXLARB bit will be set. In this case, no interrupt is available to signal the loss of arbitration.

22.10.5 Transmit Message Aborting

The system can also abort a message by clearing the TXREQ bit associated with each message buffer. Setting the ABAT bit will request an abort of all pending messages. If the message has not yet started transmission, or if the message started but is interrupted by loss of arbitration or an error; the abort will be processed. The abort is indicated when the module sets the TXABT bit, and the TXnIF flag is not automatically set.
Figure 22-19: Transmit Flowchart

The message transmission sequence begins when the device determines that the TXREQ for any of the transmit registers has been set.

Start

Are any TXREQ bits = 1?

Yes

Clear: TXABT, TXLARB, and TXERR

Examine TXPRI<1:0> to Determine Highest Priority Message

Begin transmission (SOF)

Was message transmitted successfully?

Yes

Set TXREQ=0

Generate Interrupt

Is TXIE=1?

Yes

Set TXUEF=1

No

Yes

End

No

Is CAN bus available to start transmission?

Yes

Can bus available to start transmission?

No

Does TXREQ=0 and ABAT =1?

Yes

Clearing the TXREQ bit while it is set, or setting the ABAT bit before the message has started transmission will abort the message.

No

Begin transmission (SOF)

Was message transmitted successfully?

Yes

Set TXERR=1

No

Does TXLARB=1?

Yes

Arbitration lost during transmission

No

Does TXREQ=0 or TXABT =1?

Yes

Abort Transmission: Set TXABT=1

No

A message can also be aborted if a message error or lost arbitration condition occurred during transmission.

The TXIE bit determines if an interrupt should be generated when a message is successfully transmitted.
Section 22. CAN

22.10.6 Transmit Boundary Conditions

The module handles transmit commands which are not necessarily synchronized to the CAN bus message framing time.

22.10.6.1 Clearing TXREQ bit as a Message Starts

The TXREQ bit can be cleared just when a message is starting transmission, with the intent of aborting the message. If the message is not being transmitted, the TXABT bit will be set, indicating that the Abort was successfully processed.

When the user clears the TXREQ bit and the TXABT bit is not set two cycles later, the message has already begun transmission.

If the message is being transmitted, the abort is not immediately processed, at some point later, the TXnIF Interrupt Flag or the TXABT bit is set. If transmission has begun the message will only be aborted if either an error or a loss of arbitration occurs.

22.10.6.2 Setting TXABT bit as a Message Starts

Setting the ABAT bit will abort all pending transmit buffers and has the function of clearing all of the TXREQ bits for all buffers. The boundary conditions are the same as clearing the TXREQ bit.

22.10.6.3 Clearing TXREQ bit as a Message Completes

The TXREQ bit can be cleared when a message is just about to successfully complete transmission. Even if the TXREQ bit is cleared by the Data bus a short time before it will be cleared by the successful transmission of the message, the TXnIF flag will still be set due to the successful transmission.

22.10.6.4 Setting TXABT bit as a Message Completes

The boundary conditions are the same as clearing the TXREQ bit.

22.10.6.5 Clearing TXREQ bit as a Message Loses Transmission

The TXREQ bit can be cleared when a message is just about to be lost to arbitration or an error. If the TXREQ signal falls before the loss of arbitration signal or error signal, the result will be like clearing TXREQ during transmission. When the arbitration is lost or the error is set, the TXABT bit will be set, as it will see that an error has occurred while transmitting, and that the TXREQ bit was not set.

If the TXREQ bit falls after the arbitration signal has entered the block, the result will be like clearing TXREQ during an inactive transmit time. The TXABT bit will be set.

22.10.6.6 Setting TXABT bit as a Message Loses Transmission

The boundary conditions are the same as clearing the TXREQ bit.
22.10.7 Effects of a RESET

Upon any RESET the CAN module has to be initialized. All registers are set according to the reset values. The content of a received message is lost. The initialization is discussed in Section 22.8.

22.10.8 Baud Rate Setting

All nodes on any particular CAN bus must have the same nominal bit rate. The baud rate is set once during the initialization phase of the CAN module. After that, the baud rate is not changed again. Section 22.12 explains the setting of the Baud Rate.

22.10.9 Transmit Message Aborting

The system can also abort a message by clearing the TXREQ bit associated with each message buffer. Setting the ABAT bit will request an abort of all pending messages (see Figure 22-21). A queued message is aborted by clearing the TXREQ bit. Aborting a queued message is illustrated in Figure 22-20. If the message has not yet started transmission, or if the message started but is interrupted by loss of arbitration or an error; the abort will be processed. The abort is indicated when the module sets the TXABT bit. If the message has started to transmit, it will attempt to transmit the current message fully (see Figure 22-22). If the current message is transmitted fully, and is not lost to arbitration or an error, the TXABT bit will not be set, because the message was transmitted successfully. Likewise, if a message is being transmitted during an abort request, and the message is lost to arbitration (see Figure 22-23) or an error, the message will not be re-transmitted, and the TXABT bit will be set, indicating that the message was successfully aborted.

Figure 22-20: Abort Queued Message

1 - Processor sets TXREQ while module receiving/transmitting message. Module continues with CAN message.
2 - Processor clears TXREQ while module looking for 11 recessive bits. Module aborts pending transmission, sets TXABT bit in 2 clocks.
3 - Another module takes the available transmit slot.
Figure 22-21: Abort All Messages

3. Another module takes the available transmit slot.

Figure 22-22: Failed Abort During Transmission

3. Processor clears TXREQ requesting message abort. Abort cannot be acknowledged.
4. At successful completion of transmission, TXREQ bit remains clear and TXIF bit set. TXABT remains clear.
22.10.10 Transmission Errors

The CAN module will detect the following transmission errors:

- Acknowledge Error
- Form Error
- Bit Error

These transmission errors will not necessarily generate an interrupt but are indicated by the transmission error counter. However, each of these errors will cause the transmission error counter to be incremented by one. Once the value of the error counter exceeds the value of 96, the ERRIF and the TXWARN bit are set. Once the value of the error counter exceeds the value of 96 an interrupt is generated and the TXWARN bit in the error flag register is set.

An example transmission error is illustrated in Figure 22-24.
22.10.1 Acknowledge Error

In the Acknowledge Field of a message, the transmitter checks if the Acknowledge Slot (which it has sent out as a recessive bit) contains a dominant bit. If not, no other node has received the frame correctly. An Acknowledge Error has occurred and the message has to be repeated. No Error Frame is generated.

22.10.2 Form Error

If a transmitter detects a dominant bit in one of the four segments including End of Frame, Inter-frame Space, Acknowledge Delimiter or CRC Delimiter; then a Form Error has occurred and an Error Frame is generated. The message is repeated.

22.10.3 Bit Error

A Bit Error occurs if a transmitter sends a dominant bit and detects a recessive bit. In the case where the transmitter sends a recessive bit and a dominant bit is detected during the Arbitration Field and the Acknowledge Slot, no bit error is generated because normal arbitration is occurring.

22.10.4 Rules for Modifying the Transmit Error Counter

The Transmit Error Counter is modified according to the following rules:

- When the Transmitter sends an error flag the Transmit Error Counter is increased by 8 with the following exceptions. In these two exceptions, the Transmit Error Counter is not changed.
- If the transmitter is "error passive" and detects an acknowledgment error because of not detecting a "dominant" ACK, and does not detect a "dominant" bit while sending a Passive Error Flag.
- If the Transmitter sends an Error Flag because of a bit-stuffing Error occurred during arbitration whereby the Stuffbit is located before the RTR bit, and should have been "recessive", and has been sent as "recessive" but monitored as "dominant".
- If a Transmitter detects a Bit Error while sending an Active Error Flag or an Overload Flag the Transmit Error Counter is increased by 8.
- Any Node tolerates up to 7 consecutive "dominant" bits after sending an Active Error Flag, Passive Error Flag or an Overload Flag. After detecting the 14th consecutive "dominant" bit (in case of an Active Error Flag or an Overload flag) or after detecting the 8th consecutive "dominant" following a passive error flag, and after each sequence of eight additional consecutive "dominant" bits, every Transmitter increases its Transmission Error Counter and every Receiver increases its Receive Error Counter by 8.
- After the successful transmission of a message (getting an acknowledge and no error until End of Frame is finished) the Transmit Error Counter is decreased by one unless it was already 0.
22.10.11 Transmission Interrupts

There are several interrupts linked to the message transmission. The transmission interrupts can be broken up into two groups:

- Transmission interrupts
- Transmission error interrupts

22.10.11.1 Transmit Interrupt

At least one of the three transmit buffers is empty (not scheduled) and can be loaded to schedule a message for transmission. Reading the TXIF flags will indicate which transmit buffer is available and caused the interrupt.

22.10.11.2 Transmission Error Interrupts

A transmission error interrupt will be indicated by the ERRIF flag. This flag shows that an error condition occurred. The source of the error can be determined by checking the error flags in the Communication Status register COMSTAT. The flags in this register are related to receive and transmit errors. The following subsequences will show which flags are linked to the transmit errors.

22.10.11.3 Transmitter Warning Interrupt

The TXWARN bit indicates that the Transmit Error Counter has reached the CPU Warning limit of 96. When this bit transitions from a 0 to a 1, it will cause the Error Interrupt Flag to become set. The TXWARN bit cannot be manually cleared, as it should remain as an indicator that the Transmit Error Counter has reached the CPU Warning limit of 96. The TXWARN bit will become clear automatically if the Transmit Error Counter becomes less than or equal to 95. The ERRIF flag can be manually cleared allowing the interrupt service routine to be exited without affecting the TXWARN bit.

22.10.11.4 Transmitter Error Passive

The TXEP bit indicates that the Transmit Error Counter has exceeded the Error Passive limit of 127 and the module has gone to Error Passive state. When this bit transitions from a 0 to a 1, it will cause the Error Interrupt Flag to become set. The TXEP bit cannot be manually cleared, as it should remain as an indicator that the Bus is in Error State Passive. The TXEP bit will become clear automatically if the Transmit Error Counter becomes less than or equal to 127. The ERRIF flag can be manually cleared allowing the interrupt service routine to be exited without affecting the TXEP bit.

22.10.11.5 Bus Off Interrupt

The TXBO bit indicates that the Transmit Error Counter has exceeded 255 and the module has gone to Bus Off state. When this bit transitions from a 0 to a 1, it will cause the Error Interrupt Flag to become set. The TXBO bit cannot be manually cleared, as it should remain as an indicator that the Bus is Off. The ERRIF flag can be manually cleared allowing the interrupt service routine to be exited without affecting the TXBO bit.
Section 22. CAN

22.11 Error Detection

The CAN protocol provides sophisticated error detection mechanisms. The following errors can be detected. These errors are either receive or transmit errors.

Receive errors are:
- Cyclic Redundancy Check (CRC) Error (see Section 22.9.6.1)
- Bit Stuffing Bit Error (see Section 22.9.6.2)
- Invalid Message Received Error (see Section 22.9.6.2)

The transmit errors are:
- Acknowledge Error (see Section 22.10.10.1)
- Form Error (see Section 22.10.10.2)
- Bit Error (see Section 22.10.10.3)

22.11.1 Error States

Detected errors are made public to all other nodes via Error Frames. The transmission of the erroneous message is aborted and the frame is repeated as soon as possible. Furthermore, each CAN node is in one of the three error states "error active", "error passive" or "bus off" according to the value of the internal error counters. The error-active state is the usual state where the bus node can transmit messages and active Error Frames (made of dominant bits) without any restrictions. In the error-passive state, messages and passive Error Frames (made of recessive bits) may be transmitted. The bus-off state makes it temporarily impossible for the station to participate in the bus communication. During this state, messages can neither be received nor transmitted.

22.11.2 Error Modes and Error Counters

The CAN controller contains the two error counters Receive Error Counter (RXERRCNT) and Transmit Error Counter (TXERRCNT). The values of both counters can be read by the CPU. These counters are incremented or decremented according to the CAN bus specification.

The CAN controller is error active if both error counters are below the error passive limit of 128. It is error passive if at least one of the error counters equals or exceeds 128. It goes bus off if the Transmit Error Counter equals or exceeds the bus off limit of 256. The device remains in this state, until the bus off recovery sequence is finished, which is 128 consecutive 11 recessive bit times. Additionally, there is a error state warning flag bit, EWARN, which is set if at least one of the error counters equals or exceeds the error warning limit of 96. EWARN is reset if both error counters are less than the error warning limit.
22.11.3 Error Flag Register

The values in the error flag register indicate which error(s) caused the Error Interrupt Flag. The RXXOVR Error Flags have a different function than the other Error Flag bits in this register. The RXXOVR bits must be cleared in order to clear the ERRIF interrupt flag. The other Error Flag bits in this register will cause the ERRIF interrupt flag to become set as the value of the Transmit and Receive Error Counters crosses a specific threshold. Clearing the ERRIF interrupt flag in these cases will allow the interrupt service routine to be exited without recursive interrupt occurring. It may be desirable to disable specific interrupts after they have occurred once to stop the device from interrupting repeatedly as the Error Counter moves up and down in the vicinity of a threshold value.
Section 22. CAN

22.12 Baud Rate Setting

All nodes on any particular CAN bus must have the same nominal bit rate. The CAN bus uses NRZ coding which does not encode a clock. Therefore the receivers independent clock must be recovered by the receiving nodes and synchronized to the transmitters clock.

In order to set the baud rate the following bits have to be initialized:
- Synchronization Jump Width (see Section 22.12.6.2)
- Baud rate prescaler (see Section 22.12.2)
- Phase segments (see Section 22.12.4)
- Length determination of Phase segment 2 (see Section 22.12.4)
- Sample Point (see Section 22.12.5)
- Propagation segment bits (see Section 22.12.3)

22.12.1 Bit Timing

As oscillators and transmission time may vary from node to node, the receiver must have some type of PLL synchronized to data transmission edges to synchronize and maintain the receiver clock. Since the data is NRZ coded, it is necessary to include bit-stuffing to ensure that an edge occurs at least every 6 bit times, to maintain the Digital Phase Lock Loop (DPLL) synchronization.

Bus timing functions executed within the bit time frame, such as synchronization to the local oscillator, network transmission delay compensation, and sample point positioning, are defined by the programmable bit timing logic of the DPLL.

All controllers on the CAN bus must have the same baud rate and bit length. However, different controllers are not required to have the same master oscillator clock. At different clock frequencies of the individual controllers, the baud rate has to be adjusted by adjusting the number of time quanta in each segment.

The Nominal Bit Time can be thought of as being divided into separate non-overlapping time segments. These segments are shown in Figure 22-26.
- Synchronization segment (Sync Seg)
- Propagation time segment (Prop Seg)
- Phase buffer segment 1 (Phase1 Seg)
- Phase buffer segment 2 (Phase2 Seg)

The time segments and also the nominal bit time are made up of integer units of time called time quanta or To. By definition, the Nominal Bit Time has a minimum of 8 To and a maximum of 25 To. Also, by definition the minimum nominal bit time is 1 usec, corresponding to a maximum 1 MHz bit rate.

Figure 22-26: CAN Bit Timing

![CAN Bit Timing Diagram](image-url)
22.12.2 Prescaler Setting

There is a programmable prescaler, with integral values ranging at least from 1 to 64, in addition to a fixed divide by 2 for clock generation. The Time Quanta (T_Q) is a fixed unit of time derived from the oscillator period. Time quanta is defined as:

Equation 22-1: Time Quanta for Clock Generation

\[ T_Q = 2 \cdot (\text{Baud Rate} + 1) \cdot T_{OSC} \]

Where Baud Rate is the binary value of BRP <5:0>.

Example 22-1: Calculation for \( F_{OSC} = 16 \text{ MHz} \)

If \( F_{OSC} = 16 \text{ MHz} \), BRP5:BRP0 = 00h, and Nominal Bit Time = 8 T_Q; then T_Q = 125 nsec and Nominal Bit Rate = 1 MHz.

Example 22-2: Calculation for \( F_{OSC} = 32 \text{ MHz} \)

If \( F_{OSC} = 32 \text{ MHz} \), BRP5:BRP0 = 01h, and Nominal Bit Time = 8 T_Q; then T_Q = 125 nsec and Nominal Bit Rate = 1 MHz.

Example 22-3: Calculation for \( F_{OSC} = 32 \text{ MHz} \) and 25 T_Q

If \( F_{OSC} = 32 \text{ MHz} \), BRP5:BRP0 = 3Fh, and Nominal Bit Time = 25 T_Q; then T_Q = 4 usec and Nominal Bit Rate = 10 kHz.

The frequencies of the oscillators in the different nodes must be coordinated in order to provide a system-wide specified time quantum. This means that all oscillators must have an \( F_{OSC} \) that is an integral divisor of T_Q.

22.12.3 Propagation Segment

This part of the bit time is used to compensate physical delay times within the network. These delay times consist of the signal propagation time on the bus line and the internal delay time of the nodes. The delay is calculated as the round trip from transmitter to receiver as twice the signal's propagation time on the bus line, the input comparator delay, and the output driver delay. The Propagation Segment can be programmed from 1 T_Q to 8 T_Q by setting the PRSE2:PRSEG0 bits.

22.12.4 Phase Segments

The phase segments are used to optimally locate the sampling of the received bit within the transmitted bit time. The sampling point is between Phase1 Segment and Phase2 Segment. These segments are lengthened or shortened by resynchronization. The end of the Phase1 Segment determines the sampling point within a bit period. The segment is programmable from 1 T_Q to 8 T_Q. Phase2 Segment provides delay to the next transmitted data transition. The segment is programmable from 1 T_Q to 8 T_Q or it may be defined to be equal to the greater of Phase1 Segment or the Information Processing Time. The phase segment 1 is initialized by setting bits SEG1PH2:SEG1PH0, and phase segment 2 is initialized by setting SEG2PH2:SEG2PH0.
Section 22. CAN

22.12.5 Sample Point

The sample point is the point of time at which the bus level is read and interpreted as the Value of that respective bit. The location is at the end of Phase Segment 1. If the bit timing is slow and contains many TQ, it is possible to specify multiple sampling of the bus line at the sample point. The level determined by the CAN bus then corresponds to the result from the majority decision of three values. The majority samples are taken at the sample point and twice before with a distance of TQ/2. The CAN module allows to choose between sampling three times at the same point or once at the same point. This is done by setting or clearing the SAM bit (BRG2CON register).

22.12.6 Synchronization

To compensate for phase shifts between the oscillator frequencies of the different bus stations, each CAN controller must be able to synchronize to the relevant signal edge of the incoming signal. When an edge in the transmitted data is detected, the logic will compare the location of the edge to the expected time (Synchronous Segment). The circuit will then adjust the values of Phase1 Segment and Phase2 Segment. There are 2 mechanisms used to synchronize.

22.12.6.1 Hard Synchronization

Hard Synchronization is only done whenever there is a 'recessive' to 'dominant' edge during Bus Idle, indicating the start of a message. After hard synchronization, the bit time counters are restarted with Synchronous Segment. Hard synchronization forces the edge which has caused the hard synchronization to lie within the synchronization segment of the restarted bit time. Due to the rules of synchronization, if a hard synchronization is done, there will not be a resynchronization within that bit time.

22.12.6.2 Resynchronization

As a result of resynchronization Phase Segment 1 may be lengthened or Phase Segment 2 may be shortened. The amount of lengthening or shortening (SJW1:SJW0) of the phase buffer segment has an upper bound given by the resynchronization jump width bits. The value of the synchronization jump width will be added to Phase Segment 1 or subtracted from Phase Segment 2. The resynchronization jump width is programmable between 1 TQ and 4 TQ.

Clocking information will only be derived from transitions of recessive to dominant bus states. The property that only a fixed maximum number of successive bits have the same value ensures resynchronizing a bus unit to the bit stream during a frame (e.g. bit-stuffing).

The Phase Error of an edge is given by the position of the edge relative to Synchronous Segment, measured in Time Quanta. The Phase Error is defined in magnitude of TQ as follows:

- e = 0 if the edge lies within Synchronous Segment.
- e > 0 if the edge lies before the Sample Point.
- e < 0 if the edge lies after the Sample Point of the previous bit.

If the magnitude of the phase error is less than or equal to the programmed value of the resynchronization jump width, the effect of a resynchronization is the same as that of a hard synchronization.

If the magnitude of the phase error is larger than the resynchronization jump width, and if the phase error is positive, then Phase Segment 1 is lengthened by an amount equal to the resynchronization jump width.

If the magnitude of the phase error is larger than the resynchronization jump width, and if the phase error is negative, then Phase Segment 2 is shortened by an amount equal to the resynchronization jump width.
22.12.7 Programming Time Segments

Some requirements for programming of the time segments:
- Propagation Segment + Phase1 Segment ≥ Phase2 Segment
- Phase2 Segment > Synchronous Jump Width

Example 22-4: Segment Time

CAN Baud Rate = 125 kHz
Fosc = 20 MHz
Then:
\[ \text{Tosc} = 50 \text{ nsec} \]
BRP5:BRP0 = 04h, \( \rightarrow \text{TQ} = 500 \text{ nsec} \)
For:
125 kHz
bit time = 16 TQ

Typically, the sampling of the bit should take place at about 60 - 70% of the bit time, depending on the system parameters.

Synchronous Segment = 1 TQ; Propagation Segment = 2 TQ; So setting Phase Segment 1 = 7 TQ would place the sample at 10 TQ after the transition. This would leave 6 TQ for Phase Segment 2.

Since Phase Segment 2 is 6, by the rules, the SJW1: SJW0 bits could be set to the maximum of 4 TQ. However, normally a large synchronization jump width is only necessary when the clock generation of the different nodes is inaccurate or unstable, such as using ceramic resonators. So a synchronization jump width of 1 is typically enough.
22.13 Interrupts

The module has several sources of interrupts. Each of these interrupts can be individually enabled or disabled. A PIR register contains interrupt flags. A PIE register contains the enables for the 8 main interrupts. A special set of read-only bits in the CANSTAT register (ICODE2:ICODE0) can be used in combination with a jump table for efficient handling of interrupts.

All interrupts have one source, with the exception of the Error Interrupt. Any of the Error Interrupt sources can set the Error Interrupt Flag. The source of the Error Interrupt can be determined by reading the Communication Status (COMSTAT) register.

The interrupts can be broken up into two categories: receive and transmit interrupts.

The receive related interrupts are:
- Receive Interrupt (see Section 22.9.7.1)
- Wake-up Interrupt (see Section 22.9.7.2)
- Receiver Overrun Interrupt (see Section 22.9.7.3.2)
- Receiver Warning Interrupt (see Section 22.9.7.4)
- Receiver Error Passive Interrupt (Section 22.9.7.5)

The transmit related interrupts are:
- Transmit interrupt (see Section 22.10.11.1)
- Transmitter Error Passive Interrupt (see Section 22.10.11.3)
- Transmitter Error Passive Interrupt (see Section 22.10.11.4)

22.13.1 Interrupt Acknowledge

Interrupts are directly associated with one or more status flags in either a PIR or COMSTAT registers. Interrupts are pending as long as one of the corresponding flags is set. The flags in the registers must be reset within the interrupt handler in order to handshake the interrupt. A flag cannot be cleared if the respective condition still prevails, with the exception being interrupts that are caused by a certain value being reached in one of the Error Counter Registers.
22.13.2 The ICODE Bits

The ICODE2:ICODE0 bits are a set of read-only bits designed for efficient handling of interrupts via a jump table. The ICODE2:ICODE0 bits can only display one interrupt at a time because the interrupt bits are multiplexed into this register. Therefore, the pending interrupt with the highest priority and enabled interrupt is reflected in the ICODE2:ICODE0 bits. Once the highest priority interrupt flag has been cleared, the next highest priority interrupt code is reflected in the ICODE2:ICODE0 bits. An interrupt code for a corresponding interrupt can only be displayed if both its interrupt flag and interrupt enable are set. Table 22-5 describes the operation of the ICODE2:ICODE0 bits.

Table 22-5: ICODE Bits Decode Table

<table>
<thead>
<tr>
<th>ICODE2:ICODE0</th>
<th>Boolean Expression</th>
</tr>
</thead>
<tbody>
<tr>
<td>000</td>
<td>ERR-WAK-TX0-TX1-TX2-RX0-RX1</td>
</tr>
<tr>
<td>001</td>
<td>ERR</td>
</tr>
<tr>
<td>100</td>
<td>ERR-TX0</td>
</tr>
<tr>
<td>011</td>
<td>ERR-TX0-TX1</td>
</tr>
<tr>
<td>010</td>
<td>ERR-TX0-TX1-TX2</td>
</tr>
<tr>
<td>110</td>
<td>ERR-TX0-TX1-TX2-RX0</td>
</tr>
<tr>
<td>101</td>
<td>ERR-TX0-TX1-TX2-RX0-RX1</td>
</tr>
<tr>
<td>111</td>
<td>ERR-TX0-TX1-TX2-RX0-RX1-WAK</td>
</tr>
</tbody>
</table>

Legend: ERR = ERRIF • ERRIE  
TX0 = TX0IF • TX0IE  
TX1 = TX1IF • TX1IE  
TX2 = TX2IF • TX2IE  
RX0 = RX0IF • RX0IE  
RX1 = RX1IF • RX1IE  
WAK = WAKIF • WAKIE
Section 22. CAN

22.14 Timestamping

The CAN module will generate a signal that can be selected to a timer capture input whenever a valid frame has been accepted. Because the CAN specification defines a frame to be valid if no errors occurred before the EOF field has been transmitted successfully, the timer signal will be generated right after the EOF. A pulse of one bit time is generated.

22.15 CAN Module I/O

The CAN bus module communicates on up to 3 I/O pins. There are 1 or 2 transmit pins and 1 receive pin. These pins are multiplexed with normal digital I/O functions of the device. The CIOCON register controls the functions of the I/O pins.

When the module is in the configuration mode, module disable mode or loopback mode, the I/O pins revert to a Port I/O function.

When the module is active, the TX0 pin is always dedicated to the CAN output function. If a single ended driver is needed, then only the TX0 pin is required. If a differential driver is required, then the TX1 pin must be enabled by setting the TX1EN bit. If the bus requires an active pull-up on the line, the ENDRHI bit should be cleared.

The TRIS bits associated with the transmit pins are overridden by the CAN bus modes. If the CAN module expects an output to be driving, it will be regardless of the state of the TRIS bit associated with that pin.

The output buffers for the TX0 and TX1 pin are designed such that the rise and fall rate of the output signal is approximately equal as is necessary for differential drive.

The module can receive the CAN input on one digital input line.
22.16 Design Tips

Question 1: My CAN module does not seem to work after a RESET.
Answer 1:
Ensure that you reinitialize your CAN bus module. After a RESET, the CAN bus module will automatically go into the initialization mode.

Question 2: I constantly get a Receive error warning interrupt.
Answer 2:
Ensure that your CAN module is set up correctly. Check if the Baud rate is set correctly.
Section 22. CAN

22.17 Related Application Notes

This subsection lists application notes that are related to this subsection of the manual. These application notes may not be written for the Mid-range family (that is they may be written for the Baseline, or the High-end), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to this section are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>An Introduction to the CAN Protocol</td>
<td>AN713</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
22.18  Revision History

Revision A

This is the initial released revision of this document.
Section 23. Comparator Voltage Reference

HIGHLIGHTS

This section of the manual contains the following major topics:

23.1 Introduction ............................................................................................................... 23-2
23.2 Control Register........................................................................................................... 23-3
23.3 Configuring the Voltage Reference .............................................................................. 23-4
23.4 Voltage Reference Accuracy/Error ........................................................................... 23-5
23.5 Operation During SLEEP ............................................................................................. 23-5
23.6 Effects of a RESET ...................................................................................................... 23-5
23.7 Connection Considerations ........................................................................................ 23-6
23.8 Initialization ............................................................................................................. 23-7
23.9 Design Tips ................................................................................................................. 23-8
23.10 Related Application Notes ........................................................................................ 23-9
23.11 Revision History ....................................................................................................... 23-10
23.1 Introduction

This Voltage Reference module is typically used in conjunction with the Comparator module. The Comparator module’s inputs do not require very large drive, and therefore the drive capability of this Voltage Reference is limited.

The Voltage Reference is a 16-tap resistor ladder network that provides a selectable Voltage Reference. The resistor ladder is segmented to provide two ranges of $V_{REF}$ values and has a power-down function to conserve power when the reference is not being used. The VRCON register controls the operation of the reference (shown in Register 23-1). The block diagram is given in Figure 23-1. Within each range, the 16 steps are monotonic (i.e., each increasing code will result in an increasing output).

![Voltage Reference Block Diagram](image)

**Figure 23-1:** Voltage Reference Block Diagram

**Table 23-1: Typical Voltage Reference with $V_{DD} = 5.0V$**

<table>
<thead>
<tr>
<th>VR3:VR0</th>
<th>$V_{REF}$</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>$V_{RR} = 1$</td>
</tr>
<tr>
<td>0000</td>
<td>0.00 V</td>
</tr>
<tr>
<td>0001</td>
<td>0.21 V</td>
</tr>
<tr>
<td>0010</td>
<td>0.42 V</td>
</tr>
<tr>
<td>0011</td>
<td>0.63 V</td>
</tr>
<tr>
<td>0100</td>
<td>0.83 V</td>
</tr>
<tr>
<td>0101</td>
<td>1.04 V</td>
</tr>
<tr>
<td>0110</td>
<td>1.25 V</td>
</tr>
<tr>
<td>0111</td>
<td>1.46 V</td>
</tr>
<tr>
<td>1000</td>
<td>1.67 V</td>
</tr>
<tr>
<td>1001</td>
<td>1.88 V</td>
</tr>
<tr>
<td>1010</td>
<td>2.08 V</td>
</tr>
<tr>
<td>1011</td>
<td>2.29 V</td>
</tr>
<tr>
<td>1100</td>
<td>2.50 V</td>
</tr>
<tr>
<td>1101</td>
<td>2.71 V</td>
</tr>
<tr>
<td>1110</td>
<td>2.92 V</td>
</tr>
<tr>
<td>1111</td>
<td>3.13 V</td>
</tr>
</tbody>
</table>

**Note 1:** See parameter D312 in the "Electrical Specifications" section of the device data sheet.
Section 23. Comparator Voltage Reference

23.2 Control Register

The Voltage Reference Control register (VRCON) is shown in Register 23-1.

Register 23-1: VRCON Register

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>VREN</td>
<td>VROEN</td>
<td>VRR</td>
<td>—</td>
<td>VR3</td>
<td>VR2</td>
<td>VR1</td>
<td>VR0</td>
</tr>
</tbody>
</table>

bit 7 VREN: VREF Enable
1 = VREF circuit powered on
0 = VREF circuit powered down

bit 6 VROEN: VREF Output Enable
1 = VREF is internally connected to Comparator module’s VREF. This voltage level is also output on the VREF pin
0 = VREF is not connected to the Comparator module. This voltage is disconnected from the VREF pin

bit 5 VRR: VREF Range Selection
1 = 0V to 0.75 VDD, with VDD/24 step size
0 = 0.25 VDD to 0.75 VDD, with VDD/32 step size

bit 4 Unimplemented: Read as ‘0’

bit 3:0 VR3:VR0: VREF Value Selection
0 ≤ VR3:VR0 ≤ 15
When VRR = 1:
VREF = (VR<3:0> / 24) • VDD
When VRR = 0:
VREF = 1/4 • VDD + (VR3:VR0 / 32) • VDD

Legend
R = Readable bit  W = Writable bit  U = Unimplemented bit, read as ‘0’
- n = Value at POR reset  ’1’ = bit is set  ’0’ = bit is cleared  x = bit is unknown
23.3 Configuring the Voltage Reference

The Voltage Reference can output 16 distinct voltage levels for each range. The equations used to calculate the output of the Voltage Reference are as follows:

if \( VRR = 1 \): \[ V_{REF} = \frac{VR3:VR0}{24} \times V_{DD} \]

if \( VRR = 0 \): \[ V_{REF} = \frac{V_{DD}}{4} + \frac{VR3:VR0}{32} \times V_{DD} \]

The settling time of the Voltage Reference must be considered when changing the \( V_{REF} \) output. Example 23-1 shows an example of how to configure the Voltage Reference for an output voltage of 1.25V with \( V_{DD} = 5.0V \).

Generally the \( V_{REF} \) and \( V_{DD} \) of the system will be known and you need to determine the value to load into VR3:VR0. Equation 23-1 shows how to calculate the VR3:VR0 value. There will be some error since VR3:VR0 can only be an integer, and the \( V_{REF} \) and \( V_{DD} \) levels must be chosen so that the result is not greater then 15.

\begin{equation}
V_{REF} = \frac{VR3:VR0}{V_{DD}} \times 24 \\
\text{When } VRR = 1
\end{equation}

\begin{equation}
V_{REF} = \frac{V_{DD}}{4} + \frac{VR3:VR0}{32} \times 32 \\
\text{When } VRR = 0
\end{equation}
Section 23. Comparator Voltage Reference

23.4 Voltage Reference Accuracy/Error

The full range of VSS to VDD cannot be realized due to the construction of the module. The transistors on the top and bottom of the resistor ladder network (Figure 23-1) keep VREF from approaching VSS or VDD. The Voltage Reference is VDD derived and therefore, the VREF output changes with fluctuations in VDD. The absolute accuracy of the Voltage Reference can be found in the Electrical Specifications parameter D311.

23.5 Operation During SLEEP

When the device wakes up from SLEEP through an interrupt or a Watchdog Timer time-out, the contents of the VRCON register are not affected. To minimize current consumption in SLEEP mode, the Voltage Reference should be disabled.

23.6 Effects of a RESET

A device RESET disables the Voltage Reference by clearing the VREN bit (VRCON<7>). This RESET also disconnects the reference from the VREF pin by clearing the VROEN bit (VRCON<6>) and selects the high voltage range by clearing the VRR bit (VRCON<5>). The VREF value select bits, VRCON<3:0>, are also cleared.
23.7 Connection Considerations

The Voltage Reference Module operates independently of the Comparator module. The output of the reference generator may be connected to the VREF pin if the corresponding TRIS bit is set and the VROEN bit (VRCON<6>) is set. Enabling the Voltage Reference output onto the VREF pin with an input signal present will increase current consumption. Configuring the VREF as a digital output with VREF enabled will also increase current consumption.

The VREF pin can be used as a simple D/A output with limited drive capability. Due to the limited drive capability, a buffer must be used in conjunction with the Voltage Reference output for external connections to VREF. Figure 23-2 shows an example buffering technique.

Figure 23-2: Voltage Reference Output Buffer Example

Note 1: R is the Voltage Reference Output Impedance and is dependent upon the Voltage Reference Configuration (the VR3:VR0 bits and the VRR bit).
Section 23. Comparator Voltage Reference

23.8 Initialization

Example 23-1 shows a program sequence to configure the Voltage Reference, comparator module, and PORT pins.

Example 23-1: Voltage Reference Configuration

```
MOVLW 0x02 ; 4 Inputs Muxed to 2 comparators
MOVWF CMCON ;
MOVLW PORTxout ; Select PORTx pins
MOVWF TRISx ; to be output
MOVLW 0xA6 ; enable VREF
MOVWF VRCON ; low range set VR3:VR0 = 6
CALL DELAY10 ; 10 µs delay
```
23.9 Design Tips

Question 1:  *My VREF is not what I expect.*
Answer 1:
Any variation of the device VDD will translate directly onto the VREF pin. Also ensure that you have correctly calculated (specified) the VDD divider which generates the VREF.

Question 2:  *I am connecting VREF into a low impedance circuit, and the VREF is not at the expected level.*
Answer 2:
The Voltage Reference module is not intended to drive large loads. A buffer must be used between the PICmicro's VREF pin and the load.
## Section 23. Comparator Voltage Reference

### 23.10 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is they may be written for the Base-Line, Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to Voltage Reference are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Resistance and Capacitance Meter using a PIC16C622</td>
<td>AN611</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
23.11 Revision History

Revision A

This is the initial released revision of the Voltage Reference description.
Section 24. Comparator

HIGHLIGHTS

This section of the manual contains the following major topics:

24.1 Introduction ............................................................................................................... 24-2
24.2 Control Register ........................................................................................................... 24-3
24.3 Comparator Configuration ............................................................................................ 24-4
24.4 Comparator Operation ................................................................................................. 24-6
24.5 Comparator Reference ............................................................................................... 24-6
24.6 Comparator Response Time ....................................................................................... 24-8
24.7 Comparator Outputs .................................................................................................. 24-8
24.8 Comparator Interrupts .................................................................................................. 24-9
24.9 Comparator Operation During SLEEP ........................................................................ 24-9
24.10 Effects of a RESET .................................................................................................... 24-9
24.11 Analog Input Connection Considerations ................................................................. 24-10
24.12 Initialization ............................................................................................................. 24-11
24.13 Design Tips ................................................................................................................ 24-12
24.14 Related Application Notes ....................................................................................... 24-13
24.15 Revision History ....................................................................................................... 24-14
24.1 Introduction

The comparator module contains two analog comparators. The inputs to the comparators are multiplexed with the I/O pins. The on-chip Voltage Reference (see the “Comparator Voltage Reference” section) can also be an input to the comparators.

The CMCON register, shown in Register 24-1, controls the comparator input and output multiplexers. A block diagram of the comparator is shown in Figure 24-1.
# Section 24. Comparator

## 24.2 Control Register

The Comparator Control register (CMCON) is shown in [Register 24-1](#).

### Register 24-1: CMCON Register

<table>
<thead>
<tr>
<th>bit 7</th>
<th>bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>C2OUT</td>
<td>C1OUT</td>
</tr>
<tr>
<td>C2INV</td>
<td>C1INV</td>
</tr>
<tr>
<td>CIS</td>
<td>CM2</td>
</tr>
<tr>
<td>CM1</td>
<td>CM0</td>
</tr>
</tbody>
</table>

- **bit 7 C2OUT**: Comparator2 Output State bit
  - This bit indicates the output state of comparator 2.
  - 1 = C2VIN+ > C2VIN-
  - 0 = C2VIN+ < C2VIN-

- **bit 6 C1OUT**: Comparator1 Output State bit
  - This bit indicates the output state of comparator 1.
  - 1 = C1VIN+ > C1VIN-
  - 0 = C1VIN+ < C1VIN-

- **bit 5 C2INV**: Comparator2 Inverted Output State bit
  - 1 = Invert the state of C2 output
  - 0 = State of C2 output is not inverted

- **bit 4 C1INV**: Comparator1 Inverted Output State bit
  - 1 = Invert the state of C1 output
  - 0 = State of C1 output is not inverted

- **bit 3 CIS**: Comparator Input Switch bit
  - This bit selects which analog inputs are used as the input to the comparator.
  - When CM2:CM0 = 001:
    - C1VIN– connects to ANx3
    - C2VIN– connects to ANx0
  - When CM2:CM0 = 010:
    - C1VIN– connects to ANx3
    - C2VIN– connects to ANx2
    - C2VIN– connects to ANx1

- **bit 2:0 CM2:CM0**: Comparator Mode Select bits
  - This bit selects the configuration of the two comparators with the comparator input pins and the "Comparator Voltage Reference".
  - See [Figure 24-1](#) to select the CM2:CM0 state for the desired mode. The use of ANx0 through ANx3 indicates that there are four analog inputs used with the comparator module. The actual analog inputs connected to the comparator inputs will be device dependent.

Legend

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as ‘0’
- **- n** = Value at POR reset
- **’1’** = bit is set
- **’0’** = bit is cleared
- **x** = bit is unknown
24.3 Comparator Configuration

There are eight modes of operation for the comparators. The CMCON register is used to select the mode. Figure 24-1 shows the eight possible modes. The TRIS register controls the data direction of the comparator I/O pins for each mode. If the comparator mode is changed, the comparator output level may not be valid for the new mode for the delay specified in the electrical specifications of the device.

**Note:** Comparator interrupts should be disabled during a comparator mode change, otherwise a false interrupt may occur.
Figure 24-1: Comparator I/O Operating Modes

### CM2:CM0 = 000
Comparators Reset (POR Default Value)

- ANx0: A - VIN- → C1: Off (Read as '0')
- ANx3: A - VIN+ → C1: Off (Read as '0')
- ANx1: A - VIN- → C2: Off (Read as '0')
- ANx2: A - VIN+ → C2: Off (Read as '0')

### CM2:CM0 = 010
Two Independent Comparators

- ANx0: A - VIN- → C1: C1OUT
- ANx3: A - VIN+ → C1: C1OUT
- ANx1: A - VIN- → C2: C2OUT
- ANx2: A - VIN+ → C2: C2OUT

### CM2:CM0 = 100
Two Common Reference Comparators

- ANx0: A - VIN- → C1: C1OUT
- ANx3: A - VIN+ → C1: C1OUT
- ANx1: A - VIN- → C2: C2OUT
- ANx2: D - VIN+ → C2: C2OUT

### CM2:CM0 = 011
Three Inputs Multiplexed to Two Comparators

- ANx0: A - VIN- → C1: C1OUT
- ANx3: A - VIN+ → C1: C1OUT
- ANx1: A - VIN- → C2: C2OUT
- ANx2: D - VIN+ → C2: C2OUT

### CM2:CM0 = 111
Comparators Off

- ANx0: D - VIN- → C1: Off (Read as '0')
- ANx3: D - VIN+ → C1: Off (Read as '0')
- ANx1: D - VIN- → C2: Off (Read as '0')
- ANx2: D - VIN+ → C2: Off (Read as '0')

### CM2:CM0 = 001
One Independent Comparator

- ANx1: A - VIN- → C1: C1OUT
- ANx2: A - VIN+ → C1: C1OUT
- ANx0: D - VIN- → C1: Off (Read as '0')
- ANx3: D - VIN+ → C1: Off (Read as '0')

### CM2:CM0 = 110
Four Inputs Multiplexed to Two Comparators

- ANx0: A - VIN- → C1: C1OUT
- ANx3: A - VIN+ → C1: C1OUT
- ANx1: A - VIN- → C2: C2OUT
- ANx2: A - VIN+ → C2: C2OUT

- CIS (CMCON<3>) is the Comparator Input Switch.

A = Analog Input, port reads as zeros always.
D = Digital Input.
24.4 Comparator Operation

A single comparator is shown in Figure 24-2 along with the relationship between the analog input levels and the digital output. When the analog input at \( V_{IN+} \) is less than the analog input \( V_{IN-} \), the output of the comparator is a digital low level. When the analog input at \( V_{IN+} \) is greater than the analog input \( V_{IN-} \), the output of the comparator is a digital high level. The shaded areas of the output of the comparator (shown in Figure 24-2) represent the uncertainty due to input offsets and response time.

24.5 Comparator Reference

An external or internal reference signal may be used depending on the comparator operating mode. The analog signal that is present at \( V_{IN-} \) is compared to the signal at \( V_{IN+} \), and the digital output of the comparator is adjusted accordingly (Figure 24-2).

Figure 24-2: Single Comparator
Section 24. Comparator

24.5.1 External Reference Signal

When external voltage references are used, the comparator module can be configured to have the comparators operate from the same or different reference sources. The reference signal must be between VSS and VDD, and can be applied to either pin of the comparator(s).

24.5.2 Internal Reference Signal

The comparator module also allows the selection of an internally generated voltage reference for the comparators. The “Comparator Voltage Reference” section contains a detailed description of the Voltage Reference Module that provides this signal. The internal reference signal is used when the comparators are in mode CM2:CM0 = 110 (Figure 24-1). In this mode, the internal voltage reference is applied to the VIn+ input of both comparators.

The internal voltage reference may be used in any comparator mode. The voltage reference is output to the VREF pin. Any comparator input pin may be connected externally to the VREF pin.
24.6 Comparator Response Time

Response time is the minimum time, after selecting a new reference voltage or input source, before the comparator output is guaranteed to have a valid level. If the internal reference is changed, the maximum settling time of the internal voltage reference must be considered when using the comparator outputs. Otherwise the maximum response time of the comparators should be used.

24.7 Comparator Outputs

The comparator outputs are read through the CMCON register. These bits are read only. The comparator outputs may also be directly output to the I/O pins. When CM2:CM0 = 011, multiplexors in the output path of the I/O pins will switch and the output of each pin will be the unsynchronized output of the comparator. The uncertainty of each of the comparators is related to the input offset voltage and the response time given in the specifications. Figure 24-3 shows the comparator output block diagram.

The TRIS bits will still function as the output enable/disable for the I/O pins while in this mode.

---

**Note 1:** When reading the Port register, all pins configured as analog inputs will read as a '0'. Pins configured as digital inputs will convert an analog input according to the Schmitt Trigger input specification.

**Note 2:** Analog levels on any pin that is defined as a digital input may cause the input buffer to consume more current than is specified.

---

**Figure 24-3: Comparator Output Block Diagram**

![Comparator Output Block Diagram](image-url)
24.8 Comparator Interrupts

The comparator interrupt flag is set whenever the comparators value changes relative to the last value loaded into CMxOUT bits. Software will need to maintain information about the status of the output bits, as read from CMCON<7:6>, to determine the actual change that has occurred. The CMIF bit is the comparator interrupt flag. The CMIF bit must be cleared. Since it is also possible to set this bit, a simulated interrupt may be initiated.

The CMIE bit, the PEIE/GIEL bit, and the GIE/GIEH bit must be set to enable the interrupt. If any of these bits are clear, an interrupt from the comparator module will not occur, though the CMIF bit will still be set if an interrupt condition occurs. Table 24-1 shows the state of the comparator interrupt bits to enable an interrupt to vector to the interrupt vector address. If these conditions are not met, the comparator module will set the CMIF bit, but the program execution will not go to the interrupt vector address.

The user, in the interrupt service routine, can clear the interrupt in the following manner:

a) Any read or write of the CMCON register. This will load the CMCON register with the new value with the CMxOUT bits.

b) Clear the CMIF flag bit.

An interrupt condition will continue to set the CMIF flag bit. Reading CMCON will end the interrupt condition, and allow the CMIF flag bit to be cleared.

Table 24-1: How State of Interrupt Control Bits Determine Action After Comparator Trip (CMIF is Set)

<table>
<thead>
<tr>
<th>GIE</th>
<th>GIEH</th>
<th>PEIE</th>
<th>GIEL</th>
<th>CMIE</th>
<th>IPEN</th>
<th>CMIP</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>—</td>
<td>1</td>
<td>—</td>
<td>1</td>
<td>0</td>
<td>—</td>
<td>CMIF set Branch to ISR</td>
</tr>
<tr>
<td>x</td>
<td>—</td>
<td>x</td>
<td>—</td>
<td>0</td>
<td>0</td>
<td>—</td>
<td>CMIF set</td>
</tr>
<tr>
<td>x</td>
<td>—</td>
<td>0</td>
<td>—</td>
<td>x</td>
<td>0</td>
<td>—</td>
<td>CMIF set</td>
</tr>
<tr>
<td>0</td>
<td>—</td>
<td>x</td>
<td>—</td>
<td>x</td>
<td>0</td>
<td>—</td>
<td>CMIF set</td>
</tr>
<tr>
<td>—</td>
<td>x</td>
<td>—</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>CMIF set Branch to ISR</td>
</tr>
<tr>
<td>—</td>
<td>1</td>
<td>—</td>
<td>x</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>CMIF set Branch to ISR</td>
</tr>
<tr>
<td>—</td>
<td>x</td>
<td>—</td>
<td>x</td>
<td>0</td>
<td>1</td>
<td>x</td>
<td>CMIF set</td>
</tr>
<tr>
<td>—</td>
<td>x</td>
<td>—</td>
<td>0</td>
<td>x</td>
<td>1</td>
<td>0</td>
<td>CMIF set</td>
</tr>
<tr>
<td>—</td>
<td>x</td>
<td>0</td>
<td>—</td>
<td>x</td>
<td>1</td>
<td>1</td>
<td>CMIF set</td>
</tr>
<tr>
<td>—</td>
<td>0</td>
<td>x</td>
<td>—</td>
<td>x</td>
<td>1</td>
<td>1</td>
<td>CMIF set</td>
</tr>
</tbody>
</table>

24.9 Comparator Operation During SLEEP

When a comparator is active and the device is placed in SLEEP mode, the comparator remains active and the interrupt is functional if enabled. This interrupt will wake-up the device from SLEEP mode when enabled. While the comparator is powered up, each comparator that is operational will consume additional current as shown in the comparator specifications. To minimize power consumption while in SLEEP mode, turn off the comparators (CM2:CM0 = 1111), before entering SLEEP. If the device wakes up from SLEEP, the contents of the CMCON register are not affected.

24.10 Effects of a RESET

A device RESET forces the CMCON register to its reset state. This forces the comparator module to be in the comparator reset mode, CM2:CM0 = 000. This ensures that all potential inputs are analog inputs. Device current is minimized when analog inputs are present at RESET time. The comparators will be powered down disabled during the RESET interval.
24.11 Analog Input Connection Considerations

A simplified circuit for an analog input is shown in Figure 24-4. Since the analog pins are connected to a digital output, they have reverse biased diodes to VDD and VSS. The analog input therefore, must be between VSS and VDD. If the input voltage deviates from this range by more than 0.6V in either direction, one of the diodes is forward biased and a latch-up may occur. A maximum source impedance of 10 kΩ is recommended for the analog sources.

![Figure 24-4: Analog Input Model](image)

Legend:
- CPIN = Input Capacitance
- VT = Threshold Voltage
- ILEAKAGE = Leakage Current at the pin due to various junctions
- RIC = Interconnect Resistance
- Rs = Source Impedance
- VA = Analog Voltage

Table 24-2: Registers Associated with Comparator Module

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
<th>Value on POR, BOR</th>
<th>Value on All Other Resets</th>
</tr>
</thead>
<tbody>
<tr>
<td>CMCON</td>
<td>C2OUT</td>
<td>C1OUT</td>
<td>—</td>
<td>—</td>
<td>CIS</td>
<td>CM2</td>
<td>CM1</td>
<td>CM0</td>
<td>00-- 0000</td>
<td>00-- 0000</td>
</tr>
<tr>
<td>VRCON</td>
<td>VREN</td>
<td>VROE</td>
<td>VR</td>
<td>—</td>
<td>VR3</td>
<td>VR2</td>
<td>VR1</td>
<td>VR0</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>INTCON</td>
<td>GIE/GIEH</td>
<td>PEIE/GIEL</td>
<td>TMR0IE</td>
<td>INTOIE</td>
<td>RBIE</td>
<td>TMR0IF</td>
<td>INTOIF</td>
<td>RBIF</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PIR</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>CMIF</td>
<td>—</td>
<td>—</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>PIE</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>CMIE</td>
<td>—</td>
<td>—</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>IPE</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>CMIP</td>
<td>—</td>
<td>—</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
</tbody>
</table>

Legend: x = unknown, – = unimplemented locations read as '0'.

Shaded cells are not used for Comparator Module.

Note 1: The position of this bit is device dependent.
24.12 Initialization

The code in Example 24-1 depicts example steps required to configure the comparator module. The Port registers (PORTx, LATx, and TRISx) need to be configured appropriately depending on the mode selected. For CM2:CM0 = 100, the I/O multiplied with ANx0, ANx1, and ANx2 needs to be configured for analog inputs. Other I/O may be digital.

Example 24-1: Initializing Comparator Module

```assembly
FLAG_REG EQU 0x020

; Init flag register
CLRF FLAG_REG

; Init the desired port
CLRF PORTx

; Mask comparator bits
ANDLW 0xC0

; Store bits in flag register
IORWF FLAG_REG,F

; Init comparator mode
MOVLW 0x04

; CM<2:0> = 100
MOVWF CMCON

; Initialize data direction of the ANx0, ANx1, and ANx2. Set as inputs, other I/O on port as desired (either inputs or outputs)
MOVLW PORTxDIR

; Read CMCON to end change condition
CALL DELAY10

; 10us delay
MOVF CMCON, W

; Clear pending interrupts
BCF PIR1,CMIF

; Enable comparator interrupts
BSF PIE1,CMIE

; Enable peripheral interrupts
BSF INTCON,PEIE

; Global interrupt enable
BSF INTCON,GIE
```
24.13 Design Tips

Question 1: My program appears to lock up.
Answer 1:
You may be getting stuck in an infinite loop with the comparator interrupt service routine if you did not follow the proper sequence to clear the CMIF flag bit. First, you must read the CMCON register and then you can clear the CMIF flag bit.
## Section 24. Comparator

### 24.14 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is, they may be written for the Base-Line, Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the comparator module are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Resistance and Capacitance Meter using a PIC16C622</td>
<td>AN611</td>
</tr>
</tbody>
</table>

**Note:** Please visit the Microchip Web site for additional software code examples. These code examples are stand-alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
24.15 Revision History

Revision A

This is the initial released revision of the Comparator module description.
Section 25. Compatible 10-bit A/D Converter

HIGHLIGHTS

This section of the manual contains the following major topics:

25.1 Introduction ............................................................................................................... 25-2
25.2 Control Register ..................................................................................................... 25-4
25.3 Operation ................................................................................................................. 25-7
25.4 A/D Acquisition Requirements .............................................................................. 25-8
25.5 Selecting the A/D Conversion Clock ...................................................................... 25-10
25.6 Configuring Analog Port Pins ............................................................................... 25-11
25.7 A/D Conversions .................................................................................................... 25-12
25.8 Operation During SLEEP ....................................................................................... 25-16
25.9 Effects of a RESET ............................................................................................... 25-16
25.10 A/D Accuracy/Error ............................................................................................. 25-17
25.11 Connection Considerations .................................................................................. 25-18
25.12 Transfer Function ................................................................................................ 25-18
25.13 Initialization .......................................................................................................... 25-19
25.14 Design Tips .......................................................................................................... 25-20
25.15 Related Application Notes ................................................................................... 25-21
25.16 Revision History .................................................................................................. 25-22

© 2000 Microchip Technology Inc.
25.1 Introduction

The compatible analog-to-digital (A/D) converter module is software compatible with the Standard 10-bit A/D converter and can have up to sixteen analog inputs.

The analog input charges a sample and hold capacitor. The output of the sample and hold capacitor is the input into the converter. The converter then generates a digital result of this analog level via successive approximation. This A/D conversion of the analog input signal results in a corresponding 10-bit digital number.

The analog reference voltages (positive and negative supply) are software selectable to either the device’s supply voltages (AVDD, AVss) or the voltage level on the AN3/\text{VREF+} and AN2/\text{VREF-} pins.

The A/D converter has the unique feature of being able to convert while the device is in SLEEP mode.

The A/D module has four registers. These registers are:
- A/D Result High Register (ADRESH)
- A/D Result Low Register (ADRESL)
- A/D Control Register0 (ADCON0)
- A/D Control Register1 (ADCON1)

The ADCON0 register, shown in Register 25-1, controls the operation of the A/D module. The ADCON1 register, shown in Register 25-2, configures the functions of the port pins. The port pins can be configured as analog inputs (AN3 and AN2 can also be the voltage references) or as digital I/O.
Section 25. Compatible 10-bit A/D Converter

Figure 25-1: Compatible 10-bit A/D Block Diagram

Note: Not all 16 input channels may be implemented on every device. Unimplemented selections are reserved and must not be selected.
25.2 Control Register

ADCON0 (Register 25-1) is used to select the clock and the analog channel. ADCON1 (Register 25-2) configures the port logic to either analog or digital inputs and the format of the result.

Register 25-1: ADCON0 Register

<table>
<thead>
<tr>
<th>Bit 7-6</th>
<th>ADCS1:ADCS0: A/D Conversion Clock Select bits (shown in bold)</th>
</tr>
</thead>
<tbody>
<tr>
<td>00</td>
<td>FOSC/2</td>
</tr>
<tr>
<td>001</td>
<td>FOSC/8</td>
</tr>
<tr>
<td>010</td>
<td>FOSC/32</td>
</tr>
<tr>
<td>011</td>
<td>FRC (clock derived from the internal A/D RC oscillator)</td>
</tr>
<tr>
<td>100</td>
<td>FOSC/4</td>
</tr>
<tr>
<td>101</td>
<td>FOSC/16</td>
</tr>
<tr>
<td>110</td>
<td>FOSC/64</td>
</tr>
<tr>
<td>111</td>
<td>FRC (clock derived from the internal A/D RC oscillator)</td>
</tr>
</tbody>
</table>

Note: The ADCS2 bit is located in the ADCON1 register.

bit 5-3: CHS2:CHS0: Analog Channel Select bits

There are four bits that select the A/D channel. These are CHS3:CHS0.

| 0000   | channel 0, (AN0)                                           |
| 0001   | channel 1, (AN1)                                           |
| 0010   | channel 2, (AN2)                                           |
| 0011   | channel 3, (AN3)                                           |
| 0100   | channel 4, (AN4)                                           |
| 0101   | channel 5, (AN5)                                           |
| 0110   | channel 6, (AN6)                                           |
| 0111   | channel 7, (AN7)                                           |
| 1000   | channel 8, (AN8)                                           |
| 1001   | channel 9, (AN9)                                           |
| 1010   | channel 10, (AN10)                                         |
| 1011   | channel 11, (AN11)                                         |
| 1100   | channel 12, (AN12)                                         |
| 1101   | channel 13, (AN13)                                         |
| 1110   | channel 14, (AN14)                                         |
| 1111   | channel 15, (AN15)                                         |

Note: For devices that do not implement the full 16 A/D channels, the unimplemented selections are reserved. Do not select any unimplemented channel.
Section 25. Compatible 10-bit A/D Converter

bit 2: \textbf{GO/DONE}: A/D Conversion Status bit

\begin{verbatim}
When ADON = 1
1 = A/D conversion in progress. Setting this bit starts an A/D conversion cycle.
This bit is automatically cleared by hardware when the A/D conversion is completed.
0 = A/D conversion not in progress
\end{verbatim}

bit 1: \textbf{CHS3}: Analog Channel Select bit

The CHS2:CHS0 bits are located in positions bit 5 to bit 3. See the CHS2:CHS0 description for
operational details.

bit 0: \textbf{ADON}: A/D On bit

\begin{verbatim}
1 = A/D converter module is powered up
0 = A/D converter module is shut off and consumes no operating current
\end{verbatim}

Legend

\begin{itemize}
\item R = Readable bit
\item W = Writable bit
\item U = Unimplemented bit, read as ‘0’
\item ‘n’ = Value at POR reset
\item ‘1’ = bit is set
\item ‘0’ = bit is cleared
\item x = bit is unknown
\end{itemize}
Register 25-2: ADCON1 Register

<table>
<thead>
<tr>
<th>R/W-0</th>
<th>R/W-0</th>
<th>U-0</th>
<th>—</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADFM</td>
<td>ADCS2</td>
<td>—</td>
<td>—</td>
<td>PCFG3</td>
<td>PCFG2</td>
<td>PCFG1</td>
<td>PCFG0</td>
</tr>
</tbody>
</table>

bit 7: **ADFM**: A/D Result Format Select (also see Figure 25-6)

1 = Right justified. 6 Most Significant bits of ADRESH are read as ‘0’.
0 = Left justified. 6 Least Significant bits of ADRESL are read as ‘0’.

bit 6: **ADCS2**: A/D Conversion Clock Select bits (shown in bold)

Three bits are required to select the A/D clock source. These bits are ADCS2:ADCS0.

- 000 = FOSC/2
- 001 = FOSC/8
- 010 = FOSC/32
- 011 = FRC (clock derived from the internal A/D RC oscillator)
- 100 = FOSC/4
- 101 = FOSC/16
- 110 = FOSC/64
- 111 = FRC (clock derived from the internal A/D RC oscillator)

**Note:** The ADCS1:ADCS0 bits are located in the ADCON0 register.

bit 5-4: Unimplemented: Read as ‘0’

bit 3-0: **PCFG3:PCFG0**: A/D Port Configuration Control bits

Legend

- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- n = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown

Note 1: On any device RESET, the port pins that are multiplexed with analog functions (ANx) are forced to be an analog input.
Section 25. Compatible 10-bit A/D Converter

25.3 Operation

The ADRESH:ADRESL registers contain the 10-bit result of the A/D conversion. When the A/D conversion is complete, the result is loaded into these A/D result register pair (ADRESH:ADRESL), the GO/DONE bit (ADCON0) is cleared, and A/D interrupt flag bit, ADIF, is set. The block diagram of the A/D module is shown in Figure 25-1.

After the A/D module has been configured, the signal on the selected channel must be acquired before the conversion is started. The analog input channels must have their corresponding TRIS bits selected as inputs. To determine acquisition time, see Subsection 25.4 "A/D Acquisition Requirements." After this acquisition time has elapsed, the A/D conversion can be started. The following steps should be followed for doing an A/D conversion:

1. Configure the A/D module:
   - Configure analog pins, voltage reference, and digital I/O (ADCON1)
   - Select A/D input channel (ADCON0)
   - Select A/D conversion clock (ADCON0)
   - Turn on A/D module (ADCON0)
2. Configure A/D interrupt (if desired):
   - Clear the ADIF bit
   - Set the ADIE bit
   - Set/Clear the ADIP bit
   - Set the GIE/GIEH or PEIE/GIEL bit
3. Wait the required acquisition time.
4. Start conversion:
   - Set the GO/DONE bit (ADCON0)
5. Wait for the A/D conversion to complete, by either:
   - Polling for the GO/DONE bit to be cleared or the ADIF bit to be set, or
   - Waiting for the A/D interrupt
6. Read A/D Result register pair (ADRESH:ADRESL): clear the ADIF bit, if required.
7. For next conversion, go to step 1 or step 2 as required.

Figure 25-2 shows the conversion sequence and the terms that are used. Acquisition time is the time that the A/D module's holding capacitor is connected to the external voltage level. When the GO bit is set, the conversion time of 12 TAD is started. The sum of these two times is the sampling time. There is a minimum acquisition time to ensure that the holding capacitor is charged to a level that will give the desired accuracy for the A/D conversion.

Figure 25-2: A/D Conversion Sequence
25.4 A/D Acquisition Requirements

For the A/D converter to meet its specified accuracy, the charge holding capacitor (C\text{HOLD}) must be allowed to fully charge to the input channel voltage level. The analog input model is shown in Figure 25-3. The source impedance (R\text{S}) and the internal sampling switch (R\text{SS}) impedance directly affect the time required to charge the capacitor C\text{HOLD}. The sampling switch (R\text{SS}) impedance varies over the device voltage (V\text{DD}), Figure 25-3. The source impedance affects the offset voltage at the analog input (due to pin leakage current). The maximum recommended impedance for analog sources is 2.5 k\text{\Omega}. As the impedance is decreased, the acquisition time may be decreased. After the analog input channel is selected (changed), this acquisition must pass before the conversion can be started.

To calculate the minimum acquisition time, Equation 25-1 may be used. This equation assumes that 1/2 LSB error is used (1024 steps for the A/D). The 1/2 LSB error is the maximum error allowed for the A/D to meet its specified resolution.

**Note:** When the conversion is started, the holding capacitor is disconnected from the input pin.

**Equation 25-1: Acquisition Time**

\[
T_{\text{ACQ}} = T_{\text{AMP}} + T_{\text{C}} + T_{\text{COFF}}
\]

**Equation 25-2: A/D Minimum Charging Time**

\[
V_{\text{HOLD}} = (V_{\text{REF}} - (V_{\text{REF}}/2048)) \cdot (1 - e^{(T_{\text{C}}/C_{\text{HOLD}}(R_{\text{IC}} + R_{\text{SS}} + R_{\text{S}})))}
\]

or

\[
T_{\text{C}} = -(120 \text{ pF})(1 \text{ k\Omega} + R_{\text{SS}} + R_{\text{S}}) \ln(1/2047)
\]

Example 25-1 shows the calculation of the minimum required acquisition time T_{\text{ACQ}}. This calculation is based on the following application system assumptions.

- C_{\text{HOLD}} = 120 \text{ pF}
- R_{\text{S}} = 2.5 \text{ k\Omega}
- Conversion Error \leq 1/2 \text{ LSB}
- V_{\text{DD}} = 5 \text{ V} \rightarrow R_{\text{SS}} = 7 \text{ k\Omega} \quad \text{(see graph in Figure 25-3)}
- Temperature = 50^\circ \text{C} \text{ (system max.)}
- V_{\text{HOLD}} = 0 \text{ V @ time = 0}

**Example 25-1: Calculating the Minimum Required Acquisition Time (Case 1)**

\[
T_{\text{ACQ}} = T_{\text{AMP}} + T_{\text{C}} + T_{\text{COFF}}
\]

Temperature coefficient is only required for temperatures > 25^\circ \text{C}.

\[
T_{\text{ACQ}} = 2 \mu \text{s} + T_{\text{C}} + [(\text{Temp} - 25^\circ \text{C})(0.05 \mu \text{s/}^\circ \text{C})]
\]

\[
T_{\text{C}} = C_{\text{HOLD}} (R_{\text{IC}} + R_{\text{SS}} + R_{\text{S}}) \ln(1/2047)
\]

-120 pF (1 k\text{\Omega} + 7 k\text{\Omega} + 2.5 k\text{\Omega}) \ln(0.0004885)
-120 pF (10.5 k\text{\Omega}) \ln(0.0004885)
-1.26 \mu \text{s} \quad (7.6241)
9.61 \mu \text{s}

\[
T_{\text{ACQ}} = 2 \mu \text{s} + 9.61 \mu \text{s} + [(50^\circ \text{C} - 25^\circ \text{C})(0.05 \mu \text{s/}^\circ \text{C})]
\]

11.61 \mu \text{s} + 1.25 \mu \text{s}
12.86 \mu \text{s}
Section 25. Compatible 10-bit A/D Converter

Now to get an idea what happens to the acquisition time when the source impedance is a minimal value (Rs = 50 Ω). Example 25-2 shows the same conditions as in Example 25-1 with only the source impedance changed to the minimal value.

Example 25-2: Calculating the Minimum Required Acquisition Time (Case 2)

\[ T_{ACQ} = T_{AMP} + T_C + T_{COFF} \]

Temperature coefficient is only required for temperatures > 25°C.

\[ T_{ACQ} = 2 \mu s + T_c + \left(\frac{(Temp - 25°C)(0.05 \mu s/°C)}{25°C} \right) \]

\[ T_C = \frac{-Chold (R_{ic} + R_{ss} + Rs) \ln(1/2047)}{-120 \text{ pF} (1 \text{ kΩ} + 7 \text{ kΩ} + 50 \Omega) \ln(0.0004885)} \]
\[ + \left(\frac{-120 \text{ pF} (8050 \Omega) \ln(0.0004885)}{-0.966 \mu s (-7.6241)} \right) \]
\[ + 7.36 \mu s \]

\[ T_{ACQ} = 2 \mu s + 16.47 \mu s + \left(\frac{(50°C - 25°C)(0.05 \mu s/°C)}{25°C} \right) \]
\[ + 9.36 \mu s + 1.25 \mu s \]
\[ + 10.61 \mu s \]

Note 1: The reference voltage (VREF) has no effect on the equation, since it cancels itself out.

2: The charge holding capacitor (Chold) is not discharged after each conversion.

3: The maximum recommended impedance for analog sources is 2.5 kΩ. This is required to meet the pin leakage specification.

4: After a conversion has completed, a 2 TAD delay must complete before acquisition can begin again. During this time the holding capacitor is not connected to the selected A/D input channel.

Figure 25-3: Analog Input Model

Legend
- CPIN: Input Capacitance
- VT: Threshold Voltage
- ILEAKAGE: Leakage Current at the pin due to various junctions
- RIC: Interconnect Resistance
- SS: Sampling Switch
- CHOLD: Sample/Hold Capacitance (from DAC)
25.5 Selecting the A/D Conversion Clock

The A/D conversion time per bit is defined as $T_{AD}$. The A/D conversion requires $11.5T_{AD}$ per 10-bit conversion. The source of the A/D conversion clock is software selectable. The seven possible options for $T_{AD}$ are:

- 2Tosc
- 4Tosc
- 8Tosc
- 16Tosc
- 32Tosc
- 64Tosc
- Internal A/D RC oscillator

For correct A/D conversions, the A/D conversion clock ($T_{AD}$) must be selected to ensure a minimum $T_{AD}$ time of 1.6 $\mu$s as shown in Electrical Specifications parameter 130.

Table 25-1 and Table 25-2 show the resultant $T_{AD}$ times derived from the device operating frequencies and the selected A/D clock source.

### Table 25-1: $T_{AD}$ vs. Device Operating Frequencies (for Standard, C, Devices)

<table>
<thead>
<tr>
<th>AD Clock Source (TAD)</th>
<th>Operation</th>
<th>Device Frequency</th>
</tr>
</thead>
<tbody>
<tr>
<td>2Tosc</td>
<td>000</td>
<td>100 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1.6 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>6 $\mu$s</td>
</tr>
<tr>
<td>4Tosc</td>
<td>100</td>
<td>200 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>800 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3.2 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>12 $\mu$s</td>
</tr>
<tr>
<td>8Tosc</td>
<td>001</td>
<td>400 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1.6 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>6.4 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>24 $\mu$s (3)</td>
</tr>
<tr>
<td>16Tosc</td>
<td>101</td>
<td>800 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3.2 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>12.8 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>48 $\mu$s (3)</td>
</tr>
<tr>
<td>32Tosc</td>
<td>010</td>
<td>1.6 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>6.4 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>25.6 $\mu$s (3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>96 $\mu$s (3)</td>
</tr>
<tr>
<td>64Tosc</td>
<td>110</td>
<td>3.2 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>12.8 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>51.2 $\mu$s (3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>192 $\mu$s (3)</td>
</tr>
<tr>
<td>RC</td>
<td>011</td>
<td>2 - 6 $\mu$s (1,4)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>2 - 6 $\mu$s (1,4)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>2 - 6 $\mu$s (1)</td>
</tr>
</tbody>
</table>

Legend: Shaded cells are outside of recommended range.

Note 1: The RC source has a typical $T_{AD}$ of 4 $\mu$s.

Note 2: These values violate the minimum required $T_{AD}$.

Note 3: For faster conversion times, the selection of another clock source is recommended.

Note 4: For device frequencies above 1 MHz, the device must be in SLEEP for the entire conversion, or the A/D accuracy may be out of specification.

### Table 25-2: $T_{AD}$ vs. Device Operating Frequencies (for Extended, LC, Devices)

<table>
<thead>
<tr>
<th>AD Clock Source (TAD)</th>
<th>Operation</th>
<th>Device Frequency</th>
</tr>
</thead>
<tbody>
<tr>
<td>2Tosc</td>
<td>000</td>
<td>500 ns (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1.0 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1.6 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>6 $\mu$s</td>
</tr>
<tr>
<td>4Tosc</td>
<td>100</td>
<td>1.0 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>2.0 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3.2 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>12 $\mu$s</td>
</tr>
<tr>
<td>8Tosc</td>
<td>001</td>
<td>2.0 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>4.0 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>6.4 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>24 $\mu$s (3)</td>
</tr>
<tr>
<td>16Tosc</td>
<td>101</td>
<td>4.0 $\mu$s (2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>8.0 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>12.8 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>48 $\mu$s (3)</td>
</tr>
<tr>
<td>32Tosc</td>
<td>010</td>
<td>8.0 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>16.0 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>25.6 $\mu$s (3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>96 $\mu$s (3)</td>
</tr>
<tr>
<td>64Tosc</td>
<td>110</td>
<td>16.0 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>32.0 $\mu$s</td>
</tr>
<tr>
<td></td>
<td></td>
<td>51.2 $\mu$s (3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>192 $\mu$s (3)</td>
</tr>
<tr>
<td>RC</td>
<td>011</td>
<td>3 - 9 $\mu$s (1,4)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3 - 9 $\mu$s (1,4)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3 - 9 $\mu$s (1)</td>
</tr>
</tbody>
</table>

Legend: Shaded cells are outside of recommended range.

Note 1: The RC source has a typical $T_{AD}$ of 6 $\mu$s.

Note 2: These values violate the minimum required $T_{AD}$.

Note 3: For faster conversion times, the selection of another clock source is recommended.

Note 4: For device frequencies above 1 MHz, the device must be in SLEEP for the entire conversion, or the A/D accuracy may be out of specification.
Section 25. Compatible 10-bit A/D Converter

25.6 Configuring Analog Port Pins

The ADCON1 and TRIS registers control the operation of the A/D port pins. The port pins that are desired as analog inputs must have their corresponding TRIS bits set (input). If the TRIS bit is cleared (output), the digital output level (VOH or VOL) will be converted. After a device RESET, pins that are multiplexed with analog inputs will be configured as an analog input. The corresponding TRIS bit will be set.

The A/D operation is independent of the state of the CHS2:CHS0 bits and the TRIS bits.

Note 1: When reading the port register, any pin configured as an analog input channel will read as cleared (a low level). Pins configured as digital inputs, will convert an analog input. Analog levels on a digitally configured input will not affect the conversion accuracy.

2: Analog levels on any pin that is defined as a digital input (including the AN7:AN0 pins), may cause the input buffer to consume current that is out of the devices specification.
25.7 A/D Conversions

Example 25-3 shows how to perform an A/D conversion. The port pins are configured as analog inputs. The analog references (VREF+ and VREF-) are the device AVDD and AVSS. The A/D interrupt is enabled, and the A/D conversion clock is FRC. The conversion is performed on the AN0 pin (channel 0). The result of the conversion is left justified.

**Note:** The GO/DONE bit should NOT be set in the same instruction that turns on the A/D, due to the required acquisition time.

Clearing the GO/DONE bit during a conversion will abort the current conversion. The A/D result register pair will NOT be updated with the partially completed A/D conversion sample. That is, the ADRESH:ADRESL registers will continue to contain the value of the last completed conversion (or the last value written to the ADRESH:ADRESL registers). After the A/D conversion is aborted, a 2TAD wait is required before the next acquisition is started. After this 2TAD wait, acquisition on the selected channel is automatically started.

**Example 25-3: A/D Conversion**

```
CLRF ADCON1 ; Configure A/D inputs,
; result is left justified
BSF IPR1, ADIP ; High priority
BSF PIE1, ADIE ; Enable A/D interrupts
MOVLW 0xC1 ; RC Clock, A/D is on,
MOVF ADCON0 ; Channel 0 is selected
BCF PIR1, ADIF ; Clear A/D interrupt flag bit
BSF INTCON, PEIE ; Enable peripheral interrupts
BSF INTCON, GIE ; Enable all interrupts

; Ensure that the required sampling time for the selected input
; channel has elapsed. Then the conversion may be started.
;
BSF ADCON0, GO ; Start A/D Conversion
; The ADIF bit will be set and the
; GO/DONE bit is cleared upon
; completion of the A/D Conversion.
```

**Figure 25-4: A/D Conversion TAD Cycles**

- TAD1
- TAD2
- TAD3
- TAD4
- TAD5
- TAD6
- TAD7
- TAD8
- TAD9
- TAD10
- TAD11

Conversion Starts

Holding capacitor is disconnected from analog input (typically 100 ns)

Set GO bit

Next Q4: ADRES is loaded,
GO bit is cleared,
ADIF bit is set,
holding capacitor is connected to analog input.
Section 25. Compatible 10-bit A/D Converter

Figure 25-5: Flowchart of A/D Operation
25.7.1 Faster Conversion - Lower Resolution Trade-off

Not all applications require a result with 10-bits of resolution, but may instead require a faster conversion time. The A/D module allows users to make the trade-off of conversion speed to resolution. Regardless of the resolution required, the acquisition time is the same. To speed up the conversion, the clock source of the A/D module may be switched so that the $T_{AD}$ time violates the minimum specified time (see electrical specification parameter 130). Once the $T_{AD}$ time violates the minimum specified time, all the following A/D result bits are not valid (see A/D Conversion Timing in the Electrical Specifications section). The clock sources may only be switched between the three oscillator versions (cannot be switched from/to RC). The equation to determine the time before the oscillator can be switched is as follows:

Since the $T_{AD}$ is based from the device oscillator, the user must use some method (a timer, software loop, etc.) to determine when the A/D oscillator may be changed. Example 25-4 shows a comparison of time required for a conversion with 4-bits of resolution, versus the 10-bit resolution conversion. The example is for devices operating at 20 MHz (the A/D clock is programmed for $32T_{OSC}$), and assumes that immediately after $6T_{AD}$, the A/D clock is programmed for $2T_{OSC}$.

The $2T_{OSC}$ violates the minimum $T_{AD}$ time since the last 6 bits will not be converted to correct values.

Example 25-4: 4-bit vs. 10-bit Conversion Times

<table>
<thead>
<tr>
<th></th>
<th>Resolution</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>4-bit</td>
</tr>
<tr>
<td>$T_{AD}$</td>
<td>40 1.6 $\mu$s</td>
</tr>
<tr>
<td>$T_{OSC}$</td>
<td>40 25 ns</td>
</tr>
<tr>
<td>$T_{AD} + N \cdot T_{AD} + (11 - N)(2T_{OSC})$</td>
<td>40 8.5 $\mu$s</td>
</tr>
</tbody>
</table>

Note 1: A minimum $T_{AD}$ time of 1.6 $\mu$s is required.

2: If the full 10-bit conversion is required, the A/D clock source should not be changed.

Equation 25-3: Resolution/Speed Conversion Trade-off

\[
\text{Conversion time} = T_{AD} + N \cdot T_{AD} + (11 - N)(2T_{OSC})
\]

Where: $N = \text{number of bits of resolution required}$
Section 25. Compatible 10-bit A/D Converter

25.7.2 A/D Result Registers

The ADRESH:ADRESL register pair is the location where the 10-bit A/D result is loaded at the completion of the A/D conversion. This register pair is 16-bits wide. The A/D module gives the flexibility to left or right justify the 10-bit result in the 16-bit result register. The A/D Format Select bit (ADFM) controls this justification. Figure 25-6 shows the operation of the A/D result justification. The extra bits are loaded with ‘0’s’. When the A/D module is disabled these registers may be used as two general purpose 8-bit registers.

Figure 25-6: A/D Result Justification

![Diagram showing A/D Result Justification]

Right Justified

Left Justified

25

Compatible 10-bit
A/D Converter
25.8 Operation During SLEEP

The A/D module can operate during SLEEP mode. This requires that the A/D clock source be set to RC (ADCS2:ADCS0 = x11). When the RC clock source is selected, the A/D module waits one instruction cycle before starting the conversion. This allows the SLEEP instruction to be executed, which eliminates all internal digital switching noise from the conversion. When the conversion is completed, the GO/DONE bit will be cleared and the result is loaded into the ADRESH:ADRESL registers. If the A/D interrupt is enabled, the device will wake-up from SLEEP. If the A/D interrupt is not enabled, the A/D module will be turned off, although the ADON bit will remain set.

When the A/D clock source is another clock option (not RC), a SLEEP instruction will cause the present conversion to be aborted and the A/D module to be turned off (to conserve power), though the ADON bit will remain set.

Turning off the A/D places the A/D module in its lowest current consumption state.

Note: For the A/D module to operate in SLEEP, the A/D clock source must be set to RC (ADCS2:ADCS0 = x11). To allow the conversion to occur during SLEEP, ensure the SLEEP instruction immediately follows the instruction that sets the GO/DONE bit.

25.9 Effects of a RESET

A device RESET forces all registers to their RESET state. This forces the A/D module to be turned off, and any conversion is aborted. All pins that are multiplexed with analog inputs will be configured as an analog input. The corresponding TRIS bits will be set.

The value that is in the ADRESH:ADRESL registers is not initialized from a Power-on Reset. The ADRESH:ADRESL registers will contain unknown data after a Power-on Reset.
Section 25. Compatible 10-bit A/D Converter

25.10 A/D Accuracy/Error

In systems where the device frequency is low, use of the A/D RC clock is preferred. At moderate to high frequencies, $T_{AD}$ should be derived from the device oscillator.

For a given range of analog inputs, the output digital code will be the same. This is due to the quantization of the analog input to a digital code. Quantization error is typically ± 1/2 LSb and is inherent in the analog to digital conversion process. The only way to reduce quantization error is to increase the resolution of the A/D converter.

Offset error measures the first actual transition of a code versus the first ideal transition of a code. Offset error shifts the entire transfer function. Offset error can be calibrated out of a system or introduced into a system, through the interaction of the total leakage current and source impedance at the analog input.

Gain error measures the maximum deviation of the last actual transition and the last ideal transition, adjusted for offset error. This error appears as a change in slope of the transfer function. The difference in gain error to full scale error, is that full scale does not take offset error into account. Gain error can be calibrated out in software.

Linearity error refers to the uniformity of the code changes. Linearity errors cannot be calibrated out of the system. Integral non-linearity error measures the actual code transition versus the ideal code transition, adjusted by the gain error for each code.

Differential non-linearity measures the maximum actual code width versus the ideal code width. This measure is unadjusted.

The maximum pin leakage current is specified in Electrical Specifications parameter D060. $T_{AD}$ must not violate the minimum and should be minimized to reduce inaccuracies due to noise and sampling capacitor bleed off.

In systems where the device will enter SLEEP mode after the start of the A/D conversion, the RC clock source selection is required. In this mode, the digital noise from the modules in SLEEP are stopped. This method gives high accuracy.
25.11 Connection Considerations

If the input voltage exceeds the rail values (VSS or VDD) by greater than 0.3V, then the accuracy of the conversion is out of specification.

An external RC filter is sometimes added for anti-aliasing of the input signal. The R component should be selected to ensure that the total source impedance is kept under the 2.5 k\(\Omega\) recommended specification. Any external components connected (via hi-impedance) to an analog input pin (capacitor, zener diode, etc.) should have very little leakage current at the pin.

25.12 Transfer Function

The ideal transfer function of the A/D converter is as follows: the first transition occurs when the analog input voltage \((V_{AIN})\) is 1 LSb (or Analog \(V_{REF} / 1024\)) (Figure 25-7).

![Figure 25-7: A/D Transfer Function](image)
Section 25. Compatible 10-bit A/D Converter

25.13 Initialization

Example 25-5 shows an initialization of the A/D module.

Example 25-5: A/D Initialization

```
CLRF ADCON1 ; Configure A/D inputs
BSF PIE1, ADIE ; Enable A/D interrupts
BSF IPR1, ADIP ; High Priority
MOVLW 0xC1 ; RC Clock, A/D is on,
MOVWF ADCON0 ; Channel 0 is selected
MOVLW 0x4E ; Left Justified, AN0 is analog
MOVWF ADCON1 ; Vref comes from AVDD and AVSS
BCF PIR1, ADIF ; Clear A/D interrupt flag bit
BSF INTCON, PEIE ; Enable peripheral interrupts
BSF INTCON, GIE ; Enable all interrupts
;
; Ensure that the required sampling time for the selected input
; channel has elapsed. Then the conversion may be started.
;
BSF ADCON0, GO ; Start A/D Conversion
; The ADIF bit will be set and the
; GO/DONE bit is cleared upon
; completion of the A/D conversion.
```
25.14 Design Tips

Question 1: I find that the Analog to Digital Converter result is not always accurate. What can I do to improve accuracy?

Answer 1:

1. Make sure you are meeting all of the timing specifications. If you are turning the module off and on, there is a minimum delay you must wait before taking a sample. If you are changing input channels, there is a minimum delay you must wait for this as well, and finally there is TAD, which is the time selected for each bit conversion. This is selected in ADCON0 and should be between 1.6 and 6 $\mu$s. If TAD is too short, the result may not be fully converted before the conversion is terminated, and if TAD is made too long, the voltage on the sampling capacitor can decay before the conversion is complete. These timing specifications are provided in the “Electrical Specifications” section. See the device data sheet for device specific information.

2. Often the source impedance of the analog signal is high (greater than 1 kOhms), so the current drawn from the source to charge the sample capacitor can affect accuracy. If the input signal does not change too quickly, try putting a 0.1 $\mu$F capacitor on the analog input. This capacitor will charge to the analog voltage being sampled and supply the instantaneous current needed to charge the 120 pF internal holding capacitor.

3. In systems where the device frequency is low, use of the A/D clock derived from the device oscillator is preferred...this reduces, to a large extent, the effects of digital switching noise. In systems where the device will enter SLEEP mode after start of A/D conversion, the RC clock source selection is required. This method gives the highest accuracy.

Question 2: After starting an A/D conversion may I change the input channel (for my next conversion)?

Answer 2:

After the holding capacitor is disconnected from the input channel, typically 100 ns after the GO bit is set, the input channel may be changed.

Question 3: Do you know of a good reference on A/D’s?

Answer 3:


Question 4: I migrated my code from a PIC18CXX8 device with 10-bit A/D to another device with a 10-bit A/D (such as a PIC18CXX2) and the A/D does not seem to operate the same. What’s going on?

Answer 4:

The 10-bit A/D on the PIC18CXX2 device is the compatible 10-bit A/D module. This module has its ADCON bits in the same locations as the PICmicro’s Mid-Range 10-bit A/D module. The standard PIC18CXXX 10-bit A/D module (as found on the PIC18CXX8 device) has optimized the bit locations to ease configuration of the module.
# Section 25. Compatible 10-bit A/D Converter

## 25.15 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the 10-bit A/D module are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using the Analog to Digital Converter</td>
<td>AN546</td>
</tr>
<tr>
<td>Four Channel Digital Voltmeter with Display and Keyboard</td>
<td>AN557</td>
</tr>
</tbody>
</table>

**Note:** Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXFamily. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
25.16 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Compatible 10-bit A/D module description.
Section 26. 10-bit A/D Converter

HIGHLIGHTS

This section of the manual contains the following major topics:

26.1 Introduction ............................................................................................................... 26-2
26.2 Control Register ........................................................................................................... 26-4
26.3 Operation .................................................................................................................. 26-7
26.4 A/D Acquisition Requirements ..................................................................................... 26-8
26.5 Selecting the A/D Conversion Clock ........................................................................ 26-10
26.6 Configuring Analog Port Pins .................................................................................... 26-11
26.7 A/D Conversions ........................................................................................................ 26-12
26.8 Operation During SLEEP ........................................................................................... 26-16
26.9 Effects of a RESET .................................................................................................... 26-16
26.10 A/D Accuracy/Error ............................................................................................... 26-17
26.11 Connection Considerations ...................................................................................... 26-18
26.12 Transfer Function .................................................................................................... 26-18
26.13 Initialization ............................................................................................................ 26-19
26.14 Design Tips ............................................................................................................. 26-20
26.15 Related Application Notes ...................................................................................... 26-21
26.16 Revision History ..................................................................................................... 26-22
26.1 Introduction

The 10-bit Analog-to-Digital (A/D) Converter module can have up to sixteen analog inputs. The analog input charges a sample and hold capacitor. The output of the sample and hold capacitor is the input into the converter. The converter then generates a digital result of this analog level via successive approximation. This A/D conversion of the analog input signal results in a corresponding 10-bit digital number.

The analog reference voltages (positive and negative supply) are software selectable to either the device’s supply voltages (AVDD, AVss) or the voltage level on the AN3/VREF+ and AN2/VREF- pins.

The A/D converter has the unique feature of being able to convert while the device is in SLEEP mode.

The A/D module has five registers. These registers are:

- A/D Result High Register (ADRESH)
- A/D Result Low Register (ADRESL)
- A/D Control Register0 (ADCON0)
- A/D Control Register1 (ADCON1)
- A/D Control Register2 (ADCON2)

The ADCON0 register, shown in Register 26-1, selects the input channel of the A/D module. The ADCON1 register, shown in Register 26-2, configures the functions of the port pins and the Voltage Reference for the A/D module. The port pins can be configured as analog inputs (AN3 and AN2 can also be the Voltage References) or as digital I/O. ADCON2 selects the A/D conversion clock source and the format of the A/D result.
Section 26. 10-bit A/D Converter

Figure 26-1: 10-bit A/D Block Diagram

Note: Not all 16 input channels may be implemented on every device. Unimplemented selections are reserved and must not be selected.
26.2 Control Register

ADCON0 (Register 26-1) is used to select the analog channel. ADCON1 (Register 26-2) configures the port logic to either analog or digital inputs and the voltage reference source for the A/D. ADCON2 (Register 26-3) selects the source of the A/D clock and the justification of the result.

Register 26-1: ADCON0 Register

<table>
<thead>
<tr>
<th>U-0</th>
<th>U-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>CHS3</td>
<td>CHS2</td>
<td>CHS1</td>
<td>CHS0</td>
<td>GO/DONE</td>
</tr>
<tr>
<td>bit 7</td>
<td>bit 0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7-6: Unimplemented: Read as '0'

bit 5-2: CHS3:CHS0: Analog Channel Select bits
- 0000 = channel 0, (AN0)
- 0001 = channel 1, (AN1)
- 0010 = channel 2, (AN2)
- 0011 = channel 3, (AN3)
- 0100 = channel 4, (AN4)
- 0101 = channel 5, (AN5)
- 0110 = channel 6, (AN6)
- 0111 = channel 7, (AN7)
- 1000 = channel 8, (AN8)
- 1001 = channel 9, (AN9)
- 1010 = channel 10, (AN10)
- 1011 = channel 11, (AN11)
- 1100 = channel 12, (AN12)
- 1101 = channel 13, (AN13)
- 1110 = channel 14, (AN14)
- 1111 = channel 15, (AN15)

bit 1: GO/DONE: A/D Conversion Status bit
- 1 = A/D conversion in progress. Setting this bit starts an A/D conversion cycle.
- This bit is automatically cleared by hardware when the A/D conversion is completed.
- 0 = A/D conversion not in progress

bit 0: ADON: A/D On bit
- 1 = A/D converter module is operating
- 0 = A/D converter module is shut off and consumes no operating current

Legend

- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as '0'
- n = Value at POR reset
- '1' = bit is set
- '0' = bit is cleared
- x = bit is unknown
Section 26. 10-bit A/D Converter

Register 26-2: ADCON1 Register

<table>
<thead>
<tr>
<th>U-0</th>
<th>U-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>—</td>
<td>—</td>
<td>VCFG1</td>
<td>VCFG0</td>
<td>PCFG3</td>
<td>PCFG2</td>
<td>PCFG1</td>
<td>PCFG0</td>
</tr>
</tbody>
</table>

| bit 7-6: | Unimplemented: Read as '0' |
| bit 5-4: | VCFG1:VCFG0: Voltage Reference Configuration bits |

<table>
<thead>
<tr>
<th>A/D VREFH</th>
<th>A/D VREFL</th>
</tr>
</thead>
<tbody>
<tr>
<td>00</td>
<td>AVDD</td>
</tr>
<tr>
<td>01</td>
<td>External VREF+</td>
</tr>
<tr>
<td>10</td>
<td>AVDD</td>
</tr>
<tr>
<td>11</td>
<td>External VREF+</td>
</tr>
</tbody>
</table>

<p>| bit 3-0: | PCFG3:PCFG0: A/D Port Configuration Control bits (f) |</p>
<table>
<thead>
<tr>
<th>AN15</th>
<th>AN14</th>
<th>AN13</th>
<th>AN12</th>
<th>AN11</th>
<th>AN10</th>
<th>AN9</th>
<th>AN8</th>
<th>AN7</th>
<th>AN6</th>
<th>AN5</th>
<th>AN4</th>
<th>AN3</th>
<th>AN2</th>
<th>AN1</th>
<th>AN0</th>
</tr>
</thead>
<tbody>
<tr>
<td>0000</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0001</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0010</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0011</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0100</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0101</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0110</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>0111</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>1000</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>1001</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>1010</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>1011</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>1100</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
<td>A</td>
</tr>
<tr>
<td>1101</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>A</td>
</tr>
<tr>
<td>1110</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
</tr>
<tr>
<td>1111</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
<td>D</td>
</tr>
</tbody>
</table>

A = Analog input   D = Digital I/O

**Note 1:** Selection of an unimplemented channel produces a result of 0xFFF.

Legend
- R = Readable bit
- W = Writable bit
- U = Unimplemented bit, read as ‘0’
- ' - n = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- x = bit is unknown

© 2000 Microchip Technology Inc.
Register 26-3: ADCON2 Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>R/W-0</th>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADFM</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>ADCS2</td>
<td>ADCS1</td>
<td>ADCS0</td>
</tr>
<tr>
<td>bit 7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

bit 7: **ADFM**: A/D Result Format Select bit
- **1**: Right justified
- **0**: Left justified

bit 6-3: **Unimplemented**: Read as '0'

bit 2-0: **ADCS2:ADCS0**: A/D Conversion Clock Select bits
- **000** = Fosc/2
- **001** = Fosc/8
- **010** = Fosc/32
- **011** = FrC (clock derived from an internal RC oscillator, 1 MHz maximum frequency)
- **100** = Fosc/4
- **101** = Fosc/16
- **110** = Fosc/64
- **111** = FrC (clock derived from an RC oscillator, 1 MHz maximum frequency)

Legend
- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- **n** = Value at POR reset
- **’1’** = bit is set
- **’0’** = bit is cleared
- **x** = bit is unknown
Section 26. 10-bit A/D Converter

26.3 Operation

The ADRESH:ADRESL registers contain the 10-bit result of the A/D conversion. When the A/D conversion is complete, the result is loaded into this A/D result register pair (ADRESH:ADRESL), the GO/DONE bit (ADCON0 register) is cleared, and A/D interrupt flag bit, ADIF, is set. The block diagram of the A/D module is shown in Figure 26-1.

After the A/D module has been configured, the signal on the selected channel must be acquired before the conversion is started. The analog input channels must have their corresponding TRIS bits selected as inputs. To determine acquisition time, see Subsection 26.4 “A/D Acquisition Requirements.” After this acquisition time has elapsed, the A/D conversion can be started. The following steps should be followed for doing an A/D conversion:

1. Configure the A/D module:
   - Configure analog pins, Voltage Reference, and digital I/O (ADCON1)
   - Select A/D input channel (ADCON0)
   - Select A/D conversion clock (ADCON0)
   - Turn on A/D module (ADCON0)
2. Configure A/D interrupt (if desired):
   - Clear the ADIF bit
   - Set the ADIE bit
   - Set/Clear the ADIP bit
   - Set the GIE/GIEH or PEIE/GIEL bit
3. Wait the required acquisition time.
4. Start conversion:
   - Set the GO/DONE bit (ADCON0)
5. Wait for the A/D conversion to complete, by either:
   - Polling for the GO/DONE bit to be cleared or the ADIF bit to be set, or
   - Waiting for the A/D interrupt
6. Read A/D Result register pair (ADRESH:ADRESL): clear the ADIF bit, if required.
7. For next conversion, go to step 1 or step 2 as required.

Figure 26-2 shows the conversion sequence, and the terms that are used. Acquisition time is the time that the A/D module’s holding capacitor is connected to the external voltage level. When the GO bit is set, the conversion time of 12 Ta0 is started. The sum of these two times is the sampling time. There is a minimum acquisition time to ensure that the holding capacitor is charged to a level that will give the desired accuracy for the A/D conversion.

Figure 26-2: A/D Conversion Sequence

<table>
<thead>
<tr>
<th>Acquisition Time</th>
<th>A/D Conversion Time</th>
</tr>
</thead>
<tbody>
<tr>
<td>When A/D holding capacitor starts to charge.</td>
<td>A/D conversion complete, result is loaded in ADRES register.</td>
</tr>
<tr>
<td>After A/D conversion, or when new A/D channel is selected.</td>
<td>Holding capacitor begins acquiring voltage level on selected channel; ADIF bit is set.</td>
</tr>
</tbody>
</table>
26.4  A/D Acquisition Requirements

For the A/D converter to meet its specified accuracy, the charge holding capacitor (C\text{HOLD}) must be allowed to fully charge to the input channel voltage level. The analog input model is shown in Figure 26-3. The source impedance (Rs) and the internal sampling switch (Rss) impedance directly affect the time required to charge the capacitor C\text{HOLD}. The sampling switch (Rss) impedance varies over the device voltage (V\text{DD}), Figure 26-3. The source impedance affects the offset voltage at the analog input (due to pin leakage current). The maximum recommended impedance for analog sources is 2.5 k\text{\Omega}. As the impedance is decreased, the acquisition time may be decreased. After the analog input channel is selected (changed), this acquisition must pass before the conversion can be started.

To calculate the minimum acquisition time, Equation 26-1 may be used. This equation assumes that 1/2 LSB error is used (1024 steps for the A/D). The 1/2 LSB error is the maximum error allowed for the A/D to meet its specified resolution.

**Equation 26-1:** Acquisition Time

\[
\text{TACQ} = \text{TAMP} + \text{Tc} + \text{TCOFF}
\]

Where:
- **\text{TACQ}** = Acquisition Time
- **\text{TAMP}** = Amplifier Settling Time
- **\text{Tc}** = Holding Capacitor Charging Time
- **\text{TCOFF}** = Temperature Coefficient

**Equation 26-2:** A/D Minimum Charging Time

\[
\text{VHOLD} = (\text{VREF} - (\text{VREF}/2048)) \cdot (1 - e^{(\text{Tc}/\text{C\text{HOLD}}(R\text{IC} + \text{Rss} + \text{Rs}))})
\]

or

\[
\text{Tc} = -\frac{(120 \text{ pF})(1 \text{ k\text{\Omega}} + \text{Rss} + \text{Rs})}{\text{ln}(1/2048)}
\]

Example 26-1 shows the calculation of the minimum required acquisition time TACQ. This calculation is based on the following application system assumptions.

- C\text{HOLD} = 120 pF
- Rs = 2.5 k\text{\Omega}
- Conversion Error \leq 1/2 LSB
- V\text{DD} = 5V \rightarrow \text{Rss} = 7 k\text{\Omega} (see graph in Figure 26-3)
- Temperature = 50°C (system max.)
- VHOLD = 0V @ time = 0

Example 26-1: Calculating the Minimum Required Acquisition Time (Case 1)

\[
\text{TACQ} = \text{TAMP} + \text{Tc} + \text{TCOFF}
\]

Temperature coefficient is only required for temperatures > 25°C.

\[
\text{TACQ} = 2 \mu\text{s} + \text{Tc} + [(\text{Temp} - 25^\circ\text{C})(0.05 \mu\text{s}/\text{°C})]
\]

\[
\text{Tc} = -\frac{\text{C\text{HOLD}}(R\text{IC} + \text{Rss} + \text{Rs})}{\text{ln}(1/2048)}
\]

-20 pF (1 k\text{\Omega} + 7 k\text{\Omega} + 2.5 k\text{\Omega}) ln(0.0004885)
-120 pF (10.5 k\text{\Omega}) ln(0.0004885)
-1.26 µs (-7.6241)
9.61 µs

\[
\text{TACQ} = 2 \mu\text{s} + 9.61 \mu\text{s} + [(50^\circ\text{C} - 25^\circ\text{C})(0.05 \mu\text{s}/\text{°C})]
\]

11.61 µs + 1.25 µs
12.86 µs

---

Note: When the conversion is started, the holding capacitor is disconnected from the input pin.
Section 26. 10-bit A/D Converter

Now to get an idea what happens to the acquisition time when the source impedance is a minimal value ($R_S = 50 \, \Omega$). Example 26-2 shows the same conditions as in Example 26-1 with only the source impedance changed to the minimal value.

Example 26-2: Calculating the Minimum Required Acquisition Time (Case 2)

$$T_{ACQ} = T_{AMP} + T_C + T_{COFF}$$

Temperature coefficient is only required for temperatures $> 25^\circ C$.

$$T_{ACQ} = 2 \, \mu s + T_c + [(Temp - 25^\circ C)(0.05 \, \mu s/\circ C)]$$

$$T_C = -\text{Chold} \times (R_i + R_{ss} + R_s) \ln(1/2047)$$

$$-120 \, \text{pF} (1 \, \text{k}\Omega + 7 \, \text{k}\Omega + 50 \, \Omega) \ln(0.0004885)$$

$$-120 \, \text{pF} (8050 \, \Omega) \ln(0.0004885)$$

$$-0.966 \, \mu s (-7.6241)$$

$$7.36 \, \mu s$$

$$T_{ACQ} = 2 \, \mu s + 16.47 \, \mu s + [(50^\circ C - 25^\circ C)(0.05 \, \mu s/\circ C)]$$

$$9.36 \, \mu s + 1.25 \, \mu s$$

$$10.61 \, \mu s$$

Note 1: The reference voltage (VREF) has no effect on the equation, since it cancels itself out.

2: The charge holding capacitor (CHOLD) is not discharged after each conversion.

3: The maximum recommended impedance for analog sources is 2.5 k\Omega. This is required to meet the pin leakage specification.

4: After a conversion has completed, a 2 T_{AD} delay must complete before acquisition can begin again. During this time the holding capacitor is not connected to the selected A/D input channel.

Figure 26-3: Analog Input Model

Legend:
- CHN = Input Capacitance
- V_T = Threshold Voltage
- I_{LEAKAGE} = Leakage Current at the pin due to various junctions
- R_{IC} = Interconnect Resistance
- SS = Sampling Switch
- CHOLD = Sample/hold capacitance (from DAC)
26.5 Selecting the A/D Conversion Clock

The A/D conversion time per bit is defined as \( T_{AD} \). The A/D conversion requires \( 11.5T_{AD} \) per 10-bit conversion. The source of the A/D conversion clock is software selectable. The seven possible options for \( T_{AD} \) are:

- \( 2T_{OSC} \)
- \( 4T_{OSC} \)
- \( 8T_{OSC} \)
- \( 16T_{OSC} \)
- \( 32T_{OSC} \)
- \( 64T_{OSC} \)
- Internal A/D RC oscillator

For correct A/D conversions, the A/D conversion clock (\( T_{AD} \)) must be selected to ensure a minimum \( T_{AD} \) time of \( 1.6 \mu s \) as shown in Electrical Specifications parameter 130.

Table 26-1 and Table 26-2 show the resultant \( T_{AD} \) times derived from the device operating frequencies and the selected A/D clock source.

### Table 26-1: \( T_{AD} \) vs. Device Operating Frequencies (for Standard, C, Devices)

<table>
<thead>
<tr>
<th>AD Clock Source (( T_{AD} ))</th>
<th>Device Frequency</th>
</tr>
</thead>
<tbody>
<tr>
<td>Operation</td>
<td>ADCS2:ADCS0</td>
</tr>
<tr>
<td>2TOSC</td>
<td>000</td>
</tr>
<tr>
<td>4TOSC</td>
<td>010</td>
</tr>
<tr>
<td>8TOSC</td>
<td>001</td>
</tr>
<tr>
<td>16TOSC</td>
<td>101</td>
</tr>
<tr>
<td>32TOSC</td>
<td>010</td>
</tr>
<tr>
<td>64TOSC</td>
<td>110</td>
</tr>
<tr>
<td>RC</td>
<td>011</td>
</tr>
</tbody>
</table>

Legend: Shaded cells are outside of recommended range.

**Note 1:** The RC source has a typical \( T_{AD} \) of 4 \( \mu s \).

**Note 2:** These values violate the minimum required \( T_{AD} \).

**Note 3:** For faster conversion times, the selection of another clock source is recommended.

**Note 4:** For device frequencies above 1 MHz, the device must be in SLEEP for the entire conversion, or the A/D accuracy may be out of specification.

### Table 26-2: \( T_{AD} \) vs. Device Operating Frequencies (for Extended, LC, Devices)

<table>
<thead>
<tr>
<th>AD Clock Source (( T_{AD} ))</th>
<th>Device Frequency</th>
</tr>
</thead>
<tbody>
<tr>
<td>Operation</td>
<td>ADCS2:ADCS0</td>
</tr>
<tr>
<td>2TOSC</td>
<td>000</td>
</tr>
<tr>
<td>4TOSC</td>
<td>010</td>
</tr>
<tr>
<td>8TOSC</td>
<td>001</td>
</tr>
<tr>
<td>16TOSC</td>
<td>101</td>
</tr>
<tr>
<td>32TOSC</td>
<td>010</td>
</tr>
<tr>
<td>64TOSC</td>
<td>110</td>
</tr>
<tr>
<td>RC</td>
<td>011</td>
</tr>
</tbody>
</table>

Legend: Shaded cells are outside of recommended range.

**Note 1:** The RC source has a typical \( T_{AD} \) of 6 \( \mu s \).

**Note 2:** These values violate the minimum required \( T_{AD} \).

**Note 3:** For faster conversion times, the selection of another clock source is recommended.

**Note 4:** For device frequencies above 1 MHz, the device must be in SLEEP for the entire conversion, or the A/D accuracy may be out of specification.
Section 26. 10-bit A/D Converter

26.6 Configuring Analog Port Pins

The ADCON1 and TRIS registers control the operation of the A/D port pins. The port pins that are desired as analog inputs must have their corresponding TRIS bits set (input). If the TRIS bit is cleared (output), the digital output level (VOH or VOL) will be converted. After a device RESET, pins that are multiplexed with analog inputs will be configured as an analog input. The corresponding TRIS bit will be set.

The A/D operation is independent of the state of the CHS2:CHS0 bits and the TRIS bits.

| Note 1: When reading the port register, any pin configured as an analog input channel will read as cleared (a low level). Pins configured as digital inputs, will convert an analog input. Analog levels on a digitally configured input will not affect the conversion accuracy. |
| 2: Analog levels on any pin that is defined as a digital input (including the AN7:AN0 pins), may cause the input buffer to consume current that is out of the devices specification. |
26.7 A/D Conversions

Example 26-3 shows how to perform an A/D conversion. The port pins are configured as analog inputs. The analog references (VREF+ and VREF-) are the device AVDD and AVSS. The A/D interrupt is enabled, and the A/D conversion clock is FRC. The conversion is performed on the AN0 pin (channel 0). The result of the conversion is left justified.

Note: The GO/DONE bit should NOT be set in the instruction that turns on the A/D, due to the required acquisition time.

Clearing the GO/DONE bit during a conversion will abort the current conversion. The A/D result register pair will NOT be updated with the partially completed A/D conversion sample. That is, the ADRESH:ADRESL registers will continue to contain the value of the last completed conversion (or the last value written to the ADRESH:ADRESL registers). After the A/D conversion is aborted, a 2TAD wait is required before the next acquisition is started. After this 2TAD wait, acquisition on the selected channel is automatically started.

Example 26-3: A/D Conversion

CLRF ADCON1 ; Configure A/D inputs, result is left justified
BSF IPR1, ADIP ; High Priority.
BSF PIE1, ADIE ; Enable A/D interrupts
MOVLW 0xC1 ; RC Clock, A/D is on,
MOVWF ADCON0 ; Channel 0 is selected
BCF PIR1, ADIF ; Clear A/D interrupt flag bit
BSF INTCON, PEIE ; Enable peripheral interrupts
BSF INTCON, GIE ; Enable all interrupts
; ; Ensure that the required sampling time for the selected input channel has elapsed. Then the conversion may be started.
;
BSF ADCON0, GO ; Start A/D Conversion :
; ; The ADIF bit will be set and the ; ; GO/DONE bit is cleared upon ; ; completion of the A/D Conversion.

Figure 26-4: A/D Conversion TAD Cycles

Conversion Starts
Holding capacitor is disconnected from analog input (typically 100 ns)

Set GO bit
Next Q4: ADRES is loaded, GO bit is cleared, ADIF bit is set, holding capacitor is connected to analog input.
Section 26. 10-bit A/D Converter

Figure 26-5: Flowchart of A/D Operation

- Acquire Selected Channel
- SLEEP Instruction
- Start of A/D Conversion Delayed 1 Instruction Cycle
- Finish Conversion GO = 0, ADIF = 1
- Wait 2TAD
- Power-down A/D
- Wake-up from SLEEP
- Stay in SLEEP Power-down A/D
- Instruction?
26.7.1 Faster Conversion - Lower Resolution Trade-off

Not all applications require a result with 10-bits of resolution, but may instead require a faster conversion time. The A/D module allows users to make the trade-off of conversion speed to resolution. Regardless of the resolution required, the acquisition time is the same. To speed up the conversion, the clock source of the A/D module may be switched so that the $T_{AD}$ time violates the minimum specified time (Electrical Specifications parameter 130). Once the $T_{AD}$ time violates the minimum specified time, all the following A/D result bits are not valid (see A/D Conversion Timing in the Electrical Specifications section). The clock sources may only be switched between the three oscillator versions (cannot be switched from/to RC). The equation to determine the time before the oscillator can be switched is as follows:

$$T_{AD} + (11 - N)(2T_{OSC})$$

Since the $T_{AD}$ is based from the device oscillator, the user must use some method (a timer, software loop, etc.) to determine when the A/D oscillator may be changed. Example 26-4 shows a comparison of time required for a conversion with 4-bits of resolution, versus the 10-bit resolution conversion. The example is for devices operating at 20 MHz (the A/D clock is programmed for $32T_{OSC}$), and assumes that immediately after $6T_{AD}$, the A/D clock is programmed for $2T_{OSC}$. The $2T_{OSC}$ violates the minimum $T_{AD}$ time since the last 6 bits will not be converted to correct values.

Example 26-4: 4-bit vs. 10-bit Conversion Times

<table>
<thead>
<tr>
<th>Freq. (MHz)$^{(1)}$</th>
<th>Resolution</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>4-bit</td>
</tr>
<tr>
<td>$T_{AD}$</td>
<td>40</td>
</tr>
<tr>
<td>$T_{OSC}$</td>
<td>40</td>
</tr>
<tr>
<td>$T_{AD} + N \times T_{AD} + (11 - N)(2T_{OSC})$</td>
<td>40</td>
</tr>
</tbody>
</table>

Note 1: A minimum $T_{AD}$ time of 1.6 µs is required.
Note 2: If the full 10-bit conversion is required, the A/D clock source should not be changed.

Equation 26-3: Resolution/Speed Conversion Trade-off

Conversion time = $T_{AD} + N \times T_{AD} + (11 - N)(2T_{OSC})$

Where: $N$ = number of bits of resolution required
Section 26. 10-bit A/D Converter

26.7.2 A/D Result Registers

The ADRESH:ADRESL register pair is the location where the 10-bit A/D result is loaded at the completion of the A/D conversion. This register pair is 16-bits wide. The A/D module gives the flexibility to left or right justify the 10-bit result in the 16-bit result register. The A/D Format Select bit (ADFM) controls this justification. Figure 26-6 shows the operation of the A/D result justification. The extra bits are loaded with '0's'. When the A/D module is disabled, these registers may be used as two general purpose 8-bit registers.

Figure 26-6: A/D Result Justification
26.8 Operation During SLEEP

The A/D module can operate during SLEEP mode. This requires that the A/D clock source be set to RC (ADCS2:ADCS0 = 111). When the RC clock source is selected, the A/D module waits one instruction cycle before starting the conversion. This allows the SLEEP instruction to be executed, which eliminates all internal digital switching noise from the conversion. When the conversion is completed, the GO/DONE bit will be cleared and the result is loaded into the ADRESH:ADRESL registers. If the A/D interrupt is enabled, the device will wake-up from SLEEP. If the A/D interrupt is not enabled, the A/D module will be turned off, although the ADON bit will remain set.

When the A/D clock source is another clock option (not RC), a SLEEP instruction will cause the present conversion to be aborted and the A/D module to be turned off (to conserve power), though the ADON bit will remain set.

Turning off the A/D places the A/D module in its lowest current consumption state.

Note: For the A/D module to operate in SLEEP, the A/D clock source must be set to RC (ADCS2:ADCS0 = 111). To allow the conversion to occur during SLEEP, ensure the SLEEP instruction immediately follows the instruction that sets the GO/DONE bit.

26.9 Effects of a RESET

A device RESET forces all registers to their RESET state. This forces the A/D module to be turned off, and any conversion is aborted. All pins that are multiplexed with analog inputs will be configured as an analog input. The corresponding TRIS bits will be set.

The value that is in the ADRESH:ADRESL registers is not initialized from a Power-on Reset. The ADRESH:ADRESL registers will contain unknown data after a Power-on Reset.
## Section 26. 10-bit A/D Converter

### 26.10 A/D Accuracy/Error

In systems where the device frequency is low, use of the A/D RC clock is preferred. At moderate to high frequencies, TAD should be derived from the device oscillator.

For a given range of analog inputs, the output digital code will be the same. This is due to the quantization of the analog input to a digital code. Quantization error is typically ± 1/2 LSB and is inherent in the analog to digital conversion process. The only way to reduce quantization error is to increase the resolution of the A/D converter.

Offset error measures the first actual transition of a code versus the first ideal transition of a code. Offset error shifts the entire transfer function. Offset error can be calibrated out of a system or introduced into a system, through the interaction of the total leakage current and source impedance at the analog input.

Gain error measures the maximum deviation of the last actual transition and the last ideal transition, adjusted for offset error. This error appears as a change in slope of the transfer function. The difference in gain error to full scale error, is that full scale does not take offset error into account. Gain error can be calibrated out in software.

Linearity error refers to the uniformity of the code changes. Linearity errors cannot be calibrated out of the system. Integral non-linearity error measures the actual code transition versus the ideal code transition, adjusted by the gain error for each code.

Differential non-linearity measures the maximum actual code width versus the ideal code width. This measure is unadjusted.

The maximum pin leakage current is specified in Electrical Specifications parameter D060.

TAD must not violate the minimum and should be minimized to reduce inaccuracies due to noise and sampling capacitor bleed off.

In systems where the device will enter SLEEP mode after the start of the A/D conversion, the RC clock source selection is required. In this mode, the digital noise from the modules in SLEEP are stopped. This method gives high accuracy.
26.11 Connection Considerations

If the input voltage exceeds the rail values (VSS or VDD) by greater than 0.3V, then the accuracy of the conversion is out of specification.

An external RC filter is sometimes added for anti-aliasing of the input signal. The R component should be selected to ensure that the total source impedance is kept under the 2.5 kΩ recommended specification. Any external components connected (via hi-impedance) to an analog input pin (capacitor, zener diode, etc.) should have very little leakage current at the pin.

26.12 Transfer Function

The ideal transfer function of the A/D converter is as follows: the first transition occurs when the analog input voltage (VAIN) is 1 LSb (or Analog VREF / 1024) (Figure 26-7).

Figure 26-7: A/D Transfer Function
Section 26. 10-bit A/D Converter

26.13 Initialization

Example 26-5 shows an initialization of the A/D module.

Example 26-5: A/D Initialization

```
CLRF ADCON1 ; Configure A/D inputs
BSF PIE1, ADIE ; Enable A/D interrupts
BSF IPR1, ADIP ; High Priority
MOVLW 0xC1 ; RC Clock, A/D is on,
MOVWF ADCON0 ; Channel 0 is selected
MOVLW 0x4E ; Left Justified, AN0 is analog
MOVWF ADCON1 ; Vref comes from AVDD and AVSS
BCF PIR1, ADIF ; Clear A/D interrupt flag bit
BSF INTCON, PEIE ; Enable peripheral interrupts
BSF INTCON, GIE ; Enable all interrupts
;
; Ensure that the required sampling time for the selected input
; channel has elapsed. Then the conversion may be started.
;
BSF ADCON0, GO ; Start A/D Conversion
; The ADIF bit will be set and the
; GO/DONE bit is cleared upon
; completion of the A/D conversion.
```
26.14 Design Tips

Question 1: I find that the Analog to Digital Converter result is not always accurate. What can I do to improve accuracy?

Answer 1:
1. Make sure you are meeting all of the timing specifications. If you are turning the module on and off, there is a minimum delay you must wait before taking a sample. If you are changing input channels, there is a minimum delay you must wait for this as well, and finally there is TAD, which is the time selected for each bit conversion. This is selected in ADCON0 and should be between 1.6 and 6 µs. If TAD is too short, the result may not be fully converted before the conversion is terminated, and if TAD is made too long, the voltage on the sampling capacitor can decay before the conversion is complete. These timing specifications are provided in the "Electrical Specifications" section. See the device data sheet for device specific information.

2. Often the source impedance of the analog signal is high (greater than 1 kOhms), so the current drawn from the source to charge the sample capacitor can affect accuracy. If the input signal does not change too quickly, try putting a 0.1 µF capacitor on the analog input. This capacitor will charge to the analog voltage being sampled and supply the instantaneous current needed to charge the 120 pF internal holding capacitor.

3. In systems where the device frequency is low, use of the A/D clock derived from the device oscillator is preferred...this reduces, to a large extent, the effects of digital switching noise. In systems where the device will enter SLEEP mode after start of A/D conversion, the RC clock source selection is required. This method gives the highest accuracy.

Question 2: After starting an A/D conversion may I change the input channel (for my next conversion)?

Answer 2:
After the holding capacitor is disconnected from the input channel, typically 100 ns after the GO bit is set, the input channel may be changed.

Question 3: Do you know of a good reference on A/D's?

Answer 3:

Question 4: I migrated my code from a PIC18CXX2 device with 10-bit A/D to another device with a 10-bit A/D (such as a PIC18CXX8) and the A/D does not seem to operate the same. What's going on?

Answer 4:
The 10-bit A/D on the PIC18CXX2 device is the compatible 10-bit A/D module. This module has its ADCON bits in the same locations as the PICmicro Mid-Range 10-bit A/D module. The standard PIC18CXXX 10-bit A/D module (as found on the PIC18CXX8 device) has optimized the bit locations to ease configuration of the module.
Section 26. 10-bit A/D Converter

26.15 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the 10-bit A/D module are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Using the Analog to Digital Converter</td>
<td>AN546</td>
</tr>
<tr>
<td>Four Channel Digital Voltmeter with Display and Keyboard</td>
<td>AN557</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
26.16 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Compatible 10-bit A/D module description.
Section 27. Low Voltage Detect

HIGHLIGHTS

This section of the manual contains the following major topics:

27.1 Introduction .......................................................... 27-2
27.2 Control Register ................................................. 27-4
27.3 Operation .............................................................. 27-5
27.4 Operation During SLEEP .................................... 27-6
27.5 Effects of a RESET ............................................. 27-6
27.6 Initialization ......................................................... 27-7
27.7 Design Tips ........................................................ 27-8
27.8 Related Application Notes ................................. 27-9
27.9 Revision History .................................................. 27-10
27.1 Introduction

In many applications, the ability to determine if the device voltage (VDD) is below a specified voltage level is a desirable feature. A window of operation for the application can be created where the application software can do "housekeeping tasks" before the device voltage exits the valid operating range. This can be done using the Low Voltage Detect module.

This module is software programmable circuitry, where a device voltage trip point can be specified. When the voltage of the device becomes lower than the specified point, an interrupt flag is set. If the interrupt is enabled, the program execution will branch to the interrupt vector address, and the software can then respond to that interrupt source.

The Low Voltage Detect circuitry is completely under software control. This allows the circuitry to be "turned off" by the software, which minimizes the current consumption for the device.

Figure 27-1 shows a possible application voltage curve (typically for batteries). Over time the device voltage decreases. When the device voltage equals voltage VA, the LVD logic generates an interrupt. This occurs at time TA. The application software then has until the device voltage is no longer in valid operating range to have shut down the system. Voltage point VB is the minimum valid operating voltage specification. This gives a time TB. The total time for shutdown is TB - TA.

Figure 27-1: Typical Low Voltage Detect Application
Section 27. Low Voltage Detect

Figure 27-2 shows the block diagram for the LVD module. A comparator uses an internally generated reference voltage as the set point. When the selected tap output of the device voltage crosses the set point (is lower then), the LVDIF bit is set.

Each node in the resistor divider represents a “trip point” voltage.

The “trip point” voltage is the minimum supply voltage level at which the device can operate before the LVD module asserts an interrupt. When the supply voltage is equal to the trip point, the voltage tapped off of the resistor array is equal to the voltage generated by the internal voltage reference module. The comparator then generates an interrupt signal setting the LVDIF bit. This voltage is software programmable to any one of 16 values (see Figure 27-2). The trip point is selected by programming the LVDL3:LVDL0 bits (LVDCON<3:0>).

Figure 27-2: Low Voltage Detect (LVD) Block Diagram
## 27.2 Control Register

The Low Voltage Detect Control register controls the operation of the Low Voltage Detect circuitry.

### Register 27-1: LVDCON Register

<table>
<thead>
<tr>
<th>U-0</th>
<th>U-0</th>
<th>R-0</th>
<th>R/W-0</th>
<th>R/W-0</th>
<th>R/W-1</th>
<th>R/W-0</th>
<th>R/W-1</th>
</tr>
</thead>
<tbody>
<tr>
<td>—</td>
<td>—</td>
<td>IRVST</td>
<td>LVDEN</td>
<td>LVDL3</td>
<td>LVDL2</td>
<td>LVDL1</td>
<td>LVDL0</td>
</tr>
</tbody>
</table>

- **bit 7**: Unimplemented: Read as '0'
- **bit 5**: IRVST: Internal Reference Voltage Stable Flag bit
  - 1 = Indicates that the Low Voltage Detect logic will generate the interrupt flag at the specified voltage range.
  - 0 = Indicates that the Low Voltage Detect logic will not generate the interrupt flag at the specified voltage range, and LVD Interrupt should not be enabled
- **bit 4**: LVDEN: Low-voltage Detect Power Enable bit
  - 1 = Enables LVD, powers up LVD circuit
  - 0 = Disables LVD, powers down LVD circuit
- **bit 3:0**: LVDL3:LVDL0: Low Voltage Detection Limit bits

The following shows the typical limits for the low voltage detect circuitry. Refer to the device data sheet electrical specifications for the actual tested limit.

- 1111 = External analog input is used (input comes from the LVDIN pin)
- 1110 = 4.5V min - 4.77V max.
- 1101 = 4.2V min - 4.45V max.
- 1100 = 4.0V min - 4.24V max.
- 1011 = 3.8V min - 4.03V max.
- 1010 = 3.6V min - 3.92V max.
- 1001 = 3.5V min - 3.71V max.
- 1000 = 3.3V min - 3.50V max.
- 0111 = 3.0V min - 3.18V max.
- 0110 = 2.8V min - 2.97V max.
- 0101 = 2.7V min - 2.86V max.
- 0100 = 2.5V min - 2.65V max.
- 0011 = 2.4V min - 2.54V max.
- 0010 = 2.2V min - 2.33V max.
- 0001 = 2.0V min - 2.12V max.
- 0000 = 1.8V min - 1.91V max.

**Note 1**: LVDL3:LVDL0 modes which result in a trip point below the valid operating voltage of the device are not tested.

**Note 2**: See the “Electrical Specifications” section, parameter 32 in the Device Data Sheet for tested limits.

### Legend

- **R** = Readable bit
- **W** = Writable bit
- **U** = Unimplemented bit, read as '0'
- - **n** = Value at POR reset
- ‘1’ = bit is set
- ‘0’ = bit is cleared
- **x** = bit is unknown
Section 27. Low Voltage Detect

27.3 Operation

The LVD module is useful to add robustness into the application. The device can monitor the state of the device voltage. When the device voltage enters the voltage window near the lower limit of the valid operating voltage range, the device can save values to ensure a "clean" shut-down through the brown-out.

Note: The system design should be done to ensure that the application software is given adequate time to save values before the device exits the valid operating range or is forced into a Brown-out Reset.

Depending on the power source for the device voltage, the voltage normally decreases relatively slowly. This means that the LVD module does not need to be constantly operating. To decrease the current requirements, the LVD circuitry only needs to be enabled for short periods, where the voltage is checked. After doing the check, the LVD module may be disabled.

Each time that the LVD module is enabled, the circuitry requires some time to stabilize. After the circuitry has stabilized, all status flags may be cleared. The module will then indicate the proper state of the system.

Steps to setup the LVD module:
1. Write the value to the LVDL3:LVDL0 bits (LVDCON register) which selects the desired LVD Trip Point.
2. Ensure that LVD interrupts are disabled (the LVDIE bit is cleared or the GIE bit is cleared).
3. Enable the LVD module (set the LVDEN bit in the LVDCON register).
4. Wait for the LVD module to stabilize (the IRVST bit to become set).
5. Clear the LVD interrupt flag which may have falsely become set while the LVD module stabilized (clear the LVDIF bit).
6. Enable the LVD interrupt (set the LVDIE and the GIE bits).

Figure 27-3 shows some waveforms that the LVD module may be used to detect.
27.3.1 Reference Voltage Set Point

The internal reference voltage of the LVD module may be used by other internal circuitry (e.g., the programmable Brown-out Reset). If these circuits are disabled (lower power consumption), the reference voltage circuit requires stabilization time before a low voltage condition can be reliably detected. This time is specified in electrical specification parameter # 36. The low-voltage interrupt flag will not be enabled until a stable reference voltage is reached. Refer to the timing diagram in Figure 27-3.

27.3.2 Current Consumption

When the module is enabled the LVD comparator and voltage divider are enabled and will consume static current. The voltage divider can be tapped from multiple places in the resistor array. Total current consumption when enabled is specified in electrical specification parameter D022B (typically < 50 µA).

27.4 Operation During SLEEP

When enabled, the LVD circuitry continues to operate during SLEEP. If the device voltage crosses the trip point, the LVDIF bit will be set and the device will wake-up from SLEEP. Device execution will continue from the interrupt vector address, if interrupts have been globally enabled.

27.5 Effects of a RESET

A device RESET forces all registers to their RESET state. This forces the LVD module to be turned off.
Section 27. Low Voltage Detect

27.6 Initialization

Example 27-1 shows an initialization of the LVD module.

Example 27-1: LVD Initialization

```
MOVLW 0x14  ; Enable LVD, Trip point = 2.5V
MOVF LVDCON ;
LVD_STABLE
STFSS LVDCON, IRVST    ; Has LVD circuitry stabilized?
GOTO LVD_STABLE        ; NO, Wait longer
BCF PIR, LVDIF         ; YES, clear LVD interrupt flag
BSF PIE, LVDIE         ; Enable LVD interrupt
```
27.7 Design Tips

Question 1: *The LVD circuitry seems to be generating random interrupts?*

Answer 1:
Ensure that the LVD circuitry is stable before enabling the LVD interrupt. This is done by monitoring the IRVST bit. Once the IRVST bit is set, the LVDIF bit should be cleared and then the LVDIE bit may be set.

Question 2: *How can I reduce the current consumption of the module?*

Answer 2:
Low Voltage Detect is used to monitor the device voltage. The power source is normally a battery that ramps down slowly. This means that the LVD circuitry can be disabled for most of the time, and only enabled occasionally to do the device voltage check.
Section 27. Low Voltage Detect

27.8 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the LVD module are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related application notes at this time</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faq/codeex/
27.9 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Low Voltage Detect module description.
Section 28. WDT and SLEEP Mode

HIGHLIGHTS

This section of the manual contains the following major topics:

<p>| | |</p>
<table>
<thead>
<tr>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>28.1</td>
<td>Introduction ................................................................. 28-2</td>
</tr>
<tr>
<td>28.2</td>
<td>Control Register .......................................................... 28-3</td>
</tr>
<tr>
<td>28.3</td>
<td>Watchdog Timer (WDT) Operation .................................... 28-4</td>
</tr>
<tr>
<td>28.4</td>
<td>SLEEP (Power-Down) Mode .............................................. 28-5</td>
</tr>
<tr>
<td>28.5</td>
<td>Initialization ................................................................. 28-11</td>
</tr>
<tr>
<td>28.6</td>
<td>Design Tips .................................................................. 28-12</td>
</tr>
<tr>
<td>28.7</td>
<td>Related Application Notes ............................................. 28-13</td>
</tr>
<tr>
<td>28.8</td>
<td>Revision History .......................................................... 28-14</td>
</tr>
</tbody>
</table>
28.1 Introduction

The Watchdog Timer and SLEEP functions are two functions that can enhance the system. The Watchdog Timer may be used to return to operating mode, or to cause a controller RESET if the program begins to behave erratically. This enhances the overall operation of the system.

The Watchdog Timer (WDT) is a free running on-chip RC oscillator that does not require any external components. The block diagram is shown in Figure 28-1. This RC oscillator is separate from the device RC oscillator of the OSC1/CLKI pin. This means that the WDT will run, even if the clock on the OSC1/CLKI and OSC2/CLKO pins has been stopped, for example, by execution of a SLEEP instruction.

The Watchdog Timer (WDT) is enabled/disabled by a device configuration bit. If the WDT is enabled, software execution may not disable this function. When the WDTE N configuration bit is cleared, the SWDTE N bit enables/disables the operation of the WDT.

Figure 28-1: Watchdog Timer Block Diagram

The SLEEP function halts controller activity and reduces current consumption to a minimum. The SLEEP mode is a reduced power state, where it is possible to halt almost all activity in the controller. In this mode, power consumption is very low, allowing for long term operation from battery powered applications. Normal operation may be resumed when any of several interrupts occur, the WDT times out, or a RESET occurs.
Section 28. Watchdog Timer and SLEEP Mode

28.2 Control Register

Register 28-1 shows the WDTCON register. This is a readable and writable register that contains the SWDTEN control bit. If the WDT enable configuration bit has been cleared, this software controlled bit enables or disables the WDT.

Register 28-1: WDTCON Register

<table>
<thead>
<tr>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>U-0</th>
<th>R/W-0</th>
</tr>
</thead>
<tbody>
<tr>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>SWDTEN</td>
</tr>
</tbody>
</table>

bit 7:1 Unimplemented: Read as '0'

bit 0  SWDTEN: Software Controlled Watchdog Timer Enable bit

1 = Watchdog Timer is on
0 = Watchdog Timer is turned off if the WDTEN configuration bit is '0'

Legend
R = Readable bit  W = Writable bit  U = Unimplemented bit, read as '0'
- n = Value at POR reset    '1' = bit is set    '0' = bit is cleared    x = bit is unknown
28.3 Watchdog Timer (WDT) Operation

During normal operation, a WDT time-out generates a device RESET. If the device is in SLEEP mode, a WDT time-out causes the device to wake-up and continue with normal operation. This is known as a WDT wake-up.

The WDT can be permanently enabled by setting the WDTE N configuration bit. If the WDT configuration bit disables the WDT, then software can be used to enable/disable the WDT through setting/clearing the SWDTE N bit.

28.3.1 WDT Period

The WDT has a nominal time-out period of 18 ms with no postscaler (see the "Electrical Specifications" section, parameter 31). The time-out period varies with temperature, VDD and process variations from part to part (see DC parameters in the "Electrical Specifications" section). If longer time-outs are desired, a postscaler with a division ratio of up to 1:128 can be assigned to the WDT. Thus, time-out periods of up to 2.3 seconds can be realized.

The postscaler assignment is specified at time of device programming through the device configuration bits.

The CLRWDT and SLEEP instructions clear the WDT counter and the WDT postscaler which prevents it from timing out and generating a device RESET.

When a CLRWDT instruction is executed and the prescaler is assigned to the WDT, the prescaler count will be cleared, but the prescaler assignment is not changed.

The TO bit in the RCON register will be cleared upon a Watchdog Timer time-out (WDT Reset and WDT wake-up).

28.3.2 Clearing the WDT Counter

The CLRWDT instruction will force the count value of the WDT counter to ‘0’. When the WDT is disabled (WDTE N configuration bit = ‘0’ and SWDTE N is clear), the WDT counter is forced to ‘0’ and the internal WDT clock source is disabled. Then, when the WDT is enabled (setting the SWDTE N bit when previously cleared), the WDT counter starts from a value of ‘0’.

28.3.3 WDT Considerations

It should also be taken in account that under worst case conditions (VDD = Minimum, Temperature = Maximum, WDT postscaler = Maximum), it may take several seconds before a WDT time-out occurs.

28.3.4 Effects of a RESET

When a device RESET occurs, the Watchdog Timer counter and postscaler counter are cleared and the TO bit is set.

Table 28-1: Summary of Watchdog Timer Registers

<table>
<thead>
<tr>
<th>Name</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>CONFIG2H</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>WDTPS2</td>
<td>WDTPS1</td>
<td>WDTPS0</td>
<td>WDTEN</td>
</tr>
<tr>
<td>WDTCON</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>RCON</td>
<td>IPEN</td>
<td>LWRT</td>
<td></td>
<td>RI</td>
<td>TO</td>
<td>PD</td>
<td>POR</td>
<td>BOR</td>
</tr>
</tbody>
</table>

Legend: Shaded cells are not used by the Watchdog Timer.
Section 28. Watchdog Timer and SLEEP Mode

28.4 SLEEP (Power-Down) Mode

SLEEP (Power-down) mode is the lowest current consumption state and is entered by executing a SLEEP instruction. The device oscillator is turned off, so no system clocks are occurring in the device.

If enabled, the Watchdog Timer will be cleared but keeps running, the PD bit in the RCON register is cleared, the TO bit is set, and the oscillator driver is turned off. The I/O ports maintain the status they had before the SLEEP instruction was executed (driving high, low, or hi-impedance).

For lowest current consumption in this mode, all I/O pins should be either at VDD or VSS, with no external circuitry drawing current from the I/O pin and modules that are specified to have a delta SLEEP current, should be disabled. I/O pins that are hi-impedance inputs should be pulled high or low externally, to avoid switching currents caused by floating inputs. The contribution from on-chip pull-ups on PORTB should be considered.

During SLEEP, the MCLR pin must be at a valid high level.

Some features of the device consume a delta current. These are enabled/disabled by device configuration bits. These features include the Watchdog Timer (WDT), LVD, and Brown-out Reset (BOR) circuitry modules.

28.4.1 Wake-up from SLEEP

There are several ways to wake the controller from SLEEP. The WDT can wake-up the controller when it times out. A RESET will wake the controller and cause the program to restart, and interrupts (from peripherals or external sources) will wake the controller from SLEEP.

The device can wake-up from SLEEP through one of the following events:

1. Any device RESET, such as MCLR pin = VIL, VDD = VBOR (if enabled).
2. Watchdog Timer Wake-up (if WDT was enabled).
3. Any peripheral module which can set its interrupt flag while in SLEEP, such as:
   - An external INT pin
   - Change on Port pin
   - Comparators
   - A/D
   - Timer1
   - Timer3
   - LVD
   - MSSP
   - Capture
   - PSP read or write
   - CCP1
   - CCP2
   - Addressable USART
   - PORTB Interrupt on Change
   - External Interrupts
   - Parallel Slave Port
   - Voltage Reference (bandgap)
   - WDT

The first event will RESET the device upon wake-up. However, the latter two events will wake the device and then resume program execution. The TO and PD bits in the RCON register can be used to determine the cause of device RESET. The PD bit, which is set on power-up, is cleared when SLEEP is invoked. The TO bit is cleared if WDT time-out occurred (and caused a wake-up).

When the SLEEP instruction is being executed, the next instruction (PC + 2) is pre-fetched. For the device to wake-up through an interrupt event, the corresponding interrupt enable bit must be set (enabled). Wake-up is regardless of the state of the GIE bit. If the GIE bit is clear (disabled), the device continues execution at the instruction after the SLEEP instruction. If the GIE bit is set (enabled), the device executes the instruction after the SLEEP instruction and then branches to the interrupt address. In cases where the execution of the instruction following SLEEP is not desirable, the user should have a NOP after the SLEEP instruction.
28.4.2 Wake-up Using Interrupts

When interrupts are globally disabled (GIE cleared) and any interrupt source has both its interrupt enable bit and interrupt flag bit set, one of the following events will occur:

- If an interrupt condition (interrupt flag bit and interrupt enable bits are set) occurs before the execution of a SLEEP instruction, the SLEEP instruction will complete as a NOP. Therefore, the WDT and WDT postscaler will not be cleared, the TO bit will not be set and PD bit will not be cleared.

- If the interrupt condition occurs during or after the execution of a SLEEP instruction, the device will immediately wake-up from SLEEP. The SLEEP instruction will be completely executed before the wake-up. Therefore, the WDT and WDT postscaler will be cleared, the TO bit will be set and the PD bit will be cleared.

Even if the flag bits were checked before executing a SLEEP instruction, it may be possible for flag bits to become set before the SLEEP instruction completes. To determine whether a SLEEP instruction executed, test the PD bit. If the PD bit is set, the SLEEP instruction was executed as a NOP.

To ensure that the WDT is clear, a CLRWDT instruction should be executed before a SLEEP instruction.

Figure 28-2: Wake-up from SLEEP Through Interrupt

Note 1: XT, HS or LP oscillator mode assumed.
2: TOST = 1024Tosc (drawing not to scale). This delay will not occur for RC and EC osc modes.
3: GIE = ‘1’ assumed. In this case, after wake-up, the processor jumps to the interrupt routine. If GIE = ‘0’, execution will continue in-line.
4: CLKOUT is not available in these osc modes, but shown here for timing reference.
Section 28. Watchdog Timer and SLEEP Mode

Interrupt sources can wake the controller from SLEEP without actually causing an interrupt. The interrupt source must have its interrupt enable flag set, but GIE does not need to be set. If GIE is clear, the controller will wake without vectoring to an interrupt. If GIE is set, the controller will vector to an interrupt.

If interrupt priority is not used, all interrupt priority bits are set. If interrupt priority is used (any interrupt priority bit is cleared), GIEH controls high priority interrupts and GIEL controls low priority interrupts. Table 28-2 shows the response to the interrupt flag bits depending on the state of the interrupt enable and priority bits.

Table 28-2: SLEEP Mode, Interrupt Enable Bits, and Interrupt Results

<table>
<thead>
<tr>
<th>Interrupt Source</th>
<th>GIE/GIEH</th>
<th>PEIE/GIEL</th>
<th>Interrupt Priority</th>
<th>Peripheral Interrupt Flag</th>
<th>Response to Interrupt</th>
</tr>
</thead>
<tbody>
<tr>
<td>Any interrupt source that operates during SLEEP</td>
<td>X</td>
<td>X</td>
<td>X</td>
<td>0</td>
<td>SLEEP</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>low priority</td>
<td>1</td>
<td>wake</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>1</td>
<td>high priority</td>
<td>1</td>
<td>wake</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>X</td>
<td>1</td>
<td>1</td>
<td>wake</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>High priority vector followed</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>0</td>
<td>1</td>
<td>1</td>
<td>Low priority vector followed</td>
</tr>
</tbody>
</table>

Legend: X is don’t care.
28.4.3 Effects of SLEEP Mode on the On-Chip Oscillator

When the device executes a SLEEP instruction, the Watchdog Timer and prescaler counter are cleared (if the WDT is enabled), the on-chip clocks and oscillator are turned off and the controller is held at the beginning of an instruction cycle (Q1 state). With the oscillator off, the OSC1 and OSC2 signals will stop oscillating. Since all the transistor switching currents have been removed, SLEEP mode achieves the lowest current consumption of the device (only leakage currents). Enabling any on-chip feature that will operate during SLEEP will increase the current consumed during SLEEP. The user can wake from SLEEP through external RESET, Brown-out Reset (if enabled), external interrupt, Watchdog Timer time-out or a peripheral interrupt.

Table 28-3: Oscillator Selections, SLEEP Mode, and Waking from SLEEP

<table>
<thead>
<tr>
<th>OSC Mode</th>
<th>OSC1 Pin in SLEEP</th>
<th>OSC2 Pin in SLEEP</th>
<th>Waking Delays</th>
<th>OSC1 in Run</th>
<th>OSC2 in Run</th>
</tr>
</thead>
<tbody>
<tr>
<td>RC</td>
<td>Floating, pulled high</td>
<td>At logic low</td>
<td>None</td>
<td>R and C set frequency</td>
<td>CLKO (4Tosc)</td>
</tr>
<tr>
<td>RCIO</td>
<td>Floating, pulled high</td>
<td>Configured as I/O pin</td>
<td>None</td>
<td>R and C set frequency</td>
<td>Configured as I/O pin</td>
</tr>
<tr>
<td>LP</td>
<td>At quiescent voltage level</td>
<td>At quiescent voltage level</td>
<td>TOST (1)</td>
<td>XTAL/res</td>
<td>XTAL/res</td>
</tr>
<tr>
<td>XT</td>
<td>At quiescent voltage level</td>
<td>At quiescent voltage level</td>
<td>TOST (1)</td>
<td>XTAL/res</td>
<td>XTAL/res</td>
</tr>
<tr>
<td>HS</td>
<td>At quiescent voltage level</td>
<td>At quiescent voltage level</td>
<td>TOST (1)</td>
<td>XTAL/res</td>
<td>XTAL/res</td>
</tr>
<tr>
<td>HS w/PLL</td>
<td>At quiescent voltage level</td>
<td>At quiescent voltage level</td>
<td>TOST (1) + TPLL (2)</td>
<td>XTAL/res</td>
<td>XTAL/res</td>
</tr>
<tr>
<td>EC</td>
<td>Driven by external clock source</td>
<td>At logic low</td>
<td>None</td>
<td>Driven by external clock source</td>
<td>CLKO (4Tosc)</td>
</tr>
<tr>
<td>ECIO</td>
<td>Driven by external clock source</td>
<td>Configured as I/O pin</td>
<td>None</td>
<td>Driven by external clock source</td>
<td>Configured as I/O pin</td>
</tr>
</tbody>
</table>

Note 1: TOST (Oscillator Start-up Timer) counts 1024 oscillator cycles before allowing controller clocks to resume. This provides time for the oscillator to start-up and stabilize.

Note 2: A TPLL delay is required to allow the PLL to lock to the oscillator frequency.
Section 28. Watchdog Timer and SLEEP Mode

28.4.4 Wake-up Delays

Several factors affect how much time the controller requires to return to operating mode from SLEEP. These include oscillator mode and the use of the PLL.

The Oscillator Start-up Timer, OST, counts 1024 oscillator cycles to allow the oscillator to start-up and stabilize before allowing system clocks to resume. The OST is not enabled for RC and EC oscillator modes.

28.4.4.1 Oscillator With PLL Enabled Time-out Sequence After Wake-up

The Oscillator Start-up Timer (OST) provides a 1024 oscillator cycle delay after a wake-up from SLEEP has occurred. 1024 oscillator cycles are not a sufficient amount of time to allow the PLL to lock at high frequencies. An additional TPLL time is required to allow the PLL to lock before allowing system clocks to resume.
28.4.5 Peripheral Module Operation During SLEEP

Table 28-4 gives an overview of which devices operate during SLEEP. For further details, refer to the individual sections in this reference manual.

Table 28-4: Peripheral Modules Active in SLEEP Mode

<table>
<thead>
<tr>
<th>Peripheral Module</th>
<th>Operates During SLEEP?</th>
<th>Mode of Operation</th>
<th>Wakes from SLEEP?</th>
</tr>
</thead>
<tbody>
<tr>
<td>Timer1, Timer3</td>
<td>Yes</td>
<td>External clock/U.S.C.G., Asynchronous Counter mode</td>
<td>Yes</td>
</tr>
<tr>
<td>A/D</td>
<td>Yes</td>
<td>A/D clock = RC clock</td>
<td>Yes</td>
</tr>
<tr>
<td>CCP1, CCP2</td>
<td>Yes</td>
<td>Only capture available.</td>
<td>Yes, do not rely on capture value</td>
</tr>
<tr>
<td>MSSP</td>
<td>Yes</td>
<td>I2C – Non-master modes SPI – Slave mode</td>
<td>Yes</td>
</tr>
<tr>
<td>Addressable USART</td>
<td>Yes</td>
<td>Synchronous slave mode</td>
<td>Yes</td>
</tr>
<tr>
<td>PORTB Interrupt on Change</td>
<td>Yes</td>
<td>All</td>
<td>Yes</td>
</tr>
<tr>
<td>External Interrupts</td>
<td>Yes</td>
<td>All</td>
<td>Yes</td>
</tr>
<tr>
<td>Parallel Slave Port</td>
<td>Yes</td>
<td>All</td>
<td>Yes</td>
</tr>
<tr>
<td>LVD</td>
<td>Yes</td>
<td>All</td>
<td>Yes</td>
</tr>
<tr>
<td>Volt Reference (bandgap)</td>
<td>Yes</td>
<td>If required to support LVD, and A/D</td>
<td>No</td>
</tr>
<tr>
<td>WDT</td>
<td>Yes</td>
<td>All</td>
<td>Yes</td>
</tr>
</tbody>
</table>

28.4.6 Effects of a WDT Time-out

If the WDT has been enabled, either by the WDTEN configuration bit (=1) or by the SWDTEN bit being set, the WDT will wake-up the controller from SLEEP mode and clear the TO bit.

28.4.7 Effects of a Device RESET

When MCLR is asserted, TO is set and PD is clear. All other bits in RCON are unchanged. The controller will resume code execution at the RESET vector address.
Section 28. Watchdog Timer and SLEEP Mode

28.5 Initialization

No initialization code at this time.
28.6 Design Tips

Question 1: My system voltage drops and then returns to the specified device voltage range. The device is not operating correctly and the WDT does not reset and return the device to proper operation.

Answer 1:
The WDT was not designed to be a recovery from a brown-out condition. It was designed to recover from errant software operation (the device remaining in the specified operating ranges). If your system can be subjected to brown-outs, either the on-chip brown-out circuitry should be enabled or an external brown-out circuit should be implemented.

Question 2: Device RESETS even though I do the \texttt{CLRWDT} instruction in my loop.

Answer 2:
Make sure that the loop with the \texttt{CLRWDT} instruction meets the minimum specification of the WDT (not the typical).

Question 3: Device never gets out of RESETS.

Answer 3:
On power-up, you must take into account the Oscillator Start-up time (\texttt{Tost}). Sometimes it helps to put the \texttt{CLRWDT} instruction at the beginning of the loop, since this start-up time may be variable.
Section 28. Watchdog Timer and SLEEP Mode

28.7 Related Application Notes

This section lists application notes that are related to this section of the manual. These applica-
tion notes may not be written specifically for the Enhanced family (that is, they may be written for
the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent and could be
used (with modification and possible limitations). The current application notes related to the
WDT and SLEEP Mode are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>Power-up Trouble Shooting</td>
<td>AN607</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These
code examples are stand alone examples to assist in the understanding of the
PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
28.8 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Watchdog Timer and SLEEP mode description.
Section 29. Device Configuration Bits

HIGHLIGHTS

This section of the manual contains the following major topics:

29.1 Introduction ............................................................................................................... 29-2
29.2 Configuration Word Bits ........................................................................................... 29-3
29.3 Program Verification/Code Protection ....................................................................... 29-10
29.4 ID Locations ............................................................................................................ 29-11
29.5 Device ID ................................................................................................................ 29-11
29.6 Design Tips ............................................................................................................. 29-12
29.7 Related Application Notes ....................................................................................... 29-13
29.8 Revision History ..................................................................................................... 29-14
The device configuration bits allow each user to customize certain aspects of the device to the needs of the application. When the device powers up, the state of these bits determines the modes that the device uses. Section 29.2 "Configuration Word Bits" discusses the configuration bits and the modes to which they can be configured. These bits are mapped in program memory locations starting at address 300000h. These locations are accessible during normal device operation.

The configuration bits can be programmed (read as '0') or left unprogrammed (read as '1') to select various device configurations. The ability to change these settings once they have been programmed depends on the memory technology and the package type. This is discussed below:

- Read Only Memory (ROM) devices: These bits are specified at time of ROM code submission; once the device is masked, these bits can not be changed (would require a new mask code).
- One Time Programmable (OTP) devices: Once the bit is programmed ('0'), it may not be changed.
- Windowed EPROM devices: Once these bits are programmed ('0'), the device must be UV erased to return the configuration word to the erased state. UV erasing the device also erases the program memory. Window devices are for debugging purposes.

**Note:** Microchip does not recommend code-protecting windowed devices.

- FLASH devices: These bits may be erased and reprogrammed.
Section 29. Device Configuration Bits

29.2 Configuration Word Bits

These configuration bits specify some of the device modes and are programmed by a device programmer, or by using the In-Circuit Serial Programming™ (ICSP) feature of the Enhanced Architecture devices. The placement of these configuration bits is automatically handled when you select the device in your device programmer. The desired state of the configuration bits may be specified in the source code (dependent on the language tool used), or through the programming interface. After the device has been programmed, the application software may read the configuration bit values through the Table read instructions. For additional information, please refer to the Programming Specification of the device.

Note 1: Always ensure that your device programmer has the same device selected as you are programming.

2: Microchip recommends that the desired configuration bit states be embedded into the application source code. This is easily done in the MPASM assembler by the use of the CONFIG directive. See Subsection 29.2.1 “MPASM’s CONFIG Directive.”
Register 29-1: Configuration Bits

P/R-u

Bit placement is device dependent

A19DIS: Disable A19:A16 Drivers bit
This bit disables the A19:A16 address lines of the system bus so that these pins may be used as general purpose I/O.

1 = Drivers enabled
0 = Drivers disabled

A15DIS: Disable AD15:AD12 Drivers bit
This bit disables the AD15:AD12 address lines of the system bus so that these pins may be used as general purpose I/O. The AD15:AD12 address lines may only be disabled when the system bus is configured for 8-bit data (BW16 = '0').

When BW16 = '1':
1 = Drivers enabled (state of A15DIS is ignored)
0 = Drivers enabled (state of A15DIS is ignored)

When BW16 = '0':
1 = Drivers enabled
0 = Drivers disabled

A11DIS: Disable AD11:AD8 Drivers bit
This bit disables the AD11:AD8 address lines of the system bus so that these pins may be used as general purpose I/O. The AD11:AD8 address lines may only be disabled when the system bus is configured for 8-bit data (BW16 = '0').

When BW16 = '1':
1 = Drivers enabled (state of A11DIS is ignored)
0 = Drivers enabled (state of A11DIS is ignored)

When BW16 = '0':
1 = Drivers enabled
0 = Drivers disabled

BADIS: Byte Address BA0 Disable bit

1 = Drivers enabled
0 = Drivers disabled

BSDIS: Byte Select UB, LB Disable bit

1 = Drivers enabled
0 = Drivers disabled

BOREN: Brown-out Reset Enable bit

1 = Brown-out Reset enabled
0 = Brown-out Reset disabled

BORV1:BORV0: Brown-out Reset Voltage bits
These bits specify the trip point for the Brown-out Reset circuitry. The values shown below are for a typical device. Please refer to the device data sheet “Electrical Specifications” section for the tested range.

11 = VBOR set to 2.5V or 1.8V (device dependent)
10 = VBOR set to 2.7V
01 = VBOR set to 4.2V
00 = VBOR set to 4.5V
Section 29. Device Configuration Bits

BW16: 16-bit Bus Width bit
This bit specifies the data width of the system bus.
1 = System Bus has 16-bit data bus
0 = System Bus has 8-bit data bus

CCP2MX: CCP2 Mux bit
1 = CCP2 input/output is multiplexed with an I/O
0 = CCP2 input/output is multiplexed with a different I/O

CP: Code Protection bits (apply when in Code Protected Microcontroller Mode)
1 = Program memory code protection off
0 = All of program memory code protected

DP: Data EEPROM Memory Code Protection bit
1 = Code protection off
0 = Data EEPROM memory is code protected

Note: This bit is used when a ROM program memory device or a ROMless device has Data EEPROM memory.

FOSC2:FOSC0: Oscillator Selection bits
111 = RC oscillator w/ OSC2 configured as I/O
110 = HS oscillator with PLL enabled/clock frequency = (4 x Fosc1)
101 = EC oscillator w/ OSC2 configured as I/O
100 = EC oscillator w/ OSC2 configured as divide by 4 clock output
011 = RC oscillator
010 = HS oscillator
001 = XT oscillator
000 = LP oscillator

OSCSEFN: Oscillator System Clock Switch Enable bit
1 = Oscillator system clock switch option is disabled (main oscillator is source)
0 = Oscillator system clock switch option is enabled (oscillator switching is enabled)

PM1:PM0: Processor Mode Select bits
These bits select the processor operating mode for the device. The processor operating mode specifies how the program memory is mapped (internal/external) and the default configuration of the system bus pins.
11 = Microprocessor mode
10 = Microcontroller mode
01 = Reserved
00 = Extended microcontroller mode

PWRTE: Power-up Timer Enable bit
1 = PWRT disabled
0 = PWRT enabled

STVREN: Stack Full/Underflow Reset Enable bit
1 = Stack Full/Underflow will cause RESET
0 = Stack Full/Underflow will not cause RESET
**PIC18C Reference Manual**

**WAIT:** System Bus Wait bit

This bit is used to enable wait states for table reads and table writes to external memory on the system bus:

- 1 = Wait selections unavailable, device will not wait
- 0 = Wait programmed by WAIT1 and WAIT0 bit in MEMCON register

**WDIS:** Write Select WRH, WRL Disable bit

- 1 = Drivers enabled
- 0 = Drivers disabled

**WDTPS2:WDTPS0:** Watchdog Timer Postscale Select bits

<table>
<thead>
<tr>
<th>Value</th>
<th>Postscale</th>
</tr>
</thead>
<tbody>
<tr>
<td>111</td>
<td>1:1</td>
</tr>
<tr>
<td>110</td>
<td>1:2</td>
</tr>
<tr>
<td>101</td>
<td>1:4</td>
</tr>
<tr>
<td>100</td>
<td>1:8</td>
</tr>
<tr>
<td>011</td>
<td>1:16</td>
</tr>
<tr>
<td>010</td>
<td>1:32</td>
</tr>
<tr>
<td>001</td>
<td>1:64</td>
</tr>
<tr>
<td>000</td>
<td>1:128</td>
</tr>
</tbody>
</table>

**WDTEN:** Watchdog Timer Enable bit

- 1 = WDT enabled
- 0 = WDT disabled (control is placed on the SWDTEN bit, in register WDTCON)

**Legend**

- **R** = Readable bit
- **P** = Programmable bit
- **U** = Unimplemented bit, read as '0'
- **n** = Value at POR reset
- **’1’** = bit is set
- **’0’** = bit is cleared
- **x** = bit is unknown

**Note 1:** The position of the configuration bits is device dependent. Please refer to the device programming specification for bit placement. You are not required to know the configuration bit positions when using a Microchip device programmer. This is addressed by either the configuration direction of the software tool or MPLAB’s user interface.

**Note 2:** In ROMless devices, some of the system bus configuration bits are hardwired into a set configuration. Other bits may be placed in protected SFR locations which can only be modified after properly writing to the CMLK1:CMLK0 bits.
Section 29. Device Configuration Bits

29.2.1 MPASM’s CONFIG Directive

Microchip’s assembler, MPASM, has a nice feature (directives) that allows you to specify the device configuration in the source code file. This ensures that when programming a device for an application, the required configuration is also programmed. This minimizes the risk of programming the wrong device configuration and then wondering why it no longer works in the application (unexpected operation).

Example 29-1 shows a template for using the CONFIG directive.

Example 29-1: Using the CONFIG Directive, a Source File Template

```
LIST p = p18C452 ; List Directive,
; Revision History
;
#include <P18C452.INC> ; Microchip Device Header File
;
#include <MY_STD.MAC> ; File which includes my standard macros
#include <APP.MAC> ; File which includes macros specific to this application
;
; Specify Device Configuration Bits for Program Configuration Registers 0 through 6
__config , CP_OFF_0
__config , LPSREN_OFF_2 & RCRA6_OSC_1
__config , BORV_25_2 & BOREN_ON_2 & PWRTEN_OFF_2
__config , WDPS_128_3 & WDT_ON_3
__config , CCP2MX_ON_5
__config , SVTREN_ON_6
;
org 0x00 ; Start of Program Memory
RESET_ADDR : ; First instruction to execute after a reset

end
```

The symbols currently in the Microchip Device Header files make using the CONFIG directive straightforward. These are shown in Table 29-1. The symbols available for your device are listed in the Microchip Include file for that device.

**Note:** As long as the correct device is specified (in the LIST and INCLUDE file directives), the correct polarity of all bits is ensured.
<table>
<thead>
<tr>
<th>Feature</th>
<th>Configuration Byte</th>
<th>Symbols</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>Oscillators</td>
<td>_CONFIG1H</td>
<td>_LP_OSC_1</td>
<td>LP mode for device</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_XT_OSC_1</td>
<td>XT mode for device</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_HS_OSC_1</td>
<td>HS mode for device oscillator</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_RC_OSC_1</td>
<td>RC mode for device oscillator</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_RCIO_OSC_1</td>
<td>RC mode for device oscillator with clockout</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_EC_OSC_1</td>
<td>EC (external clock) mode for device oscillator</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_ECIO_OSC_1</td>
<td>EC (external clock) mode for device oscillator</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_HS4_OSC_1</td>
<td>HS mode with 4 x PLL for device oscillator</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_HSPLL_OSC_1</td>
<td>HS mode with 4 x PLL for device oscillator</td>
</tr>
<tr>
<td>Oscillator Switch</td>
<td>_CONFIG1H</td>
<td>_LPSCEN_ON_1</td>
<td>Oscillator switch enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_LPSCEN_OFF_1</td>
<td>Oscillator switch disabled</td>
</tr>
<tr>
<td>Code Protect</td>
<td>_CONFIG1L</td>
<td>_CP_ON_0</td>
<td>Code Protect enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_CP_OFF_0</td>
<td>Code Protect disabled</td>
</tr>
<tr>
<td>Watchdog Timer</td>
<td>_CONFIG2H</td>
<td>_WDT_ON_3</td>
<td>Watchdog Timer enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDT_OFF_3</td>
<td>Watchdog Timer disabled</td>
</tr>
<tr>
<td>Watchdog Timer Postscale Assignment</td>
<td>_CONFIG2H</td>
<td>_WDTPS_128_3</td>
<td>WDT prescaler set to 1:128</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_64_3</td>
<td>WDT prescaler set to 1:64</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_32_3</td>
<td>WDT prescaler set to 1:32</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_16_3</td>
<td>WDT prescaler set to 1:16</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_8_3</td>
<td>WDT prescaler set to 1:8</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_4_3</td>
<td>WDT prescaler set to 1:4</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_2_3</td>
<td>WDT prescaler set to 1:2</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDTPS_1_3</td>
<td>WDT prescaler set to 1:1</td>
</tr>
<tr>
<td>Power-up Timer</td>
<td>_CONFIG2L</td>
<td>_PWRTEN_ON_2</td>
<td>Power-up Timer enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_PWRTEN_OFF_2</td>
<td>Power-up Timer disabled</td>
</tr>
<tr>
<td>Brown-out Reset</td>
<td>_CONFIG2L</td>
<td>_BOREN_ON_2</td>
<td>Brown-out Reset enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BOREN_OFF_2</td>
<td>Brown-out Reset disabled</td>
</tr>
<tr>
<td>BOR Trip-Point Voltage</td>
<td>_CONFIG2L</td>
<td>_BORV_18_2</td>
<td>BOR trip point = 1.8V min (2, 3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BORV_25_2</td>
<td>BOR trip point = 2.5V min (2, 3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BORV_27_2</td>
<td>BOR trip point = 2.7V min (3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BORV_42_2</td>
<td>BOR trip point = 4.2V min (3)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BORV_45_2</td>
<td>BOR trip point = 4.5V min (3)</td>
</tr>
</tbody>
</table>

Note 1: Not all configuration bit symbols may be available on any one device. Please refer to the Microchip include file of that device for available symbols.

2: The option for a 1.8V or 2.5V BOR trip point is device dependent. Only one of these symbols will be available in the Microchip supplied header file.

3: These are the trip points for a typical device. Other trip points (symbols) may be specified for the device.

4: Symbol obsoleted and may not be available in header file.
# Section 29. Device Configuration Bits

## Table 29-1: _CONFIG Directive Symbols (from Microchip Header Files) (1) (Continued)

<table>
<thead>
<tr>
<th>Feature</th>
<th>Configuration Byte</th>
<th>Symbols</th>
<th>Comment</th>
</tr>
</thead>
<tbody>
<tr>
<td>CCP2 pin multiplex</td>
<td>_CONFIG3H</td>
<td>_CCP2MX_ON_5</td>
<td>CCP2 out multiplexer enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_CCP2MX_OFF_5</td>
<td>CCP2 out multiplexer disabled</td>
</tr>
<tr>
<td>Processor Mode</td>
<td>_CONFIG3L</td>
<td>_MP_MODE_4</td>
<td>Microprocessor mode</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_MC_MODE_4</td>
<td>Microcontroller mode</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_XMC_MODE_4</td>
<td>Extended Microcontroller mode</td>
</tr>
<tr>
<td>External Data Bus Wait</td>
<td>_CONFIG3L</td>
<td>_WAIT_ON_4</td>
<td>Wait on external data bus enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WAIT_OFF_4</td>
<td>Wait on external data bus disabled</td>
</tr>
<tr>
<td>System Bus A19 Disable</td>
<td>_CONFIG4H</td>
<td>_A19DIS_ON_7</td>
<td>System bus A19 disabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_A19DIS_OFF_7</td>
<td>System bus A19 enabled</td>
</tr>
<tr>
<td>System Bus A15 Disable</td>
<td>_CONFIG4H</td>
<td>_A15DIS_ON_7</td>
<td>System bus A15 disabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_A15DIS_OFF_7</td>
<td>System bus A15 enabled</td>
</tr>
<tr>
<td>System Bus A11 Disable</td>
<td>_CONFIG4H</td>
<td>_A11DIS_ON_7</td>
<td>System bus A11 disabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_A11DIS_OFF_7</td>
<td>System bus A11 enabled</td>
</tr>
<tr>
<td>Byte Address BA0 Disable</td>
<td>_CONFIG4H</td>
<td>_BADIS_ON_7</td>
<td>Byte Address BA0 disabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BADIS_OFF_7</td>
<td>Byte Address BA0 enabled</td>
</tr>
<tr>
<td>Byte Select Disable</td>
<td>_CONFIG4H</td>
<td>_BSDIS_ON_7</td>
<td>Byte Select enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_BSDIS_OFF_7</td>
<td>Byte Select enabled</td>
</tr>
<tr>
<td>Write Select Disable</td>
<td>_CONFIG4H</td>
<td>_WDIS_ON_7</td>
<td>Write Select enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_WDIS_OFF_7</td>
<td>Write Select enabled</td>
</tr>
<tr>
<td>Stack Full/Overflow</td>
<td>_CONFIG4L</td>
<td>_STVREN_ON_6</td>
<td>Stack full/overflow rest enabled</td>
</tr>
<tr>
<td></td>
<td></td>
<td>_STVREN_OFF_6</td>
<td>Stack full/overflow rest disabled</td>
</tr>
<tr>
<td>Code Protect</td>
<td>TBD</td>
<td>_DP_ON</td>
<td>Data EEPROM protect enabled</td>
</tr>
<tr>
<td>Data EEPROM</td>
<td></td>
<td>_DP_OFF</td>
<td>Data EEPROM protect disabled</td>
</tr>
</tbody>
</table>

**Note 1:** Not all configuration bit symbols may be available on any one device. Please refer to the Microchip include file of that device for available symbols.

**Note 2:** The option for a 1.8V or 2.5V BOR trip point is device dependent. Only one of these symbols will be available in the Microchip supplied header file.
29.3 Program Verification/Code Protection

If the code protection bit(s) have not been programmed, the on-chip program memory can be read out for verification purposes.

**Note:** Microchip does not recommend code protecting windowed devices.

29.3.1 ROM Devices

When a ROM device also has Data EEPROM memory, an additional code protect configuration bit may be implemented. The program memory configuration bit is submitted as part of the ROM code submittal. The Data EEPROM memory code protect configuration bit will be an EEPROM bit. When ROM devices complete testing, the EEPROM data memory code protect bit will be programmed to the same state as the program memory code protect bit. That is, Data EEPROM code protect is off when program memory code protect is off and Data EEPROM code protect is on for all other selections.

In applications where the device is code protected and the Data EEPROM needs to be programmed before the application can be released, the Data EEPROM memory must have the entire Data EEPROM memory erased. The device programming specification details the steps to do this. Microchip device programmers implement the specified sequence. Once this sequence is complete, the Data EEPROM memory code protect is disabled. This allows the desired data to be programmed into the device. After programming the Data EEPROM memory array, the Data EEPROM memory code protect configuration bit should be programmed as desired.
Section 29. Device Configuration Bits

29.4 ID Locations

Five memory locations (200000h - 200004h) are designated as ID locations where the user can store checksum or other code-identification numbers. These locations are accessible during normal execution through the TBLRD instruction, or during program/verify. The ID locations can be read when the device is code protected.

29.5 Device ID

One memory location (two bytes) (3FFFFFFEh-3FFFFFFFh) is designated as the Device ID location. The value at this location is specified by Microchip and is useful in determining the device.

Device ID bits can be used by a device programmer to retrieve information about what device is being programmed and what the revision of the device is.

The Device ID can be accessed by a TBLRD instruction or via serial program/verify. The Device ID can be read when the part is code protected.

The 5 LSbs are the device revision information, and the remaining 11 bits contain the device ID number. This is shown in Table 29-2.

Table 29-2: Device ID Registers

<table>
<thead>
<tr>
<th>Register</th>
<th>Bit 7</th>
<th>Bit 6</th>
<th>Bit 5</th>
<th>Bit 4</th>
<th>Bit 3</th>
<th>Bit 2</th>
<th>Bit 1</th>
<th>Bit 0</th>
</tr>
</thead>
<tbody>
<tr>
<td>DEVID2</td>
<td>DEV10</td>
<td>DEV9</td>
<td>DEV8</td>
<td>DEV7</td>
<td>DEV6</td>
<td>DEV5</td>
<td>DEV4</td>
<td>DEV3</td>
</tr>
<tr>
<td>DEVID1</td>
<td>DEV2</td>
<td>DEV1</td>
<td>DEV0</td>
<td>REV4</td>
<td>REV3</td>
<td>REV2</td>
<td>REV1</td>
<td>REV0</td>
</tr>
</tbody>
</table>
29.6 Design Tips

Question 1: *I have a JW device and I can no longer program it (reads scrambled data or all '0's). What's wrong with the device?*

Answer 1:
Nothing. You probably code protected the device. If this is the case, the device is no longer usable. See Section 29.3 “Program Verification/Code Protection” for more details.

Question 2: *In converting from a PIC16C74 to a PIC18C452, my application no longer works.*

Answer 2:
1. Did you re-assemble the source file specifying the PIC18C452 in the INCLUDE file and LIST directives? The use of the CONFIG directive is highly recommended.
2. On the device programmer, did you specify the PIC18C452, and were all the configuration bits as desired?

Question 3: *When I erase the device, the program memory is blank but the configuration word is not yet erased.*

Answer 3:
That is by design. Also remember that Microchip does not recommend code protecting windowed devices.
Section 29. Device Configuration Bits

29.7 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is, they may be written for the Base-Line, Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to Configuration Word are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related Application Notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faq/codeex/
29.8 Revision History

Revision A

This is the initial released revision of the Configuration Word description.
Section 30. In-Circuit Serial Programming™ (ICSP™)

HIGHLIGHTS

This section of the manual contains the following major topics:

30.1 Introduction ............................................................................................................... 30-2
30.2 Entering In-Circuit Serial Programming Mode ............................................................. 30-3
30.3 Application Circuit .................................................................................................... 30-4
30.4 Programmer .............................................................................................................. 30-6
30.5 Programming Environment ....................................................................................... 30-6
30.6 Other Benefits ........................................................................................................... 30-7
30.7 Field Programming of PICmicro OTP MCUs ............................................................... 30-8
30.8 Field Programming of FLASH PICmicro MCUs ......................................................... 30-10
30.9 Design Tips .............................................................................................................. 30-12
30.10 Related Application Notes ...................................................................................... 30-13
30.11 Revision History ..................................................................................................... 30-14
30.1 Introduction

All Enhanced MCU devices can be In-Circuit Serial Programmed (ICSP™) while in the end application circuit. This is simply done with two lines for clock and data, and three other lines for power, ground and the programming voltage.

In-Circuit Serial Programming (ICSP™) is a great way to reduce your inventory overhead and time-to-market for your product. By assembling your product with a blank Microchip microcontroller (MCU), you can stock one design. When an order has been placed, these units can be programmed with the latest revision of firmware, tested and shipped in a very short time. This method also reduces scrapped inventory due to old firmware revisions. This type of manufacturing system can also facilitate quick turnarounds on custom orders for your product.

Most people would think to use ICSP with PICmicro® OTP MCUs only on an assembly line where the device is programmed once. However, there is a method by which an OTP device can be programmed several times depending on the size of the firmware. This method, explained in Section 30.7, provides a way to field upgrade your firmware in a way similar to EEPROM- or FLASH-based devices.
Section 30. ICSP

30.2 Entering In-Circuit Serial Programming Mode

The device is placed into a program/verify mode by holding the CLOCK (typically on the RB6 pin) and DATA (typically on the RB7 pin) pins low, while raising the MCLR (VPP) pin from VIL to VIHH (see programming specification) and having VDD at the programming voltage. Both the CLOCK and DATA pins are Schmitt Trigger inputs in this mode. When in I/O mode and RB7 is driving data, it is a CMOS output driver.

After RESET, to place the device into programming/verify mode, the program counter (PC) is at location 00h. A command is then supplied to the device. Some commands then specify that 16-bits of program data are then supplied to or read from the device, depending on whether the command was a load or a read. For complete details of serial programming, please refer to the device specific Programming Specifications.

During the In-Circuit Serial Programming Mode, the WDT circuitry is disabled from generating a device RESET.
30.3 Application Circuit

The application circuit must be designed to allow all the programming signals to be directly connected to the PICmicro MCU. Figure 30-1 shows a typical circuit that is a starting point when designing with ICSP. The application must compensate for the following:

1. Isolation of the MCLR/VPP pin from the rest of the circuit
2. Loading of pins CLOCK and DATA
3. Capacitance on each of the VDD, MCLR/VPP, CLOCK and DATA pins
4. Minimum and maximum operating voltage for VDD
5. PICmicro oscillator

30.3.1 Isolation of the MCLR/VPP Pin from the Rest of the Circuit

The MCLR/VPP pin is normally connected to an RC circuit. The pull-up resistor is tied to VDD and a capacitor is tied to ground. This circuit can affect the operation of ICSP depending on the size of the capacitor, since the VPP voltage must be isolated from the rest of the circuit. The resistor (R1) should be greater than 10kΩ to provide isolation between VDD and VPP. It is, therefore, recommended that the circuit in Figure 30-1 be used when an RC is connected to MCLR/VPP. Another consideration with MCLR/VPP is that when the PICmicro device is programmed, this pin is driven to approximately 13V and also to ground. Therefore, the application circuit must be isolated from this voltage provided by the programmer.

30.3.2 Loading of Pins CLOCK and DATA

The CLOCK and DATA pins are used by the PICmicro MCU for serial programming. CLOCK is driven by the programmer. DATA is a bi-directional pin that is driven by the programmer when programming, and driven by the MCU when verifying. These pins must be isolated from the rest of the application circuit so as not to affect the signals during programming. You must take into consideration the output impedance of the programmer when isolating CLOCK and DATA from the rest of the circuit. This isolation circuit must account for CLOCK being an input on the MCU and for DATA being bi-directional (can be driven by both the MCU and the programmer). For instance, PRO MATE® II has an output impedance of 1kΩ. If the design permits, these pins should not be used by the application. This is not the case with most applications, so it is recommended that the designer evaluate whether these signals need to be buffered. As a designer, you must consider what type of circuitry is connected to CLOCK and DATA and then make a decision on how to isolate these pins. Figure 30-1 does not show any circuitry to isolate CLOCK and DATA on the application circuit, because this is very application dependent.

To simplify this interface, the optimal usage of these I/O pins in the application are (in order):

1. Dedicate the CLOCK/DATA pins for the ICSP interface (not connected to other circuitry).
2. Use these pins as outputs with minimal loading on the signal line.
3. Use isolation circuitry so these signals can be driven to the ICSP specifications.

Figure 30-1: Typical In-Circuit Serial Programming (ICSP) Application Circuit
30.3.3 Capacitance on Each of the VDD, MCLR/VPP, CLOCK and DATA Pins

The total capacitance on the programming pins affects the rise rates of these signals as they are driven out of the programmer. Typical circuits use several hundred microfarads of capacitance on VDD, which helps to dampen noise and ripple. However, this capacitance requires a fairly strong driver in the programmer to meet the rise rate timings for VDD. Most programmers are designed to simply program the MCU itself and don’t have strong enough drivers to power the application circuit. One solution is to use a driver board between the programmer and the application circuit. The driver board requires a separate power supply that is capable of driving the VPP and VDD pins with the correct rise rates and should also provide enough current to power the application circuit. CLOCK and DATA are not buffered on this schematic, but may require buffering depending upon the application. A sample driver board schematic is shown in Figure 30-2.

| Note: | The driver board design MUST be tested in the user’s application to determine the effects of the application circuit on the programming signals timing. Changes may be required if the application places a significant load on the VDD, Vpp, CLOCK or DATA pins. |

30.3.4 Minimum and Maximum Operating Voltage for VDD

The Microchip programming specification states that the device should be programmed at 5V. Special considerations must be made if your application circuit operates at 3V only. These considerations may include totally isolating the MCU during programming. Another consideration is the device must be verified at the minimum and maximum voltages at which the application circuit will be operating. For instance, a battery operated system may operate from three 1.5V cells giving an operating voltage range of 2.7V to 4.5V. The programmer must program the device at 5V and must verify the program memory contents at both 2.7V and 4.5V to ensure that proper programming margins have been achieved. This ensures the PICmicro MCU operation over the voltage range of the system.

30.3.5 PICmicro Oscillator

The final consideration deals with the oscillator circuit on the application board. The voltage on MCLR/VPP must rise to the specified program mode entry voltage before the device executes any code. The crystal modes available on the device are not affected by this, because the Oscillator Start-up Timer waits for 1024 oscillations before any code is executed. However, RC or EC oscillators do not require any start-up time; therefore, the Oscillator Start-up Timer is not used. The programmer must drive MCLR/VPP to the program mode entry voltage before the RC or EC oscillator toggles four times. If the RC or EC oscillator toggles four or more times, the program counter will be incremented to some value X. When the device enters programming mode, the program counter will not be zero and the programmer will start programming your code at an offset of X. There are several alternatives that can compensate for a slow rise rate on MCLR/VPP.

The first method is to not populate the resistor (R1) in Figure 30-1, program the device, and then insert the resistor (R1). The other method is to have the programming interface drive the OSC1 pin of the PICmicro MCU to ground while programming. This will prevent any oscillations from occurring during programming.

Connecting the application circuit to the programmer is dependent on the programming environment. Refer to **Section 30.5 "Programming Environment"** for more details.
30.4 Programmer

PIC18CXXX MCUs only use serial programming and, therefore, all programmers supporting these devices will support ICSP. One area of consideration with the programmer is the drive capability. As discussed before, it must be able to provide the specified rise rates on the ICSP signals and also provide enough current to power the application circuit. Figure 30-2 shows an example driver board. This driver schematic does not show any buffer circuitry for CLOCK and DATA. It is recommended that an evaluation be performed to determine if buffering is required.

Another consideration with the programmer is what VDD levels are used to verify the memory contents of the device. For instance, the PRO MATE II verifies program memory at the minimum and maximum VDD levels for the specified device and is, therefore, considered a production quality programmer. On the other hand, the PICSTART® Plus only verifies at 5V and is for prototyping use only. The Microchip programming specifications state that the program memory contents should be verified at both the minimum and maximum VDD levels that the application circuit will be operating. This implies that the application circuit must be able to handle the varying VDD voltages.

There are also several third party programmers that are available. You should select a programmer based on the features it has and how it fits into your programming environment. The Microchip Development Systems Ordering Guide (DS30177) provides detailed information on all our development tools. The Microchip Third Party Guide (DS00104) provides information on all of our third party tool developers. Please consult these two references when selecting a programmer. Many options exist, including serial or parallel PC host connection, stand-alone operation, and single or gang programmers. Some of the third party developers include Advanced Transdata Corporation, BP Microsystems, Data I/O, Emulation Technology, and Logical Devices.

30.5 Programming Environment

The programming environment affects the type of programmer used, the programmer cable length and the application circuit interface. Some programmers are well suited for a manual assembly line, while others are desirable for an automated assembly line. You may want to choose a gang programmer to program multiple systems at a time.

The physical distance between the programmer and the application circuit affects the load capacitance on each of the programming signals. This directly affects the drive strength needed to provide the correct signal rise rates and current. This programming cable must also be as short as possible and properly terminated and shielded, or the programming signals may be corrupted by ringing or noise.

Finally, the application circuit interface to the programmer depends on the size constraints of the application circuit itself and the assembly line. A simple header can be used to interface the application circuit to the programmer. This might be more desirable for a manual assembly line where a technician plugs the programmer cable into the board. A different method uses spring loaded test pins (commonly referred to as pogo pins). The application circuit has pads on the board for each of the programming signals, and there is a fixture that has pogo pins in the corresponding configuration. The application circuit or fixture is moved into position, such that the pogo pins come into contact with the board. This method might be more suitable for an automated assembly line.

After taking into consideration the various points with the application circuit, the programmer and the programming environment, anyone can build a high quality, reliable manufacturing line based on ICSP.
30.6 Other Benefits

ICSP provides other benefits, such as calibration and serialization.

If program memory permits, it would be cheaper and more reliable to store calibration constants in program memory instead of using an external serial EEPROM. For example, if your system has a thermistor that can vary from one system to another, storing some calibration information in a table format allows the microcontroller to compensate (in software) for external component tolerances. System cost can be reduced without affecting the required performance of the system by using software calibration techniques. But how does this relate to ICSP? The PICmicro MCU has already been programmed with firmware that performs a calibration cycle. The calibration data is transferred to a calibration fixture. When all calibration data has been transferred, the fixture places the PICmicro MCU in programming mode and programs the PICmicro MCU with the calibration data. Application note AN656, “In-Circuit Serial Programming of Calibration Parameters Using a PICmicro Microcontroller,” shows exactly how to implement this type of calibration data programming.

The other benefit of ICSP is serialization. Each individual system can be programmed with a unique or random serial number. One such application of a unique serial number would be for security systems. A typical system might use DIP switches to set the serial number. Instead, this number can be burned into program memory, thus reducing the overall system cost and lowering the risk of tampering.
30.7 Field Programming of PICmicro OTP MCUs

An OTP device is not normally capable of being reprogrammed, but the PICmicro architecture gives you this flexibility, provided the size of your firmware is at least half that of the desired device, and the device is not code protected. If your target device does not have enough program memory, Microchip provides a wide spectrum of devices from 0.5K to 16K word program memory with the same set of peripheral features that will help meet the criteria.

The Enhanced MCU devices have three vectors; RESET and two interrupt vector addresses. When the PICmicro device encounters a RESET or interrupt condition, the code located at one of these locations in program memory is executed.

For an example of reprogramming an OTP device, we will use an example from our Mid-Range family. This technology is applicable to all EPROM based PICmicro devices. The first listing of Example 30-1 shows the code that is first programmed into the PICmicro device. The second listing of Example 30-1 shows the code that is programmed into the PICmicro device for the second time.

Example 30-1 shows that to program the device a second time, the memory location 0x0000 (originally goto Main (0x2808)), is reprogrammed to all 0's. This happens to be a NOP instruction. This location cannot be reprogrammed to the new opcode (0x2860), because the bits that are 0's cannot be reprogrammed to 1's. Only bits that are 1's can be reprogrammed to 0's. The next memory location, 0x0001, was originally blank (all 1's) and now becomes a goto Main (0x2860). When a RESET condition occurs, the MCU executes the instruction at location 0x0000, which is the NOP (a completely benign instruction), and then executes the goto Main to start the execution of code. The example also shows that all program memory locations after 0x005A are blank in the original program, so that the second time the PICmicro device is programmed, the revised code can be programmed at these locations. The same descriptions can be given for the interrupt vector locations.

Now your one-time programmable Enhanced MCU is exhibiting EEPROM- or FLASH-like qualities.
### Section 30. ICSP

#### Example 30-1: Programming Cycle Listing Files

<table>
<thead>
<tr>
<th>First Program Cycle</th>
<th>Second Program Cycle</th>
</tr>
</thead>
<tbody>
<tr>
<td>Prog Opcode</td>
<td>Assembly</td>
</tr>
<tr>
<td>0000 2808</td>
<td>goto Main ;Main loop</td>
</tr>
<tr>
<td>0001 3FFF</td>
<td>&lt;blank&gt; ; at 0x0008</td>
</tr>
<tr>
<td>0002 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>0004 28AE</td>
<td>goto ISR ; ISR at</td>
</tr>
<tr>
<td>0005 3FFF</td>
<td>&lt;blank&gt; ; 0x0048</td>
</tr>
<tr>
<td>0006 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>0008 1683</td>
<td>btfss PIR1,RBIF</td>
</tr>
<tr>
<td>0009 3007</td>
<td>movlw 0x07</td>
</tr>
<tr>
<td>000A 009F</td>
<td>movwf ADCON1</td>
</tr>
<tr>
<td>0048 1C0C</td>
<td>btfss PIR1,RBIF</td>
</tr>
<tr>
<td>0049 28AE</td>
<td>goto EndISR</td>
</tr>
<tr>
<td>004A 1806</td>
<td>btfsc PORTB,0</td>
</tr>
<tr>
<td>0060 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>0061 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>0062 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>00A8 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>00A9 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>00AA 3FFF</td>
<td>&lt;blank&gt;</td>
</tr>
<tr>
<td>.</td>
<td>.</td>
</tr>
</tbody>
</table>
30.8 Field Programming of FLASH PICmicro MCUs

With the ICSP interface circuitry already in place, FLASH-based PICmicro MCUs can be easily reprogrammed in the field. These FLASH devices allow you to reprogram them even if they are code protected. A portable ICSP programming station might consist of a laptop computer and programmer. The technician plugs the ICSP interface cable into the application circuit and downloads the new firmware into the device. The next thing you know, the system is up and running without those annoying “bugs.” Another instance would be that you want to add an additional feature to your system. All of your current inventory can be converted to the new firmware, and field upgrades can be performed to bring your installed base of systems up to the latest revision of firmware.
Figure 30-2: Example Driver Board Schematic

Note: All resistors are in Ohms, all capacitors are in µF.
30.9 Design Tips

Question 1: When I try to do ICSP, the entire program is shifted (offset) in the device program memory.

Answer 1:
If the MCLR pin does not rise fast enough, while the device's voltage is in the valid operating range, the internal Program Counter (PC) can increment. This means that the PC is no longer pointing to the address that you expected. The exact location depends on the number of device clocks that occurred in the valid operating region of the device.

Question 2: I am using a PRO MATE II with a socket that I designed to bring the programming signal to my application board. Sometimes when I try to do ICSP, the program memory is programmed wrong.

Answer 2:
The voltages / timings may be violated at the device. This could be due to the:
• Application board circuitry
• Cable length from programmer to target
• Large capacitance on VDD that affects levels / timings
Section 30. ICSP

30.10 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is, they may be written for the Base-Line, the Mid-Range or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to In-Circuit Serial Programming are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>In-Circuit Serial Programming of Calibration Parameters using a PICmicro® Microcontroller</td>
<td>AN656</td>
</tr>
<tr>
<td>In-Circuit Serial Programming Guide</td>
<td>DS30277</td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faq/codeex/
30.11 Revision History

Revision A
This is the initial released revision of the Enhanced MCU In-Circuit Serial Programming module description.
Section 31. Instruction Set

HIGHLIGHTS

This section of the manual contains the following major topics:

31.1 Introduction ............................................................................................................... 31-2
31.2 Data Memory Map ..................................................................................................... 31-3
31.3 Instruction Formats .................................................................................................. 31-9
31.4 Special Function Registers as Source/Destination ................................................ 31-12
31.5 Fast Register Stack ................................................................................................... 31-13
31.6 Q Cycle Activity ....................................................................................................... 31-13
31.7 Instruction Descriptions .......................................................................................... 31-14
31.8 Design Tips .............................................................................................................. 31-136
31.9 Related Application Notes ...................................................................................... 31-137
31.10 Revision History ..................................................................................................... 31-138
31.1 Introduction

The PIC18CXXX instruction set adds many enhancements to the previous PICmicro instruction sets, while maintaining an easy migration from these PICmicro instruction sets. Most instructions are a single program memory word (16-bits), but to address customer requests, some new instructions have been added that require two program memory locations. The Instruction Set Summary, shown in Table 31-1, lists the instructions recognized by the Microchip assembler (MPASM). Table 31-2 gives the instruction description conventions.

Each instruction is divided into an OPCODE that specifies the instruction type and one or more operands which further specify the operation of the instruction.

The instruction set is highly orthogonal and is grouped into four basic categories:

- **Byte-oriented** operations
- **Bit-oriented** operations
- **Literal** operations
- **Control** operations

Most byte-oriented instructions have three operands:

1. The File Register (specified by the value of 'f')
2. The destination of the result (specified by the value of 'd')
3. The accessed memory (specified by the value of 'a')

'f' represents a File Register Designator and 'd' represents a Destination Designator. The File Register Designator specifies which File Register is to be used by the instruction. The access indicator 'a' specifies if the BSR selects the bank or if the access bank is used.

The destination designator specifies where the result of the operation is to be placed. If 'd' is zero, the result is placed in the WREG Register. If 'd' is one, the result is placed in the File Register specified in the instruction.

All bit-oriented instructions have three operands:

1. The File Register (specified by the value of 'f')
2. The bit in the File Register (specified by the value of 'b')
3. The accessed memory (specified by the value of 'a')

'b' represents a bit field designator that selects the number of the bit affected by the operation, while 'f' represents the number of the file in which the bit is located. The access indicator 'a' specifies if the BSR selects the bank or if the access bank is used.

The literal instructions may use some of the following operands:

- A literal value to be loaded into a File Register (specified by the value of 'k')
- The desired FSR Register to load the literal value into (specified by the value of 'f')
- No operand required (specified by the value of '—')

The control instructions may use some of the following operands:

- A program memory address (specified by the value of 'n')
- The mode of the CALL or RETURN instructions (specified by the value of 's')
- The mode of the Table Read and Table Write instructions (specified by the value of 'm')
- No operand required (specified by the value of '—')

All instructions are a single word except for three double word instructions. These three instructions were made double word instructions so that all the required information is available in these 32 bits. In the second word, the 4-MSb's, are '1's. If this second word is executed as an instruction (by itself), it will execute as a NOP.

All single word instructions are executed in a single instruction cycle, unless a conditional test is true or the program counter is changed as a result of the instruction. In these cases, the execution takes two instruction cycles with the additional instruction cycle(s) executed as a NOP.

The double word instructions (that do not modify the PC) execute in two instruction cycles.

One instruction cycle consists of four oscillator periods. Thus, for an oscillator frequency of 4 MHz, the normal instruction execution time is 1 µs. If a conditional test is true or the program counter is changed as a result of an instruction, the instruction execution time is 2 µs. Two word branch instructions (if true) would take 3 µs.
Section 31. Instruction Set

31.2 Data Memory Map

The Data Memory Map has 16 banks of 256 bytes. The Instruction Set and architecture allows operations across all banks (such as MOVFF). A Segment of Bank 0 and a segment of Bank 15 comprise the access bank. See Section 31.2.1 for description of the access bank.

Figure 31-1: The Data Memory Map and the Access Bank

<table>
<thead>
<tr>
<th>BSR&lt;3:0&gt;</th>
<th>Data Memory Map</th>
</tr>
</thead>
<tbody>
<tr>
<td>= 0000b</td>
<td>Segment 0</td>
</tr>
<tr>
<td></td>
<td>Bank 0</td>
</tr>
<tr>
<td>= 0001b</td>
<td>Bank 1</td>
</tr>
<tr>
<td></td>
<td>Bank n</td>
</tr>
<tr>
<td>= 1110b</td>
<td>Bank 14</td>
</tr>
<tr>
<td>= 1111b</td>
<td>Bank 15 Segment 1</td>
</tr>
</tbody>
</table>

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

31.2.1 Access Bank

The access bank is an architectural enhancement that is very useful for C compiler code optimization. The techniques used by the C compiler may also be useful for programs written in assembly.

This data memory region can be used for

- Intermediate computational values
- Local variables of subroutine
- Faster context saving/switching of variables
- Common variables
- Faster evaluation/control of SFRs (no banking)

The access bank is comprised of 2 segments: Segment 0 and Segment 1. Segment 0 is the RAM that is mapped in Bank 0. Segment 1 is the SFRs that are mapped in Bank 15. Each Segment can be of different sizes. The sum of RAM mapped by Segment 0 and Segment 1 is 256 bytes.

When forced in the access bank (a = '0'), the last address in Segment 0 is followed by the first address in Segment 1. Segment 1 maps the Special Function Registers so that these registers can be accessed without any software overhead. This is useful for testing status flags and modifying control bits.
Example 31-1 shows how registers are affected depending on the value of the access bit. Register MYREG has an 8-bit address. This address specifies the location in the specified bank to perform the operation. The specified bank is either the bank specified by the Bank Select Register (BSR) (when \( a = 1 \)), or the access bank (when \( a = 0 \)).

Example 31-1: Operation of Destination and Access Bits

<table>
<thead>
<tr>
<th>For destination bit:</th>
<th>For access bit:</th>
</tr>
</thead>
<tbody>
<tr>
<td>( F = 1 ); Result is placed in File Register</td>
<td>( B = 1 ); Register used specified by BSR Bank Register</td>
</tr>
<tr>
<td>( W = 0 ); Result is placed in WREG Register</td>
<td>( A = 0 ); Register used is in Access Bank</td>
</tr>
</tbody>
</table>

MYREG is a register with an 8-bit address value between 0h and FFh. For this example we will assign MYREG to bank 5, though it could be in any (or all) banks.

\[
\begin{align*}
\text{MOVLB} & \quad 5 \\
\text{DECF} & \quad \text{MYREG, F, B} \\
\text{DECF} & \quad \text{MYREG, F, A} \\
\text{DECF} & \quad \text{MYREG, W, B} \\
\text{DECF} & \quad \text{MYREG, W, A}
\end{align*}
\]

Note: If the register is specified with the full 12-bit address, the assembler will automatically force the access bit to a '0' (when the address is in the access RAM area) or a '1' (for all other addresses).
Section 31. Instruction Set

The common assembler usage should be that all RAM and SFR addresses are 12-bit. This means that the assembler can determine from the address of the register, whether the access bit needs to be set or cleared. Example 31-2 shows this, as well as forcing the use of the access bank.

Example 31-2: Code

```assembly
; The following symbols are defined in the Microchip supplied device header file:

; For destination bit:
; F = 1 ; Result is placed in File Register
; W = 0 ; Result is placed in WREG Register

; For access bit:
; B = 1 ; Register used specified by BSR Bank Register
; A = 0 ; Register used is in Access Bank

Register Name    Address
Loop_CNTR        0x000
MYREG            0x524
SFR1             0xA9F

MOVLB 5           ; BSR points to RAM bank 5
                  ; Addr 24
                  ; a-bit MYREG Loop_CNTR SFR1 WREG Bank0
                  ; Starting Value   0x7F 0x7F 0x7F x 0xA9
                  ;
                  ; DECF Loop_CNTR, F ; 0 --- 0x7F --- --- --- ---
                  ; DECF MYREG, F     ; 1 0x7E --- --- --- ---
                  ; DECF SFR1, F      ; 0 --- --- 0x7E --- ---
                  ;
                  ; DECF Loop_CNTR, W ; 0 --- --- --- 0x7D ---
                  ; DECF MYREG, W     ; 1 --- --- --- 0x7D ---
                  ; DECF SFR1, W      ; 0 --- --- --- 0x7D ---
                  ;
                  ; INCF MYREG, F, A  ; 0 --- --- --- --- 0xAA
                  ; INCF MYREG, W, A  ; 0 --- --- --- 0x7F ---
```
### Table 31-1: PIC18CXXX Instruction Set Summary

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>MSb</td>
<td>LSb</td>
<td></td>
</tr>
<tr>
<td><strong>BYTE-ORIENTED FILE REGISTER OPERATIONS</strong></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ADDWF f, d, a</td>
<td>Add WREG and f</td>
<td>1</td>
<td>0010 01da</td>
<td>00da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>ADDWF f, d, a</td>
<td>Add WREG and Carry bit to f</td>
<td>1</td>
<td>0010 00da</td>
<td>00da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>ANDWF f, d, a</td>
<td>AND WREG with f</td>
<td>1</td>
<td>0001 01da</td>
<td>00da</td>
<td>Z, N</td>
</tr>
<tr>
<td>CLR f, a</td>
<td>Clear f</td>
<td>1</td>
<td>0110 101a</td>
<td>00da</td>
<td>Z</td>
</tr>
<tr>
<td>COMF f, d, a</td>
<td>Complement f</td>
<td>1</td>
<td>0001 11da</td>
<td>00da</td>
<td>Z, N</td>
</tr>
<tr>
<td>CPFS EQ f, a</td>
<td>Compare f with WREG, skip f</td>
<td>1 (2 or 3)</td>
<td>0110 001a</td>
<td>001a</td>
<td>None</td>
</tr>
<tr>
<td>CPFS GT f, a</td>
<td>Compare f with WREG, skip &gt;</td>
<td>1 (2 or 3)</td>
<td>0110 010a</td>
<td>010a</td>
<td>None</td>
</tr>
<tr>
<td>CPFS LT f, a</td>
<td>Compare f with WREG, skip &lt;</td>
<td>1 (2 or 3)</td>
<td>0110 000a</td>
<td>000a</td>
<td>None</td>
</tr>
<tr>
<td>DECF f, d, a</td>
<td>Decrement f</td>
<td>1</td>
<td>0000 01da</td>
<td>01da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>DECFNZ f, d, a</td>
<td>Decrement f, Skip if 0</td>
<td>1 (2 or 3)</td>
<td>0100 11da</td>
<td>11da</td>
<td>None</td>
</tr>
<tr>
<td>INC f, d, a</td>
<td>Increment f</td>
<td>1</td>
<td>0010 10da</td>
<td>10da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>INCFNZ f, d, a</td>
<td>Increment f, Skip if 0</td>
<td>1 (2 or 3)</td>
<td>0011 11da</td>
<td>11da</td>
<td>None</td>
</tr>
<tr>
<td>INFSNZ f, d, a</td>
<td>Increment f, Skip if Not f</td>
<td>1 (2 or 3)</td>
<td>0100 10da</td>
<td>10da</td>
<td>None</td>
</tr>
<tr>
<td>IORWF f, d, a</td>
<td>Inclusive OR WREG with f</td>
<td>1</td>
<td>0001 00da</td>
<td>00da</td>
<td>Z, N</td>
</tr>
<tr>
<td>MOV f, d, a</td>
<td>Move f</td>
<td>1</td>
<td>0101 00da</td>
<td>00da</td>
<td>Z, N</td>
</tr>
<tr>
<td>MOVFF f, d, a</td>
<td>Move f (source) to 1st word f (destination) 2nd word</td>
<td>2</td>
<td>1100 11da</td>
<td>11da</td>
<td>None</td>
</tr>
<tr>
<td>MOVWF f, a</td>
<td>Move WREG to f</td>
<td>1</td>
<td>0110 111a</td>
<td>111a</td>
<td>None</td>
</tr>
<tr>
<td>MULWF f, a</td>
<td>Multiply WREG with f</td>
<td>1</td>
<td>0000 011a</td>
<td>011a</td>
<td>None</td>
</tr>
<tr>
<td>NEG f, a</td>
<td>Negate f</td>
<td>1</td>
<td>0110 110a</td>
<td>110a</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>RLCF f, d, a</td>
<td>Rotate Left f through Carry</td>
<td>1</td>
<td>0011 01da</td>
<td>01da</td>
<td>C, Z, N</td>
</tr>
<tr>
<td>RLNCF f, d, a</td>
<td>Rotate Left f (No Carry)</td>
<td>1</td>
<td>0100 01da</td>
<td>01da</td>
<td>Z, N</td>
</tr>
<tr>
<td>RRCF f, d, a</td>
<td>Rotate Right f through Carry</td>
<td>1</td>
<td>0011 00da</td>
<td>00da</td>
<td>C, Z, N</td>
</tr>
<tr>
<td>RRNCF f, d, a</td>
<td>Rotate Right f (No Carry)</td>
<td>1</td>
<td>0100 00da</td>
<td>00da</td>
<td>Z, N</td>
</tr>
<tr>
<td>SETF f, a</td>
<td>Set f</td>
<td>1</td>
<td>0110 100a</td>
<td>100a</td>
<td>None</td>
</tr>
<tr>
<td>SUBFWB f, d, a</td>
<td>Subtract f from WREG with borrow</td>
<td>1</td>
<td>0101 01da</td>
<td>01da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>SUBWF f, d, a</td>
<td>Subtract WREG from f</td>
<td>1</td>
<td>0101 11da</td>
<td>11da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>SUBWF f, d, a</td>
<td>Subtract WREG from f with borrow</td>
<td>1</td>
<td>0101 10da</td>
<td>10da</td>
<td>C, DC, Z, OV, N</td>
</tr>
<tr>
<td>SWAPF f, d, a</td>
<td>Swap nibbles in f</td>
<td>1</td>
<td>0011 10da</td>
<td>10da</td>
<td>None</td>
</tr>
<tr>
<td>TS TFS Z f, a</td>
<td>Test f, skip if 0</td>
<td>1 (2 or 3)</td>
<td>0110 011a</td>
<td>011a</td>
<td>None</td>
</tr>
<tr>
<td>XORWF f, d, a</td>
<td>Exclusive OR WREG with f</td>
<td>1</td>
<td>0001 10da</td>
<td>10da</td>
<td>Z, N</td>
</tr>
<tr>
<td><strong>BIT-ORIENTED FILE REGISTER OPERATIONS</strong></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>BCF f, b, a</td>
<td>Bit Clear f</td>
<td>1</td>
<td>1001 bbbf</td>
<td>111f</td>
<td>None</td>
</tr>
<tr>
<td>BSR f, b, a</td>
<td>Bit Set f</td>
<td>1</td>
<td>1000 bbbf</td>
<td>111f</td>
<td>None</td>
</tr>
<tr>
<td>BTFS C f, b, a</td>
<td>Bit Test f, Skip if Clear</td>
<td>1 (2 or 3)</td>
<td>1011 bbbf</td>
<td>111f</td>
<td>None</td>
</tr>
<tr>
<td>BTFSS f, b, a</td>
<td>Bit Test f, Skip if Set</td>
<td>1 (2 or 3)</td>
<td>1101 11bb</td>
<td>11bb</td>
<td>None</td>
</tr>
<tr>
<td>BTG f, d, a</td>
<td>Bit Toggle f</td>
<td>1</td>
<td>0111 bbbf</td>
<td>111f</td>
<td>None</td>
</tr>
</tbody>
</table>

**Note 1:** When a PORT Register is modified as a function of itself (e.g., MOVF PORTB, 1, 0), the value used will be that value present on the pins themselves. For example, if the data latch is '1' for a pin configured as input and is driven low by an external device, the data will be written back with a '0'.

**Note 2:** If this instruction is executed on the TMR0 Register (and, where applicable, d = 1), the prescaler will be cleared if assigned.

**Note 3:** If Program Counter (PC) is modified or a conditional test is true, the instruction requires two cycles. The second cycle is executed as a NOP.

**Note 4:** Some instructions are 2 word instructions. The second word of these instructions will be executed as a NOP, unless the first word of the instruction retrieves the information embedded in these 16-bits. This ensures that all program memory locations have a valid instruction.

**Note 5:** If the table write starts the write cycle to internal program memory, the write continues until terminated.
## Section 31. Instruction Set

### Table 31-1: PIC18CXXX Instruction Set Summary (Continued)

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>CONTROL OPERATIONS</strong></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>BC n</td>
<td>Branch if Carry</td>
<td>1 (2)</td>
<td>1110 0010 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BN n</td>
<td>Branch if Negative</td>
<td>1 (2)</td>
<td>1110 0110 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BNC n</td>
<td>Branch if Not Carry</td>
<td>1 (2)</td>
<td>1110 0011 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BNN n</td>
<td>Branch if Not Negative</td>
<td>1 (2)</td>
<td>1110 0111 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BNOV n</td>
<td>Branch if Not Overflow</td>
<td>1 (2)</td>
<td>1110 0101 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BNZ n</td>
<td>Branch if Not Zero</td>
<td>1 (2)</td>
<td>1110 0001 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BOV n</td>
<td>Branch if Overflow</td>
<td>1 (2)</td>
<td>1110 0100 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BRA n</td>
<td>Branch Unconditionally</td>
<td>2</td>
<td>1101 00nn nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>BZ n</td>
<td>Branch if Zero</td>
<td>1 (2)</td>
<td>1110 0000 nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>CALL n, s</td>
<td>Call subroutine 1st word</td>
<td>2</td>
<td>1110 110s kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td></td>
<td>2nd word</td>
<td></td>
<td>1111 kkkk kkkk kkkk</td>
<td></td>
<td></td>
</tr>
<tr>
<td>CLRWD</td>
<td>—</td>
<td>Clear Watchdog Timer</td>
<td>1</td>
<td>0000 0000 0000 0100</td>
<td>TO, PD</td>
</tr>
<tr>
<td>DAW —</td>
<td>Decimal Adjust WREG</td>
<td>1</td>
<td>0000 0000 0000 0111</td>
<td>C</td>
<td></td>
</tr>
<tr>
<td>GOTO n</td>
<td>Go to address 1st word</td>
<td>2</td>
<td>1110 1111 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td></td>
<td>2nd word</td>
<td></td>
<td>1111 kkkk kkkk kkkk</td>
<td></td>
<td></td>
</tr>
<tr>
<td>NOP —</td>
<td>No Operation</td>
<td>1</td>
<td>0000 0000 0000 0000</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>NOP —</td>
<td>No Operation</td>
<td>1</td>
<td>1111 xxxx xxxx xxxx</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>POP —</td>
<td>Pop top of return stack (TOS)</td>
<td>1</td>
<td>0000 0000 0000 0110</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>PUSH —</td>
<td>Push top of return stack (TOS)</td>
<td>1</td>
<td>0000 0000 0000 0101</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RCALL n</td>
<td>Relative Call</td>
<td>2</td>
<td>1101 1nnn nnnn nnnn</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RESET</td>
<td>Software device RESET</td>
<td>1</td>
<td>0000 0000 1111 1111</td>
<td>All</td>
<td></td>
</tr>
<tr>
<td>RETFIE s</td>
<td>Return from interrupt enable</td>
<td>2</td>
<td>0000 0000 0000 000a</td>
<td>GIE/GIEH, PEIE/GIEL</td>
<td></td>
</tr>
<tr>
<td>RETLW k</td>
<td>Return with literal in WREG</td>
<td>2</td>
<td>0000 1100 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RETURN s</td>
<td>Return from Subroutine</td>
<td>2</td>
<td>0000 0000 0000 001s</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>SLEEP —</td>
<td>Go into standby mode</td>
<td>1</td>
<td>0000 0000 0000 0011</td>
<td>TO, PD</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** When a PORT Register is modified as a function of itself (e.g., MOVF PORTB, 1, 0), the value used will be that value present on the pins themselves. For example, if the data latch is ‘1’ for a pin configured as input and is driven low by an external device, the data will be written back with a ‘0’.

**2:** If this instruction is executed on the TMR0 Register (and, where applicable, d = 1), the prescaler will be cleared if assigned.

**3:** If Program Counter (PC) is modified or a conditional test is true, the instruction requires two cycles. The second cycle is executed as a NOP.

**4:** Some instructions are 2 word instructions. The second word of these instructions will be executed as a NOP, unless the first word of the instruction retrieves the information embedded in these 16-bits. This ensures that all program memory locations have a valid instruction.

**5:** If the table write starts the write cycle to internal program memory, the write continues until terminated.
### Table 31-1: PIC18CXXX Instruction Set Summary (Continued)

<table>
<thead>
<tr>
<th>Mnemonic, Operands</th>
<th>Description</th>
<th>Cycles</th>
<th>16-Bit Instruction Word</th>
<th>Status Affected</th>
<th>Notes</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>LITERAL OPERATIONS</strong></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ADDLW k</td>
<td>Add literal and WREG</td>
<td>1</td>
<td>0000 1111 kkkk kkkk</td>
<td>C, DC, Z, OV, N</td>
<td></td>
</tr>
<tr>
<td>ANDLW k</td>
<td>AND literal with WREG</td>
<td>1</td>
<td>0000 1011 kkkk kkkk</td>
<td>Z, N</td>
<td></td>
</tr>
<tr>
<td>IORLW f, k</td>
<td>Inclusive OR literal with WREG</td>
<td>1</td>
<td>0000 1001 kkkk kkkk</td>
<td>Z, N</td>
<td></td>
</tr>
<tr>
<td>LFSR f, k</td>
<td>Move literal (12-bit) 1st word to FSRx 2nd word</td>
<td>2</td>
<td>1110 1110 00ff kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>MOVLB k</td>
<td>Move literal to BSR&lt;3:0&gt;</td>
<td>1</td>
<td>0000 0001 0000 kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>MOVLW k</td>
<td>Move literal to WREG</td>
<td>1</td>
<td>0000 1110 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>MULLW k</td>
<td>Multiply literal with WREG</td>
<td>1</td>
<td>0000 1101 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>RETLW k</td>
<td>Return with literal in WREG</td>
<td>2</td>
<td>0000 1100 kkkk kkkk</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>SUBLW k</td>
<td>Subtract WREG from literal</td>
<td>1</td>
<td>0000 1000 kkkk kkkk</td>
<td>C, DC, Z, OV, N</td>
<td></td>
</tr>
<tr>
<td>XORLW k</td>
<td>Exclusive OR literal with WREG</td>
<td>1</td>
<td>0000 1010 kkkk kkkk</td>
<td>Z, N</td>
<td></td>
</tr>
<tr>
<td><strong>DATA MEMORY ↔ PROGRAM MEMORY OPERATIONS</strong></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TBLRD+</td>
<td>Table Read</td>
<td>2</td>
<td>0000 0000 0000 1000</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLRD++</td>
<td>Table Read with post-increment</td>
<td>2</td>
<td>0000 0000 0000 1001</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLRD+-</td>
<td>Table Read with post-decrement</td>
<td>2</td>
<td>0000 0000 0000 1010</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLRD++</td>
<td>Table Read with pre-increment</td>
<td>2</td>
<td>0000 0000 0000 1011</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLWT</td>
<td>Table Write</td>
<td>2 (5)</td>
<td>0000 0000 0000 1100</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLWT++</td>
<td>Table Write with post-increment</td>
<td>2 (5)</td>
<td>0000 0000 0000 1101</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLWT+-</td>
<td>Table Write with post-decrement</td>
<td>2 (5)</td>
<td>0000 0000 0000 1110</td>
<td>None</td>
<td></td>
</tr>
<tr>
<td>TBLWT++</td>
<td>Table Write with pre-increment</td>
<td>2 (5)</td>
<td>0000 0000 0000 1111</td>
<td>None</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** When a PORT Register is modified as a function of itself (e.g., MOVF PORTB, 1, 0), the value used will be that value present on the pins themselves. For example, if the data latch is '1' for a pin configured as input and is driven low by an external device, the data will be written back with a '0'.

2: If this instruction is executed on the TMR0 Register (and, where applicable, d = 1), the prescaler will be cleared if assigned.

3: If Program Counter (PC) is modified or a conditional test is true, the instruction requires two cycles. The second cycle is executed as a NOP.

4: Some instructions are 2 word instructions. The second word of these instructions will be executed as a NOP, unless the first word of the instruction retrieves the information embedded in these 16-bits. This ensures that all program memory locations have a valid instruction.

5: If the table write starts the write cycle to internal program memory, the write continues until terminated.
### Section 31. Instruction Set

#### 31.3 Instruction Formats

Figure 31-2 shows the three general formats that the instructions can have. As can be seen from the general format of the instructions, the opcode portion of the instruction word varies from 3-bits to 6-bits of information. This is what allows the Enhanced Instruction Set to have 75 instructions.

**Note:** Any unused opcode is Reserved. Use of any reserved opcode may cause unexpected operation.

All instruction examples use the following format to represent a hexadecimal number:

```
0xhh
```

where **h** signifies a hexadecimal digit.

To represent a binary number:

```
00000100b
```

where **b** is a binary string identifier.

**Figure 31-2: General Format for Instructions**

<table>
<thead>
<tr>
<th>Byte-oriented File Register operations</th>
<th>Example Instruction</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>OPCODE</td>
<td>d</td>
</tr>
<tr>
<td>d = 0 for result destination to be WREG Register</td>
<td></td>
</tr>
<tr>
<td>d = 1 for result destination to be File Register (f)</td>
<td></td>
</tr>
<tr>
<td>a = 0 to force Access Bank</td>
<td></td>
</tr>
<tr>
<td>a = 1 for BSR to select bank</td>
<td></td>
</tr>
<tr>
<td>f = 8-bit File Register address</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>Byte to Byte move operations (2-word)</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>OPCODE</td>
<td>f (Source FILE #)</td>
</tr>
<tr>
<td>15</td>
<td>12</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>1111</td>
<td>f (Destination FILE #)</td>
</tr>
<tr>
<td>15</td>
<td>12</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>Bit-oriented File Register operations</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>OPCODE</td>
<td>b (BIT #)</td>
</tr>
<tr>
<td>b = 3-bit position of bit in File Register (f)</td>
<td></td>
</tr>
<tr>
<td>a = 0 to force Access Bank</td>
<td></td>
</tr>
<tr>
<td>a = 1 for BSR to select bank</td>
<td></td>
</tr>
<tr>
<td>f = 8-bit File Register address</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>Literal operations</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>OPCODE</td>
<td>k (literal)</td>
</tr>
<tr>
<td>15</td>
<td>8</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>Control operations</td>
<td></td>
</tr>
<tr>
<td>CALL, GOTO, and Branch operations</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>OPCODE</td>
<td>n&lt;7:0&gt; (literal)</td>
</tr>
<tr>
<td>15</td>
<td>8</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>1111</td>
<td>n&lt;19:8&gt; (literal)</td>
</tr>
<tr>
<td>15</td>
<td>12</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
<tr>
<td>n = 20-bit immediate value</td>
<td></td>
</tr>
</tbody>
</table>
Table 31-2: Instruction Description Conventions

<table>
<thead>
<tr>
<th>Field</th>
<th>Description</th>
</tr>
</thead>
</table>
| a     | RAM access bit  
       | a = 0: RAM location in access bank (BSR Register is ignored)  
       | a = 1: RAM bank is specified by BSR Register |
| bbb   | Bit address within an 8-bit File Register (0 to 7) |
| BSR   | Bank Select Register. Used to select the current RAM bank. |
| d     | Destination select bit;  
       | d = 0: store result in WREG,  
       | d = 1: store result in File Register f. |
| dest  | Destination either the WREG Register or the specified register file location |
| f     | 8-bit Register file address (0x00 to 0xFF) |
| fs    | 12-bit Register file address (0x000 to 0xFFF). This is the source address. |
| fd    | 12-bit Register file address (0x000 to 0xFFF). This is the destination address. |
| k     | Literal field, constant data or label (may be either an 8-bit, 12-bit or a 20-bit value) |
| label | Label name |
| mm    | The mode of the TBLPTR Register for the Table Read and Table Write instructions  
       | Only used with Table Read and Table Write instructions:  
       | *: No Change to Register (such as TBLPTR with Table reads and writes)  
       | ++: Post-Increment Register (such as TBLPTR with Table reads and writes)  
       | -: Post-Decrement Register (such as TBLPTR with Table reads and writes)  
       | *+: Pre-Increment Register (such as TBLPTR with Table reads and writes) |
| n     | The relative address (2’s complement number) for relative branch instructions, or  
       | the direct address for Call/Branch and Return instructions |
| PRODH | Product of Multiply high byte |
| PRODL | Product of Multiply low byte |
| s     | Fast Call / Return mode select bit.  
       | s = 0: do not update into/from Shadow Registers  
       | s = 1: certain registers loaded into/from Shadow Registers |
| u     | Unused or Unchanged |
| WREG  | Working Register (accumulator) |
| x     | Don’t care (0 or 1)  
       | The assembler will generate code with x = 0. It is the recommended form of use for  
       | compatibility with all Microchip software tools. |
| TBLPTR| 21-bit Table Pointer (points to a Program Memory location) |
| TABLAT| 8-bit Table Latch |
| TOS   | Top of Stack |
| INDF  | Any one of the indirect addressing registers, such as INDF0, INDF1, or INDF2 |
| FSR   | Any one of the file select register pairs, such as FSR0H:FSR0L, FSR1H:FSR1L, or  
       | FSR2H:FSR2L |
Section 31. Instruction Set

Table 31-2: Instruction Description Conventions (Continued)

<table>
<thead>
<tr>
<th>Field</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PC</td>
<td>Program Counter</td>
</tr>
<tr>
<td>PCL</td>
<td>Program Counter Low Byte</td>
</tr>
<tr>
<td>PCH</td>
<td>Program Counter High Byte</td>
</tr>
<tr>
<td>PCLATH</td>
<td>Program Counter High Byte Latch</td>
</tr>
<tr>
<td>PCLATU</td>
<td>Program Counter Upper Byte Latch</td>
</tr>
<tr>
<td>GIE</td>
<td>Global Interrupt Enable bit</td>
</tr>
<tr>
<td>WDT</td>
<td>Watchdog Timer</td>
</tr>
<tr>
<td>TO</td>
<td>Time-out bit</td>
</tr>
<tr>
<td>PD</td>
<td>Power-down bit</td>
</tr>
<tr>
<td>C, DC, Z, OV, N</td>
<td>ALU status bits Carry, Digit Carry, Zero, Overflow, Negative</td>
</tr>
<tr>
<td>[ ]</td>
<td>Optional</td>
</tr>
<tr>
<td>( )</td>
<td>Contents of</td>
</tr>
<tr>
<td>→</td>
<td>Assigned to</td>
</tr>
<tr>
<td>&lt; &gt;</td>
<td>Register bit field</td>
</tr>
<tr>
<td>e</td>
<td>In the set of</td>
</tr>
<tr>
<td>italics</td>
<td>User defined term (font is courier)</td>
</tr>
</tbody>
</table>

Table 31-3: Indirect Addressing Symbols

<table>
<thead>
<tr>
<th>Field</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>*FSRn</td>
<td>Selects INDFn Register</td>
</tr>
<tr>
<td>*FSRn++</td>
<td>Selects POSTINCn Register</td>
</tr>
<tr>
<td>*FSRn--</td>
<td>Selects POSTDECn Register</td>
</tr>
<tr>
<td>*(++FSRn)</td>
<td>Selects PREINCn Register</td>
</tr>
<tr>
<td>*(FSRn+W)</td>
<td>Selects PLUSWn Register</td>
</tr>
</tbody>
</table>
31.4 Special Function Registers as Source/Destination

The Section 31, Instruction Set's Orthogonal Instruction Set allows read and write of all File Registers, including Special Function Registers. The user should be aware of some special situations which are explained in the following subsections.

31.4.1 STATUS Register as Destination

If an instruction writes to the STATUS Register, the Z, C, DC, OV, and N bits may be set or cleared as a result of the instruction and overwrite the original data bits. For example, executing CLRF STATUS will clear Register STATUS, and then set the Z bit leaving 0000 0100b in the register.

31.4.2 Bit Manipulation

All bit manipulation instructions will first read the entire register, operate on the selected bit and then write the result back to (read-modify-write (R-M-W)) the specified register. The user should keep this in mind when operating on some Special Function Registers, such as the Port Pin Register.

Note: Status bits that are manipulated by the device (including the interrupt flag bits) are set or cleared in the Q1 cycle, so there is no issue with executing R-M-W instructions on registers that contain these bits.

31.4.3 PCL as Source or Destination

Read, write or read-modify-write (R-M-W) on PCL may have the following results:

Read PCL: PCL → destination ; Reading PCL causes the following
PCH → PCLATH
PCU → PCLATU

Write PCL: 8-bit destination value → PCL ; Writing PCL causes the following
PCLATH → PCH
PCLATU → PCU

Read-Modify-Write: PCL → ALU operand ; R-M-W of PCL causes the following
PCH → PCLATH
PCU → PCLATU

8-bit result → PCL ; result is written back to PCL
PCLATH → PCH
PCLATU → PCU

Where PCH = program counter high byte (not an addressable register),
PCLATH = Program counter high holding latch,
PCU = program counter upper byte (not an addressable register),
PCLATU = Program counter upper holding latch,
destination = Register file 'T'.
Section 31. Instruction Set

31.5 Fast Register Stack

At times it is desirable to be able to quickly access and return from a function. This function may be called as a subroutine, or an interrupt routine of the device. To reduce the overhead for accessing/returning from these functions, the architecture has the ability to save three key registers in a one deep Register Stack. These registers are:

- WREG Register
- BSR (Bank Select Register) Register
- STATUS Register

The two events that cause these registers to be loaded onto the Fast Register Stack are:

- A fast call (CALL K, fast) (where the fast bit is set ('1'))
- Any interrupt occurs

These Fast Stack Registers are not accessible for reading or writing. When doing the return from these subroutine, the values can be restored into their registers executing the fast return:

- RETFIE fast (where the fast bit is set ('1'))
- RETURN fast (where the fast bit is set ('1'))

When s(fast) = '0', the Fast Register Stack is not used, when s(fast) = '1', the Fast Register Stack is used.

31.6 Q Cycle Activity

Each instruction cycle (Tcy) is comprised of four Q clocks (also called Q cycles). These are referred to as Q1, Q2, Q3, or Q4. The Q cycles provide the timing/designation for the Decode, Read, Process Data, Write etc., of each instruction cycle. The Figure 31-3 shows the relationship of the Q cycles to the instruction cycle.

The four Q cycles that make up an instruction cycle (Tcy) can be generalized as:

Q1: Instruction Decode Cycle or forced No Operation
Q2: Instruction Read Cycle or No Operation
Q3: Process the Data
Q4: Instruction Write Cycle or No Operation

Some actions occur on the edge between the end of one Q cycle and the start of the next Q cycle. An example would be a Q2-Q3 action. This action occurs on the clock edge between the end of Q2 cycle and the beginning of the Q3 cycle.

The clock source for the Q cycle is normally the device oscillator clock (TOSC). But the clock source is software selectable. So the Q cycle may be independent of the device oscillator cycle (TOSC).

In the full description of each instruction, the detailed Q cycle operation for the instruction will be shown.

Figure 31-3: Q Cycle Activity
31.7 Instruction Descriptions

**ADDLW** Add Literal to WREG

Syntax: \[ label \] ADDLW \ k

Operands: \( 0 \leq k \leq 255 \)

Operation: \( (WREG) + k \rightarrow WREG \)

Status Affected: C, DC, Z, OV, N

Encoding: 0000 1111 kkkk kkkk

Description: The eight bit literal \( 'k' \) is added to the contents of the WREG and the result is placed in the WREG.

Words: 1

Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'k'</td>
<td>Process data</td>
<td>Write to WREG Register</td>
</tr>
</tbody>
</table>

**Example 1**

ADDLW 0x19 ; Add 19h to value in WREG

Before Instruction

| WREG = 0x18 |
| C, DC, Z, OV, N = x |

After Instruction

| WREG = 0x31 |
| C = 0 |
| DC = 1 |
| Z = 0 |
| OV = 0 |
| N = 0 |

**Example 2**

ADDLW MYREG ; Add the value of the address for MYREG Register to WREG

Before Instruction

| WREG = 0x60 |

Address of MYREG† = 0x37

†MYREG is a symbol for a data memory location
| C, DC, Z, OV, N = x |

After Instruction

| WREG = 0x97 |
| C = 0 |
| DC = 0 |
| Z = 0 |
| OV = 1 |
| N = 1 |
Section 31. Instruction Set

Example 3

ADDLW HIGH (LU_TABLE) ; Add high byte of address
; LU_TABLE to WREG

Before Instruction
WREG = 0x10
Address of LU_TABLE † = 0x9375
† LU_TABLE is a label for an address in program memory
C, DC, Z, OV, N = x

After Instruction
WREG = 0xA3
C = 0
DC = 0
Z = 0
OV = 0
N = 1

Example 4

ADDLW PCL ; Add value of the address
; of Program Counter Low
; byte (PCL) to WREG

Before Instruction
WREG = 0x02
Address of PCL † = 0xFF8 (only low 8-bits are used)
† PCL is the symbol for the Program Counter low byte location
C, DC, Z, OV, N = x

After Instruction
WREG = 0xFA
C = 0
DC = 0
Z = 0
OV = 0
N = 0

Example 5

ADDLW Offset ; Add the value of symbol
; Offset to WREG

Before Instruction
WREG = 0x10
Offset = 0x02
C, DC, Z, OV, N = x

After Instruction
WREG = 0x12
Offset = 0x02
C = 0
DC = 0
Z = 0
OV = 0
N = 0
ADDWF
Add WREG and f

Syntax: \[
\text{[ label]}\ ADDWF \ f, d, a
\]

Operands: \[ 0 \leq f \leq 255 \]
\[ d \in [0, 1] \]
\[ a \in [0, 1] \]

Operation: \((\text{WREG}) + (f) \rightarrow \text{destination}\)

Status Affected: C, DC, Z, OV, N

Encoding: \[ \text{0010 01da fffe fffe} \]

Description: Add the contents of the WREG Register to the contents of Register 'f'.

The 'd' bit selects the destination for the operation.
- If 'd' is 1: the result is stored back in the File Register 'f'.
- If 'd' is 0: the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1: the bank specified by the BSR Register is used.
- If 'a' is 0: the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1
ADDWF FSR0L, 1, 1 ; Add value in WREG to value in the FSR0H:FSR0L Register

Case 1: Before Instruction
- WREG = 0x17
- FSR0H:FSR0L = 0x2C2
- C, DC, Z, OV, N = x

After Instruction
- WREG = 0x17
- FSR0H:FSR0L = 0x2D9
- C = 0
- DC = 0
- Z = 0
- OV = 0
- N = 1

Case 2: Before Instruction
- WREG = 0x17
- FSR0H:FSR0L = 0xFF
- C, DC, Z, OV, N = x

After Instruction
- WREG = 0x17
- FSR0H:FSR0L = 0x316
- C = 0
- DC = 0
- Z = 0
- OV = 0
- N = 0
Section 31. Instruction Set

Example 2

```
ADDWF INDF0, 1, 1 ; Add value in WREG to
; value in the register
; pointed to (addressed)
; by the FSR0H:FSR0L
; Register
```

Before Instruction:

- WREG = 0x17
- FSR0H:FSR0L = 0x6C2
- Contents of Address (FSR0) = 0x20
- C, DC, Z, OV, N = x

After Instruction:

- WREG = 0x17
- FSR0H:FSR0L = 0x6C2
- Contents of Address (FSR0) = 0x37
- C = 0
- DC = 0
- Z = 0
- OV = 0
- N = 0

Example 3

```
ADDWF INDF0, 1, 0 ; Add value in WREG to
; value in the register
; pointed to (addressed)
; by the FSR0H:FSR0L
; Register
```

Before Instruction:

- WREG = 0x17
- FSR0H:FSR0L = 0x0C2
- Contents of Address (FSR0) = 0x20
- C, DC, Z, OV, N = x

After Instruction:

- WREG = 0x17
- FSR0H:FSR0L = 0x0C2
- Contents of Address (FSR0) = 0x37
- C = 0
- DC = 0
- Z = 0
- OV = 0
- N = 0
Example 4
   ADDWF PCL, 1, 1 ; Add the value in WREG to
   ; the current value in the
   ; low byte of the program
   ; counter (PCL)

Case 1: Before Instruction
   WREG = 0x10
   PCL = 0x37
   C, DC, Z, OV, N = x
   After Instruction
   WREG = 0x10
   PCL = 0x47
   C = 0
   DC = 0
   Z = 0
   OV = 0
   N = 0

Case 2: Before Instruction
   WREG = 0x10
   PCL = 0xF7
   PCH = 0x08
   C, DC, Z, OV, N = x
   After Instruction
   WREG = 0x10
   PCL = 0x07
   PCH = 0x08
   C = 1
   DC = 0
   Z = 0
   OV = 0
   N = 0
Section 31. Instruction Set

Example 5  
ADDWF MYREG, 1 ; Add the value in WREG to  
; the current value in  
; MYREG  
; (assembler determines  
; that MYREG requires  
; access bit to be set)

Case 1: Before Instruction  
BSR = 0x01  
WREG = 0x10  
MYREG = 0x37 ; In Bank 1  
C, DC, Z, OV, N = x

After Instruction  
BSR = 0x01  
WREG = 0x10  
MYREG = 0x47 ; In Bank 1  
C = 0  
DC = 0  
Z = 0  
OV = 0  
N = 0

Case 2: Before Instruction  
BSR = 0x01  
WREG = 0x10  
MYREG = 0xF7 ; In Bank 1  
C, DC, Z, OV, N = x

After Instruction  
BSR = 0x01  
WREG = 0x10  
MYREG = 0x07 ; In Bank 1  
C = 1  
DC = 0  
Z = 0  
OV = 0  
N = 0
ADDWFC Add WREG and Carry bit to f

Syntax: \([ \text{label} ] \) ADDWF f, d, a

Operands: \(0 \leq f \leq 255\)
\(d \in \{0,1\}\)
\(a \in \{0,1\}\)

Operation: \((\text{WREG}) + (f) + (C) \rightarrow \text{destination}\)

Status Affected: C, DC, Z, OV, N

Encoding:

\[
\begin{array}{c|c|c|c|c}
 & 0 & 0 & 1 & 0 \\
\hline
0 & 0 & x & x & x \\
\hline
\end{array}
\]

Description: Add the contents of the WREG Register and the Carry bit to the contents of Register f.

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register f.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Example 1

ADDWF FSR0L, 0, 1 ; Add WREG, C bit, and FSR0L
; value (Destination WREG)

Case 1: Before Instruction

\begin{align*}
\text{WREG} & = 0x17 \\
\text{FSR0H:FSR0L} & = 0x9C2 \\
C & = 0 \\
DC, Z, OV, N & = x
\end{align*}

After Instruction

\begin{align*}
\text{WREG} & = 0xD9 \\
\text{FSR0H:FSR0L} & = 0x9C2 \\
C & = 0 \\
DC & = 0 \\
Z & = 0 \\
OV & = 0 \\
N & = 1
\end{align*}

Case 2: Before Instruction

\begin{align*}
\text{WREG} & = 0x17 \\
\text{FSR0H:FSR0L} & = 0x7C2 \\
C & = 1 \\
DC, Z, OV, N & = x
\end{align*}

After Instruction

\begin{align*}
\text{WREG} & = 0xDA \\
\text{FSR0H:FSR0L} & = 0x7C2 \\
C & = 0 \\
DC & = 0 \\
Z & = 0 \\
OV & = 0 \\
N & = 1
\end{align*}
Section 31. Instruction Set

Example 2

ADDWFC INDF0, 1, 1 ; Add WREG and the Carry bit to the value pointed to by the FSR0H:FSR0L (Destination: File Register)

Before Instruction
WREG = 0x17
FSR0H:FSR0L = 0x0C2
Contents of Address (FSR0H:FSR0L) = 0x20
C = 0
DC, Z, OV, N = x

After Instruction
WREG = 0x17
FSR0H:FSR0L = 0x0C2
Contents of Address (FSR0H:FSR0L) = 0x37
C = 0
DC = 0
Z = 0
OV = 0
N = 0

Example 3

ADDWFC PCL, 1, 1 ; Add WREG and the Carry bit to the PCL Register

Case 1: Before Instruction
WREG = 0x10
PCL = 0x38
C = 0
DC, Z, OV, N = x

After Instruction
WREG = 0x10
PCL = 0x48
C = 0
DC = 0
Z = 0
OV = 0
N = 0

Case 2: Before Instruction
WREG = 0x10
PCL = 0xF8
PCH = 0x08
C = 0
DC, Z, OV, N = x

After Instruction
WREG = 0x10
PCL = 0x08
PCH = 0x08
C = 1
DC = 0
Z = 0
OV = 0
N = 0
ANDLW AND Literal with WREG

Syntax: \[ \text{label} \] ANDLW \( k \)

Operands: \( 0 \leq k \leq 255 \)

Operation: \((\text{WREG}).\text{AND.} (k) \rightarrow W\)

Status Affected: Z, N

Encoding: 0000 1011 kkkk kkkk

Description: The contents of WREG Register are AND’ed with the eight bit literal ‘k’. The result is placed in the WREG Register.

Words: 1

Cycles: 1

Q Cycle Activity:

\[\begin{array}{|c|c|c|c|}
\hline
\text{Q1} & \text{Q2} & \text{Q3} & \text{Q4} \\
\hline
\text{Decode} & \text{Read literal ‘k’} & \text{Process data} & \text{Write to WREG Register} \\
\hline
\end{array}\]

Example 1

ANDLW 0x5F ; And constant to WREG

Before Instruction

WREG = 0xA3
Z, N = x

After Instruction

WREG = 0x03
Z = 0
N = 0

Example 2

ANDLW MYREG ; And address of MYREG to WREG

Before Instruction

WREG = 0xA3
Address of MYREG\( ^\dagger \) = 0x37
Z, N = x

† MYREG is a symbol for a data memory location

After Instruction

WREG = 0x23
Z = 0
N = 0
Section 31. Instruction Set

Example 3

```
ANDLW HIGH (LU_TABLE); And the high byte of

(ADDRESS) with WREG

Before Instruction
WREG = 0xA3
Address of LU_TABLE† = 0x9375
Z, N = x

† LU_TABLE is a label for an
address in program memory

After Instruction
WREG = 0x83
Z = 0
N = 1

Example 4

ANDLW LOW (LU_TABLE); And the low byte of

ADDRESS with WREG

Before Instruction
WREG = 0xA3
Address of LU_TABLE† = 0x9375
Z, N = x

† LU_TABLE is a label for an
address in program memory

After Instruction
WREG = 0x21
Z = 0
N = 0
```
ANDWF AND WREG with f

Syntax: \[ \text{label} \] ANDWF f, d, a

Operands: 0 \leq f \leq 255
\ d \in [0,1]
\ a \in [0,1]

Operation: (WREG).AND. (f) \rightarrow \text{destination}

Status Affected: Z, N

Encoding: 0001 01da ffff ffff

Description: The contents of the WREG Register are AND'd with the contents of Register 'f'.

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register 'f'.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Register 'f'</td>
<td>Process</td>
<td>data</td>
</tr>
</tbody>
</table>

Example 1

ANDWF REG1, 1, 1 ; And WREG with REG1

Before Instruction
WREG = 0x17 ; 0001 0111 (0x17)
REG1 = 0xC2 ; 1100 0010 (0xC2)
Z, N = x ; ------------ --

After Instruction
WREG = 0x17
REG1 = 0x02
Z = 0
N = 0

Example 2

ANDWF REG1, 0, 1 ; And WREG with REG1

Before Instruction
WREG = 0x17 ; 0001 0111 (0x17)
REG1 = 0xC2 ; 1100 0010 (0xC2)
Z, N = x ; ------------ --

After Instruction
WREG = 0x02 ; 0000 0010 (0x02)
REG1 = 0xC2
Z = 0
N = 0
Section 31. Instruction Set

Example 3

ANDWF  INDF0, 1, 1 ; And WREG with value pointed
; by FSR0H:FSR0L (FSR0)

Case 1: Before Instruction
WREG = 0x17
FSR0H:FSR0L = 0xFC 2
Contents of Address (FSR0) = 0x5A
Z, N = x

After Instruction
WREG = 0x17
FSR0H:FSR0L = 0xFC 2
Contents of Address (FSR0) = 0x12
Z = 0
N = 0

Case 2: Before Instruction
WREG = 0x00
FSR0H:FSR0L = 0x4C 2
Contents of Address (FSR0) = 0x5A
Z, N = x

After Instruction
WREG = 0x00
FSR0H:FSR0L = 0x4C 2
Contents of Address (FSR0) = 0x00
Z = 1
N = 0
BC  Branch if Carry

Syntax: \[ \text{label} \] BC \ n
Operands: \[-128 \leq f \leq 127\]
Operation: If carry bit is '1'
\[(PC + 2) + 2n \rightarrow PC\]
Status Affected: None
Encoding:

| 1110 0010 nnnn nnnn |

Description: If the Carry bit is '1', then the program will branch.
The 2's complement number '2n' (the offset) is added to the PC. Since the
PC will have incremented to fetch the next instruction, the new address will
be \((PC+2)+2n\). This instruction is then a two-cycle instruction.

Words: 1
Cycles: 1 (2)
Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

If Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to</td>
</tr>
<tr>
<td>No operation</td>
<td>literal 'n'</td>
<td>data</td>
<td>PC</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>No operation</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td>No operation</td>
<td></td>
</tr>
</tbody>
</table>

Example 1

```
HERE BC C_CODE
NOT_C

; If C bit is not set
; execute this code.

GOTO MORE_CODE

; else if C bit is set
; this code will execute
```

Case 1: Before Instruction

\[
\begin{align*}
\text{PC} & = \text{address HERE} \\
C & = 0 \\
\end{align*}
\]
After Instruction

\[
\begin{align*}
\text{PC} & = \text{address NOT_C} \\
\end{align*}
\]

Case 2: Before Instruction

\[
\begin{align*}
\text{PC} & = \text{address HERE} \\
C & = 1 \\
\end{align*}
\]
After Instruction

\[
\begin{align*}
\text{PC} & = \text{address C_CODE} \\
\end{align*}
\]
Section 31. Instruction Set

Example 2

\[ \text{HERE} \quad \text{BC} \quad \$ + \text{OFFSET} \quad ; \text{If carry bit is set,} \]
\[ \text{NO}_C \quad \text{GOTO} \quad \text{PROCESS\_CODE} \quad ; \text{branch to HERE+OFFSET} \]
\[ \text{PLUS0} \quad \cdot \]
\[ \text{PLUS1} \quad \cdot \]
\[ \text{PLUS2} \quad \cdot \]
\[ \text{PLUS3} \quad \cdot \]
\[ \text{PLUS4} \quad \cdot \]
\[ \text{PLUS5} \quad \cdot \]
\[ \text{PLUS6} \quad \cdot \]

Case 1: Before Instruction
\[ \text{PC} = \text{address} \quad \text{HERE} \]
\[ \text{C} = 0 \]

After Instruction
\[ \text{PC} = \text{address} \quad \text{NO\_C} \]

Case 2: Before Instruction
\[ \text{PC} = \text{address} \quad \text{HERE} \]
\[ \text{C} = 1 \]

After Instruction
\[ \text{PC} = \text{address} \quad \text{HERE} + \text{OFFSET} \]

Example 3

\[ \text{MIN6} \quad \cdot \]
\[ \text{MIN5} \quad \cdot \]
\[ \text{MIN4} \quad \cdot \]
\[ \text{MIN3} \quad \cdot \]
\[ \text{MIN2} \quad \cdot \]
\[ \text{MIN1} \quad \cdot \]
\[ \text{MIN0} \quad \cdot \quad ; \text{If carry bit is set,} \]
\[ \text{HERE} \quad \text{BC} \quad \$ - \text{OFFSET} \quad ; \text{branch to HERE-OFFSET} \]
\[ \text{NO}_C \quad \text{GOTO} \quad \text{PROCESS\_CODE} \]

Case 1: Before Instruction
\[ \text{PC} = \text{address} \quad \text{HERE} \]
\[ \text{C} = 0 \]

After Instruction
\[ \text{PC} = \text{address} \quad \text{NO\_C} \]

Case 2: Before Instruction
\[ \text{PC} = \text{address} \quad \text{HERE} \]
\[ \text{C} = 1 \]

After Instruction
\[ \text{PC} = \text{address} \quad \text{HERE} - \text{OFFSET} \]

Note: Assembler will convert the specified address label into the offset to be used.
BCF Bit Clear f

Syntax: [label] BCF f, b, a

Operands: 0 ≤ f ≤ 255
0 ≤ b ≤ 7
a ∈ [0, 1]

Operation: 0 → f<b>

Status Affected: None

Encoding: 1001 bba fff ffff

Description: Bit 'b' in Register 'f' of the specified bank is cleared.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1

Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write Register 'f'</td>
</tr>
</tbody>
</table>

Example 1

BCF MYREG, 7, 1 ; Clear bit 7 in Register MYREG

Before Instruction

MYREG = 0xC7 ; 1100 0111

After Instruction

MYREG = 0x47 ; 0100 0111
Section 31. Instruction Set

Example 2

BCF INDF0, 3, 0 ; Clear bit 7 in the register
; pointed to by the FSR0
; (FSR0H:FSR0L) Register

Before Instruction
FSR0 = 0x3C2
Contents of Address
(FSR0) = 0x2F ; 0010 1111

After Instruction
FSR0 = 0x3C2
Contents of Address
(FSR0) = 0x27 ; 0010 0111
BN  Branch if Negative

Syntax: \[
\text{(label)} \ BN \ n
\]

Operands: \[-128 \leq f \leq 127\]

Operation: If negative bit is '1'

\[(PC + 2) + 2n \rightarrow PC\]

Status Affected: None

Encoding: 1110 0110 nnnn nnnn

Description: If the Negative bit is '1', then the program will branch.

The 2's complement number '2n' (the offset) is added to the PC. Since the
PC will have incremented to fetch the next instruction, the new address will
be (PC+2)+2n. This instruction is then a two-cycle instruction.

Words: 1

Cycles: 1 (2)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to</td>
</tr>
<tr>
<td>No</td>
<td>literal 'n'</td>
<td>data</td>
<td>PC</td>
</tr>
<tr>
<td>operation</td>
<td>No</td>
<td>operation</td>
<td>No</td>
</tr>
<tr>
<td>operation</td>
<td>operation</td>
<td>operation</td>
<td>operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>No</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td>operation</td>
<td>operation</td>
</tr>
</tbody>
</table>

Example 1

HERE \ BN \ N_CODE ; If N bit is not set
NOT_N \ * \ ; execute this code.
* \ ;
GOTO \ MORE_CODE ;
N_CODE \ * \ ; else if N bit is set
* \ ; this code will execute

Case 1: Before Instruction

PC = address \ HERE
N = 0

After Instruction

PC = address \ NOT_N

Case 2: Before Instruction

PC = address \ HERE
N = 1

After Instruction

PC = address \ N_CODE
Section 31. Instruction Set

Example 2

\[
\begin{align*}
\text{HERE} & \quad \text{BN} \quad \$ + \text{OFFSET} \quad ; \text{If negative bit is set,} \\
\text{NOT\_N} & \quad \text{GOTO} \quad \text{PROCESS\_CODE} \quad ; \text{branch to \text{HERE} + \text{OFFSET}} \\
\text{PLUS0} & \cdot \\
\text{PLUS1} & \cdot \\
\text{PLUS2} & \cdot \\
\text{PLUS3} & \cdot \\
\text{PLUS4} & \cdot \\
\text{PLUS5} & \cdot \\
\text{PLUS6} & \cdot \\
\end{align*}
\]

Case 1: Before Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{HERE}} \\
\text{N} & = 0
\end{align*}
\]
After Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{NOT\_N}} \\
\end{align*}
\]
Case 2: Before Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{HERE}} \\
\text{N} & = 1
\end{align*}
\]
After Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{HERE} + \text{OFFSET}}
\end{align*}
\]

Example 3

\[
\begin{align*}
\text{MIN6} & \cdot \\
\text{MIN5} & \cdot \\
\text{MIN4} & \cdot \\
\text{MIN3} & \cdot \\
\text{MIN2} & \cdot \\
\text{MIN1} & \cdot \\
\text{MIN0} & \cdot \\
\text{HERE} & \quad \text{BN} \quad \$ - \text{OFFSET} \quad ; \text{If negative bit is set,} \\
\text{NO\_N} & \quad \text{GOTO} \quad \text{PROCESS\_CODE} \\
\end{align*}
\]

Case 1: Before Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{HERE}} \\
\text{N} & = 0
\end{align*}
\]
After Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{NO\_N}} \\
\end{align*}
\]
Case 2: Before Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{HERE}} \\
\text{N} & = 1
\end{align*}
\]
After Instruction
\[
\begin{align*}
\text{PC} & = \text{address \text{HERE} - \text{OFFSET}}
\end{align*}
\]

Note: Assembler will convert the specified address label into the offset to be used.
**BNC**  
Branch if Not Carry

**Syntax:**  
\[ \text{[label]} \text{ BNC } n \]

**Operands:**  
\(-128 \leq f \leq 127\)

**Operation:**  
If carry bit is '0'  
\((\text{PC} + 2) + 2n \rightarrow \text{PC}\)

**Status Affected:**  
None

**Encoding:**  
\[ \begin{array}{c}
1110 0011 \ nnnn \ nnnn
\end{array} \]

**Description:**  
If the Carry bit is '0', then the program will branch.  
The 2's complement number '2n' (the offset) is added to the PC. Since the  
PC will have incremented to fetch the next instruction, the new address will  
be \((\text{PC}+2)+2n\). This instruction is then a two-cycle instruction.

**Words:**  
1

**Cycles:**  
1 (2)

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td>PC</td>
<td></td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>No operation</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

**Example 1**

```
HERE BNC NC_CODE ; If C bit is set
CARRY   * ; execute this code.
* ;
GOTO MORE_CODE ;
NC_CODE * ; else if C bit is clear
* ; this code will execute
```

**Case 1:** Before Instruction  
\( \text{PC} = \text{address HERE} \)
\( C = 0 \)

After Instruction  
\( \text{PC} = \text{address NC_CODE} \)

**Case 2:** Before Instruction  
\( \text{PC} = \text{address HERE} \)
\( C = 1 \)

After Instruction  
\( \text{PC} = \text{address CARRY} \)
Section 31. Instruction Set

Example 2

Here BNC $ + OFFSET ; If carry bit is clear,
CARRY GOTO PROCESS_CODE ; branch to HERE + OFFSET
PLUS0 •
PLUS1 •
PLUS2 •
PLUS3 •
PLUS4 •
PLUS5 •
PLUS6 •

Case 1: Before Instruction
PC = address HERE
C = 0
After Instruction
PC = address HERE + OFFSET
Case 2: Before Instruction
PC = address HERE
C = 1
After Instruction
PC = address CARRY

Example 3

MIN6 •
MIN5 •
MIN4 •
MIN3 •
MIN2 •
MIN1 •
MIN0 •
HERE BNC $ - OFFSET ; If carry bit is clear,
CARRY GOTO PROCESS_CODE

Case 1: Before Instruction
PC = address HERE
C = 0
After Instruction
PC = address HERE - OFFSET
Case 2: Before Instruction
PC = address HERE
C = 1
After Instruction
PC = address CARRY

Note: Assembler will convert the specified address label into the offset to be used.
BNN Branch if Not Negative

Syntax: [ label ] BNN n

Operands: -128 ≤ f ≤ 127

Operation: If negative bit is '0'
( PC + 2 ) + 2n → PC

Status Affected: None

Encoding: 1110 0111 nnnn nnnn

Description: If the Negative bit is '0', then the program will branch.
The 2's complement number '2n' (the offset) is added to the PC. Since the
PC will have incremented to fetch the next instruction, the new address will
be (PC+2)+2n. This instruction is then a two-cycle instruction.

Words: 1
Cycles: 1 (2)

Q Cycle Activity:

If Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>Write to PC</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

HERE BNN POS_CODE ; If N bit is set
NEG • ; execute this code.
•
•
GOTO MORE_CODE ;
POS_CODE • ; else if N bit is clear
•
•
; this code will execute

Case 1: Before Instruction

PC = address HERE
N = 0

After Instruction

PC = address POS_CODE

Case 2: Before Instruction

PC = address HERE
N = 1

After Instruction

PC = address NEG

PIC18C Reference Manual
Section 31. Instruction Set

Example 2

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Example</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>HERE</td>
<td>BNN $ + OFFSET</td>
<td>; If negative bit is clear, branch to HERE + OFFSET</td>
</tr>
<tr>
<td>NEG GOTO PROCESS_CODE</td>
<td></td>
<td>; branch to HERE + OFFSET</td>
</tr>
<tr>
<td>PLUS0</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PLUS1</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PLUS2</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PLUS3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PLUS4</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PLUS5</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PLUS6</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Case 1: Before Instruction

PC = address HERE

N = 0

After Instruction

PC = address HERE + OFFSET

Case 2: Before Instruction

PC = address HERE

N = 1

After Instruction

PC = address NEG

Example 3

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Example</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>MIN6</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MIN5</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MIN4</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MIN3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MIN2</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MIN1</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MIN0</td>
<td></td>
<td></td>
</tr>
<tr>
<td>HERE</td>
<td>BNN $ - OFFSET</td>
<td>; If negative bit is clear, branch to HERE - OFFSET</td>
</tr>
<tr>
<td>NEG GOTO PROCESS_CODE</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Case 1: Before Instruction

PC = address HERE

N = 0

After Instruction

PC = address HERE - OFFSET

Case 2: Before Instruction

PC = address HERE

N = 1

After Instruction

PC = address NEG

Note: Assembler will convert the specified address label into the offset to be used.
BNOV Branch if Not Overflow

Syntax: \[
\text{[label]} \ B NOV \ n
\]

Operands: \[-128 \leq f \leq 127\]

Operation: If overflow bit is '0'

\[(PC + 2) + 2n \rightarrow PC\]

Status Affected: None

Encoding: \[1110 \ 0101 \ nnnn \ nnnn\]

Description: If the Overflow bit is '0', then the program will branch.

The 2's complement number '2n' (the offset) is added to the PC. Since the PC will have incremented to fetch the next instruction, the new address will be (PC+2)+2n. This instruction is then a two-cycle instruction.

Words: 1

Cycles: 1 (2)

Q Cycle Activity:

If Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to PC</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>No operation</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td>No operation</td>
<td></td>
</tr>
</tbody>
</table>

Example 1

```
HERE BNOV NOV_CODE ; If overflow bit is set
OVFL *
* ; execute this code.
* ;
GOTO MORE_CODE ;
NOV_CODE * ; else if overflow bit is
* ; clear this code will
; execute
```

Case 1: Before Instruction

PC = address HERE

OV = 0

After Instruction

PC = address NOV_CODE

Case 2: Before Instruction

PC = address HERE

OV = 1

After Instruction

PC = address OVFL
Section 31. Instruction Set

Example 2

\[
\text{HERE} \quad \text{BNOV} \quad \# + \text{OFFSET} \quad ; \text{If overflow bit is clear,}
\]
\[
\text{OVFL} \quad \text{GOTO} \quad \text{PROCESS\_CODE} \quad ; \text{branch to HERE + OFFSET}
\]
\[
\# \quad \text{PLUS0} \quad \)
\]
\[
\# \quad \text{PLUS1} \quad \)
\]
\[
\# \quad \text{PLUS2} \quad \)
\]
\[
\# \quad \text{PLUS3} \quad \)
\]
\[
\# \quad \text{PLUS4} \quad \)
\]
\[
\# \quad \text{PLUS5} \quad \)
\]
\[
\# \quad \text{PLUS6} \quad \)
\]

Case 1: Before Instruction
\[
PC = \text{address} \quad \text{HERE}
\]
\[
OV = 0
\]

After Instruction
\[
PC = \text{address} \quad \text{HERE + OFFSET}
\]

Case 2: Before Instruction
\[
PC = \text{address} \quad \text{HERE}
\]
\[
OV = 1
\]

After Instruction
\[
PC = \text{address} \quad \text{OVFL}
\]

Example 3

\[
\text{MIN6} \quad \)
\]
\[
\text{MIN5} \quad \)
\]
\[
\text{MIN4} \quad \)
\]
\[
\text{MIN3} \quad \)
\]
\[
\text{MIN2} \quad \)
\]
\[
\text{MIN1} \quad \)
\]
\[
\text{MIN0} \quad \); \text{If overflow bit is clear,}
\]
\[
\text{HERE} \quad \text{BNOV} \quad \# - \text{OFFSET} \quad ; \text{branch to HERE - OFFSET}
\]
\[
\text{OVFL} \quad \text{GOTO} \quad \text{PROCESS\_CODE}
\]

Case 1: Before Instruction
\[
PC = \text{address} \quad \text{HERE}
\]
\[
OV = 0
\]

After Instruction
\[
PC = \text{address} \quad \text{HERE - OFFSET}
\]

Case 2: Before Instruction
\[
PC = \text{address} \quad \text{HERE}
\]
\[
OV = 1
\]

After Instruction
\[
PC = \text{address} \quad \text{OVFL}
\]

Note: Assembler will convert the specified address label into the offset to be used.
BNZ  

Branch if Not Zero

Syntax: \[ \text{[label]} \text{ BNZ } n \]

Operands: \(-128 \leq f \leq 127\)

Operation: If zero bit is '0'

\((PC + 2) + 2n \rightarrow PC\)

Status Affected: None

Encoding: \[ 1110 0001 nnnn nnnn \]

Description: If the Zero bit is '0', then the program will branch. The 2's complement number '2n' (the offset) is added to the PC. Since the PC will have incremented to fetch the next instruction, the new address will be \((PC+2)+2n\). This instruction is then a two-cycle instruction.

Words: 1

Cycles: 1 (2)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>Write to PC</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

HERE BNZ Z_CODE ; If Z bit is set
ZERO ; ; execute this code.
* ;
GOTO MORE_CODE ;
Z_CODE ; else if Z bit is clear
* ; this code will execute

Case 1: Before Instruction  
  PC = address HERE  
  Z = 0  
After Instruction  
  PC = address Z_CODE

Case 2: Before Instruction  
  PC = address HERE  
  Z = 1  
After Instruction  
  PC = address ZERO
Section 31. Instruction Set

Example 2

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>HERE</td>
<td>If zero bit is clear, branch to HERE + OFFSET</td>
</tr>
<tr>
<td>ZERO</td>
<td>GOTO PROCESS_CODE; branch to HERE + OFFSET</td>
</tr>
<tr>
<td>PLUS0</td>
<td></td>
</tr>
<tr>
<td>PLUS1</td>
<td></td>
</tr>
<tr>
<td>PLUS2</td>
<td></td>
</tr>
<tr>
<td>PLUS3</td>
<td></td>
</tr>
<tr>
<td>PLUS4</td>
<td></td>
</tr>
<tr>
<td>PLUS5</td>
<td></td>
</tr>
<tr>
<td>PLUS6</td>
<td></td>
</tr>
</tbody>
</table>

Case 1: Before Instruction
PC = address HERE
Z = 0
After Instruction
PC = address HERE + OFFSET

Case 2: Before Instruction
PC = address HERE
Z = 1
After Instruction
PC = address ZERO

Example 3

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>MIN6</td>
<td></td>
</tr>
<tr>
<td>MIN5</td>
<td></td>
</tr>
<tr>
<td>MIN4</td>
<td></td>
</tr>
<tr>
<td>MIN3</td>
<td></td>
</tr>
<tr>
<td>MIN2</td>
<td></td>
</tr>
<tr>
<td>MIN1</td>
<td></td>
</tr>
<tr>
<td>MIN0</td>
<td>If zero bit is clear, branch to HERE - OFFSET</td>
</tr>
<tr>
<td>HERE</td>
<td>BNZ $ - OFFSET; branch to HERE - OFFSET</td>
</tr>
<tr>
<td>ZERO</td>
<td>GOTO PROCESS_CODE</td>
</tr>
</tbody>
</table>

Case 1: Before Instruction
PC = address HERE
Z = 0
After Instruction
PC = address HERE - OFFSET

Case 2: Before Instruction
PC = address HERE
Z = 1
After Instruction
PC = address ZERO

Note: Assembler will convert the specified address label into the offset to be used.
BOV  Branch if Overflow

Syntax:  \([\text{label}]\ \text{BOV}\ \ n\)

Operands: \(-128 \leq f \leq 127\)

Operation:  If overflow bit is '1'
\((\text{PC} + 2) + 2n \rightarrow \text{PC}\)

Status Affected:  None

Encoding: \(\text{Code}\): \(1110\ 0100\ nnnn\ nnnn\)

Description:  If the Overflow bit is '1', then the program will branch.

The 2's complement number \(2n\) (the offset) is added to the PC. Since the PC will have incremented to fetch the next instruction, the new address will be \((\text{PC}+2)+2n\). This instruction is then a two-cycle instruction.

Words:  1

Cycles:  1 (2)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td>PC</td>
<td></td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>No operation</td>
</tr>
<tr>
<td>literal 'n'</td>
<td>data</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Example 1

HERE    BOV      OV_CODE    ; If OV bit is clear
OVFL    •         •          ; execute this code.
•        •         ;
GOTO    MORE_CODE ;
OV_CODE  •         ; else if OV bit is set
•         ; this code will execute

Case 1: Before Instruction

PC = address HERE
OV = 0

After Instruction

PC = address OVFL

Case 2: Before Instruction

PC = address HERE
OV = 1

After Instruction

PC = address OV_CODE
Example 2

HERE  BOV  $ + OFFSET ; If overflow bit is set,
OVFL  GOTO  PROCESS_CODE ; branch to HERE + OFFSET
PLUS0 •
PLUS1 •
PLUS2 •
PLUS3 •
PLUS4 •
PLUS5 •
PLUS6 •

Case 1: Before Instruction
PC = address  HERE
OV = 0
After Instruction
PC = address  OVFL

Case 2: Before Instruction
PC = address  HERE
OV = 1
After Instruction
PC = address  HERE + OFFSET

Example 3

MIN6 •
MIN5 •
MIN4 •
MIN3 •
MIN2 •
MIN1 •
MIN0 •

HERE  BOV  $ - OFFSET ; If OV bit is set,
OVFL  GOTO  PROCESS_CODE ; branch to HERE - OFFSET

Case 1: Before Instruction
PC = address  HERE
OV = 0
After Instruction
PC = address  OVFL

Case 2: Before Instruction
PC = address  HERE
OV = 1
After Instruction
PC = address  HERE - OFFSET

Note: Assembler will convert the specified address label into the offset to be used.
BRA  

Branch Unconditional

Syntax:  

[ label ] BRA n

Operands:  

-1024 ≤ f ≤ 1023

Operation:  

(PC + 2) + 2n → PC

Status Affected: None

Encoding:  

1111 0nnn nnnn nnnn

Description: The 2's complement number 2n (the offset) is added to the PC. Since the PC will have incremented to fetch the next instruction, the new address will be (PC+2)+2n. This instruction is a two-cycle instruction.

Words: 1

Cycles: 2

Q Cycle Activity:

If Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>Write to PC</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If No Branch

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

HERE BRA THERE ; Branch to a program memory

; location (THERE)

; this location must be

; < 1023 locations forward

THERE

Before Instruction

PC = address HERE

After Instruction

PC = address THERE
Section 31. Instruction Set

Example 2

THERE
• ; Branch to a program memory
• ; location (THERE)
• ; this location must be
• ; < 1024 locations backward

HERE BRA THERE

Before Instruction
PC = address HERE

After Instruction
PC = address THERE

Example 3

HERE BRA $ ; Branch to program memory

; location (HERE).

; Infinite Loop

Before Instruction
PC = address HERE

After Instruction
PC = address HERE

Note: Assembler will convert the specified address label into the offset to be used.
BSF Bit Set f

Syntax: [label] BSF f, b, a

Operands: 
0 ≤ f ≤ 255 
0 ≤ b ≤ 7 
a ∈ [0,1]

Operation: 
1 → f<b>

Status Affected: None

Encoding: 
1000 bbba ffff ffff

Description: Bit 'b' in Register 'f' is set.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1: the bank specified by the BSR Register is used.
If 'a' is 0: the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write Register 'f'</td>
</tr>
</tbody>
</table>

Example 1
BSF FLAG_REG, 7 ; Set bit 7 in Register FLAG_REG
(FSRO_H:FSR0_L Register)

Before Instruction
FLAG_REG = 0x0A ; 0000 1010

After Instruction
FLAG_REG = 0x8A ; 1000 1010

Example 2
BSF INDF0, 3, 0 ; Set bit 3 in the register pointed to by the FSR0
(FSRO_H:FSR0_L Register)

Before Instruction
WREG = 0x17
FSR0 = 0x0C2
Contents of Address (FSR0) = 0x20 ; 0010 0000

After Instruction
WREG = 0x17
FSR0 = 0x0C2
Contents of Address (FSR0) = 0x28 ; 0010 1000
Section 31. Instruction Set

BTFSC  Bit Test File, Skip if Clear

Syntax:  [ label] BTFSC f, b, a
Operands:  0 ≤ f ≤ 255
          0 ≤ b ≤ 7
          a ∈ [0,1]
Operation:  Skip if (f<b>) = 0
Status Affected:  None
Encoding:  1011 bbba ffff ffff
Description:  If bit ‘b’ in Register ‘f’ is ‘0’ then the next instruction is skipped.
If bit ‘b’ is ‘0’ then the next instruction (fetched during the current instruction execution) is discarded, and a NOP is executed instead, making this a 2-cycle instruction.
The ‘a’ bit selects which bank is accessed for the operation.
If ‘a’ is 1; the bank specified by the BSR Register is used.
If ‘a’ is 0; the access bank is used.

Words:  1
Cycles:  1 (2 or 3)
Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register ‘f’</td>
<td>Process data</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by a two word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

HERE BTFSC FLAG, 4, 1 ; Test bit 4 of Register
FALSE GOTO PROCESS_CODE ; FLAG, and skip if
TRUE • ; clear
•

Case 1:  Before Instruction
•
PC = address HERE
FLAG = xxxx0 xxxx
After Instruction
Since FLAG<4> = 0
PC = address TRUE

Case 2:  Before Instruction
•
PC = address HERE
FLAG = xxxx1 xxxx
After Instruction
Since FLAG<4> = 1
PC = address FALSE
BTFSS Bit Test File, Skip if Set

Syntax: \[
\text{[label\]} \text{BTFSS} f, b, a
\]

Operands: \[
0 \leq f \leq 255 \\
0 \leq b < 7 \\
a \in \{0,1\}
\]

Operation: Skip if \((f<b>) = 1\)

Status Affected: None

Encoding: \[
\begin{array}{cccc}
1010 & bbba & fffe & fffe \\
\end{array}
\]

Description: If bit 'b' in Register 'f' is '1' then the next instruction is skipped. If bit 'b' is '1', then the next instruction (fetched during the current instruction execution) is discarded and a \texttt{NOP} is executed instead, making this a 2-cycle instruction.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

Words: 1

Cycles: 1 (2 or 3)

Q Cycle Activity:

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by a two word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

\begin{verbatim}
HERE BTFSS FLAG, 4, 0 ; Test bit 4 of Register FALSE GOTO PROCESS_CODE ; FLAG, and skip if set TRUE 

Case 1: Before Instruction
    PC = address HERE
    FLAG = xxx0 xxxx
After Instruction
    Since FLAG<4> = 0
    PC = address FALSE

Case 2: Before Instruction
    PC = address HERE
    FLAG = xxx1 xxxx
After Instruction
    Since FLAG<4> = 1
    PC = address TRUE
\end{verbatim}
Section 31. Instruction Set

BTG 

Bit Toggle f

Syntax: [label] BTG f, b, a

Operands: 
0 ≤ f ≤ 255
0 ≤ b ≤ 7
a ∈ {0, 1}

Operation: \( (f \text{<} b) \rightarrow f\text{<} b) \)

Status Affected: None

Encoding: 
0111 bbba ffff ffff

Description: Bit 'b' in Register 'f' is toggled.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1, the bank specified by the BSR Register is used.
If 'a' is 0, the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write Register 'f'</td>
</tr>
</tbody>
</table>

Example 1

BTG LATC, 7, 1 ; Toggle the value of bit 7 in the LATC Register

Before Instruction
LATC = 0x0A ; 0000 1010

After Instruction
LATC = 0x8A ; 1000 1010

Example 2

BTG INDF0, 3, 1 ; Toggle the value of bit 3 in the register pointed to by the value in the FSR0 ; (FSR0H:FSR0L) Register

Before Instruction
FSR0 = 0xAC2
Contents of Address (FSR0)= 0x20 ; 0010 0000

After Instruction
FSR0 = 0xAC2
Contents of Address (FSR0)= 0x28 ; 0010 1000
BZ  Branch if Zero

Syntax:    {label} BZ n
Operands:  -128 ≤ f ≤ 127
Operation: If zero bit is ‘1’
           (PC + 2) + 2n → PC
Status Affected: None
Encoding:  1110 0000 nnnn nnnn
Description: If the Zero bit is ‘1’, then the program will branch.
The 2's complement number ‘2n’ (the offset) is added to the PC. Since the
PC will have incremented to fetch the next instruction, the new address will
be PC+2+2n. This instruction is then a two-cycle instruction.

Words: 1
Cycles: 1 (2)
Q Cycle Activity:
  If Branch
<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>Write to PC</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
  If No Branch
<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'n'</td>
<td>Process data</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1
HERE BZ Z_CODE; If zero bit is clear
ZERO         ; execute this code.
             ;
             ;
GOTO MORE_CODE; else if zero bit is set
Z_CODE       ; this code will execute
             ;

Case 1: Before Instruction
PC = address HERE
Z = 0
After Instruction
PC = address ZERO
Case 2: Before Instruction
PC = address HERE
Z = 1
After Instruction
PC = address Z_CODE
Section 31. Instruction Set

Example 2

HERE  BZ  $ + OFFSET ; If zero bit is set,
NZERO GOTO PROCESS_CODE ; branch to HERE + OFFSET
PLUS6 *
PLUS5 *
PLUS4 *
PLUS3 *
PLUS2 *
PLUS1 *
PLUS0 *

Case 1: Before Instruction
PC = address HERE
Z = 0
After Instruction
PC = address NZERO

Case 2: Before Instruction
PC = address HERE
Z = 1
After Instruction
PC = address HERE + OFFSET

Note: Assembler will convert the specified address label into the offset to be used.

Example 3

MIN6 *
MIN5 *
MIN4 *
MIN3 *
MIN2 *
MIN1 *
MIN0 *

HERE  BZ  $ - OFFSET ; If zero bit is set,
NZERO GOTO PROCESS_CODE

Case 1: Before Instruction
PC = address HERE
Z = 0
After Instruction
PC = address NZERO

Case 2: Before Instruction
PC = address HERE
Z = 1
After Instruction
PC = address HERE - OFFSET
CALL Call Subroutine

Syntax: \[ \text{label} \] CALL \text{k, s} \\
Operands: \(0 \leq k \leq 1048575\) \\
\(s \in \{0, 1\}\) \\
Operation: \(\text{(PC)+4} \rightarrow \text{TOS},\) \\
\(k \rightarrow \text{PC<20:1>},\) \\
\(0 \rightarrow \text{PC<0>},\) \\
if \(s = 1\) \\
\((\text{WREG}) \rightarrow \text{WREGS},\) \\
\((\text{STATUS}) \rightarrow \text{STATUSS},\) \\
\((\text{BSR}) \rightarrow \text{BSRS}\) \\
Status Affected: None \\
Encoding: \\
1st word (k<7:0>): \(1110 \text{ k<7:0>}\) \\
2nd word (k<19:8>): \(1111 \text{ k<19:8>}\) \\
Description: Subroutine call of entire 2M byte memory range. First, return address (PC+4) is pushed onto the return stack (20-bits wide).

If \(s = 1\), the WREG, STATUS and BSR Registers are also pushed into their respective Shadow Registers, WREGS, STATUSS and BSRS.

If \(s = 0\), no update occurs.

Then the 20-bit value \(k\) is loaded into PC<20:1>. CALL is a two-cycle instruction.

Words: 2 \\
Cycles: 2 \\
Q Cycle Activity: \\
1st cycle: \\
\[
\begin{array}{cccc}
\text{Q1} & \text{Q2} & \text{Q3} & \text{Q4} \\
\text{Decode} & \text{Read literal } 'k' & \text{Process data} & \text{No operation} \\
\end{array}
\]
2nd cycle: \\
\[
\begin{array}{cccc}
\text{Q1} & \text{Q2} & \text{Q3} & \text{Q4} \\
\text{No operation} & \text{No operation} & \text{No operation} & \text{No operation} \\
\end{array}
\]
Section 31. Instruction Set

Example 1

HERE CALL THERE, 1 ; Call subroutine THERE.

; This is a fast call so
; the BSR, WREG, and STATUS
; Registers are forced onto
; the Fast Register Stack

Before Instruction
PC = Address HERE

After Instruction
TOS = Address HERE+4
PC = Address THERE
WREGS = WREG
BSRS = BSR
STATUSS = STATUS

Example 2

HERE CALL THERE, 0 ; Call subroutine THERE.

; This is NOT a fast call

Before Instruction
PC = Address HERE
WREGS = 0x45
BSRS = 0x29
STATUS = 0x01

After Instruction
TOS = Address HERE+4
PC = Address THERE
WREGS = 0x45 (unchanged)
BSRS = 0x29 (unchanged)
STATUS = 0x01 (unchanged)
CLRF

Clear f

Syntax: \[ label \] CLRF f, a

Operands: \( 0 \leq f \leq 255 \)
\( a \in [0,1] \)

Operation: \( 00h \rightarrow f \)
\( 1 \rightarrow Z \)

Status Affected: Z

Encoding: 0110 101a ffff ffff

Description: The contents of Register 'f' are cleared and the Z bit is set.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1, the bank specified by the BSR Register is used.
If 'a' is 0, the access bank is used.

Words: 1

Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write Register 'f'</td>
</tr>
</tbody>
</table>

Example 1

CLRF FLAG_REG, 1 ; Clear Register FLAG_REG

Before Instruction

FLAG_REG = 0x5A
Z = x

After Instruction

FLAG_REG = 0x00
Z = 1

Example 2

CLRF INDF0, 1 ; Clear the register pointed to by the FSR0

Before Instruction

FSR0 = 0x0C2
Contents of Address (FSR0) = 0xAA
Z = x

After Instruction

FSR0 = 0x0C2
Contents of Address (FSR0) = 0x00
Z = 1
Section 31. Instruction Set

**CLRWDT**  
Clear Watchdog Timer

**Syntax:**  
[ label ] CLRWDT

**Operands:** None

**Operation:**  
00h → WDT  
0 → WDT prescaler count,  
1 → TO  
1 → PD

**Status Affected:** TO, PD

**Encoding:**  
0000 0000 0000 0100

**Description:** CLRWDT instruction clears the Watchdog Timer. It also clears the postscaler count of the WDT. Status bits TO and PD are set.

**Words:** 1

**Cycles:** 1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>Process data</td>
<td>Clear WDT Counter</td>
</tr>
</tbody>
</table>

**Example**

CLRWDT ; Clear the Watchdog  
; Timer count value

**Before Instruction**

- WDT counter = x
- WDT postscaler count = 0
- WDT postscaler = 1:128
- TO = x
- PD = x

**After Instruction**

- WDT counter = 0x00
- WDT postscaler count = 0
- WDT postscaler = 1:128
- TO = 1
- PD = 1

**Note:** The CLRWDT instruction does not affect the assignment of the WDT postscaler.
COMF

Complement f

Syntax:  [ label]  COMF f, d, a

Operands:
0 ≤ f ≤ 255

d ∈ [0, 1]

a ∈ [0, 1]

Operation: (f°) → destination

Status Affected:
Z, N

Encoding:
0001 1lda ffff ffff

Description:
The contents of Register f are 1's complemented.

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register f.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1

COMF REG1, 0, 1 ; Complement the value in
; Register REG1 and place the
; result in the WREG Register

Case 1: Before Instruction
REG1 = 0x13
Z, N = x

After Instruction
REG1 = 0x14
WREG = 0xEC
Z = 0
N = 1

Case 2: Before Instruction
REG1 = 0xFF
Z, N = x

After Instruction
REG1 = 0xFF
WREG = 0x00
Z = 1
N = 0

Case 3: Before Instruction
REG1 = 0x00
Z, N = x

After Instruction
REG1 = 0x00
WREG = 0xFF
Z = 0
N = 1
Section 31. Instruction Set

Example 2

COMF INDF0, 1, 1 ; Complement the value in the register pointed to by the FSR0 (FSR0H:FSR0L); Register, placing the result in that register

Before Instruction
FSR0 = 0xFC2
Contents of Address (FSR0) = 0xAA ; 1010 1010
Z, N = x

After Instruction
FSR0 = 0xFC2
Contents of Address (FSR0) = 0x55 ; 0101 0101
Z = 0
N = 0

Example 3

COMF REG1, 1, 1 ; Complement the value in Register REG1 and place the result in Register REG1

Before Instruction
REG1 = 0xFF ; 1111 1111
Z, N = x

After Instruction
REG1 = 0x00 ; 0000 0000
Z = 1
N = 0
CPFSEQ

Compare f with WREG, Skip if Equal (f = WREG)

Syntax: [label] CPFSEQ f,a

Operands: 0 ≤ f ≤ 255
          a ∈ [0,1]

Operation: (f) - (WREG)
            skip if (f) = (WREG)

Status Affected: None

Encoding: 0110 001a ffff ffff

Description: Compares the contents of Register 'f' to the contents of WREG Register by performing an unsigned subtraction.

If 'f' = WREG then the fetched instruction is discarded and a NOP is executed instead, making this a two-cycle instruction.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1 (2 or 3)

Q Cycle Activity:

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by a two word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example

HERE CPFSEQ FLAG, 1 ; Compare the value in
NEQUAL GOTO PROCESS_CODE ; Register FLAG to the
EQUAL • • • the next program memory
• • • ; location if they are
equal

Case 1: Before Instruction
PC = address HERE
FLAG = 0x5A
WREG = 0x5A ; FLAG - WREG = 0x00
After Instruction
PC = address EQUAL ; The two values were
FLAG - WREG = 0x00

Case 2: Before Instruction
PC = address HERE
FLAG = 0xA5
WREG = 0x5A ; FLAG - WREG = 0x4B
After Instruction
PC = address NEQUAL ; The two values were
FLAG - WREG = 0x4B

\[ \text{PC} = \text{address HERE} \]
\[ \text{FLAG} = 0x5A\] ; FLAG - WREG = 0x00
\[ \text{PC} = \text{address EQUAL} ; \text{The two values were} \]
\[ \text{PC} = \text{address NEQUAL} ; \text{The two values were} \]
CPFSGT  Compare f with WREG, Skip if Greater Than (f > WREG)

Syntax:  \[ label \] CPFSGT \ t, a
Operands:  0 ≤ f ≤ 255
           a ∈ \{0, 1\}
Operation:  \((f) - (WREG)\)  
skip if \((f) > (WREG)\); (unsigned comparison)
Status Affected:  None
Encoding:  0110 010a  ffff  ffff
Description:  Compares the contents of data memory location \(f\) to the contents of WREG Register by performing an unsigned subtraction.
              If \(f > WREG\) then the fetched instruction is discarded and a NOP is executed instead, making this a two-cycle instruction.

The ‘a’ bit selects which bank is accessed for the operation.  
If ‘a’ is 1; the bank specified by the BSR Register is used.
If ‘a’ is 0; the access bank is used.

Words:  1
Cycles:  1 (2 or 3)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register ‘f’</td>
<td>Process data</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by a two word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example

`HERE CPFSGT FLAG, 1 ; Compare the value in`  
`NGT GOTO PROCESS_CODE ; Register FLAG to the`  
`GT ; WREG Register and skip`  
`; the next program memory`  
`; location if`  
`; FLAG > WREG`  

Case 1: Before Instruction
`PC = address HERE`  
`FLAG = 0x5A`  
`WREG = 0x5A ; FLAG - WREG = 0x00`  

Case 2: Before Instruction
`PC = address HERE`  
`FLAG = 0x5A`  
`WREG = 0x5A ; FLAG - WREG = 0x4B`  

After Instruction
`PC = address NGT ; The two values were`  
`; Equal (Not Greater Than)`  

After Instruction
`PC = address GT ; FLAG > WREG, Skip`  
`; the next instruction`
CPFSLT Compare f with WREG, Skip if Less Than (f < WREG)

Syntax: \[ label \] CPFSLT \ f, a

Operands: 0 ≤ f ≤ 255
a ∈ \{0, 1\}

Operation: (f) - (WREG); (unsigned comparison)
skip if (f) < (WREG)

Status Affected: None

Encoding: 0110 000a ffff ffff

Description: Compares the contents of data memory location \( f \) to the contents of WREG Register by performing an unsigned subtraction.
If \( f < WREG \) then the fetched instruction is discarded and an \textit{NOP} is executed instead making this a two-cycle instruction.

The \( 'a' \) bit selects which bank is accessed for the operation.
If \( 'a' = 1 \) the bank specified by the BSR Register is used.
If \( 'a' = 0 \) the access bank is used.

Words: 1
Cycles: 1 (2 or 3)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register ( f )</td>
<td>Process data</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by a two word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example

HERE  CPFSLT  FLAG, i  ; Compare the value in
NLT  GOTO  PROCESS_CODE  ; Register FLAG to the
LT  ; WREG Register and skip
  ; the next program memory
  ; location if
  ; FLAG < WREG

Case 1: Before Instruction
  PC  =  address HERE
  FLAG  =  0x5A
  WREG  =  0x5A  ; FLAG - WREG = 0x00

After Instruction
  PC  =  address NLT  ; the two values were
  ; Equal (Not less than)

Case 2: Before Instruction
  PC  =  address HERE
  FLAG  =  0x5A
  WREG  =  0xA5  ; FLAG - WREG = 0x4B

After Instruction
  PC  =  address LT  ; FLAG < WREG, Skip
  ; the next instruction
**DAW**

**Decimal Adjust WREG Register**

**Syntax:**

```
[ label] DAW
```

**Operands:** None

**Operation:**

- If \([WREG<3:0> >9]\) or \([DC = 1]\) then
  \((WREG<3:0>) + 6 \rightarrow WREG<3:0>\);
- Else
  \((WREG<3:0>) \rightarrow WREG<3:0>\);

- If \([WREG<7:4> >9]\) or \([C = 1]\) then
  \((WREG<7:4>) + 6 \rightarrow WREG<7:4>\);
- Else
  \((WREG<7:4>) \rightarrow WREG<7:4>\);

**Status Affected:** C

**Encoding:**

```
0000 0000 0000 0111
```

**Description:** DAW adjusts the eight bit value in WREG resulting from the earlier addition of two variables (each in packed BCD format) and produces a correct packed BCD result.

**Words:** 1

**Cycles:** 1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to</td>
</tr>
<tr>
<td>Register</td>
<td>data</td>
<td>REG</td>
<td></td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example: HERE DAW ; Decimal Adjust WREG

Case 1: Before Instruction
  WREG = 0x0F ; 0x0F is 15 decimal
  C = x
After Instruction
  WREG = 0x15
  C = 0

Case 2: Before Instruction
  WREG = 0x68 ; 0x68 is 104 decimal
  C = x
After Instruction
  PC = 0x04 ; Carry to indicate decimal rollover
  C = 1

Case 3: Before Instruction
  WREG = C6 ; 0xC6 is 198 decimal
  C = x
After Instruction
  PC = 98 ; Carry to indicate decimal rollover
  C = 1
DECF  Decrement f

Syntax:       [ label ] DECF f, d, a
Operands:    0 ≤ f ≤ 255
             d ∈ [0,1]   
             a ∈ [0,1]   

Operation:   (f) - 1 → destination
Status Affected:   C, DC, Z, OV, N
Encoding:    0000 01da fffe fffe

Description: Decrement the contents of Register f.

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register f.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words:       1
Cycles:       1
Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register f</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1

DECF CNT, 1, 1 ; Decrement Register CNT

Before Instruction

CNT  =  0x01
C, DC, OV, N = x
Z = 0

After Instruction

CNT  =  0x00
C = 0
DC = 0
Z = 1
OV = 0
N = 0
Section 31. Instruction Set

Example 2

DECF INDF0, 1, 1 ; Decrement the register
; pointed to by the FSR
; (FSR0H:FSR0L) Register

Before Instruction
FSR0 = 0x1C2
Contents of Address
(FSR0) = 0x01
C, DC, OV, N = x
Z = 0

After Instruction
FSR0 = 0x1C2
Contents of Address
(FSR0) = 0x00
C = 0
DC = 0
Z = 1
OV = 0
N = 0

Example 3

DECF CNT, 0, 1 ; Decrement Register CNT
; WREG is destination

Before Instruction
CNT = 0x10
WREG = x
Z = 0

After Instruction
CNT = 0x10
WREG = 0x0F
C = 0
DC = 1
Z = 0
OV = 0
N = 0
### DECFSZ

**Decrement f, Skip if 0**

**Syntax:**

\[
\text{[ label ] DECFSZ } f, d, a
\]

**Operands:**

- \(0 \leq f \leq 255\)
- \(d \in \{0, 1\}\)
- \(a \in \{0, 1\}\)

**Operation:**

\((f) - 1 \rightarrow \text{destination}; \text{skip if result = 0}\)

**Status Affected:** None

**Encoding:**

```
0010 11da ffff ffff
```

**Description:**

The contents of Register 'f' are decremented. If the result is 0, then the next instruction (fetched during the current instruction execution) is discarded and a **NOP** is executed instead, making this a 2 cycle instruction.

The 'd' bit selects the destination for the operation.
- If 'd' is 1: the result is stored back in the File Register 'f'.
- If 'd' is 0: the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1: the bank specified by the BSR Register is used.
- If 'a' is 0: the access bank is used.

**Words:** 1

**Cycles:** 1 (2 or 3)

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by 2 word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Example

HERE  DECFSZ CNT, 1, 1  ; Decrement Register CNT,
      GOTO    LOOP         ; if CNT then equals 0
      CONTINUE  ; skip the next
              ; instruction

Case 1:  Before Instruction
         PC = address HERE
         CNT = 0x01

After Instruction
         CNT = 0x00
         PC = address CONTINUE

Case 2:  Before Instruction
         PC = address HERE
         CNT = 0x02

After Instruction
         CNT = 0x01
         PC = address HERE + 2
## DCFSNZ

Decrement f, Skip if Not 0

**Syntax:**

```plaintext
[ label ] DCFSNZ f, d, a
```

**Operands:**

- `0 ≤ f ≤ 255`
- `d ∈ [0,1]`
- `a ∈ [0,1]`

**Operation:**

```plaintext
(f) - 1 → destination; skip if result ≠ 0
```

**Status Affected:** None

**Encoding:**

```
0100  11da  eeff  eeff
```

**Description:**

The contents of Register 'f' are decremented. If the result is not 0, then the next instruction (fetched during the current instruction execution) is discarded and a **NOP** is executed instead, making this a 2-cycle instruction.

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register 'f'.
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

**Words:** 1

**Cycles:** 1 (2 or 3)

### Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by 2 word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example  HERE   DCFANZ  CNT, 1, 1 ; Decrement Register CNT,   GOTO   LOOP   ; if CNT does not equal 0   CONTINUE ; skip the next instruction  

Case 1: Before Instruction  
PC = address HERE  
CNT = 0x01  
After Instruction  
CNT = 0x00  
PC = address HERE + 2  

Case 2: Before Instruction  
PC = address HERE  
CNT = 0x02  
After Instruction  
CNT = 0x01  
PC = address CONTINUE
GOTO

**Unconditional Branch**

Syntax: \[ \text{label} \] \text{GOTO} \ k

Operands: \( 0 \leq k \leq 1048575 \)

Operation: \( k \rightarrow \text{PC}<20:1> \)
\( 0 \rightarrow \text{PC}<0> \),

Status Affected: None

Encoding:

<table>
<thead>
<tr>
<th>1st word (k&lt;7:0&gt;)</th>
<th>2nd word (k&lt;19:8&gt;)</th>
</tr>
</thead>
<tbody>
<tr>
<td>1110 kkkk</td>
<td>1111 kkkk kkkk</td>
</tr>
</tbody>
</table>

Description:

GOTO allows an unconditional branch anywhere within the entire 2M byte memory range. The 20-bit immediate value \( k \) is loaded into PC<20:1>. GOTO is always a two-cycle instruction.

Words: 2

Cycles: 2

Q Cycle Activity:

1st cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal ( k&lt;7:0&gt; )</td>
<td>Process data</td>
<td>No operation</td>
</tr>
</tbody>
</table>

2nd cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example 1
HERE GOTO THERE ; Goto address THERE
After Instruction
PC = Address THERE

Example 2
HERE GOTO $-2 ; GOTO address HERE - 2
After Instruction
PC = Address HERE -2

Example 3
HERE GOTO $ ; GOTO address HERE ; (infinite loop)
After Instruction
PC = Address HERE

Example 4
HERE GOTO HERE ; GOTO address HERE ; (infinite loop)
After Instruction
PC = Address HERE
INCF

Increment f

Syntax:  [ label] INCF f, d, a

Operands:
0 ≤ f ≤ 255
d ∈ [0, 1]
a ∈ [0, 1]

Operation:  (f) + 1 → destination

Status Affected:  C, DC, Z, OV, N

Encoding:  0010 10da ffff ffff

Description:  The contents of Register f are incremented.

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register f.
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

Words:  1
Cycles:  1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register f</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1
INCF CNT, 1, 1 ; Increment Register CNT

Before Instruction
CNT = 0xFF  ; 1111 1111
C, DC, Z, OV, N = x

After Instruction
CNT = 0x00  ; 0000 0000
C = 1
DC = 1
Z = 1
OV = 0
N = 0
Section 31. Instruction Set

Example 2

INCF INDF0, 1, 1 ; Increment Register
间接地
(FSR0 (FSR0H:FSR0L) points
to address to increment)

Before Instruction
FSR0 = 0x0C2
Contents of Address
(FSR0) = 0xFF 1111 1111
C, DC, Z, OV, N = x

After Instruction
FSR0 = 0x0C2
Contents of Address
(FSR0) = 0x00 0000 0000
C = 1
DC = 1
Z = 1
OV = 0
N = 0

Example 3

INCF CNT, 0, 1 ; Increment Register CNT
把结果放在 WREG

Before Instruction
CNT = 0x10 0001 0000
WREG = x
C, DC, OV, N = x
Z = 0

After Instruction
CNT = 0x10 0001 0001
WREG = 0x11 0001 0001
C = 0
DC = 0
Z = 0
OV = 0
N = 0
INCFSZ  

Increment f, Skip if 0

Syntax: \[
\text{[label]} \quad \text{INCFSZ } f, d, a
\]

Operands: 
- \(0 \leq f \leq 255\)
- \(d \in [0, 1]\)
- \(a \in [0, 1]\)

Operation: 
\(f + 1 \rightarrow \text{destination}, \text{skip if result} = 0\)

Status Affected: None

Encoding: 
\[0011 \quad 1\text{id}a \quad f\text{ff}f \quad f\text{ff}f\]

Description: The contents of Register 'f' are incremented. If the result is 0, then the next instruction (fetched during the current instruction execution) is discarded and a NOP is executed instead, making this a 2-cycle instruction.

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register 'f'.
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1 (2 or 3)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read (\text{Register 'f'})</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by 2 word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example 1

HERE INCFSZ CNT, 1, 1 ; Increment Register CNT, NZERO GOTO LOOP ; if CNT then equals 0 ZERO ; skip the next

Case 1: Before Instruction
PC = address HERE
CNT = 0xFF
After Instruction
CNT = 0x00
PC = address ZERO

Case 2: Before Instruction
PC = address HERE
CNT = 0x00
After Instruction
CNT = 0x01
PC = address NZERO

Example 2

HERE INCFSZ CNT, 1, 0 ; Increment Register CNT, NZERO GOTO LOOP ; if CNT equals 0 ZERO ; skip the next

Case 1: Before Instruction
PC = address HERE
CNT = 0xFF ; In access bank
After Instruction
CNT = 0x00
PC = address ZERO

Case 2: Before Instruction
PC = address HERE
CNT = 0x00 ; In access bank
After Instruction
CNT = 0x01
PC = address NZERO
INFSNZ  
Increment f, Skip if Not 0

Syntax:  
\[ \text{[label]} \infsnz f, d, a \]

Operands:  
0 ≤ f ≤ 255  
d ∈ [0,1]  
a ∈ [0,1]

Operation:  
(f) + 1 → destination, skip if result ≠ 0

Status Affected:  None

Encoding:  
0100 10da ffff ffff

Description:  
The contents of Register 'f' are incremented. If the result is not 0, then the
next instruction (fetched during the current instruction execution) is dis-
carded and a NOP is executed instead, making this a 2-cycle instruction.

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register 'f'.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1 (2 or 3)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by 2 word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example 1

```
HERE INFSNZ CNT, 1, 1 ; Increment Register CNT,
ZERO GOTO LOOP ; if CNT does not equal 0
NZERO ; skip the next instruction
```

Case 1: Before Instruction
- PC = address HERE
- CNT = 0xFF

After Instruction
- CNT = 0x00
- PC = address ZERO

Case 2: Before Instruction
- PC = address HERE
- CNT = 0x00

After Instruction
- CNT = 0x01
- PC = address NZERO

Example 2

```
HERE INFSNZ CNT, 1, 0 ; Increment Register CNT,
ZERO GOTO LOOP ; if CNT does not equal 0
NZERO ; skip the next instruction
```

Case 1: Before Instruction
- PC = address HERE
- CNT = 0xFF ; In access bank

After Instruction
- CNT = 0x00
- PC = address ZERO

Case 2: Before Instruction
- PC = address HERE
- CNT = 0x00 ; In access bank

After Instruction
- CNT = 0x01
- PC = address NZERO
IORLW  Inclusive OR Literal with WREG

Syntax:  [ label] IORLW k

Operands:  0 ≤ k ≤ 255

Operation:  (WREG).OR. k → WREG

Status Affected:  Z, N

Encoding:  0000 1001 kkkk kkkk

Description:  The contents of the WREG Register is OR'd with the eight bit literal 'k'. The result is placed in the WREG Register.

Words:  1

Cycles:  1

Q Cycle Activity:

Example 1  IORLW 0x35 ; bit wise OR 35h with the WREG Register

Before Instruction
WREG = 0x9A
Z, N = x

After Instruction
WREG = 0xBF
Z = 0
N = 1

Example 2  IORLW MYREG ; bit wise OR the value of the address of Register MYREG with the WREG Register

Before Instruction
WREG = 0x9A
Address of MYREG † = 0x37
Z, N = x

After Instruction
WREG = 0xBF
Z = 0
N = 1

† MYREG is a symbol for a data memory location.
Section 31. Instruction Set

Example 3

IORLW HIGH (LU_TABLE) ; bit wise OR the value of
; the high byte of address
; LU_TABLE with the WREG
; Register

Before Instruction
WREG = 0x9A  ; 1001 1010
Address of LU_TABLE† = 0x9375  ; 1001 0011 (93h)
† LU_TABLE is a label for an
address in program memory
Z, N = x

After Instruction
WREG = 0x9B  ; 1001 1011
Z = 0
N = 1

Example 4

IORLW 0x00 ; bit wise OR 00h with the
; WREG Register

Before Instruction
; 0000 0000 (literal)
WREG = 0x00  ; 0000 0000
Z, N = x

After Instruction
WREG = 0x00  ; 0000 0000
Z = 1
N = 0
IORWF

**Inclusive OR WREG with f**

**Syntax:**

\[
\text{[label]} \quad \text{IORWF} \quad f, d, a
\]

**Operands:**

\[
0 \leq f \leq 255 \\
d \in [0, 1] \\
a \in [0, 1]
\]

**Operation:**

\[
(WREG).OR. (f) \rightarrow \text{destination}
\]

**Status Affected:**

Z, N

**Encoding:**

\[
0001 \quad 00da \quad fffe \quad fffe
\]

**Description:**

Inclusive OR the WREG Register with the contents of Register 'f'.

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register 'f'.
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

**Words:**

1

**Cycles:**

1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

**Example 1**

\[
\text{IORWF INDF0, 1, 1} \quad ; \quad \text{Comment}
\]

**Before Instruction**

- WREG = 0x17
- FSR0 = 0xDC2
- Contents of Address (FSR0) = 0x30
- Z, N = x

**After Instruction**

- WREG = 0x17
- FSR0 = 0xDC2
- Contents of Address (FSR0) = 0x37
- Z = 0
- N = 0
Section 31. Instruction Set

Example 2
IORWF RESULT, 1, 1 ; bit wise OR the WREG Register with the Register RESULT.

Case 1: Before Instruction
RESULT = 0x13 ; 0001 0011
WREG = 0x91 ; 1001 0001
Z, N = x

After Instruction
RESULT = 0x93 ; 1001 0011
WREG = 0x91
Z = 0
N = 1

Case 2: Before Instruction
RESULT = 0x00 ; 0000 0000
WREG = 0x00 ; 0000 0000
Z, N = x

After Instruction
RESULT = 0x00 ; 0000 0000
WREG = 0x00
Z = 1
N = 0

Example 3
IORWF RESULT, 1, 0 ; bit wise OR the WREG Register with the register in the Access bank at address of RESULT Register.

Case 1: Before Instruction
RESULT = 0x13
(RRESULT in access bank) = 0xC8 ; 1100 1000
WREG = 0x91 ; 1001 0001
Z, N = x

After Instruction
RESULT = 0x13
(RRESULT in access bank) = 0xD9
WREG = 0x91
Z = 0
N = 1

Case 2: Before Instruction
RESULT = 0x00
(RRESULT in access bank) = 0x11 ; 0001 0001
WREG = 0x00 ; 0000 0000
Z, N = x

After Instruction
RESULT = 0x00
(RRESULT in access bank) = 0x11
WREG = 0x00
Z = 0
N = 0
LFSR

Load 12-bit Literal to FSR

Syntax: \[
\text{[ label] LFSR } f, k
\]

Operands: 
- \(0 \leq f \leq 2\)
- \(0 \leq k \leq 4095\)

Operation: \(k \rightarrow \text{FSRx}\)

Status Affected: None

Encoding:

<table>
<thead>
<tr>
<th>1st word</th>
<th>2nd word</th>
</tr>
</thead>
<tbody>
<tr>
<td>1110</td>
<td>1110</td>
</tr>
<tr>
<td>1111</td>
<td>0000</td>
</tr>
<tr>
<td>(k_{11})</td>
<td>(k_{07})</td>
</tr>
<tr>
<td>(k_{06})</td>
<td>(k_{05})</td>
</tr>
</tbody>
</table>

Description: The 12-bit literal 'k' is loaded into the File Select Register (FSR) pointed to by 'f':
- \(f = 00 \rightarrow \text{FSR0}\)
- \(f = 01 \rightarrow \text{FSR1}\)
- \(f = 10 \rightarrow \text{FSR2}\)
- \(f = 11 \rightarrow \text{Reserved}\)

Words: 2

Cycles: 2

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal (k_{11}):(k_{08})</td>
<td>Process data</td>
<td>Write to FSRxH Register (k_{11}):(k_{08})</td>
</tr>
<tr>
<td>Decode</td>
<td>Read literal (k_{07}):(k_{03})</td>
<td>Process data</td>
<td>Write to FSRxL Register (k_{07}):(k_{03})</td>
</tr>
</tbody>
</table>

Example 1

\[
\text{LFSR } 2, 0x123 \quad ; \text{Load the 12-bit FSR2 with } 123h
\]

Before Instruction
- FSR0H = 0x05
- FSR0L = 0xA5
- FSR1H = 0x05
- FSR1L = 0xA5
- FSR2H = 0x05
- FSR2L = 0xA5

After Instruction
- FSR0H = 0x05
- FSR0L = 0xA5
- FSR1H = 0x05
- FSR1L = 0xA5
- FSR2H = 0x01
- FSR2L = 0x23
Section 31. Instruction Set

Example 2
LFSR 0, 0xFE3 ; Load the 12-bit FSR0 with FE3h

Before Instruction
FSR0H = 0x05
FSR0L = 0xA5
FSR1H = 0x05
FSR1L = 0xA5
FSR2H = 0x05
FSR2L = 0xA5

After Instruction
FSR0H = 0x0F
FSR0L = 0xE3
FSR1H = 0x05
FSR1L = 0xA5
FSR2H = 0x05
FSR2L = 0xA5

Example 3
LFSR 1, 0xFE3 ; Load the 12-bit FSR1 with FE3h

Before Instruction
FSR0H = 0x05
FSR0L = 0xA5
FSR1H = 0x05
FSR1L = 0xA5
FSR2H = 0x05
FSR2L = 0xA5

After Instruction
FSR0H = 0x05
FSR0L = 0xA5
FSR1H = 0x0F
FSR1L = 0xE3
FSR2H = 0x05
FSR2L = 0xA5
MOVF

Move f

Syntax:  [ label]  MOVF  f, d, a
Operands:  0 ≤ f ≤ 255
           d ∈ [0,1]
           a ∈ [0,1]
Operation:  f → destination
Status Affected:  Z, N
Encoding:  0101 00da ffff ffff
Description:  The contents of Register 'f' is moved to a destination dependent upon the status of the 'd' and 'a' bits.

The 'd' bit selects the destination for the operation.
  If 'd' is 1; the result is stored back in the File Register 'f'.
  If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
  If 'a' is 1; the bank specified by the BSR Register is used.
  If 'a' is 0; the access bank is used.

Words:  1
Cycles:  1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'k'</td>
<td>Process data</td>
<td>Write to WREG Register</td>
</tr>
</tbody>
</table>

Example 1  MOVF MYREG, 0, 1 ; Copy the contents of
            Register MYREG to the WREG
            Register

Before Instruction
    MYREG = 0x22
    WREG = 0xFF
    Z, N = x

After Instruction
    MYREG = 0x22
    WREG = 0x22
    Z = 0
    N = 0

Example 2  MOVF MYREG, 1, 1 ; Copy the contents of
            Register MYREG to itself
            (affects the status bits)

Before Instruction
    MYREG = 0x00
    WREG = 0x10
    Z, N = x

After Instruction
    MYREG = 0x00
    WREG = 0x10
    Z = 1
    N = 0
Section 31. Instruction Set

Example 3

```
MOVEF MYREG, 1, 0 ; Copy the contents of
                   ; Register MYREG in the
                   ; access bank to itself
                   ; (affects the status bits)
```

<table>
<thead>
<tr>
<th></th>
<th>Case 1: Before Instruction</th>
<th>Case 2: Before Instruction</th>
</tr>
</thead>
<tbody>
<tr>
<td>MYREG</td>
<td>0x00</td>
<td>0x80</td>
</tr>
<tr>
<td>WREG</td>
<td>0x10</td>
<td>0x10</td>
</tr>
<tr>
<td>Z, N</td>
<td>x</td>
<td>x</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th></th>
<th>After Instruction</th>
<th>After Instruction</th>
</tr>
</thead>
<tbody>
<tr>
<td>MYREG</td>
<td>0x00</td>
<td>0x80</td>
</tr>
<tr>
<td>WREG</td>
<td>0x10</td>
<td>0x10</td>
</tr>
<tr>
<td>Z</td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td>N</td>
<td>0</td>
<td>1</td>
</tr>
</tbody>
</table>

Note: Access bank indicates the register bank in which the registers are located.
MOVFF Move f to f

Syntax: \[ label \] MOVFF f_s, f_d

Operands: \( 0 \leq f_s \leq 4095 \)
\( 0 \leq f_d \leq 4095 \)

Operation: \((f_s) \rightarrow f_d\)

Status Affected: None

Encoding:

<table>
<thead>
<tr>
<th>1st word (source)</th>
<th>2nd word (destination)</th>
</tr>
</thead>
<tbody>
<tr>
<td>1100</td>
<td>1111</td>
</tr>
<tr>
<td>ffff</td>
<td>ffff</td>
</tr>
<tr>
<td>ffff</td>
<td>ffff</td>
</tr>
<tr>
<td>ffff</td>
<td>ffff</td>
</tr>
<tr>
<td>ffff</td>
<td>ffff</td>
</tr>
</tbody>
</table>

Description: The contents of Source Register 'f_s' are moved to Destination Register 'f_d'. Location of source 'f_s' can be anywhere in the 4096 byte data space (000h to FFFh), and location of destination 'f_d' can also be anywhere from 000h to FFFh.

MOVFF is particularly useful for transferring a data memory location to a Peripheral Register (such as the transmit buffer or an I/O port) without affecting the WREG Register.

Note: The MOVFF instruction cannot use the PCL, TOSU, TOSH, and TOSL as the Destination Register.

Words: 2
Cycles: 2

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Source Register f</td>
<td>Process data</td>
<td>No operation</td>
</tr>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>No operation</td>
<td>Write to destination Register f</td>
</tr>
</tbody>
</table>

Example 1

MOVFF REG1, REG2 ; Copy the contents of Register REG1 to Register REG2

Before Instruction
REG1 = 0x33
REG2 = 0x11

After Instruction
REG1 = 0x33
REG2 = 0x33

Example 2

MOVFF REG2, REG1 ; Copy the contents of Register REG2 to Register REG1

Before Instruction
REG1 = 0x33
REG2 = 0x11

After Instruction
REG1 = 0x11
REG2 = 0x11
Section 31. Instruction Set

**MOVLB**  
Move Literal to low nibble in BSR

**Syntax:** \[ label \] MOV LB k

**Operands:** 0 ≤ k ≤ 15

**Operation:** k → BSR<3:0>

**Status Affected:** None

**Encoding:**
0000 0001 0000 kkkk

**Description:** The 4-bit literal ‘k’ is loaded into the Bank Select Register (BSR).

**Words:** 1

**Cycles:** 1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal ‘k’</td>
<td>Process data</td>
<td>Write literal ‘k’ to BSR</td>
</tr>
</tbody>
</table>

**Example 1**

MOVLB 5 ; Modify Least Significant nibble of BSR Register to value 5

Before Instruction
BSR = 0x02

After Instruction
BSR = 0x05

**Example 2**

MOVLB 9 ; Modify Least Significant nibble of BSR Register to value 9

Before Instruction
BSR = 0xF

After Instruction
BSR = 0x9
MOVLW  Move Literal to WREG

Syntax: \[ label \] MOVLW \( \text{k} \)

Operands: \( 0 \leq \text{k} \leq 255 \)

Operation: \( \text{k} \rightarrow \text{WREG} \)

Status Affected: None

Encoding: \( \begin{array}{c|c|c|c|c} 0000 & 1110 & \text{kkkk} & \text{kkkk} \end{array} \)

Description: The eight bit literal \( \text{k} \) is loaded into WREG Register.

Words: 1

Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to</td>
</tr>
<tr>
<td>literal ( \text{k} )</td>
<td>data</td>
<td></td>
<td>WREG</td>
</tr>
</tbody>
</table>

Example 1

\[ \text{MOVLW 0x5A} ; \text{Load the WREG Register} \]
\[ ; \text{with the value 5Ah} \]

Before Instruction

\[ \text{WREG} = x \]

After Instruction

\[ \text{WREG} = 0x5A \]

Example 2

\[ \text{MOVLW MYREG} ; \text{Load the WREG Register} \]
\[ ; \text{with the value of the} \]
\[ ; \text{address of MYREG} \]

Before Instruction

\[ \text{WREG} = 0x10 \]
\[ \text{Address of MYREG} \dagger = 0x37 \]
\[ \dagger \text{MYREG is a symbol for a data memory location} \]

After Instruction

\[ \text{WREG} = 0x37 \]

Example 3

\[ \text{MOVLW HIGH (LU_TABLE)} ; \text{Load the WREG Register} \]
\[ ; \text{with the value of the high} \]
\[ ; \text{byte of address LU_TABLE} \]

Before Instruction

\[ \text{WREG} = 0x10 \]
\[ \text{Address of LU_TABLE} \dagger = 0x9375 \]
\[ \dagger \text{LU_TABLE is a label for an address in program memory} \]

After Instruction

\[ \text{WREG} = 0x93 \]
Section 31. Instruction Set

MOVWF Move WREG to f

Syntax: \[ label\] MOVWF f, a

Operands: \(0 \leq f \leq 255\)
\(a \in [0,1]\)

Operation: \((WREG) \rightarrow f\)

Status Affected: None

Encoding: \[0110\ 111a\ \text{fff}f\ \text{fff}\]

Description: Move data from WREG Register to Register 'f'.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

Example 1

MOVWF OPTION_REG, 1 ; Copy the value in the WREG Register to the OPTION_REG Register

Before Instruction

\(\text{OPTION\_REG} = 0xFF\)
\(\text{WREG} = 0x4F\)

After Instruction

\(\text{OPTION\_REG} = 0x4F\)
\(\text{WREG} = 0x4F\)

Example 2

MOVWF INDF0, 1 ; Copy the value in the WREG Register to the FSR0 (FSR0H:FSR0L) Register

Before Instruction

\(\text{WREG} = 0x17\)
\(\text{FSR0} = 0x5C2\)
Contents of Address (FSR0) = 0x00

After Instruction

\(\text{WREG} = 0x17\)
\(\text{FSR0} = 0x5C2\)
Contents of Address (FSR0) = 0x17
MULLW Multiply Literal with WREG

Syntax: \[ \text{[label]} \text{MULLW } k \]

Operands: \[ 0 \leq k \leq 255 \]

Operation: \((\text{WREG}) \times k \rightarrow \text{PRODH:PRODL}\)

Status Affected: None

Encoding: 0000 1101 kkkk kkkk

Description: An unsigned multiplication is carried out between the contents of WREG and the 8-bit literal 'k'. The 16-bit result is placed in PRODH:PRODL Register Pair. PRODH contains the high byte.

WREG is unchanged.

None of the status flags are affected.

Neither an overflow nor carry is possible in this operation. A zero result is possible but not detected.

Words: 1

Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'k'</td>
<td>Process data</td>
<td>Write Registers PRODH:PRODL</td>
</tr>
</tbody>
</table>

Example 1

\[ \text{MULLW } 0xC4 \]

; Multiply the WREG Register with the constant value C4h

Before Instruction

\[
\begin{align*}
\text{WREG} &= 0xE2 \\
\text{PRODH} &= \_x \_x \\
\text{PRODL} &= \_x \_x
\end{align*}
\]

After Instruction

\[
\begin{align*}
\text{WREG} &= 0xE2 \\
\text{PRODH} &= 0xAD \\
\text{PRODL} &= 0x08
\end{align*}
\]

Example 2

\[ \text{MULLW } \text{FACTOR} \]

; Multiply the WREG Register with the constant value FACTOR

Before Instruction

\[
\begin{align*}
\text{FACTOR} &= 0xC4 \\
\text{WREG} &= 0xE2 \\
\text{PRODH} &= \_x \_x \\
\text{PRODL} &= \_x \_x
\end{align*}
\]

After Instruction

\[
\begin{align*}
\text{WREG} &= 0xE2 \\
\text{PRODH} &= 0xAD \\
\text{PRODL} &= 0x08
\end{align*}
\]
Section 31. Instruction Set

MULWF Multiply WREG with f

Syntax: [label] MULWF f, a

Operands: 0 ≤ f ≤ 255
           a ∈ [0,1]

Operation: (WREG) x (f) → PRODH:PRODL

Status Affected: None

Encoding: 0000 001a ffff ffff

Description: An unsigned multiplication is carried out between the contents of WREG and the value in Register File Location 'f'. The 16-bit result is placed in the PRODH:PRODL Register Pair. PRODH contains the high byte.
Both WREG and 'f' are unchanged.
None of the status flags are affected.
Neither an overflow nor carry is possible in this operation. A zero result is possible but not detected.
The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

Q1 Q2 Q3 Q4
Decode Read Process Write
Register 'f' data Registers PRODH:

Example: MULWF MYREG, 1 ; Multiple the WREG
          ; Register with the value
          ; in MYREG Register

Before Instruction
WREG = 0xE2
MYREG = 0xB5
PRODH = x
PRODL = x

After Instruction
WREG = 0xE2
MYREG = 0xB5
PRODH = 0x9F
PRODL = 0xCA
NEGF

Negate f

Syntax:  \[
\text{[label]} \text{NEGF} \ f, a
\]

Operands: \[0 \leq f \leq 255 \]
\[a \in [0,1] \]

Operation: \[(f) + 1 \rightarrow (f)\]

Status Affected: C, DC, Z, OV, N

Encoding: \[0110 \ 110a \ ffff \ ffff\]

Description: Location ‘f’ is negated using two’s complement. The result is placed in the data memory location ‘f’.

The ‘a’ bit selects which bank is accessed for the operation.

If ‘a’ is 1; the bank specified by the BSR Register is used.
If ‘a’ is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register f</td>
<td>Process data</td>
<td>Write Register f</td>
</tr>
</tbody>
</table>

Example 1

NEGF MYREG, 1  ; 2’s complement the value in MYREG

Case 1: Before Instruction

MYREG = 0x3A  ; 0011 1010
C, DC, Z, OV, N = x

After Instruction

MYREG = 0xC6  ; 1100 0110
C = 0
DC = 0
Z = 0
OV = 0
N = 1

Case 2: Before Instruction

MYREG = 0xB0  ; 1011 0000
C, DC, Z, OV, N = x

After Instruction

MYREG = 0x50  ; 0101 0000
C = 0
DC = 1
Z = 0
OV = 0
N = 0

Case 3: Before Instruction

MYREG = 0x00  ; 0000 0000
C, DC, Z, OV, N = x

After Instruction

MYREG = 0x00  ; 0000 0000
C = 1
DC = 1
Z = 1
OV = 0
N = 0
Section 31. Instruction Set

NOP

No Operation

Syntax: [ label ] NOP
Operands: None
Operation: No operation
Status Affected: None
Encoding:

<table>
<thead>
<tr>
<th>Default</th>
<th>Used with 2 word instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>0000 0000 0000 0000</td>
<td>1111 xxxx xxxx xxxx</td>
</tr>
</tbody>
</table>

Description: No operation.
Words: 1
Cycles: 1
Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example

HERE NOP ; This instruction cycle
; does nothing
; (No Operation)

Before Instruction
PC = address HERE

After Instruction
PC = address HERE + 2
## POP

**POP Top of Return Stack**

**Syntax:**

\[
\text{[ label]} \text{ POP}
\]

**Operands:** None

**Operation:** (TOS) → bit bucket

**Status Affected:** None

**Encoding:**

\[
0000 0000 0000 0110
\]

**Description:**

The Top of Stack (TOS) value is pulled off the return stack and is discarded. The TOS value then becomes the previous value that was pushed onto the return stack.

This instruction is provided to enable the user to manage the return stack to incorporate a software stack.

**Words:** 1

**Cycles:** 1

### Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>No operation</td>
<td>POP TOS value</td>
</tr>
</tbody>
</table>

**Example**

HERE POP ; Modify the Top of Stack (TOS). The TOS points to what was one level down

**Before Instruction**

- TOS = 0x0031A2
- Stack (1 level down) = 0x014332

**After Instruction**

- TOS = 0x014332
- PC = HERE + 2

---

PIC18C Reference Manual

© 2000 Microchip Technology Inc.
Section 31. Instruction Set

### PUSH

PUSH Top of Return Stack

<table>
<thead>
<tr>
<th>Syntax:</th>
<th>[ label ] PUSH</th>
</tr>
</thead>
<tbody>
<tr>
<td>Operands:</td>
<td>None</td>
</tr>
<tr>
<td>Operation:</td>
<td>PC → (TOS)</td>
</tr>
<tr>
<td>Status Affected:</td>
<td>None</td>
</tr>
<tr>
<td>Encoding:</td>
<td>0000 0000 0000 0101</td>
</tr>
<tr>
<td>Description:</td>
<td>The previous Top of Stack (TOS) value is pushed down on the stack. The PC is pushed onto the top of the return stack. This instruction is provided to enable the user to manage the return stack to incorporate a software stack.</td>
</tr>
<tr>
<td>Words:</td>
<td>1</td>
</tr>
<tr>
<td>Cycles:</td>
<td>1</td>
</tr>
<tr>
<td>Q Cycle Activity:</td>
<td></td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>PUSH PC onto return stack</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example

HERE PUSH ; PUSH current Program ; Counter value onto the ; hardware stack

Before Instruction

PC = 0x000124
TOS = 0x00345A

After Instruction

PC = 0x000126
TOS = 0x000124
Stack (1 level down) = 0x00345A
**RCALL** Relative Call

Syntax: \[ label \] RCALL \( n \)

Operands: \(-1024 \leq n \leq 1023\)

Operation: \((PC + 2) \rightarrow TOS, (PC + 2) + 2n \rightarrow PC\)

Status Affected: None

Encoding: \(1101 1nnn nnnn nnnn\)

Description: Subroutine call with a jump up to 1K from the current location. First, the return address \((PC+2)\) is pushed onto the stack. Then the 2's complement number \(2n\) is added to the PC. Since the PC will have incremented to fetch the next instruction, the new address will be \(PC+2+2n\). This instruction is a two-cycle instruction.

Words: 1

Cycles: 2

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

```lang
HERE RCALL Sub1 ; Call a program memory
; location (Sub1)
; this location must be
; < 1024 locations forward or
; > 1025 locations backward
```

Before Instruction

- PC = Address (HERE)
- TOS = 0x0031A2

After Instruction

- PC = Address (Sub1)
- TOS = Address (HERE + 2)
- Stack (1 level down) = 0x0031A2

Example 2

```lang
HERE RCALL $ + OFFSET ; Call to HERE+OFFSET
PLUS0 *
PLUS1 *
PLUS2 *
PLUS3 *
PLUS4 *
PLUS5 *
PLUS6 *
```

Before Instruction

- PC = address HERE

After Instruction

- PC = address HERE + OFFSET
Section 31. Instruction Set

Example 3

```
MIN6  ·
MIN5  ·
MIN4  ·
MIN3  ·
MIN2  ·
MIN1  ·
MIN0  · ; Call to HERE-OFFSET
HERE  RCALL $ - OFFSET
NEXT
Before Instruction
PC = address  HERE
After Instruction
PC = address  HERE - OFFSET
```
## RESET

### Reset Device

<table>
<thead>
<tr>
<th>Syntax:</th>
<th>[label] RESET</th>
</tr>
</thead>
<tbody>
<tr>
<td>Operands:</td>
<td>None</td>
</tr>
<tr>
<td>Operation:</td>
<td>Force all registers and flag bits that are affected by a MCLR reset to their reset condition.</td>
</tr>
<tr>
<td>Status Affected:</td>
<td>All</td>
</tr>
<tr>
<td>Encoding:</td>
<td>0000 0000 1111 1111</td>
</tr>
<tr>
<td>Description:</td>
<td>This instruction provides a way to execute a software reset.</td>
</tr>
<tr>
<td>Words:</td>
<td>1</td>
</tr>
<tr>
<td>Cycles:</td>
<td>1</td>
</tr>
</tbody>
</table>

### Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>No operation</td>
<td>Start reset</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example

HERE  RESET

; Do a software reset

Before Instruction
PC = address HERE
C, DC, Z, OV, N = x

After Instruction
PC = 0x000000
SFRs = See reset section
GPRs = u (unchanged)
RETIE  Return from Interrupt

Syntax:  
\[
\text{[label]} \quad \text{RETIE} \quad s
\]

Operands:  
\[s \in [0,1]\]

Operation:  
\((\text{TOS}) \rightarrow \text{PC}, \text{if IPEN} = 0 \) (compatibility mode)
\[1 \rightarrow \text{GIE}\]
\(\text{if IPEN} = 1\)
\((\text{GIEH} \ \text{GEIL})\)
\[1 \ 1 \ 1 \rightarrow \text{Invalid}\]
\[1 \ 0 \ 1 \rightarrow \text{GIEL}\]
\[0 \ 1 \ 1 \rightarrow \text{GIEH}\]
\[0 \ 0 \ 1 \rightarrow \text{GIEH}\]
\(\text{if } s = 1\)
\((\text{WREGS}) \rightarrow \text{WREG}\)
\((\text{STATUS}) \rightarrow \text{STATUS}\)
\((\text{BSRS}) \rightarrow \text{BSR}\)
\(\text{if } s = 0\)
\((\text{WREGS}) \rightarrow \text{unchanged}\)
\((\text{STATUS}) \rightarrow \text{unchanged}\)
\((\text{BSRS}) \rightarrow \text{unchanged}\)

In both cases PCLATU, PCLATH are unchanged.

Status Affected:  
\(\text{GIE/GIEH, PEIE/GIEL}\)

Encoding:  
\[0000 \ 0000 \ 0001 \ 000s\]

Description:  
Return from Interrupt. Stack is popped and Top of Stack (TOS) is loaded into the PC. Interrupts are enabled by setting either the high or low priority global interrupt enable bits (GIEH or GIEL).

If \(s = 1\), the contents of the Shadow Registers WREGS, STATUS and BSRS are loaded into their corresponding registers, WREG, STATUS and BSR.

If \(s = 0\), no update of these registers occurs (default).

Words:  
1

Cycles:  
2

Q Cycle Activity:

1st cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>Set GIEH or GIEL</td>
<td>POP PC from stack</td>
</tr>
</tbody>
</table>

2nd cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

Example 1

HERE RETFIE 0 ; Return from interrupt, ; enable interrupts

Before Instruction
PC = address HERE
GIE/GIEH, PEIE/GIEL = x
WREG = x
BSR = x
STATUS = x

After Instruction
PC = TOS
GIE/GIEH, PEIE/GIEL = 1
WREG = unchanged
BSR = unchanged
STATUS = unchanged

Example 2

HERE RETFIE 1 ; Return from interrupt, ; enable interrupts.
; This is a fast return so ; the BSR, WREG, and STATUS ; Registers are restored ; with the values in the ; Fast Register Stack

Before Instruction
PC = address HERE
GIE/GIEH, PEIE/GIEL = x
WREG = x
BSR = x
STATUS = x

After Instruction
PC = TOS
GIE/GIEH, PEIE/GIEL = 1
WREG = WREGS
BSR = BSRS
STATUS = STATUSS
RETLW Return with Literal in W

Syntax: \([\text{label}]\) RETLW \(k\)
Operands: \(0 \leq k \leq 255\)
Operation: \(k \rightarrow \text{WREG};\)
\(\text{TOS} \rightarrow \text{PC}\)
\(\text{PCLA TU and PCLA TH are unchanged}\)
Status Affected: None
Encoding: \(0000\ 1100\ kkkk\ kkkk\)

Description: The \(\text{WREG}\) Register is loaded with the eight bit literal \(k\). The program counter is loaded from the Top of Stack (the return address). The upper and high address latches (PCLA TU:PCLA TH) remain unchanged. This is a two-cycle instruction.

Words: 1
Cycles: 2
Q Cycle Activity:
1st cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal (k)</td>
<td>Process data</td>
<td>POP PC from stack, write to WREG</td>
</tr>
</tbody>
</table>

2nd cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>
Section 31. Instruction Set

### Example

```c
HERE CALL TABLE ; WREG contains table offset
    ; value WREG now has table
    ; value

TABLE ADDWF PC ; WREG = offset
RETLW k1 ; Begin table,
RETLW k2 ; Return with constant in WREG

RETLW kn ; End of table
```

<table>
<thead>
<tr>
<th>Before Instruction</th>
<th>After Instruction</th>
</tr>
</thead>
<tbody>
<tr>
<td>WREG = x</td>
<td>WREG = value of k</td>
</tr>
<tr>
<td>PC = TOS = Address</td>
<td>HERE + 2</td>
</tr>
</tbody>
</table>

© 2000 Microchip Technology Inc.
RETURN

Return from Subroutine

Syntax:  \[ label \] RETURN \ s

Operands:  \ s \in [0,1]

Operation:  (TOS) \rightarrow \text{PC}

- \text{if } s = 1: (\text{WREGS}) \rightarrow \text{WREG}
  (\text{STATUS}) \rightarrow \text{STATUS}
  (\text{BSRS}) \rightarrow \text{BSR}

- \text{if } s = 0: (\text{WREGS}) \rightarrow \text{unchanged}
  (\text{STATUS}) \rightarrow \text{unchanged}
  (\text{BSRS}) \rightarrow \text{unchanged}

In both cases PCLATU and PCLATH are unchanged

Status Affected:  \text{None}

Encoding:  \begin{array}{cccc}
0000 & 0000 & 0001 & 001s
\end{array}

Description:  Return from subroutine. The stack is popped and the Top of Stack (TOS) is loaded into the program counter.
If \( s = 1 \), the contents of the Shadow Registers WREGS, STATUS and BSRS are loaded into their corresponding registers, WREG, STATUS, and BSR.
If \( s = 0 \), no update of these registers occurs (default).

Words:  1
Cycles:  2

Q Cycle Activity:

1st cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>Process data</td>
<td>POP PC from stack</td>
</tr>
</tbody>
</table>

2nd cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1

\begin{verbatim}
HERE RETURN 0 ; Return from subroutine.
\end{verbatim}

Before Instruction

- \text{PC} = \text{address HERE}
- \text{WREG} = x
- \text{BSR} = x
- \text{STATUS} = x

After Instruction

- \text{PC} = \text{TOS}
- \text{WREG} = \text{unchanged}
- \text{BSR} = \text{unchanged}
- \text{STATUS} = \text{unchanged}
Section 31. Instruction Set

Example 2

```
HERE  RETURN 1 ; Return from subroutine.
; This is a fast return so
; the BSR, WREG, and STATUS
; Registers are restored
; with the values in the
; Fast Register Stack
```

Before Instruction

<table>
<thead>
<tr>
<th>PC</th>
<th>WREG</th>
<th>BSR</th>
<th>STATUS</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

After Instruction

<table>
<thead>
<tr>
<th>PC</th>
<th>WREG</th>
<th>BSR</th>
<th>STATUS</th>
</tr>
</thead>
<tbody>
<tr>
<td>TOS</td>
<td>WREGS</td>
<td>BSRS</td>
<td>STATUSS</td>
</tr>
</tbody>
</table>
RLCF  Rotate Left f through Carry

Syntax: [ label] R L C F f, d, a

Operands: 0 ≤ f ≤ 127
d ∈ [0,1]
a ∈ [0,1]

Operation: See description below

Status Affected: C, Z, N

Encoding: 0011 01da ffff ffff

Description: The contents of Register 'f' are rotated one bit to the left through the Carry Flag.

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register 'f'.
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1
Q Cycle Activity:

Example 1

```
RLCF REG1, 0, 1 ; Rotate the value in REG1
; 1 bit position left and
; the carry bit loads into
; bit 0. Then place the
; result in the WREG
; Register
```

Before Instruction

REG1 = 1110 0110
C = 0
Z, N = x

After Instruction

REG1 = 1110 0110
WREG = 1100 1100
C = 1
Z = 0
N = 1
Section 31. Instruction Set

Example 2

RLF INDF0, 1, 1 ; Rotate the value in the
; register pointed by the
; FSR0 (FSR0H:FSR0L)
; Register 1 bit position
; left and place the result
; back into that register
; Carry loads into bit 0

Case 1: Before Instruction
FSR0 = 0x0C2
Contents of Address
(FSR0) = 0011 1010
C = 1
Z, N = x

After Instruction
FSR0 = 0x0C2
Contents of Address
(FSR0) = 0111 0101
C = 0
Z = 0
N = 0

Case 2: Before Instruction
FSR0 = 0x0C2
Contents of Address
(FSR0) = 1011 1001
C = 0
Z, N = x

After Instruction
FSR0 = 0x0C2
Contents of Address
(FSR0) = 0111 0010
C = 1
Z = 0
N = 0
RLNCF  
**Rotate Left f (No Carry)**

Syntax:  

```
[label]  RLNCF  f, d, a
```

Operands:  

- \(0 \leq f \leq 127\)
- \(d \in [0,1]\)
- \(a \in [0,1]\)

Operation:  
See description below

Status Affected:  
- Z
- N

Encoding:  

```
0100 01da ffff ffff
```

Description:  
The contents of Register ‘f’ are rotated one bit to the left. The Carry Flag bit is not affected.

The ‘d’ bit selects the destination for the operation.
- If ‘d’ is 1; the result is stored back in the File Register ‘f’.
- If ‘d’ is 0; the result is stored in the WREG Register.

The ‘a’ bit selects which bank is accessed for the operation.
- If ‘a’ is 1; the bank specified by the BSR Register is used.
- If ‘a’ is 0; the access bank is used.

Words:  
1

Cycles:  
1

Q Cycle Activity:  

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1  

```
RLNCF  REG1, 0, 1 ; Rotate the value in REG1
; 1 bit position left and
; bit 7 loads into bit 0.
; Then place the result in
; the WREG Register
```

Before Instruction  

```
REG1 = 1110 0110
Z, N = x
```

After Instruction  

```
REG1 = 1110 0110
WREG = 1100 1101
Z = 0
N = 1
```
Section 31. Instruction Set

Example 2

```
RLNCF INDF0, 1, 1 ; Rotate the value in the
; register pointed by the
; FSR0 (FSR0H:FSR0L)
; Register 1 bit position left
; and place the result in the
; back into that register.
; bit 7 loads into bit 0.
```

Case 1: Before Instruction

- FSR0 = 0x1C2
- Contents of Address
  - (FSR0) = 0011 1010
- Z, N = x

After Instruction

- FSR0 = 0x1C2
- Contents of Address
  - (FSR0) = 0111 0100
- Z = 0
- N = 0

Case 2: Before Instruction

- FSR0 = 0x1C2
- Contents of Address
  - (FSR0) = 1011 1001
- Z, N = x

After Instruction

- FSR0 = 0x1C2
- Contents of Address
  - (FSR0) = 0111 0011
- Z = 0
- N = 0
## RRCF

### Rotate Right f through Carry

**Syntax:**

\[ \text{[label]} \quad \text{RRCF} \quad f, \quad d, \quad a \]

**Operands:**

- \(0 \leq f \leq 127\)
- \(d \in [0,1]\)
- \(a \in [0,1]\)

**Operation:**

See description below

**Status Affected:**

C, Z, N

**Encoding:**

\[
\begin{align*}
0011 & \quad 00da & \quad fffe & \quad fffe \\
\end{align*}
\]

**Description:**

The contents of Register 'f' are rotated one bit to the right through the Carry Flag.

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register 'f'.
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

**Words:**

1

**Cycles:**

1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

**Example 1**

\[ \text{RRCF REG1, 0, 1} \quad ; \text{Rotate the value in REG1} \]
\[ ; \text{1 bit position right and} \]
\[ ; \text{the carry bit loads into} \]
\[ ; \text{bit 7. Then place the} \]
\[ ; \text{result in the WREG} \]
\[ ; \text{Register} \]

**Before Instruction**

\[
\begin{align*}
\text{REG1} & = \quad 1110 \quad 0110 \\
\text{WREG} & = \quad xxxx \quad xxxx \\
\text{C} & = \quad 0 \\
\text{Z, N} & = \quad x \\
\end{align*}
\]

**After Instruction**

\[
\begin{align*}
\text{REG1} & = \quad 1110 \quad 0110 \\
\text{WREG} & = \quad 0111 \quad 0011 \\
\text{C} & = \quad 0 \\
\text{Z} & = \quad 0 \\
\text{N} & = \quad 0 \\
\end{align*}
\]
Section 31. Instruction Set

Example 2

RRCF INDF0, 1, 1  ; Rotate the value in the
; register pointed by the
; FSR0 (FSR0H:FSR0L)
; Register 1 bit position
; right and place the result
; in the back into that
; register.
; Carry loads into bit 7.

Case 1: Before Instruction
FSR0     = 0x2C2
Contents of Address
(FSR0)   = 0011 1010
C         = 1
Z, N      = x

After Instruction
FSR0     = 0x2C2
Contents of Address
(FSR0)   = 1001 1101
C         = 0
Z          = 0
N          = 1

Case 2: Before Instruction
FSR0     = 0x2C2
Contents of Address
(FSR0)   = 0011 1001
C         = 0
Z, N      = x

After Instruction
FSR0     = 0x2C2
Contents of Address
(FSR0)   = 0001 1100
C         = 1
Z          = 0
N          = 0
RRNCF  

**Rotate Right f (No Carry)**

<table>
<thead>
<tr>
<th>Syntax:</th>
<th>[label] RRNCF f, d, a</th>
</tr>
</thead>
</table>
| Operands: | 0 ≤ f ≤ 127  
|    | d ∈ [0, 1]  
|    | a ∈ [0, 1] |
| Operation: | See description below |
| Status Affected: | Z, N |
| Encoding: | 0100 00da ffff ffff |
| Description: | The contents of Register 'f' are rotated one bit to the right. The Carry Flag bit is not affected. |

The 'd' bit selects the destination for the operation.  
If 'd' is 1; the result is stored back in the File Register 'f'.  
If 'd' is 0; the result is stored in the WREG Register.  

The 'a' bit selects which bank is accessed for the operation.  
If 'a' is 1; the bank specified by the BSR Register is used.  
If 'a' is 0; the access bank is used.  

Words:  
1  
Cycles:  
1  

**Q Cycle Activity:**  
<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

**Example 1**  
RRNCF REG1, 0, 1 ; Rotate the value in REG1  
; 1 bit position right and  
; bit 0 loads into bit 7.  
; Then place the result in  
; the WREG Register  

**Before Instruction**  
REG1 = 1110 0110  
WREG = x  
Z, N = 1  

**After Instruction**  
REG1 = 1110 0110  
WREG = 0111 0011  
Z = 0  
N = 0
Example 2

RRNCF INDF0, 1, 1 ; Rotate the value in the
; register pointed by the
; FSR (FSR0H:FSR0L)
; Register 1 bit
; position right and place
; the result back into
; that register.
; bit 0 loads into bit 7.

Case 1: Before Instruction
FSR0 = 0x3C2
Contents of Address
(FSR0) = 0011 1010
Z, N = x

After Instruction
FSR0 = 0x3C2
Contents of Address
(FSR0) = 0001 1101
Z = 0
N = 0

Case 2: Before Instruction
FSR0 = 0x3C2
Contents of Address
(FSR0) = 0011 1001
Z, N = x

After Instruction
FSR0 = 0x3C2
Contents of Address
(FSR0) = 1001 1100
Z = 0
N = 1
## SETF

### Set f

**Syntax:**

\[
\text{[label]} \text{SETF } f, a
\]

**Operands:**

\[
0 \leq f \leq 255 \\
a \in [0, 1]
\]

**Operation:**

\[
\text{FFh } \rightarrow f
\]

**Status Affected:**

None

**Encoding:**

\[
0110 \ 100a \ ffff \ ffff
\]

**Description:**

The contents of the specified register are set.

The 'a' bit selects which bank is accessed for the operation.

- If 'a' is 1: the bank specified by the BSR Register is used.
- If 'a' is 0: the access bank is used.

**Words:**

1

**Cycles:**

1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write Register 'f'</td>
</tr>
</tbody>
</table>

**Example 1**

\[
\text{SETF FLAG_REG, 1 } \ ; \ \text{Set all the bits in Register FLAG_REG}
\]

**Before Instruction**

- FLAG_REG = 0x5A

**After Instruction**

- FLAG_REG = 0xFF

**Example 2**

\[
\text{SETF INDF0, 1 } \ ; \ \text{Set all the bits in the register pointed to by the FSR (FSR0H:FSR0L) Register}
\]

**Before Instruction**

- FSR0 = 0x4C2
- Contents of Address (FSR0) = 0xAA

**After Instruction**

- FSR0 = 0x4C2
- Contents of Address (FSR0) = 0xFF
Section 31. Instruction Set

**SLEEP** Enter SLEEP mode

Syntax: `[ label ] SLEEP`  
Operands: None  
Operation: 00h → WDT,  
0 → WDT prescaler count,  
1 → TO,  
0 → PD  
Status Affected: TO, PD  
Encoding: 0000 0000 0000 0011  
Description: The power-down status bit, PD is cleared. Time-out status bit, TO is set. Watchdog Timer and its prescaler count are cleared. The processor is put into SLEEP mode with the oscillator stopped.  
Words: 1  
Cycles: 1  
Q Cycle Activity:  
<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>No operation</td>
<td>Go to sleep</td>
</tr>
</tbody>
</table>

Example: 

```
SLEEP ; Turn off the device
; oscillator. This is the
; lowest power mode
```

Before Instruction

<table>
<thead>
<tr>
<th>TO</th>
<th>PD</th>
</tr>
</thead>
<tbody>
<tr>
<td>?</td>
<td>?</td>
</tr>
</tbody>
</table>

After Instruction

<table>
<thead>
<tr>
<th>TO</th>
<th>PD</th>
</tr>
</thead>
<tbody>
<tr>
<td>1†</td>
<td>0†</td>
</tr>
</tbody>
</table>

† If WDT causes wake-up, this bit is cleared

**Note:** The SLEEP instruction does not affect the assignment of the WDT prescaler.
SUBFWB Subtract f from WREG with borrow

Syntax: [ label] SUBFWB f, d, a

Operands: 0 ≤ f ≤ 255
d ∈ [0, 1]
a ∈ [0, 1]

Operation: (WREG) − (f) − (C) → destination

Status Affected: C, DC, Z, OV, N

Encoding: 0101 01da ffff ffff

Description: Subtract Register 'f' and carry flag (borrow) from W (2's complement method).

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register 'f'.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1
SUBFWB MYREG, 1, 1 ; WREG - MYREG - borrow bit

Before Instruction
MYREG = 0x37
WREG = 0x10
C, DC, Z, OV, N = x
C = 0

After Instruction
MYREG = 0xA8
WREG = 0x10 ; result is negative
C = 0
DC = 0
Z = 0
OV = 0
N = 1
Section 31. Instruction Set

Example 2: \texttt{SUBFWB MYREG, 1, 1} ; WREG – MYREG – borrow bit

Case 1: Before Instruction
\begin{align*}
\text{MYREG} &= 0x03 \\
\text{WREG} &= 0x02 \\
C &= 1 \\
\text{DC}, \text{Z}, \text{OV}, \text{N} &= x
\end{align*}

After Instruction
\begin{align*}
\text{MYREG} &= 0xFF \\
\text{WREG} &= 0x02 \\
C &= 0 \\
\text{DC} &= 0 \\
Z &= 0 \\
\text{OV} &= 0 \\
N &= 1 ; \text{result is negative}
\end{align*}

Case 2: Before Instruction
\begin{align*}
\text{MYREG} &= 0x02 \\
\text{WREG} &= 0x02 \\
C &= 1 \\
\text{DC}, \text{Z}, \text{OV}, \text{N} &= x
\end{align*}

After Instruction
\begin{align*}
\text{MYREG} &= 0x00 \\
\text{WREG} &= 0x02 ; \text{result is zero} \\
C &= 1 \\
\text{DC} &= 1 \\
Z &= 1 \\
\text{OV} &= 0 \\
N &= 0
\end{align*}

Case 3: Before Instruction
\begin{align*}
\text{MYREG} &= 0x01 \\
\text{WREG} &= 0x03 \\
C &= 1 \\
\text{DC}, \text{Z}, \text{OV}, \text{N} &= x
\end{align*}

After Instruction
\begin{align*}
\text{MYREG} &= 0x02 \\
\text{WREG} &= 0x03 ; \text{result is positive} \\
C &= 1 \\
\text{DC} &= 1 \\
Z &= 0 \\
\text{OV} &= 0 \\
N &= 0
\end{align*}
SUBLW  Subtract W from Literal

Syntax: \[ \text{label} \] SUBLW k

Operands: \( 0 \leq k \leq 255 \)

Operation: \( k \cdot (\text{WREG}) \rightarrow \text{WREG} \)

Status Affected: C, DC, Z, OV, N

Encoding: \( 0000 1000 kkkk kkkk \)

Description: The \( \text{WREG} \) Register is subtracted (2's complement method) from the eight bit literal 'k'. The result is placed in the \( \text{WREG} \) Register.

Words: 1
Cycles: 1

Q Cycle Activity:

Example 1  

\begin{verbatim}
SUBLW OFFSET ; Subtract the value in WREG from the constant OFFSET

Before Instruction
\begin{verbatim}
WREG = 0x37
OFFSET = 0x10
C, DC, Z, OV, N = x
\end{verbatim}

After Instruction
\begin{verbatim}
WREG = 0xD9
C = 0 ; result is negative
DC = 0
Z = 0
OV = 0
N = 1
\end{verbatim}
\end{verbatim}
Section 31. Instruction Set

Example 2: \textbf{SUBLW 0x02 ; Subtract WREG Register from 2h}

Case 1: Before Instruction

\begin{itemize}
  \item WREG = 0x01
  \item C, DC, Z, OV, N = x
\end{itemize}

After Instruction

\begin{itemize}
  \item WREG = 0x01
  \item C = 1 \quad ; \text{result is positive}
  \item DC = 1
  \item Z = 0
  \item OV = 0
  \item N = 0
\end{itemize}

Case 2: Before Instruction

\begin{itemize}
  \item WREG = 0x02
  \item C, DC, Z, OV, N = x
\end{itemize}

After Instruction

\begin{itemize}
  \item WREG = 0x00
  \item C = 1 \quad ; \text{result is zero}
  \item DC = 1
  \item Z = 0
  \item OV = 0
  \item N = 0
\end{itemize}

Case 3: Before Instruction

\begin{itemize}
  \item WREG = 0x03
  \item C, DC, Z, OV, N = x
\end{itemize}

After Instruction

\begin{itemize}
  \item WREG = 0xFF \quad ; \text{result is negative}
  \item C = 0
  \item DC = 0
  \item Z = 0
  \item OV = 0
  \item N = 1
\end{itemize}
SUBWF Subtract W from f

Syntax: [ label] SUBWF f, d, a

Operands: $0 \leq f \leq 255$
$d \in [0,1]$
$a \in [0,1]$

Operation: $(f) - (\text{WREG}) \rightarrow \text{destination}$

Status Affected: C, DC, Z, OV, N

Encoding: 0101 11da ffff ffff

Description: Subtract (2's complement method) WREG Register from Register 'f'.

The 'd' bit selects the destination for the operation.
If 'd' is 1; the result is stored back in the File Register 'f'.
If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
If 'a' is 1; the bank specified by the BSR Register is used.
If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1: SUBWF REG1, 1, 1 ; Subtract the value in the WREG Register from REG1, placing the result in REG1

Case 1: Before Instruction
REG1 = 3
WREG = 2
C, DC, Z, OV, N = x

After Instruction
REG1 = 1
WREG = 2
C = 1 ; result is positive
DC = 1
Z = 0
OV = 0
N = 0
Section 31. Instruction Set

Case 2: Before Instruction

REG1 = 2
WREG = 2
C, DC, Z, OV, N = x

After Instruction

REG1 = 0
WREG = 2
C = 1 ; result is zero
DC = 1
Z = 1
OV = 0
N = 0

Case 3: Before Instruction

REG1 = 1
WREG = 2
C, DC, Z, OV, N = x

After Instruction

REG1 = 0xFF
WREG = 2
C = 0 ; result is negative
DC = 0
Z = 0
OV = 0
N = 1
SUBWFB

Subtract W from f with Borrow

Syntax: [label] SUBWFB f, d, a

Operands: $0 \leq f \leq 255$
$\text{d} \in [0,1]$
$\text{a} \in [0,1]$

Operation: $(f) - (\text{WREG}) - (\text{C}) \rightarrow \text{destination}$

Status Affected: C, DC, Z, OV, N

Encoding: 0101 10da ffff ffff

Description: Subtract (2’s complement method) WREG Register from Register ‘f’ with borrow.

The ‘d’ bit selects the destination for the operation.
If ‘d’ is 1; the result is stored back in the File Register ‘f’.
If ‘d’ is 0; the result is stored in the WREG Register.

The ‘a’ bit selects which bank is accessed for the operation.
If ‘a’ is 1; the bank specified by the BSR Register is used.
If ‘a’ is 0; the access bank is used.

Words: 1
Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

Example 1: SUBWF REG1, 1, 1 ; Subtract the value in the WREG Register from REG1, ; placing the result in REG1

Case 1: Before Instruction

REG1 = 3
WREG = 2
C = 1
DC, Z, OV, N = x

After Instruction

REG1 = 1
WREG = 2
C = 1 ; result is positive
DC = 1
Z = 0
OV = 0
N = 0
Section 31. Instruction Set

Case 2: Before Instruction
REG1 = 2
WREG = 2
C = 1
DC, Z, OV, N = x

After Instruction
REG1 = 0
WREG = 2
C = 1
DC = 1
Z = 1
OV = 0
N = 0
; result is zero

Case 3: Before Instruction
REG1 = 1
WREG = 2
C = 1
DC, Z, OV, N = x

After Instruction
REG1 = 0xFF
WREG = 2
C = 0
DC = 0
Z = 0
OV = 0
N = 1
; result is negative
**SWAPF**

**Swap Nibbles in f**

**Syntax:**

\[
\text{[ label]} \text{ SWAPF } f, d, a
\]

**Operands:**

- \(0 \leq f \leq 255\)
- \(d \in [0,1]\)
- \(a \in [0,1]\)

**Operation:**

\[
(f<3:0>) \rightarrow \text{destination}<7:4>,
(f<7:4>) \rightarrow \text{destination}<3:0>
\]

**Status Affected:** None

**Encoding:**

\[
0011 \text{ 10da ffff ffff}
\]

**Description:**

The upper and lower nibbles of Register 'f' are exchanged.

The 'd' bit selects the destination for the operation:
- If 'd' is 1: the result is stored back in the File Register 'f'.
- If 'd' is 0: the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation:
- If 'a' is 1: the bank specified by the BSR Register is used.
- If 'a' is 0: the access bank is used.

**Words:** 1

**Cycles:** 1

**Q Cycle Activity:**

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

**Example 1**

```
SWAPF REG1, 0, 1 ; Swap the high and low
; nibble of Register REG1
; and place the result in
; the WREG Register
```

**Before Instruction**

- REG1 = 0xA5
- WREG = x

**After Instruction**

- REG1 = 0xA5
- WREG = 0x5A
Section 31. Instruction Set

Example 2

SWAPF INDF0, 1, 1 ; Swap the high and low
  ; nibble of register pointed
  ; to by the FSR
  ; (FSR0H:FSR0L) Register,
  ; placing the result back
  ; into that register

Before Instruction
  FSR0 = 0x5C2
  Contents of Address (FSR0) = 0x20

After Instruction
  FSR0 = 0x5C2
  Contents of Address (FSR0) = 0x02

Example 3

SWAPF REG1, 1, 1 ; Swap the high and low
  ; nibble of Register REG1
  ; placing the result back
  ; into that register

Before Instruction
  REG1 = 0xA5

After Instruction
  REG1 = 0x5A
TBLRD  Table Read

Syntax:  
  [ label ]  TBLRD[*, *+, *-, or +*]

Operands:  
  0 ≤ m ≤ 3

Operation:  
  if TBLRD *,
    (Prog Mem (TBLPTR)) → TABLAT;
    TBLPTR - No Change;
  if TBLRD *+,
    (Prog Mem (TBLPTR)) → TABLAT;
    (TBLPTR) +1 → TBLPTR;
  if TBLRD *,
    (Prog Mem (TBLPTR)) → TABLAT;
    (TBLPTR) -1 → TBLPTR;
  if TBLRD +*,
    (TBLPTR) +1 → TBLPTR;
    (Prog Mem (TBLPTR)) → TABLAT;

Status Affected:  None

Encoding:  
  0000 0000 0000 10mm *
  →  m m=0 0
  *+  →  m m=0 1
  *-  →  m m=1 0
  +*  →  m m=1 1

Description:  This instruction is used to read the contents of Program Memory. To
  address the program memory a pointer called Table Pointer (TBLPTR) is
  used.

  The TBLPTR (a 21-bit pointer) points to each byte in the program memory.
  TBLPTR has a 2 Mbyte address range. The LSb of the TBLPTR selects
  which byte of the program memory location to access.

  TBLPTR[0] = 0: Least Significant byte of Program Memory Word
  TBLPTR[0] = 1: Most Significant byte of Program Memory Word

  The Program Memory word address is the same as the TBLPTR address,
  except that the LSb of TBLPTR (TBLPTR[0]) is always forced to '0'.

  The TBLRD instruction can modify the value of TBLPTR as follows:

  - no change
  - post-increment
  - post-decrement
  - pre-increment

Words: 1
Cycles: 2

Q Cycle Activity:
1st cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

2nd cycle:

<p>| | | | |</p>
<table>
<thead>
<tr>
<th></th>
<th></th>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
<tr>
<td>(Table Pointer on Address bus)</td>
<td>(OE goes low)</td>
<td>(OE goes low)</td>
<td>(OE goes low)</td>
</tr>
</tbody>
</table>
### Section 31. Instruction Set

#### Example 31-3: Program Memory Contents for Examples

<table>
<thead>
<tr>
<th>Word Address (points to low byte)</th>
<th>Data Word</th>
<th>Word Address (points to low byte)</th>
<th>Data Word</th>
</tr>
</thead>
<tbody>
<tr>
<td>0x00A356</td>
<td>0x12 0x34</td>
<td>0x000000</td>
<td>0x01 0x00</td>
</tr>
<tr>
<td>0x00A358</td>
<td>0x56 0x28</td>
<td>0x000002</td>
<td>0x03 0x02</td>
</tr>
<tr>
<td>0x00A35A</td>
<td>0xAA 0x55</td>
<td>0x000004</td>
<td>0x05 0x04</td>
</tr>
<tr>
<td>0x00A35C</td>
<td>0xFF 0xFE</td>
<td>0x000006</td>
<td>0x07 0x06</td>
</tr>
<tr>
<td>0x00A35E</td>
<td>0xB1 0x00</td>
<td>0x000008</td>
<td>0x08 0x07</td>
</tr>
</tbody>
</table>

#### Example 1

**TBLRD**<sup>++</sup>  
; Read byte addressed by  
; TBLPTR, then increment  
; TBLPTR

**Before Instruction**

- TABLAT = x
- TBLPTR = 0x00A356
- Contents of Address (TBLPTR) = 0x34

**After Instruction**

- TABLAT = 0x34
- TBLPTR = 0x00A357

#### Example 2

**TBLRD**<sup>++</sup>  
; Increment TBLPTR, then  
; Read byte addressed by  
; TBLPTR

**Before Instruction**

- TABLAT = x
- TBLPTR = 0x00A357
- Contents of Address (TBLPTR) = 0x12
- Contents of Address (TBLPTR + 1) = 0x28

**After Instruction**

- TABLAT = 0x28
- TBLPTR = 0x00A358

#### Example 3

**TBLRD**<sup>-</sup>  
; Read byte addressed by  
; TBLPTR, then decrement  
; TBLPTR

**Before Instruction**

- TABLAT = x
- TBLPTR = 0x00A357
- Contents of Address (TBLPTR) = 0x12

**After Instruction**

- TABLAT = 0x12
- TBLPTR = 0x00A356

#### Example 4

**TBLRD**<sup>*</sup>  
; Read byte addressed by  
; TBLPTR, TBLPTR is unchanged

**Before Instruction**

- TABLAT = x
- TBLPTR = 0x00A357
- Contents of Address (TBLPTR) = 0x12

**After Instruction**

- TABLAT = 0x12
- TBLPTR = 0x00A357
TBLWT

Table Write

Syntax: \[ label \] TBLWT\[*, +, -, +\]

Operands: \[0 \leq m \leq 3\]

Operation:
- if TBLWT*, (TABLAT) \(\rightarrow\) Prog Mem (TBLPTR) or Holding Register; TBLPTR - No Change;
- if TBLWT*+, (TABLAT) \(\rightarrow\) Prog Mem (TBLPTR) or Holding Register;
- (TBLPTR) +1 \(\rightarrow\) TBLPTR;
- if TBLWT*-,
- (TABLAT) \(\rightarrow\) Prog Mem (TBLPTR) or Holding Register;
- (TBLPTR) -1 \(\rightarrow\) TBLPTR;
- if TBLWT+*,
- (TBLPTR) +1 \(\rightarrow\) TBLPTR;
- (TABLAT) \(\rightarrow\) Prog Mem (TBLPTR) or Holding Register;

Note 1: The use of a Holding Register(s) is device specific. Please refer to the Device Data Sheet for information on the operation of the TBLWT instruction with the Program Memory.

Status Affected: None

Encoding:

<table>
<thead>
<tr>
<th>m</th>
<th>mm</th>
<th>Encoding</th>
</tr>
</thead>
</table>
| 0 | 00 | 0000 0000 0000 11mm *
| 1 | 01 | +* \(\rightarrow\) mm = 01
| 1 | 10 | -* \(\rightarrow\) mm = 10
| 1 | 11 | ++ \(\rightarrow\) mm = 11

Description:
This instruction is used to program the contents of Program Memory. To address the program memory a pointer called Table Pointer (TBLPTR) is used.

The TBLPTR (a 21-bit pointer) points to each byte in the program memory. TBLPTR has a 2 MByte address range. The LSb of the TBLPTR selects which byte of the program memory location to access.

TBLPTR[0] = 0: Least Significant byte of Program Memory Word
TBLPTR[0] = 1: Most Significant byte of Program Memory Word

The Program Memory word address is the same as the TBLPTR address, except that the LSb of TBLPTR (TBLPTR[0]) is always forced to '0'.

The TBLWT instruction can modify the value of TBLPTR as follows:
- no change
- post-increment
- post-decrement
- pre-increment

Words: 1

Cycles: 2 (many if long write to internal program memory)

Q Cycle Activity:

1st cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

2nd cycle:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation (Table Pointer on Address bus)</td>
<td>No operation</td>
<td>No operation (Table Latch on Address bus, WR goes low)</td>
</tr>
</tbody>
</table>

© 2000 Microchip Technology Inc.
Section 31. Instruction Set

Example 31-4: Program Memory Contents for Examples

<table>
<thead>
<tr>
<th>Word Address (points to low byte)</th>
<th>Original Data Word (before)</th>
<th>Example 1 Data Word (after)</th>
<th>Example 2 Data Word (after)</th>
<th>Example 3 Data Word (after)</th>
<th>Example 4 Data Word (after)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>M</td>
<td>Sb</td>
<td>Lsb</td>
<td>M</td>
<td>Sb</td>
</tr>
<tr>
<td>0x00A356</td>
<td>0x12</td>
<td>0x34</td>
<td>0x12</td>
<td>0x55</td>
<td>0x12</td>
</tr>
<tr>
<td>0x00A358</td>
<td>0x56</td>
<td>0x28</td>
<td>0x56</td>
<td>0xDD</td>
<td>0x56</td>
</tr>
<tr>
<td>0x00A35A</td>
<td>0xAA</td>
<td>0x55</td>
<td>0xAA</td>
<td>0x55</td>
<td>0xAA</td>
</tr>
</tbody>
</table>

Example 1

TBLWT**+ ; Write byte addressed by TBLPTR, then increment TBLPTR

Before Instruction

<table>
<thead>
<tr>
<th>TBLAT</th>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0xAA</td>
<td>0x00A356</td>
<td>0x34</td>
</tr>
</tbody>
</table>

After Instruction

<table>
<thead>
<tr>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0x00A357</td>
<td>0x55</td>
</tr>
</tbody>
</table>

Example 2

TBLWT**+ ; Increment TBLPTR, then Write byte addressed by TBLPTR

Before Instruction

<table>
<thead>
<tr>
<th>TBLAT</th>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
<th>Contents of (TBLPTR + 1)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0xAA</td>
<td>0x00A357</td>
<td>0x12</td>
<td>0x28</td>
</tr>
</tbody>
</table>

After Instruction

<table>
<thead>
<tr>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
<th>Contents of (TBLPTR + 1)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0x00A358</td>
<td>0x12</td>
<td>0xAA</td>
</tr>
</tbody>
</table>

Example 3

TBLWT**- ; Write byte addressed by TBLPTR, then decrement TBLPTR

Before Instruction

<table>
<thead>
<tr>
<th>TBLAT</th>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0xAA</td>
<td>0x00A357</td>
<td>0x12</td>
</tr>
</tbody>
</table>

After Instruction

<table>
<thead>
<tr>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0x00A356</td>
<td>0xAA</td>
</tr>
</tbody>
</table>

Example 4

TBLWT* ; Write byte addressed by TBLPTR. TBLPTR is unchanged

Before Instruction

<table>
<thead>
<tr>
<th>TBLAT</th>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0x5A</td>
<td>0x00A359</td>
<td>0x56</td>
</tr>
</tbody>
</table>

After Instruction

<table>
<thead>
<tr>
<th>TBLPTR</th>
<th>Contents of (TBLPTR)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0x00A359</td>
<td>0x5A</td>
</tr>
</tbody>
</table>
TSTFSZ  Test f, Skip if 0

Syntax:  \[ \text{label} \] TSTFSZ f, a

Operands: 0 ≤ f ≤ 255
a ∈ [0,1]

Operation: Skip if (f) = 0

Status Affected: None

Encoding:

0110 011a ffff ffff

Description: If Register 'f' = 0, the next instruction fetched is discarded and a NOP is executed (two NOPs if the fetched instruction is a two-cycle instruction).

The 'a' bit selects which bank is accessed for the operation.

- If 'a' is 1: the bank specified by the BSR Register is used.
- If 'a' is 0: the access bank is used.

Words: 1
Cycles: 1 (2 or 3)

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read Register 'f'</td>
<td>Process data</td>
<td>Write to destination</td>
</tr>
</tbody>
</table>

If skip (2nd cycle):

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

If skip and followed by a two word instruction:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Example 1A

HERE  TSTFSZ REG1, 1  ; If Register REG1 is zero
NZERO  ; then skip the next
ZERO   ; program memory address

Before Instruction

REG1 = 0xAF
PC = Address (HERE)

After Instruction

PC = Address (NZERO)
Section 31. Instruction Set

Example 2

HERE TSTFSZ REG1, 1 ; If Register REG1 is zero
NZERO • ; then skip the next
ZERO • ; program memory address

Before Instruction
REG1 = 0x00
PC = Address (HERE)

After Instruction
PC = Address (ZERO)

Example 3

HERE TSTFSZ REG1, 0 ; If Register REG1 is zero
NZERO • ; then skip the next
ZERO • ; program memory address

Case 1

Before Instruction
REG1 = 0xAF
Address of REG1 = 0x9A, Bank 3
0x9A, Bank 15 = 0x00
PC = Address (HERE)

After Instruction
PC = Address (ZERO)

Case 2

Before Instruction
REG1 = 0x00
Address of REG1 = 0x9A, Bank 3
0x9A, Bank 15 = 0xAF
PC = Address (HERE)

After Instruction
PC = Address (NZERO)

Example 4

HERE TSTFSZ INDF0, 1 ; If Register pointed to by
NZERO • ; FSR0 (FSR0H:FSR0L) is
ZERO • ; zero, then skip the next
; program memory address

Case 1

Before Instruction
FSR0 = 0x6C2
Contents of Address (FSR0) = 0xAF
PC = Address (HERE)

After Instruction
FSR0 = 0x6C2
Contents of Address (FSR0) = 0xAF
PC = Address (NZERO)

Case 2

Before Instruction
FSR0 = 0x6C2
Contents of Address (FSR0) = 0x00
PC = Address (HERE)

After Instruction
FSR0 = 0x6C2
Contents of Address (FSR0) = 0x00
PC = Address (ZERO)
XORLW

Exclusive OR Literal with W

Syntax: \([\text{label}]\ \text{XORLW} \ k\)

Operands: \(0 \leq k \leq 255\)

Operation: \((WREG).\text{XOR.} \ k \rightarrow W\)

Status Affected: \(Z, N\)

Encoding:

```
0000 1010 kkkk kkkk
```

Description: The contents of the WREG Register are XOR'ed with the eight bit literal 'k'. The result is placed in the WREG Register.

Words: 1

Cycles: 1

Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read literal 'k'</td>
<td>Process data</td>
<td>Write to WREG Register</td>
</tr>
</tbody>
</table>

Example 1

XORLW 0xAF ; Exclusive OR the value in WREG with AFh

Before Instruction

\[
\begin{align*}
\text{WREG} & = 0xB5 \quad ; 1011 0101 \quad (0xB5) \\
\text{Z, N} & = x \quad ; 1010 1111 \quad (0xAF)
\end{align*}
\]

After Instruction

\[
\begin{align*}
\text{WREG} & = 0x1A \quad ; 0001 1010 \quad (0x1A) \\
\text{Z} & = 0 \\
\text{N} & = 0
\end{align*}
\]

Example 2

XORLW MYREG ; Exclusive OR the value in WREG with the address of MYREG

Before Instruction

\[
\begin{align*}
\text{WREG} & = 0xAF \quad ; 1010 1111 \\
\text{Address of MYREG} & = 0x37 \quad ; 0011 0111 \\
\text{Z, N} & = x \quad ; ----- ----
\end{align*}
\]

\(\dagger\) MYREG is a symbol for a data memory location

After Instruction

\[
\begin{align*}
\text{WREG} & = 0x98 \\
\text{Z} & = 0 \\
\text{N} & = 1
\end{align*}
\]
Section 31. Instruction Set

Example 3

XORLW HIGH (LU_TABLE) ; Exclusive OR the value
; in WREG with the high
; byte of the address of
; LU_TABLE

Before Instruction

WREG = 0xAF ; 1010 1111
Address of LU_TABLE † ; 1001 0011
= 0x9375 ; 1001 0011
Z, N = x ; ---- ----
† LU_TABLE is a label for an
address in program memory

After Instruction

WREG = 0x3C ; 0011 1100
Z = 0
N = 0
XORWF Exclusive OR W with f

Syntax: \[
\begin{align*}
&\text{label} \quad \text{XORWF} \quad f, d, a \\
\end{align*}
\]
Operands: 
- \(0 \leq f \leq 255\)
- \(d \in [0,1]\)
- \(a \in [0,1]\)
Operation: \((\text{WREG}) \times \text{OR} \times \text{ Register } f\) → destination
Status Affected: \(Z, N\)
Encoding: \[
\begin{array}{cccc}
0001 & 10da & fffe & fffe
\end{array}
\]
Description: Exclusive OR the contents of the WREG Register with Register \(f\).

The 'd' bit selects the destination for the operation.
- If 'd' is 1; the result is stored back in the File Register \(f\).
- If 'd' is 0; the result is stored in the WREG Register.

The 'a' bit selects which bank is accessed for the operation.
- If 'a' is 1; the bank specified by the BSR Register is used.
- If 'a' is 0; the access bank is used.

Words: 1
Cycles: 1
Q Cycle Activity:

<table>
<thead>
<tr>
<th>Q1</th>
<th>Q2</th>
<th>Q3</th>
<th>Q4</th>
</tr>
</thead>
<tbody>
<tr>
<td>Decode</td>
<td>Read</td>
<td>Process</td>
<td>Write</td>
</tr>
<tr>
<td>Register (f)</td>
<td>data</td>
<td>to destination</td>
<td></td>
</tr>
</tbody>
</table>

Example 1

XORWF REG1, 1, 1 ; Exclusive OR the value
; in WREG with the value in
; REG1

Before Instruction

\[
\begin{align*}
\text{REG1} &= 0xAF \quad ; \quad 1010\ 1111 \quad (0xAF) \\
\text{WREG} &= 0xB5 \quad ; \quad 1011\ 0101 \quad (0xB5) \\
Z, N &= x \quad ; \quad --------- \quad ------
\end{align*}
\]

After Instruction

\[
\begin{align*}
\text{REG1} &= 0x1A \quad ; \quad 0001\ 1010 \quad (0x1A) \\
\text{WREG} &= 0xB5 \\
Z &= 0 \\
N &= 0
\end{align*}
\]
Section 31. Instruction Set

Example 2

XORWF REG1, 0, 1 ; Exclusive OR the value
; in WREG with the value in
; REG1. Place result in
; WREG

Before Instruction

REG1 = 0xAF ; 1010 1111 (0xAF)
WREG = 0xB5 ; 1011 0101 (0xB5)
Z, N = x ; --------- ------

After Instruction

REG1 = 0xAF ;
WREG = 0x1A ; 0001 1010 (0x1A)
Z = 0
N = 0

Example 3

XORWF INDF0, 1, 1 ; Exclusive OR the value
; in WREG with the value
; pointed to by the FSR0
; (FSR0H:FSR0L) Register

Before Instruction

WREG = 0xB5 ; 1011 0101
FSR0 = 0x7C2 ;
Contents of Address
(FSR0) = 0xAF ; 1010 1111
Z, N = x ;

After Instruction

WREG = 0xB5 ;
FSR0 = 0x7C2 ;
Contents of Address
(FSR0) = 0x1A ; 0001 1010
Z = 0
N = 0

Example 4

XORWF REG1, 1, 0 ; Exclusive OR the value
; in WREG with the value
; at address REG1 in the
; access bank. Place
; result in access bank

Before Instruction

WREG = 0xF8 ; 1111 1000
REG1 = 0x01 ; 0000 0001
Contents of Address
(REG1) in access bank
= 0xAA ; 1010 1010
Z, N = x ;

After Instruction

WREG = 0xF8 ;
REG1 = 0x01 ;
Contents of Address
(REG1) in access bank
= 0x52 ; 0101 0010
Z = 0
N = 0
31.8 Design Tips

Question 1: I have seen references to “Read-Modify-Write” instructions in your data sheet, but I do not know what that is. Can you explain what it is and why I need to know this?

Answer 1:
An easy example of a Read-Modify-Write (or R-M-W) instruction is the bit clear instruction BCF. You might think that the processor just clears the bit, which on a port output pin would clear the pin. What actually happens is the whole port (or register) is first read, THEN the bit is cleared, then the new modified value is written back to the port (or register). Actually, any instruction that depends on a value currently in the register is going to be a Read-Modify-Write instruction. This includes ADDWF, SUBWF, BCF, BSF, INCW, XORWF, etc... Instructions that do not depend on the current register value, like MOVWF, CLRF, and so on are not R-M-W instructions.

One situation where you would want to consider the affects of a R-M-W instruction is a port that is continuously changed from input to output and back. For example, say you have TRISB set to all outputs, and write all ones to the PORTB Register, all of the PORTB pins will go high. Now, say you turn pin RB3 into an input, which happens to go low. A BCF PORTB, 6 is then executed to drive pin RB6 low. If you then turn RB3 back into an output, it will now drive low, even though the last value you put there was a one. What happened was that the BCF of the other pin (RB6) caused the whole port to be read, including the zero on RB3 when it was an input. Then, bit 6 was changed as requested, but since RB3 was read as a zero, zero will also be placed back into that port latch, overwriting the one that was there before. When the pin is turned back into an output, the new value was reflected. Try using the LATx register instead of the PORTx register for this read-modify-write operation.

Question 2: When I perform a BCF, other pins get cleared in the port. Why?

Answer 2:
There are a few possibilities, two are:
1. Another case where a R-M-W instruction may seem to change other pin values unexpectedly can be illustrated as follows: Suppose you make PORTC all outputs and drive the pins low. On each of the port pins is an LED connected to ground, such that a high output lights it. Across each LED is a 100 µF capacitor. Let’s also suppose that the processor is running very fast, say 20 MHz. Now if you go down the port setting each pin in order; BSF PORTC, 0 then BSF PORTC, 1 then BSF PORTC, 2 and so on, you may see that only the last pin was set, and only the last LED actually turns on. This is because the capacitors take a while to charge. As each pin was set, the pin before it was not charged yet and so was read as a zero. This zero is written back out to the port latch (R-M-W, remember) which clears the bit you just tried to set the instruction before. This is usually only a concern at high speeds and for successive port operations, but it can happen, so take it into consideration.
2. If this is on a PIC16C7X device, you may not have configured the I/O pins properly in the ADCON1 Register. If a pin is configured for analog input, any read of that pin will read a zero, regardless of the voltage on the pin. This is an exception to the normal rule that the pin state is always read. You can still configure an analog pin as an output in the TRIS Register, and drive the pin high or low by writing to it, but you will always read a zero. Therefore, if you execute a Read-Modify-Write instruction (see previous question), all analog pins are read as zero, and those not directly modified by the instruction will be written back to the port latch as zero. A pin configured as analog is expected to have values that may be neither high nor low to a digital pin, or floating. Floating inputs on digital pins are a no-no, and can lead to high current draw in the input buffer, so the input buffer is disabled.
## Section 31. Instruction Set

### 31.9 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced family (that is they may be written for the Baseline, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the Instruction Set are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related Application Notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

**Note:** Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
31.10 Revision History

Revision A

This is the initial released revision of the Enhanced MCU Instruction Set description.
# Section 32. Electrical Specifications

## HIGHLIGHTS

- **32.1 Introduction**
- **32.2 Absolute Maximums**
- **32.3 Voltage vs Frequency Graph**
- **32.4 Device Voltage Specifications**
- **32.5 Device Current Specifications**
- **32.6 Input Threshold Levels**
- **32.7 I/O Current Specifications**
- **32.8 Output Drive Levels**
- **32.9 I/O Capacitive Loading**
- **32.10 Low Voltage Detect (LVD)**
- **32.11 EPROM/FLASH/Data EEPROM**
- **32.12 Comparators and Voltage Reference**
- **32.13 Timing Parameter Symbology**
- **32.14 Example External Clock Timing Waveforms and Requirements**
- **32.15 Example Phase Lock Loop (PLL) Timing Waveforms and Requirements**
- **32.16 Example Power-up and RESET Timing Waveforms and Requirements**
- **32.17 Example Timer0 and Timer1 Timing Waveforms and Requirements**
- **32.18 Example CCP Timing Waveforms and Requirements**
- **32.19 Example Parallel Slave Port (PSP) Timing Waveforms and Requirements**
- **32.20 Example SSP and Master SSP SPI Mode Timing Waveforms and Requirements**
- **32.21 Example SSP I²C Mode Timing Waveforms and Requirements**
- **32.22 Example Master SSP I²C Mode Timing Waveforms and Requirements**
- **32.23 Example USART/SCI Timing Waveforms and Requirements**
- **32.24 CAN Specifications**
- **32.25 Example 8-bit A/D Timing Waveforms and Requirements**
- **32.26 Example 10-bit A/D Timing Waveforms and Requirements**
- **32.27 Design Tips**
- **32.28 Related Application Notes**
- **32.29 Revision History**
32.1 Introduction

This section is intended to present the electrical specifications that may be specified in a particular device data sheet and their meaning. This section is NOT intended to give the values of these specifications. For the device specific values you must refer to the device’s data sheet. All values shown in this section should be considered as Example Values.

In the description of the device and the functional modules (previous sections), there have been references to electrical specification parameters. These references have been hyperlinked in the electronic version to aid in the use of this manual.

Note: Before starting any design, Microchip HIGHLY recommends that you acquire the most recent copy of the device data sheet and review the electrical specifications to ensure that they will meet your requirements.

Throughout this section, certain terms will be used. Table 32-1 shows the conventions that will be used.

Table 32-1: Term Conventions

<table>
<thead>
<tr>
<th>Term</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PIC18CXXX</td>
<td>For EPROM Program Memory devices tested to standard voltage range</td>
</tr>
<tr>
<td>PIC18LCXXX</td>
<td>For EPROM Program Memory devices tested to extended voltage range</td>
</tr>
<tr>
<td>PIC18FXXX</td>
<td>For FLASH Program Memory devices tested to standard voltage range</td>
</tr>
<tr>
<td>PIC18LFXXX</td>
<td>For FLASH Program Memory devices tested to extended voltage range</td>
</tr>
<tr>
<td>PIC18CRXXX</td>
<td>For ROM Program Memory devices tested to standard voltage range</td>
</tr>
<tr>
<td>PIC18LCRXXX</td>
<td>For ROM Program Memory devices tested to extended voltage range</td>
</tr>
<tr>
<td>LP osc</td>
<td>For devices configured with the LP device oscillator selected</td>
</tr>
<tr>
<td>XT osc</td>
<td>For devices configured with the XT device oscillator selected</td>
</tr>
<tr>
<td>HS osc</td>
<td>For devices configured with the HS device oscillator selected</td>
</tr>
<tr>
<td>HS+PLL osc</td>
<td>For devices configured with the HS+PLL device oscillator selected</td>
</tr>
<tr>
<td>RC osc</td>
<td>For devices configured with the RC device oscillator selected</td>
</tr>
<tr>
<td>RCIO osc</td>
<td>For devices configured with the RCIO device oscillator selected</td>
</tr>
<tr>
<td>EC osc</td>
<td>For devices configured with the EC device oscillator selected</td>
</tr>
<tr>
<td>ECIO osc</td>
<td>For devices configured with the ECIO device oscillator selected</td>
</tr>
<tr>
<td>Commercial</td>
<td>For devices with the commercial temperature range grading (0°C ≤ TA ≤ +70°C)</td>
</tr>
<tr>
<td>Industrial</td>
<td>For devices with the industrial temperature range grading (-40°C ≤ TA ≤ +85°C)</td>
</tr>
<tr>
<td>Extended</td>
<td>For devices with the extended temperature range grading (-40°C ≤ TA ≤ +125°C)</td>
</tr>
</tbody>
</table>

Note 1: In Electrical Specification examples, we will use PIC18CXXX for the standard voltage range devices and PIC18LCXXX for the extended voltage range devices.
Section 32. Electrical Specifications

32.2 Absolute Maximums

The Absolute Maximum Ratings specify the worst case conditions that can be applied to the device. These ratings are not meant as operational specifications. Stresses above the listed values may cause damage to the device. Specifications are not always stand-alone, that is, the specification may have other requirements as well.

An example of this is the "maximum current sourced/sunk by any I/O pin". The number of I/O pins that can be sinking/sourcing current, at any one time, is dependent upon the maximum current sunk/source by the port(s) (combined) and the maximum current into the VDD pin or out of the VSS pin. In this example, the physical reason is the Power and Ground bus width to the I/O ports and internal logic. If these specifications are exceeded, then electromigration may occur on these Power and Ground buses. Over time, electromigration would cause these buses to open (be disconnected from the pin) and, therefore, cause the logic attached to these buses to stop operating. So, exceeding the absolute specifications may cause device reliability issues.

Input Clamp Current is defined as the current through the diode to VSS/VDD if pin voltage exceeds specification.

Example Absolute Maximum Ratings †

<table>
<thead>
<tr>
<th>Specification</th>
<th>Value</th>
</tr>
</thead>
<tbody>
<tr>
<td>Ambient temperature under bias</td>
<td>-55 to +125 °C</td>
</tr>
<tr>
<td>Storage temperature</td>
<td>-65°C to +150°C</td>
</tr>
<tr>
<td>Voltage on any pin with respect to VSS (except VDD, MCLR, and RA4)</td>
<td>-0.3V to (VDD - 0.3V)</td>
</tr>
<tr>
<td>Voltage on VDD with respect to VSS</td>
<td>-0.3V to +7.5V</td>
</tr>
<tr>
<td>Voltage on MCLR with respect to VSS (2)</td>
<td>0 to +13.25V</td>
</tr>
<tr>
<td>Voltage on RA4 with respect to VSS</td>
<td>0 to +8.5V</td>
</tr>
<tr>
<td>Total power dissipation (1)</td>
<td>1.0W</td>
</tr>
<tr>
<td>Maximum current out of VSS pin</td>
<td>300 mA</td>
</tr>
<tr>
<td>Maximum current into VDD pin</td>
<td>250 mA</td>
</tr>
<tr>
<td>Input clamp current, Ii (VI &lt; 0 or VI &gt; VDD)</td>
<td>± 20 mA</td>
</tr>
<tr>
<td>Output clamp current, IiC (VO &lt; 0 or VO &gt; VDD)</td>
<td>± 20 mA</td>
</tr>
<tr>
<td>Maximum output current sunk by any I/O pin</td>
<td>25 mA</td>
</tr>
<tr>
<td>Maximum output current sourced by any I/O pin</td>
<td>25 mA</td>
</tr>
<tr>
<td>Maximum current sunk by PORTA, PORTB, and PORTE (combined)</td>
<td>200 mA</td>
</tr>
<tr>
<td>Maximum current sourced by PORTA, PORTB, and PORTE (combined)</td>
<td>200 mA</td>
</tr>
<tr>
<td>Maximum current sunk by PORTC and PORTD (combined)</td>
<td>200 mA</td>
</tr>
<tr>
<td>Maximum current sourced by PORTC and PORTD (combined)</td>
<td>200 mA</td>
</tr>
<tr>
<td>Maximum current sunk by PORTF and PORTG (combined)</td>
<td>100 mA</td>
</tr>
<tr>
<td>Maximum current sourced by PORTF and PORTG (combined)</td>
<td>100 mA</td>
</tr>
</tbody>
</table>

Note 1: Power dissipation is calculated as follows:

\[ P_{dis} = VDD \times \left( \frac{I_{DD} - \sum I_{OH}}{2} \right) + \sum \left( \frac{(VDD - VOH) \times I_{OH}}{2} + \sum (VOH \times IOL) \right) \]

Note 2: Voltage spikes below VSS at the MCLR/VPP pin, inducing currents greater than 80 mA, may cause latch-up. Thus, a series resistor of 50-100Ω should be used when applying a "low" level to the MCLR/VPP pin, rather than pulling this pin directly to VSS.

† NOTICE: Stresses above those listed under "Absolute Maximum Ratings" may cause permanent damage to the device. This is a stress rating only and functional operation of the device at those or any other conditions above those indicated in the operation listings of this specification is not implied. Exposure to maximum rating conditions for extended periods may affect device reliability.
32.3 Voltage vs Frequency Graph

Windowed devices are superset devices with all oscillator configurations tested to the specification ranges of the C and LC devices. The temperature range that the device is tested to should be considered commercial, though at a later time, they may be tested to industrial or extended temperature levels.

Figure 32-1 and Figure 32-2 show proposed voltage vs frequency graphs for the C and LC devices.

**Note:** Devices that are designated Engineering Sample are tested to the current engineering test program at time of the device testing. There is no implied warranty that these devices have been tested to any or all specifications in the Device Data Sheet.

Battery applications usually require an extended voltage range. Devices marked LC have an extended voltage range.
Section 32. Electrical Specifications

The voltage vs frequency graphs show what is the maximum frequency of operation for a given voltage. Figure 32-1 is for a \textit{C} device, while Figure 32-2 is for an \textit{LC} device. Notice that for Figure 32-2, there is a slope from 6MHz to 40 MHz. An equation is given in the figure which will allow you to calculate the maximum frequency of operation for a given voltage.

**Figure 32-1: Example PIC18CXXX Voltage Frequency Graph**

**Figure 32-2: Example PIC18LCXXX Voltage Frequency Graph**

\[ F_{\text{MAX}} = (20.0 \text{ MHz/V}) (V_{\text{DDAPPMIN}} - 2.5 \text{ V}) + 6 \text{ MHz} \]

**Note:** \( V_{\text{DDAPPMIN}} \) is the minimum voltage of the PICmicro® device in the application.
32.4 Device Voltage Specifications

These specifications relate to the device VDD, power-up, and function.

Supply Voltage is the voltage level that must be applied to the device VDD pins for the proper functional operation.

RAM Data Retention Voltage is the minimum level that the device voltage may be at and still retain the RAM’s data value.

VDD Start Voltage to ensure the internal Power-on Reset signal, is the level that VDD must start from, to ensure that the POR circuitry will operate properly.

VDD Rise Rate to ensure internal Power-on Reset signal, is the minimum slope that VDD must rise to cause the POR circuitry to trip.

Brown-out Reset Voltage is the voltage range where the brown-out circuitry may trip. When the BOR circuitry trips, the device will either be in Brown-out Reset, or has just come out of Brown-out Reset.

Table 32-2: Example DC Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ†</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D001</td>
<td>VDD</td>
<td>Supply Voltage</td>
<td>4.2</td>
<td>—</td>
<td>5.5</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td>PIC18CXXX</td>
<td></td>
<td>2.5</td>
<td>—</td>
<td>5.5</td>
<td>V</td>
<td>HS, XT, RC and LP osc mode</td>
</tr>
<tr>
<td>D002</td>
<td>VDR</td>
<td>RAM Data Retention Voltage (1)</td>
<td>1.5</td>
<td>—</td>
<td>—</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D003</td>
<td>VPOR</td>
<td>VDD Start Voltage to ensure internal Power-on Reset signal</td>
<td>—</td>
<td>—</td>
<td>0.7</td>
<td>V</td>
<td>See section on Power-on Reset for details</td>
</tr>
<tr>
<td>D004</td>
<td>SVD</td>
<td>VDD Rise Rate to ensure internal Power-on Reset signal</td>
<td>0.05</td>
<td>—</td>
<td>—</td>
<td>V/ms</td>
<td>See section on Power-on Reset for details</td>
</tr>
<tr>
<td>D005</td>
<td>VBOR</td>
<td>Brown-out Reset Voltage</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>BORV1:BORV0 = 11</td>
<td>1.8</td>
<td>—</td>
<td>1.91</td>
<td>V</td>
<td>For PIC18LCXXX VDDMIN = 1.8V</td>
</tr>
<tr>
<td></td>
<td></td>
<td>BORV1:BORV0 = 11</td>
<td>2.5</td>
<td>—</td>
<td>2.66</td>
<td>V</td>
<td>For PIC18LCXXX VDDMIN &gt; 1.8V</td>
</tr>
<tr>
<td></td>
<td></td>
<td>BORV1:BORV0 = 10</td>
<td>2.7</td>
<td>—</td>
<td>2.86</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>BORV1:BORV0 = 01</td>
<td>4.2</td>
<td>—</td>
<td>4.46</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>BORV1:BORV0 = 00</td>
<td>4.5</td>
<td>—</td>
<td>4.78</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D007</td>
<td>VBHYS</td>
<td>Brown-out Hysteresis</td>
<td>30</td>
<td>—</td>
<td>100</td>
<td>mV</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: This is the limit to which VDD can be lowered in SLEEP mode or during a device RESET without losing RAM data.
Section 32. Electrical Specifications

32.5 Device Current Specifications

$\text{IDD}$ is referred to as supply current and is the current ($I$) consumed by the device when in operating mode. This test is taken with all I/O as inputs, either pulled high or low. That is, there are no floating inputs, nor are any pins driving an output (with a load).

$\text{IPD}$ is referred to as power-down current and is the current ($I$) consumed by the device when in SLEEP mode (power-down), referred to as power-down current. These tests are taken with all I/O as inputs, either pulled high or low. That is, there are no floating inputs, nor are any pins driving an output (with a load), weak pull-ups are disabled.

A device may have certain features and modules that can operate while the device is in SLEEP mode. Some of these modules are:

- Watchdog Timer (WDT)
- Low Voltage Detect (LVD)
- Brown-out Reset (BOR) circuitry
- Timer1 Oscillator
- Analog to Digital converter
- Comparators
- Voltage Reference
- CAN Module

When all features are disabled, the device will consume the lowest possible current (the leakage current). If any of these features are operating while the device is in SLEEP, a higher current will occur. The difference in current between the lowest power mode (everything off) and only that one feature enabled (such as the WDT), is what we call the Module Differential Current. If more than one feature is enabled, then the expected current can easily be calculated as: the base current (everything disabled and in SLEEP mode) plus all Module Differential Currents (delta currents). Example 32-1 shows an example of calculating the typical currents for a device at 5V, with the WDT and Timer1 oscillator enabled.

**Example 32-1:** $\text{IPD}$ Calculations with WDT and Timer1 Oscillator Enabled (@ 5V)

<table>
<thead>
<tr>
<th>Feature</th>
<th>Delta Current</th>
</tr>
</thead>
<tbody>
<tr>
<td>Base Current</td>
<td>14 nA</td>
</tr>
<tr>
<td>WDT Delta Current</td>
<td>14 $\mu$A</td>
</tr>
<tr>
<td>TMR1 Delta Current</td>
<td>22 $\mu$A</td>
</tr>
<tr>
<td>Total SLEEP Current</td>
<td>36 $\mu$A</td>
</tr>
</tbody>
</table>

**Note:** Some modules (such as the Brown-out Reset and Low Voltage Detect) use a common resource (an internal reference voltage generator). This resource may consume a significant percentage of the total modules current when enabled. Since 2 modules are using this, the total current will be less than the calculation.
## Table 32-3: Example DC Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D010</td>
<td>IΩD</td>
<td>Supply Current(^{(2,4)})</td>
<td>—</td>
<td>—</td>
<td>4</td>
<td>µA</td>
<td>XT, RC, RCIO osc configurations, Fosc = 4 MHz, VDD = 4.2V</td>
</tr>
<tr>
<td>D010A</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>48</td>
<td>mA</td>
<td>LP osc configuration, Fosc = 32 kHz, VDD = 4.2V</td>
</tr>
<tr>
<td>D010C</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>45</td>
<td>mA</td>
<td>EC, ECIO osc configurations, Fosc = 40 MHz, VDD = 5.5V</td>
</tr>
<tr>
<td>D013</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>50</td>
<td>mA</td>
<td>HS osc configurations, Fosc = 40 MHz, VDD = 5.5V</td>
</tr>
<tr>
<td>D013</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>50</td>
<td>mA</td>
<td>HS4 osc configuration, Fosc = 10 MHz, VDD = 5.5V</td>
</tr>
<tr>
<td>D014</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>48</td>
<td>µA</td>
<td>OSCB osc configuration, Fosc = 32 kHz, VDD = 4.2V</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>Fosc = 32 kHz, VDD = 4.2V, 25°C</td>
</tr>
<tr>
<td>IPD</td>
<td>Power-down Current(^{(3)})</td>
<td>—</td>
<td>—</td>
<td>&lt;1</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>36</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, 25°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, -40°C to +125°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>42</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +125°C</td>
</tr>
</tbody>
</table>

**Legend:** Shading of rows is to assist in readability of the table.

**Note 1:** Not applicable.

2: The supply current is mainly a function of the operating voltage and frequency. Other factors, such as I/O pin loading and switching rate, oscillator type, internal code execution pattern and temperature, also have an impact on the current consumption.

The test conditions for all IΩD measurements in active operation mode are:
- OSC1 = external square wave, from rail to rail; all I/O pins set to inputs, pulled to VDD or VSS.
- MCVR = VDD; WDT enabled/disabled as specified.

3: The power-down current in SLEEP mode does not depend on the oscillator type. Power-down current is measured with the part in SLEEP mode, with all I/O pins set to inputs, tied to VDD or VSS, and all features that add delta current disabled (such as WDT, Timer1 Oscillator, ...).

4: For RC osc configuration, current through REXT is not included. The current through the resistor can be estimated by the formula \(I_r = \frac{VDD}{2 \times REXT}\) (mA) with REXT in kOhm.
### Table 32-3: Example DC Characteristics (Continued)

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D022</td>
<td>∆IWDT</td>
<td>Watchdog Timer</td>
<td>—</td>
<td>—</td>
<td>25</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +125°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, 25°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>12</td>
<td>µA</td>
<td>VDD = 2.5V</td>
</tr>
<tr>
<td>D022A</td>
<td>∆BOR</td>
<td>Brown-out Reset</td>
<td>—</td>
<td>—</td>
<td>50</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +125°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, 25°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>12</td>
<td>µA</td>
<td>VDD = 2.5V</td>
</tr>
<tr>
<td>D023B</td>
<td>∆ILVD</td>
<td>Low Voltage Detect</td>
<td>—</td>
<td>—</td>
<td>50</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +125°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, 25°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>50</td>
<td>µA</td>
<td>VDD = 2.5V</td>
</tr>
<tr>
<td>D022C</td>
<td>∆IDDC</td>
<td>A/D Converter Current</td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +125°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, 25°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 2.5V</td>
</tr>
<tr>
<td>D025</td>
<td>∆IOSC8</td>
<td>Timer1 Oscillator</td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +85°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 5.5V, -40°C to +125°C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>TBD</td>
<td>µA</td>
<td>VDD = 4.2V, 25°C</td>
</tr>
</tbody>
</table>

Legend: Shading of rows is to assist in readability of the table.

**Note 1:** Not applicable.

2: The supply current is mainly a function of the operating voltage and frequency. Other factors, such as I/O pin loading and switching rate, oscillator type, internal code execution pattern and temperature, also have an impact on the current consumption.

The test conditions for all I/DD measurements in active operation mode are:
- OSC1 = external square wave, from rail to rail; all I/O pins set to inputs, pulled to VDD
- MCLR = VDD; WDT enabled/disabled as specified.

3: The power-down current in SLEEP mode does not depend on the oscillator type. Power-down current is measured with the part in SLEEP mode, with all I/O pins set to inputs, tied to VDD or VSS, and all features that add delta current disabled (such as WDT, Timer1 Oscillator, ...).

4: For RC osc configuration, current through REXT is not included. The current through the resistor can be estimated by the formula $I_r = \frac{V_{DD}}{2R_{EXT}}$ (mA) with REXT in kΩ.
32.6 Input Threshold Levels

The Input Low Voltage ($V_{IL}$) is the maximum voltage level that will be read as a logic '0'. An input may not read a '0' at a voltage level above this. All designs should be to the specification, since device to device (and to a much lesser extent pin to pin) variations will cause this level to vary.

The Input High Voltage ($V_{IH}$) is the minimum voltage level that will be read as a logic '1'. An input may not read a '1' at a voltage level below this. All designs should be to the specification, since device to device (and to a much lesser extent pin to pin) variations will cause this level to vary.

The I/O pins with TTL levels are shown with two specifications. One is the industry standard TTL specification, which is specified for the voltage range of 4.5V to 5.5V. The other specifies operation over the entire voltage range of the device. The better of these two specifications may be used in the design (see Note 2 in Table 32-4).

Table 32-4: Example DC Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>VIL</td>
<td>I/O ports: with TTL buffer</td>
<td>Vss</td>
<td>0.15VDD</td>
<td>V</td>
<td>For entire VDD range (2)</td>
<td></td>
</tr>
<tr>
<td>D030</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D030A</td>
<td></td>
<td></td>
<td></td>
<td>0.8</td>
<td>V</td>
<td>4.5V ≤ VDD ≤ 5.5V (2)</td>
</tr>
<tr>
<td>D031</td>
<td>with Schmitt Trigger buffer</td>
<td>Vss</td>
<td>0.2VDD</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>RC3 and RC4</td>
<td></td>
<td></td>
<td></td>
<td>0.3VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D032</td>
<td>MCLR</td>
<td></td>
<td>Vss</td>
<td>0.2VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D032A</td>
<td></td>
<td></td>
<td></td>
<td>2.0</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D033</td>
<td></td>
<td></td>
<td>Vss</td>
<td>0.3VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>VHH</td>
<td>Input High Voltage I/O ports: with TTL buffer</td>
<td>0.25VDD + 0.8V</td>
<td>VDD</td>
<td>V</td>
<td>For entire VDD range (2)</td>
<td></td>
</tr>
<tr>
<td>D040</td>
<td></td>
<td></td>
<td></td>
<td>2.0</td>
<td>V</td>
<td>4.5V ≤ VDD ≤ 5.5V (2)</td>
</tr>
<tr>
<td>D040A</td>
<td></td>
<td></td>
<td></td>
<td>0.8VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>RC3 and RC4</td>
<td></td>
<td></td>
<td></td>
<td>0.7VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D042</td>
<td>MCLR</td>
<td></td>
<td>0.8VDD</td>
<td>VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D042A</td>
<td></td>
<td></td>
<td></td>
<td>0.7VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D043</td>
<td>OSC1 (RC mode) (1)</td>
<td>0.9VDD</td>
<td>VDD</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>D050</td>
<td>VHYS</td>
<td>Hysteresis of Schmitt Trigger Inputs</td>
<td>TBD</td>
<td>TBD</td>
<td>V</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: In RC oscillator configuration, the OSC1/CLKIN pin is a Schmitt Trigger input. It is not recommended that the PICmicro be driven with an external clock while in RC mode.

2: The better of the two specifications may be used. For VIL, this would be the higher voltage and for VHH, this would be the lower voltage.
Section 32. Electrical Specifications

32.7 I/O Current Specifications

The PORT Weak Pull-up Current is the additional current consumed when the weak pull-ups are enabled.

Leakage Currents are the currents that the device consumes, since the devices are manufactured in the real world and do not adhere to their ideal characteristics. Ideally, there should be no current on an input, but due to the real world, there is always some parasitic path that consumes negligible current.

Table 32-5: Example DC Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D060</td>
<td>IIL</td>
<td>Input Leakage Current (2,3)</td>
<td>—</td>
<td>±1</td>
<td>µA</td>
<td>Vss ≤ VPIN ≤ VDD, Pin at hi-impedance</td>
</tr>
<tr>
<td>D061</td>
<td>MCLR</td>
<td>—</td>
<td>±5</td>
<td>µA</td>
<td>Vss ≤ VPIN ≤ VDD</td>
<td></td>
</tr>
<tr>
<td>D063</td>
<td>OSC1</td>
<td>—</td>
<td>±5</td>
<td>µA</td>
<td>Vss ≤ VPIN ≤ VDD</td>
<td></td>
</tr>
<tr>
<td>D070</td>
<td>IPURB</td>
<td>PORTB weak pull-up current</td>
<td>50</td>
<td>400</td>
<td>µA</td>
<td>VDD = 5V, VPIN = VSS</td>
</tr>
</tbody>
</table>

Note 1: Not applicable.
2: The leakage current on the MCLR pin is strongly dependent on the applied voltage level. The specified levels represent normal operating conditions. Higher leakage current may be measured at different input voltages.
3: Negative current is defined as current sourced by the pin.
32.8 Output Drive Levels

The Output Low Voltage (VOL) is the pin's output voltage for a low level. The VOL of an I/O pin is dependent on the current sunk by that pin. If an I/O pin is shorted to VDD, no matter the drive capability of the I/O pin, a low level would not be reached (and the device would consume excessive drive current). The VOL is the output voltage that the I/O pin will drive, given the I/O does not need to sink more than the IOL current (at the specified device voltage), as specified in the conditions portion of the specification.

The Output High Voltage (VOH) is the pin's output voltage for a high level. The VOH of an I/O pin is dependent on the current sourced by that pin. If an I/O pin is shorted to VSS, no matter the drive capability of the I/O pin, a high level would not be reached (and the device would consume excessive drive current). The VOH is the output voltage that the I/O pin will drive, given the I/O does not need to source more than the IOH current (at the specified device voltage), as specified in the conditions portion of the specification.

Table 32-6: Example DC Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D080</td>
<td>VOL</td>
<td>Output Low Voltage</td>
<td>—</td>
<td>0.6 V</td>
<td>V</td>
<td>IOL = 8.5 mA, VDD = 4.5V, -40°C to +85°C</td>
</tr>
<tr>
<td>D080A</td>
<td>VOL</td>
<td>Output Low Voltage</td>
<td>—</td>
<td>0.6 V</td>
<td>V</td>
<td>IOL = 7.0 mA, VDD = 4.5V, -40°C to +125°C</td>
</tr>
<tr>
<td>D083</td>
<td>VOH</td>
<td>Output High Voltage (1)</td>
<td>—</td>
<td>0.6 V</td>
<td>V</td>
<td>IOH = -1.6 mA, VDD = 4.5V, -40°C to +85°C</td>
</tr>
<tr>
<td>D083A</td>
<td>VOH</td>
<td>Output High Voltage (1)</td>
<td>—</td>
<td>0.6 V</td>
<td>V</td>
<td>IOH = -1.2 mA, VDD = 4.5V, -40°C to +125°C</td>
</tr>
<tr>
<td>D090</td>
<td>VDD</td>
<td>Open-drain High Voltage</td>
<td>—</td>
<td>7.5 V</td>
<td>V</td>
<td>RA4 pin</td>
</tr>
</tbody>
</table>

Note 1: Negative current is defined as current sourced by the pin.
32.9 I/O Capacitive Loading

These loadings affect the specifications for the timing specifications. If the loading in your application is different, then you will need to determine how this will affect the characteristics of the device in your system. Capacitances less than these specifications should not have an effect on a system.

Table 32-7: Example DC Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ†</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D101</td>
<td>CIO</td>
<td>Capacitive Loading Specs on Output Pins</td>
<td>—</td>
<td>—</td>
<td>50</td>
<td>pF</td>
<td>To meet the Timing Specifications of the Device</td>
</tr>
<tr>
<td>D102</td>
<td>CB</td>
<td>SCL, SDA</td>
<td>—</td>
<td>—</td>
<td>400</td>
<td>pF</td>
<td>In I²C mode</td>
</tr>
</tbody>
</table>

Standard Operating Conditions (unless otherwise stated)
Operating temperature: -40°C ≤ TA ≤ +85°C for industrial and
-40°C ≤ TA ≤ +125°C for extended
Operating voltage VDD range as described in DC spec Table 32-2.
32.10 Low Voltage Detect (LVD)

Low Voltage Detect is internal circuitry which will set a flag when the device voltage crosses the specified trip point.

Figure 32-3: Low-Voltage Detect Characteristics

![Diagram of Low Voltage Detect Circuit]

Table 32-8: Example Low Voltage Detect Requirements

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D420</td>
<td>V_LVD</td>
<td>LVD Voltage</td>
<td>2.5</td>
<td>2.86</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 0100</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 0101</td>
<td>2.7</td>
<td>2.86</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 0110</td>
<td>2.8</td>
<td>2.98</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 0111</td>
<td>3.0</td>
<td>3.2</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1000</td>
<td>3.3</td>
<td>3.52</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1001</td>
<td>3.5</td>
<td>3.72</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1010</td>
<td>3.6</td>
<td>3.84</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1011</td>
<td>3.8</td>
<td>4.04</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1100</td>
<td>4.0</td>
<td>4.26</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1101</td>
<td>4.2</td>
<td>4.46</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>LVDL3:LVDL0 = 1110</td>
<td>4.5</td>
<td>4.78</td>
<td>V</td>
<td></td>
</tr>
</tbody>
</table>
Section 32. Electrical Specifications

32.11 EPROM/FLASH/Data EEPROM

Table 32-9 shows the specifications for programming of the internal EPROM program memory. Table 32-10 shows the specifications of the FLASH program memory and Data EEPROM.

Table 32-9: Example Program Memory Programming Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Sym.</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D110</td>
<td>VPP</td>
<td>Voltage on MCLR/VPP pin</td>
<td>12.75</td>
<td>13.25</td>
<td>V</td>
<td>(Note 2)</td>
</tr>
<tr>
<td>D111</td>
<td>VDDP</td>
<td>Supply voltage during programming</td>
<td>4.75</td>
<td>5.25</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D112</td>
<td>VPP</td>
<td>Current into MCLR/VPP pin</td>
<td>—</td>
<td>50</td>
<td>mA</td>
<td></td>
</tr>
<tr>
<td>D113</td>
<td>IIDP</td>
<td>Supply current during programming</td>
<td>—</td>
<td>30</td>
<td>mA</td>
<td></td>
</tr>
<tr>
<td>D114</td>
<td>TPROG</td>
<td>Programming pulse width</td>
<td>100</td>
<td>1000</td>
<td>µs</td>
<td>Terminated via internal/external interrupt or a RESET</td>
</tr>
<tr>
<td>D115</td>
<td>TERASE</td>
<td>EPROM erase time</td>
<td>4</td>
<td>—</td>
<td>hrs</td>
<td>See Table 32-3</td>
</tr>
</tbody>
</table>

Note 1: These specifications are for the programming of the on-chip program memory EEPROM through the use of the table write instructions. The complete programming specifications can be found in: PIC18CXXX Programming Specifications (Literature number DS39028).

2: The MCLR/VPP pin may be kept in this range at times other than programming, but is not recommended.

Table 32-10: Example Data EEPROM/Flash Characteristics

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ†</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D120</td>
<td>ED</td>
<td>Data EEPROM Memory Endurance</td>
<td>1M</td>
<td>10M</td>
<td>—</td>
<td>E/W</td>
<td>25°C at 5V</td>
</tr>
<tr>
<td>D121</td>
<td>VDWR</td>
<td>VDD for read/write</td>
<td>VMIN</td>
<td>VMIN</td>
<td>VMAX</td>
<td>V</td>
<td>VMIN = Minimum operating voltage</td>
</tr>
<tr>
<td>D122</td>
<td>TDEW</td>
<td>Erase/Write cycle time</td>
<td>—</td>
<td>—</td>
<td>10</td>
<td>ms</td>
<td>VMAX = Maximum operating voltage</td>
</tr>
<tr>
<td>D130</td>
<td>EP</td>
<td>Endurance</td>
<td>100</td>
<td>1000</td>
<td>—</td>
<td>E/W</td>
<td>25°C at 5V</td>
</tr>
<tr>
<td>D131</td>
<td>VPR</td>
<td>VDD for read</td>
<td>VMIN</td>
<td>VMIN</td>
<td>VMAX</td>
<td>V</td>
<td>VMIN = Minimum operating voltage</td>
</tr>
<tr>
<td>D132</td>
<td>VPEW</td>
<td>VDD for erase/write</td>
<td>4.5</td>
<td>—</td>
<td>5.5</td>
<td>V</td>
<td>VMAX = Maximum operating voltage</td>
</tr>
<tr>
<td>D133</td>
<td>TPWE</td>
<td>Erase/Write cycle time</td>
<td>—</td>
<td>—</td>
<td>10</td>
<td>ms</td>
<td></td>
</tr>
</tbody>
</table>

† Data in "Typ" column is at 5.0V, 25°C unless otherwise stated. These parameters are for design guidance only and are not tested.

Legend: E/W means Erase/Write cycles.
32.12 Comparators and Voltage Reference

Table 32-11: Example Comparator Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristics</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Comments</th>
</tr>
</thead>
<tbody>
<tr>
<td>D300</td>
<td>VIOFF</td>
<td>Input offset voltage</td>
<td>—</td>
<td>± 10 mV</td>
<td></td>
<td></td>
</tr>
<tr>
<td>D301</td>
<td>VICM</td>
<td>Input common mode voltage</td>
<td>0</td>
<td>VDD - 1.5</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D302</td>
<td>CMRR</td>
<td>Common Mode Rejection Ratio</td>
<td>35</td>
<td>—</td>
<td>db</td>
<td></td>
</tr>
<tr>
<td>300</td>
<td>TRESP</td>
<td>Response Time (1)</td>
<td>—</td>
<td>400 ns</td>
<td></td>
<td></td>
</tr>
<tr>
<td>300A</td>
<td></td>
<td></td>
<td>—</td>
<td>600 ns</td>
<td></td>
<td></td>
</tr>
<tr>
<td>301</td>
<td>TMC2OV</td>
<td>Comparator Mode Change to Output Valid</td>
<td>—</td>
<td>10 µs</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Note 1: Response time measured with one comparator input at (VDD - 1.5)/2, while the other input transitions from VSS to VDD.

Table 32-12: Example Voltage Reference Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristics</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Comments</th>
</tr>
</thead>
<tbody>
<tr>
<td>D310</td>
<td>VRES</td>
<td>Resolution</td>
<td>VDD/24</td>
<td>—</td>
<td>VDD/32</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>D311</td>
<td>VRAA</td>
<td>Absolute Accuracy</td>
<td>—</td>
<td>1/4</td>
<td>1/2</td>
<td>LSb</td>
<td>Low Range (VRR = 1)</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>1/2</td>
<td>—</td>
<td>LSb</td>
<td>High Range (VRR = 0)</td>
</tr>
<tr>
<td>D312</td>
<td>VRUR</td>
<td>Unit Resistor Value (R)</td>
<td>—</td>
<td>2k</td>
<td>—</td>
<td>Ω</td>
<td></td>
</tr>
<tr>
<td>310</td>
<td>TSET</td>
<td>Setting Time (1)</td>
<td>—</td>
<td>—</td>
<td>10</td>
<td>µs</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: Settling time measured while VRR = 1 and VR3:VR0 transitions from 0000 to 1111.
## Section 32. Electrical Specifications

Table 32-13: Example Fixed Voltage Reference Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>D400</td>
<td>VRL</td>
<td>Output Voltage</td>
<td>2.0</td>
<td>2.048</td>
<td>2.1</td>
<td>V</td>
<td>VDD ≥ 2.5V</td>
</tr>
<tr>
<td></td>
<td>VRH</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D402</td>
<td>TCVOUT</td>
<td>Output Voltage Drift</td>
<td></td>
<td>15</td>
<td>50</td>
<td>ppm/°C</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>D403</td>
<td>En</td>
<td>Output Noise Voltage</td>
<td></td>
<td>TBD</td>
<td></td>
<td>µVp-p</td>
<td>0.1 Hz to 10 Hz</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td></td>
<td>TBD</td>
<td></td>
<td></td>
<td>10 Hz to 10 kHz</td>
</tr>
<tr>
<td>D404</td>
<td>IVREFSO</td>
<td>External Load Source</td>
<td></td>
<td>—</td>
<td>5</td>
<td>mA</td>
<td></td>
</tr>
<tr>
<td>D405</td>
<td>IVREFSI</td>
<td>External Load Sink</td>
<td></td>
<td>—</td>
<td>-5</td>
<td>mA</td>
<td></td>
</tr>
<tr>
<td>D406</td>
<td></td>
<td>Load Regulation</td>
<td></td>
<td>TBD</td>
<td></td>
<td>mV/mA</td>
<td>Isource = 0 mA to 5 mA</td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td></td>
<td>TBD</td>
<td></td>
<td></td>
<td>Isink = 0 mA to 5 mA</td>
</tr>
<tr>
<td>D407</td>
<td></td>
<td>Line Regulation</td>
<td></td>
<td>—</td>
<td>50</td>
<td>µV/V</td>
<td></td>
</tr>
<tr>
<td>D401A</td>
<td></td>
<td>VRL Quiescent Supply Current</td>
<td>—</td>
<td>30</td>
<td>50</td>
<td>µA</td>
<td>VRH, BOR, and LVD disabled. No load on VRL.</td>
</tr>
<tr>
<td>D401B</td>
<td></td>
<td>VRH Quiescent Supply Current</td>
<td>—</td>
<td>30</td>
<td>50</td>
<td>µA</td>
<td>VRL, BOR, and LVD disabled. No load on VRH.</td>
</tr>
</tbody>
</table>
32.13 Timing Parameter Symbology

The timing parameter symbols have been created with one of the following formats:

1. TppS2ppS 3. Tcc:st (I2C specifications only)
2. TppS 4. Ts (I2C specifications only)

<table>
<thead>
<tr>
<th>T</th>
<th>F</th>
<th>Frequency</th>
<th>T</th>
<th>Time</th>
</tr>
</thead>
</table>

Lowercase letters (pp) and their meanings:

- pp
  - cc: CCP1
  - ck: CLKOUT
  - cs: CS
  - di: SDI
  - do: SDO
  - dt: Data in
  - io: I/O port
  - mc: MCLR

Uppercase letters and their meanings:

- S
  - F: Fall
  - H: High
  - I: Invalid (Hi-impedance)
  - L: Low
  - AA: Output Access
  - BUF: Bus free
  - CU: Output Access
  - BUF: Bus free

- I2C only
  - HD: Hold
  - ST: DATA input hold
  - STA: START condition
  - SU: Setup
  - STO: STOP condition

Table 32-4: Example Load Conditions

- Load Condition 1
  - VDD/2
  - RL = 464Ω
  - CL = 50 pF

- Load Condition 2
  - VSS
  - (works for I/O pin multiplexed on to OSC2/CLKOUT)
Section 32. Electrical Specifications

32.14 Example External Clock Timing Waveforms and Requirements

Figure 32-5: Example External Clock Timing Waveforms

Table 32-14: Example External Clock Timing Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>1A</td>
<td>Fosc</td>
<td>External CLKIN Frequency</td>
<td>DC</td>
<td>4</td>
<td>MHz</td>
<td>XT osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>DC</td>
<td>40</td>
<td>MHz</td>
<td>HS osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>4</td>
<td>10</td>
<td>MHz</td>
<td>HS4 osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>DC</td>
<td>40</td>
<td>kHz</td>
<td>LP osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>DC</td>
<td>40</td>
<td>MHz</td>
<td>EC</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Oscillator Frequency</td>
<td>DC</td>
<td>4</td>
<td>MHz</td>
<td>RC osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>0.1</td>
<td>4</td>
<td>MHz</td>
<td>XT osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>4</td>
<td>25</td>
<td>MHz</td>
<td>HS osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>4</td>
<td>10</td>
<td>MHz</td>
<td>HS4 osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>5</td>
<td>200</td>
<td>kHz</td>
<td>LP osc mode</td>
</tr>
<tr>
<td>1</td>
<td>Tosc</td>
<td>External CLKIN Period</td>
<td>250</td>
<td>—</td>
<td>— ns</td>
<td>XT and RC osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>40</td>
<td>—</td>
<td>— ns</td>
<td>HS osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>100</td>
<td>—</td>
<td>— ns</td>
<td>HS4 osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>5</td>
<td>—</td>
<td>— μs</td>
<td>LP osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Oscillator Period</td>
<td>250</td>
<td>—</td>
<td>— ns</td>
<td>RC osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>250</td>
<td>10,000</td>
<td>— ns</td>
<td>XT osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>100</td>
<td>10,000</td>
<td>— ns</td>
<td>HS osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>40</td>
<td>100</td>
<td>— ns</td>
<td>HS4 osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>5</td>
<td>—</td>
<td>— μs</td>
<td>LP osc</td>
</tr>
<tr>
<td>2</td>
<td>TCy</td>
<td>Instruction Cycle Time</td>
<td>100</td>
<td>—</td>
<td>— ns</td>
<td>TCy = 4/FOSC</td>
</tr>
<tr>
<td>3</td>
<td>Tosl,</td>
<td>External Clock in (OSC1)</td>
<td>30</td>
<td>—</td>
<td>— ns</td>
<td>XT osc</td>
</tr>
<tr>
<td></td>
<td>ToSH</td>
<td>High or Low Time</td>
<td>2.5</td>
<td>—</td>
<td>— μs</td>
<td>LP osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>10</td>
<td>—</td>
<td>— ns</td>
<td>HS osc</td>
</tr>
<tr>
<td>4</td>
<td>TosR,</td>
<td>External Clock in (OSC1)</td>
<td>—</td>
<td>20</td>
<td>— ns</td>
<td>XT osc</td>
</tr>
<tr>
<td></td>
<td>TosF</td>
<td>Rise or Fall Time</td>
<td>—</td>
<td>50</td>
<td>— ns</td>
<td>LP osc</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>7.5</td>
<td>— ns</td>
<td>HS osc</td>
</tr>
</tbody>
</table>

Note 1: Instruction cycle period (TCy) equals four times the input oscillator time-base period. All specified values are based on characterization data for that particular oscillator type under standard operating conditions with the device executing code. Exceeding these specified limits may result in an unstable oscillator operation and/or higher than expected current consumption. All devices are tested to operate at "min." values with an external clock applied to the OSC1/CLKIN pin. When an external clock input is used, the "Max." cycle time limit is "DC" (no clock) for all devices.
32.15 Example Phase Lock Loop (PLL) Timing Waveforms and Requirements

Table 32-15: Example PLL Clock Timing Specification (VDD = 4.2V - 5.5V)

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>T_{PLL}</td>
<td>PLL Start-up Time (Lock Time)</td>
<td>—</td>
<td>2</td>
<td>ms</td>
<td></td>
</tr>
<tr>
<td></td>
<td>Δ_{CLK}</td>
<td>CLKOUT Stability (Jitter) using PLL</td>
<td>-2</td>
<td>+2</td>
<td>%</td>
<td></td>
</tr>
</tbody>
</table>
Section 32. Electrical Specifications

Figure 32-6: Example CLKOUT and I/O Timing Waveforms

Table 32-16: Example CLKOUT and I/O Timing Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>10</td>
<td>TosH2ckL</td>
<td>OSC1↑ to CLKOUT↓</td>
<td>—</td>
<td>75</td>
<td>200</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>11</td>
<td>TosH2ckH</td>
<td>OSC1↑ to CLKOUT↑</td>
<td>—</td>
<td>75</td>
<td>200</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>12</td>
<td>TckR</td>
<td>CLKOUT rise time</td>
<td>—</td>
<td>35</td>
<td>100</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>13</td>
<td>TckF</td>
<td>CLKOUT fall time</td>
<td>—</td>
<td>35</td>
<td>100</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>14</td>
<td>TckL2ioV</td>
<td>CLKOUT ↓ to Port out valid</td>
<td>—</td>
<td>—</td>
<td>0.5TCY + 20 ns</td>
<td>(Note 1)</td>
<td></td>
</tr>
<tr>
<td>15</td>
<td>TioV2ckH</td>
<td>Port in valid before CLKOUT↑</td>
<td>0.25TCY + 25 ns</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>16</td>
<td>TckH2iol</td>
<td>Port in hold after CLKOUT↑</td>
<td>0</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>17</td>
<td>TosH2iolV</td>
<td>OSC1↑ (Q1 cycle) to Port out valid</td>
<td>—</td>
<td>50</td>
<td>150</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>18</td>
<td>TosH2iol</td>
<td>OSC1↑ (Q2 cycle) to Port input invalid (I/O in hold time)</td>
<td>PIC18CXXX</td>
<td>100</td>
<td>—</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>19</td>
<td>TioV2osH</td>
<td>Port input valid to OSC1↑ (I/O in setup time)</td>
<td>PIC18LCXXX</td>
<td>200</td>
<td>—</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>20</td>
<td>TioR</td>
<td>Port output rise time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>10</td>
<td>25</td>
<td>ns</td>
</tr>
<tr>
<td>20A</td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>—</td>
<td>60</td>
<td>ns</td>
</tr>
<tr>
<td>21</td>
<td>TioF</td>
<td>Port output fall time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>10</td>
<td>25</td>
<td>ns</td>
</tr>
<tr>
<td>21A</td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>—</td>
<td>60</td>
<td>ns</td>
</tr>
<tr>
<td>22††</td>
<td>Tinp</td>
<td>INT pin high or low time</td>
<td>TCY</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>23††</td>
<td>Tinp</td>
<td>RB&lt;7:4&gt; change INT high or low time</td>
<td>TCY</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>24††</td>
<td>Tinp</td>
<td>RC&lt;7:4&gt; change INT high or low time</td>
<td>20 ns</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>

†† These parameters are asynchronous events not related to any internal clock edges.

Note 1: Measurements are taken in RC Mode where CLKOUT output is 4 x Tosc.
32.16 Example Power-up and RESET Timing Waveforms and Requirements

Figure 32-7: Example RESET, Watchdog Timer, Oscillator Start-up Timer and Power-up Timer Timing Waveforms

Figure 32-8: Brown-out Reset Timing

Table 32-17: Example RESET, Watchdog Timer, Oscillator Start-up Timer, Brown-out Reset, and Power-up Timer Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>30</td>
<td>TmCL</td>
<td>MCLR Pulse Width (low)</td>
<td>2</td>
<td></td>
<td></td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>31</td>
<td>TWD T</td>
<td>Watchdog Timer Time-out Period (No Prescaler)</td>
<td>7</td>
<td>18</td>
<td>33</td>
<td>ms</td>
<td></td>
</tr>
<tr>
<td>32</td>
<td>TOST</td>
<td>Oscillation Start-up Timer Period</td>
<td>1024Tosc</td>
<td></td>
<td>1024Tosc</td>
<td>ns</td>
<td>Tosc = OSC1 period</td>
</tr>
<tr>
<td>33</td>
<td>PWRT</td>
<td>Power up Timer Period</td>
<td>28</td>
<td>72</td>
<td>132</td>
<td>ms</td>
<td></td>
</tr>
<tr>
<td>34</td>
<td>IOZ</td>
<td>I/O Hi-impedance from MCLR Low or Watchdog Timer Reset</td>
<td>—</td>
<td>2</td>
<td></td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>35</td>
<td>BOR</td>
<td>Brown-out Reset Pulse Width</td>
<td>200</td>
<td>—</td>
<td></td>
<td>µs</td>
<td>VDD ≤ VBOR (See D005)</td>
</tr>
<tr>
<td>36</td>
<td>IVRST</td>
<td>Time for Internal Reference Voltage to become stable</td>
<td>—</td>
<td>20</td>
<td>50</td>
<td>µs</td>
<td></td>
</tr>
</tbody>
</table>
Section 32. Electrical Specifications

32.17 Example Timer0 and Timer1 Timing Waveforms and Requirements

Table 32-18: Example Timer0 and Timer1 External Clock Requirements

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>40</td>
<td>T0H</td>
<td>T0CKI High Pulse Width</td>
<td>No Prescaler</td>
<td>0.5T CY + 20</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>10</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>41</td>
<td>T0L</td>
<td>T0CKI Low Pulse Width</td>
<td>No Prescaler</td>
<td>0.5T CY + 20</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>10</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>42</td>
<td>T0P</td>
<td>T0CKI Period</td>
<td>No Prescaler</td>
<td>T CY + 10</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>Greater of:</td>
<td>20 ns or T CY + 10</td>
<td>N</td>
<td>N = prescale value (1, 2, 4, 8)</td>
</tr>
<tr>
<td>45</td>
<td>T1H</td>
<td>T1CKI High Time</td>
<td>Synchronous, no prescaler</td>
<td>0.5T CY + 20</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Synchronous, with prescaler</td>
<td>PIC18cxx</td>
<td>10</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Asynchronous</td>
<td>PIC18Lcxx</td>
<td>25</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>46</td>
<td>T1L</td>
<td>T1CKI Low Time</td>
<td>Synchronous, no prescaler</td>
<td>0.5T CY + 5</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Synchronous, with prescaler</td>
<td>PIC18cxx</td>
<td>10</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Asynchronous</td>
<td>PIC18Lcxx</td>
<td>25</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>47</td>
<td>T1P</td>
<td>T1CKI Input Period</td>
<td>Synchronous</td>
<td>Greater of:</td>
<td>20 ns or T CY + 40</td>
<td>N</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Asynchronous</td>
<td>60</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>48</td>
<td>Tcke2tmr1</td>
<td>Delay from external T1CKI clock edge to</td>
<td>2T osc</td>
<td>7T osc</td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>
32.18 Example CCP Timing Waveforms and Requirements

Figure 32-10: Example Capture/Compare/PWM Timings Waveforms

![CCP Waveforms](image)

Table 32-19: Example Capture/Compare/PWM Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>TccL</td>
<td>CCPx</td>
<td>input low time</td>
<td>No Prescaler</td>
<td>(0.5T_{CY} + 20) ns</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>PIC18CXXX</td>
<td>10 ns</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>PIC18LCXXX</td>
<td>20 ns</td>
<td>—</td>
</tr>
<tr>
<td>TccH</td>
<td>CCPx</td>
<td>input high time</td>
<td>No Prescaler</td>
<td>(0.5T_{CY} + 20) ns</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>PIC18CXXX</td>
<td>10 ns</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>With Prescaler</td>
<td>PIC18LCXXX</td>
<td>20 ns</td>
<td>—</td>
</tr>
<tr>
<td>TccP</td>
<td>CCPx</td>
<td>input period</td>
<td>(3T_{CY} + 40) ns</td>
<td>—</td>
<td>ns</td>
<td>(N = \text{prescale value} \ (1, 4 \text{ or } 16))</td>
</tr>
<tr>
<td>TccR</td>
<td>CCPx</td>
<td>output rise time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45</td>
<td>ns</td>
</tr>
<tr>
<td>TccF</td>
<td>CCPx</td>
<td>output fall time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45</td>
<td>ns</td>
</tr>
</tbody>
</table>

Note: Refer to Figure 32-4 for load conditions.
Section 32. Electrical Specifications

32.19 Example Parallel Slave Port (PSP) Timing Waveforms and Requirements

Figure 32-11: Example Parallel Slave Port Timing Waveforms

Table 32-20: Example Parallel Slave Port Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>62</td>
<td>TdtV2wrH</td>
<td>Data-in valid before WR↑ or CS↑ (setup time)</td>
<td>20</td>
<td>25</td>
<td>ns</td>
<td>Extended Temp range</td>
</tr>
<tr>
<td>63</td>
<td>TwrH2dtI</td>
<td>WR↑ or CS↑ to data-in invalid (hold time)</td>
<td>PIC18CXXX</td>
<td>20</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>35</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>64</td>
<td>TrdL2dtV</td>
<td>RD↓ and CS↓ to data-out valid</td>
<td>—</td>
<td>80</td>
<td>ns</td>
<td>Extended Temp range</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>90</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>65</td>
<td>TrdH2dtI</td>
<td>RD↑ or CS↑ to data-out invalid</td>
<td>10</td>
<td>30</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>66</td>
<td>TibINH</td>
<td>Inhibit of the IBF flag bit being cleared from WR↑ or CS↑</td>
<td>—</td>
<td>3TCY</td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>

Note: Refer to Figure 32-4 for load conditions.
32.20 Example SSP and Master SSP SPI Mode Timing Waveforms and Requirements

Figure 32-12: Example SPI Master Mode Timing (CKE = 0)

Table 32-21: Example SPI Mode Requirements (Master Mode, CKE = 0)

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>70</td>
<td>TssL2scH, TssL2scL</td>
<td>SS↓ to SCK↓ or SCK↑ input</td>
<td>TCY</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>71</td>
<td>TscH</td>
<td>SCK input high time (slave mode)</td>
<td>Continuous</td>
<td>1.25TCY + 30</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>71A</td>
<td></td>
<td>Single Byte</td>
<td>40</td>
<td>—</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>72</td>
<td>TscL</td>
<td>SCK input low time (slave mode)</td>
<td>Continuous</td>
<td>1.25TCY + 30</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>72A</td>
<td></td>
<td>Single Byte</td>
<td>40</td>
<td>—</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>73</td>
<td>TdiV2scH, TdiV2scL</td>
<td>Setup time of SDI data input to SCK edge</td>
<td>100</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>73A</td>
<td>Tl2b</td>
<td>Last clock edge of Byte1 to the 1st clock edge of Byte2</td>
<td>1.5TCY + 40</td>
<td>—</td>
<td>ns</td>
<td>(Note 2)</td>
</tr>
<tr>
<td>74</td>
<td>Tsch2dl, Tsch2dlL</td>
<td>Hold time of SDI data input to SCK edge</td>
<td>100</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>75</td>
<td>TdoR</td>
<td>SDO data output rise time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>75</td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45 ns</td>
<td></td>
<td></td>
</tr>
<tr>
<td>76</td>
<td>TdoF</td>
<td>SDO data output fall time</td>
<td>—</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>78</td>
<td>TscR</td>
<td>SCK output rise time (master mode)</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>78</td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45 ns</td>
<td></td>
<td></td>
</tr>
<tr>
<td>79</td>
<td>TscF</td>
<td>SCK output fall time (master mode)</td>
<td>—</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>80</td>
<td>Tsh2doV, Tsh2doV</td>
<td>SDO data output valid after SCK edge</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>50 ns</td>
<td></td>
</tr>
<tr>
<td>80</td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>100 ns</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Note 1: Requires the use of Parameter # 73A.
Note 2: Only if Parameter #s 71A and 72A are used.
Section 32. Electrical Specifications

Figure 32-13: Example SPI Master Mode Timing (CKE = 1)

Table 32-22: Example SPI Mode Requirements (Master Mode, CKE = 1)

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>71</td>
<td>TscH</td>
<td>SCCK input high time (slave mode)</td>
<td>Continuous</td>
<td>1.25Tcy + 30</td>
<td>— ns</td>
<td></td>
</tr>
<tr>
<td>71A</td>
<td></td>
<td>Single Byte</td>
<td>—</td>
<td>40</td>
<td>— ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>72</td>
<td>TscL</td>
<td>SCCK input low time (slave mode)</td>
<td>Continuous</td>
<td>1.25Tcy + 30</td>
<td>— ns</td>
<td></td>
</tr>
<tr>
<td>72A</td>
<td></td>
<td>Single Byte</td>
<td>—</td>
<td>40</td>
<td>— ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>73</td>
<td>Tdiv2scH, Tdiv2scL</td>
<td>Setup time of SDI data input to SCCK edge</td>
<td>100</td>
<td>—</td>
<td>— ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>73A</td>
<td></td>
<td>Last clock edge of Byte1 to the 1st clock edge of Byte2</td>
<td>1.5Tcy + 40</td>
<td>—</td>
<td>— ns</td>
<td>(Note 2)</td>
</tr>
<tr>
<td>74</td>
<td>TschH2dll, TschL2dll</td>
<td>Hold time of SDI data input to SCCK edge</td>
<td>100</td>
<td>—</td>
<td>— ns</td>
<td></td>
</tr>
<tr>
<td>75</td>
<td>TdoR</td>
<td>SDO data output rise time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>76</td>
<td>TdoF</td>
<td>SDO data output fall time</td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45 ns</td>
<td></td>
</tr>
<tr>
<td>78</td>
<td>TscR</td>
<td>SCCK output rise time (master mode)</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>79</td>
<td>TscF</td>
<td>SCCK output fall time (master mode)</td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45 ns</td>
<td></td>
</tr>
<tr>
<td>80</td>
<td>TschH2doV, TschL2doV</td>
<td>SDO data output valid after SCCK edge</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>50 ns</td>
<td></td>
</tr>
<tr>
<td>81</td>
<td>TdoV2scH, TdoV2scL</td>
<td>SDO data output setup to SCCK edge</td>
<td>Tcy</td>
<td>—</td>
<td>— ns</td>
<td></td>
</tr>
</tbody>
</table>

Note 1: Requires the use of Parameter # 73A.
Note 2: Only if Parameter #s 71A and 72A are used.
**Figure 32-14: Example SPI Slave Mode Timing (CKE = 0)**

![Example SPI Slave Mode Timing Diagram]

**Table 32-23: Example SPI Mode Requirements (Slave Mode Timing (CKE = 0))**

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>70</td>
<td>TssL2scH, TssL2scL</td>
<td>SS↓ to SCK↓ or SCK↑ input</td>
<td>TCy</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>71</td>
<td>TscH</td>
<td>SCK input high time (slave mode)</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>71A</td>
<td></td>
<td>Continuous</td>
<td>1.25TCy + 30</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Single Byte</td>
<td>40</td>
<td></td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>72</td>
<td>TscL</td>
<td>SCK input low time (slave mode)</td>
<td></td>
<td></td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>72A</td>
<td></td>
<td>Continuous</td>
<td>1.25TCy + 30</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Single Byte</td>
<td>40</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>73</td>
<td>TdiV2scH, TdiV2scL</td>
<td>Setup time of SDI data input to SCK edge</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>73A</td>
<td>T2b</td>
<td>Last clock edge of Byte1 to the 1st clock edge of Byte2</td>
<td>1.5TCy + 40</td>
<td></td>
<td>ns</td>
<td>(Note 2)</td>
</tr>
<tr>
<td>74</td>
<td>Tsch2dIL, TscL2dIL</td>
<td>Hold time of SDI data input to SCK edge</td>
<td>100</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>75</td>
<td>TdoR</td>
<td>SDO data output rise time</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>76</td>
<td>TdoF</td>
<td>SDO data output fall time</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>77</td>
<td>TshH2doZ</td>
<td>SS↑ to SDO output hi-impedance</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>78</td>
<td>TscR</td>
<td>SCK output rise time (master mode)</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>45</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>79</td>
<td>TscF</td>
<td>SCK output fall time (master mode)</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>25</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>80</td>
<td>TschH2doV, TscL2doV</td>
<td>SDO data output valid after SCK edge</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18CXXX</td>
<td>—</td>
<td>50</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>100</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>83</td>
<td>TschH2ssH, TscL2ssH</td>
<td>SS↑ after SCK edge</td>
<td></td>
<td></td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** Requires the use of Parameter # 73A.

**Note 2:** Only if Parameter #s 71A and 72A are used.
**Section 32. Electrical Specifications**

### Figure 32-15: Example SPI Slave Mode Timing (CKE = 1)

### Table 32-24: Example SPI Slave Mode Mode Requirements (CKE = 1)

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>70</td>
<td>TssL2scH, TssL2scL</td>
<td>SS↓ to SCK↑ or SCK↑ input</td>
<td>Tcy</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>71</td>
<td>TscH</td>
<td>SCK input high time (slave mode)</td>
<td>Continuous</td>
<td>1.25Tcy + 30</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>71A</td>
<td></td>
<td>SCK input high time (slave mode)</td>
<td>Single Byte</td>
<td>40</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>72</td>
<td>TscL</td>
<td>SCK input low time (slave mode)</td>
<td>Continuous</td>
<td>1.25Tcy + 30</td>
<td>ns</td>
<td>(Note 1)</td>
</tr>
<tr>
<td>72A</td>
<td></td>
<td>SCK input low time (slave mode)</td>
<td>Single Byte</td>
<td>40</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>73A</td>
<td>Ta2b</td>
<td>Last clock edge of Byte1 to the 1st clock edge of Byte2</td>
<td>1.5Tcy + 40</td>
<td></td>
<td>ns</td>
<td>(Note 2)</td>
</tr>
<tr>
<td>74</td>
<td>Tsch2dlL, Tsch2dlL</td>
<td>Hold time of SDI data input to SCK edge</td>
<td>100</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>75</td>
<td>TdoR</td>
<td>SDO data output rise time</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td></td>
<td>45 ns</td>
<td></td>
</tr>
<tr>
<td>76</td>
<td>TdoF</td>
<td>SDO data output fall time</td>
<td></td>
<td></td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>77</td>
<td>TssH2doZ</td>
<td>SS↑ to SDO output hi-impedance</td>
<td>10</td>
<td>50</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>78</td>
<td>TscR</td>
<td>SCX output rise time (master mode)</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td></td>
<td>45 ns</td>
<td></td>
</tr>
<tr>
<td>79</td>
<td>TscF</td>
<td>SCX output fall time (master mode)</td>
<td></td>
<td></td>
<td>25 ns</td>
<td></td>
</tr>
<tr>
<td>80</td>
<td>Tsch2doV, Tsch2doV</td>
<td>SDO data output valid after SCK edge</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>50 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td></td>
<td>100 ns</td>
<td></td>
</tr>
<tr>
<td>82</td>
<td>TssL2doV</td>
<td>SDO data output valid after SSI edge</td>
<td>PIC18CXXX</td>
<td>—</td>
<td>50 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td></td>
<td>100 ns</td>
<td></td>
</tr>
<tr>
<td>83</td>
<td>Tsch2ssH, Tsch2ssH</td>
<td>SS↑ after SCK edge</td>
<td>1.5Tcy + 40</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** Requires the use of Parameter # 73A.
**Note 2:** Only if Parameter #s 71A and 72A are used.
32.21 Example SSP I\(^2\)C Mode Timing Waveforms and Requirements

Figure 32-16: Example SSP I\(^2\)C Bus Start/Stop Bits Timing Waveforms

![Timing Waveforms Diagram]

**Note:** Refer to Figure 32-4 for load conditions.

Table 32-25: Example SSP I\(^2\)C Bus Start/Stop Bits Requirements (Slave Mode)

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>90</td>
<td>Tsu:STA</td>
<td>START condition</td>
<td>100 kHz mode</td>
<td>4700</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Setup time</td>
<td>400 kHz mode</td>
<td>600</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td>91</td>
<td>Thd:STA</td>
<td>START condition</td>
<td>100 kHz mode</td>
<td>4000</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Hold time</td>
<td>400 kHz mode</td>
<td>600</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td>92</td>
<td>Tsu:STO</td>
<td>STOP condition</td>
<td>100 kHz mode</td>
<td>4700</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Setup time</td>
<td>400 kHz mode</td>
<td>600</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td>93</td>
<td>Thd:STO</td>
<td>STOP condition</td>
<td>100 kHz mode</td>
<td>4000</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Hold time</td>
<td>400 kHz mode</td>
<td>600</td>
<td>—</td>
<td></td>
</tr>
</tbody>
</table>
## Section 32. Electrical Specifications

### Figure 32-17: Example SSP I²C Bus Data Timing Waveforms

![Waveform Diagram]

### Table 32-26: Example SSP I²C bus Data Requirements (Slave Mode)

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>100</td>
<td>T\text{HIGH}</td>
<td>Clock high time</td>
<td>100 kHz mode</td>
<td>4.0</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>SSP Module</td>
<td>1.5T\text{CY}</td>
<td>—</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>101</td>
<td>T\text{LOW}</td>
<td>Clock low time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>1.3</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>SSP Module</td>
<td>1.5T\text{CY}</td>
<td>—</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>102</td>
<td>T\text{R}</td>
<td>SDA and SCL rise time</td>
<td>100 kHz mode</td>
<td>—</td>
<td>1000</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>20 + 0.1Cb</td>
<td>300</td>
<td>ns</td>
<td>Cb is specified to be from 10 to 400 pF</td>
</tr>
<tr>
<td>103</td>
<td>T\text{F}</td>
<td>SDA and SCL fall time</td>
<td>100 kHz mode</td>
<td>—</td>
<td>300</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>20 + 0.1Cb</td>
<td>300</td>
<td>ns</td>
<td>Cb is specified to be from 10 to 400 pF</td>
</tr>
<tr>
<td>90</td>
<td>T\text{SU:STA}</td>
<td>START condition setup time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>91</td>
<td>T\text{HD:STA}</td>
<td>START condition hold time</td>
<td>100 kHz mode</td>
<td>4.0</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>106</td>
<td>T\text{HD:DAT}</td>
<td>Data input hold time</td>
<td>100 kHz mode</td>
<td>0</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0</td>
<td>0.9</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>107</td>
<td>T\text{SU:DAT}</td>
<td>Data input setup time</td>
<td>100 kHz mode</td>
<td>250</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>100</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>92</td>
<td>T\text{SU:STO}</td>
<td>STOP condition setup time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>109</td>
<td>T\text{AA}</td>
<td>Output valid from clock</td>
<td>100 kHz mode</td>
<td>—</td>
<td>3500</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>110</td>
<td>T\text{BUF}</td>
<td>Bus free time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>1.3</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td>D102</td>
<td>C\text{b}</td>
<td>Bus capacitive loading</td>
<td>—</td>
<td>400</td>
<td>pF</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** As a transmitter, the device must provide this internal minimum delay time to bridge the undefined region (min. 300 ns) of the falling edge of SCL to avoid unintended generation of START or STOP conditions.

**Note 2:** A fast-mode I²C-bus device can be used in a standard-mode I²C-bus system, but the requirement \( \text{tsu;DAT} \geq 250 \text{ ns} \) must then be met. This will automatically be the case if the device does not stretch the LOW period of the SCL signal. If such a device does stretch the LOW period of the SCL signal, it must output the next data bit to the SDA line.

TR max. + tsu;DAT = 1000 + 250 = 1250 ns (according to the standard-mode I²C bus specification) before the SCL line is released.
### Example Master SSP I²C Mode Timing Waveforms and Requirements

#### Table 32-27: Example Master SSP I²C Bus Start/Stop Bits Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>90</td>
<td>TSU/STA</td>
<td>START condition Setup time</td>
<td>100 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>91</td>
<td>THD/STA</td>
<td>START condition Hold time</td>
<td>100 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>92</td>
<td>TSU/StO</td>
<td>STOP condition Setup time</td>
<td>100 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>93</td>
<td>THD/StO</td>
<td>STOP condition Hold time</td>
<td>100 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode</td>
<td>$2(T_{OSC})(BRG + 1)$ §</td>
<td>—</td>
<td>—</td>
</tr>
</tbody>
</table>

§  For the value required by the I²C specification, please refer to Figure A-11 of the “Appendix.”

**Note 1:** Maximum pin capacitance = 10 pF for all I²C pins.
Section 32. Electrical Specifications

Figure 32-19: Example Master SSP I2C Bus Data Timing

Table 32-28: Example Master SSP I2C Bus Data Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>100</td>
<td>T&lt;sub&gt;H&lt;/sub&gt;</td>
<td>Clock high time</td>
<td>100 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>400 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>101</td>
<td>T&lt;sub&gt;L&lt;/sub&gt;</td>
<td>Clock low time</td>
<td>100 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>400 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>102</td>
<td>TR</td>
<td>SDA and SCL rise time</td>
<td>100 kHz mode — 1000 ns</td>
<td>400 kHz mode 20 + 0.1C&lt;sub&gt;b&lt;/sub&gt; 300 ns</td>
<td>—</td>
<td>C&lt;sub&gt;b&lt;/sub&gt; is specified to be from 10 to 400 pF</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) — 300 ns</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>103</td>
<td>TF</td>
<td>SDA and SCL fall time</td>
<td>100 kHz mode — 300 ns</td>
<td>400 kHz mode 20 + 0.1C&lt;sub&gt;b&lt;/sub&gt; 300 ns</td>
<td>—</td>
<td>C&lt;sub&gt;b&lt;/sub&gt; is specified to be from 10 to 400 pF</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) — 100 ns</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>90</td>
<td>T&lt;sub&gt;SU&lt;/sub&gt;:STA</td>
<td>START condition setup time</td>
<td>100 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>400 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>—</td>
<td>Only relevant for repeated START condition</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>91</td>
<td>T&lt;sub&gt;HD&lt;/sub&gt;:STA</td>
<td>START condition hold time</td>
<td>100 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>400 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>—</td>
<td>After this period the first clock pulse is generated</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>106</td>
<td>T&lt;sub&gt;HD&lt;/sub&gt;:DAT</td>
<td>Data input hold time</td>
<td>100 kHz mode 0 — ns</td>
<td>400 kHz mode 0 0.3 ms</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) TBD — ns</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>107</td>
<td>T&lt;sub&gt;SU&lt;/sub&gt;:DAT</td>
<td>Data input setup time</td>
<td>100 kHz mode 250 — ns</td>
<td>400 kHz mode 100 — ns</td>
<td>—</td>
<td>(Note 2)</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) TBD — ns</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>92</td>
<td>T&lt;sub&gt;SU&lt;/sub&gt;:STO</td>
<td>STOP condition setup time</td>
<td>100 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>400 kHz mode 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) 2(T&lt;sub&gt;osc&lt;/sub&gt;)(BRG + 1) § — ms</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>109</td>
<td>TAA</td>
<td>Output valid from clock</td>
<td>100 kHz mode — 3500 ns</td>
<td>400 kHz mode — 1000 ns</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td>110</td>
<td>TBUF</td>
<td>Bus free time</td>
<td>100 kHz mode 4.7 ms</td>
<td>400 kHz mode 1.3 ms</td>
<td>—</td>
<td>Time the bus must be free before a new transmission can start</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1 MHz mode (1) TBD — ms</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D102</td>
<td>C&lt;sub&gt;b&lt;/sub&gt;</td>
<td>Bus capacitive loading</td>
<td>— 400 pF</td>
<td>—</td>
<td>—</td>
<td></td>
</tr>
</tbody>
</table>

§ For the value required by the I2C specification, please refer to Figure A-11 of the “Appendix.”

**Note 1:** Maximum pin capacitance = 10 pF for all I2C pins.

2: A fast-mode I2C-bus device can be used in a standard-mode I2C-bus system, but parameter 107 ≥ 250 ns must then be met. This will automatically be the case if the device does not stretch the LOW period of the SCL signal. If such a device does stretch the LOW period of the SCL signal, it must output the next data bit to the SDA line. Parameter 107 + parameter 107 = 1000 + 250 = 1250 ns (for 100 kHz-mode) before the SCL line is released.

© 2000 Microchip Technology Inc.
32.23 Example USART/SCI Timing Waveforms and Requirements

Figure 32-20: Example USART Synchronous Transmission (Master/Slave) Timing Waveforms

Table 32-29: Example USART Synchronous Transmission Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>120</td>
<td>TckH2dtV</td>
<td>Clock high to data out valid</td>
<td>—</td>
<td>40</td>
<td>ns</td>
<td>PIC18CXXX</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>100</td>
<td>ns</td>
<td>PIC18LCXXX</td>
</tr>
<tr>
<td>121</td>
<td>Tckrf</td>
<td>Clock out rise time and fall time (Master Mode)</td>
<td>—</td>
<td>20</td>
<td>ns</td>
<td>PIC18CXXX</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>50</td>
<td>ns</td>
<td>PIC18LCXXX</td>
</tr>
<tr>
<td>122</td>
<td>Tdtrf</td>
<td>Data out rise time and fall time</td>
<td>—</td>
<td>20</td>
<td>ns</td>
<td>PIC18CXXX</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>50</td>
<td>ns</td>
<td>PIC18LCXXX</td>
</tr>
</tbody>
</table>

Note: Refer to Figure 32-4 for load conditions.

Figure 32-21: Example USART Synchronous Receive (Master/Slave) Timing Waveforms

Table 32-30: Example USART Synchronous Receive Requirements

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>125</td>
<td>TdV2cki</td>
<td>Data hold before CK ↓ (DT hold time)</td>
<td>10</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>126</td>
<td>TckL2dtl</td>
<td>Data hold after CK ↓ (DT hold time)</td>
<td>15</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>

Note: Refer to Figure 32-4 for load conditions.
Section 32. Electrical Specifications

32.24 CAN Specifications

Figure 32-22: Example CAN Timing Waveforms

Table 32-31: Example CAN Timing

<table>
<thead>
<tr>
<th>Param. No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ †</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>20, 20A</td>
<td>TioR</td>
<td>Port output rise time (1)</td>
<td>PIC18CXXX</td>
<td>10</td>
<td>25</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>21, 21A</td>
<td>TioF</td>
<td>Port output fall time (1)</td>
<td>PIC18CXXX</td>
<td>10</td>
<td>25</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>500</td>
<td>Tcanclk2ioV</td>
<td>CANCLK ↓ or CANCLK ↑ to Port out valid</td>
<td>-20</td>
<td>—</td>
<td>20</td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td>501</td>
<td>TrxcanL</td>
<td>Wake-up noise filter</td>
<td>50</td>
<td>—</td>
<td>—</td>
<td>ns</td>
<td></td>
</tr>
</tbody>
</table>

† These parameters are asynchronous events not related to any internal clock edges.

Note 1: The CAN Clock is driven by the I/O pin drivers, so it has the same timing specification.
# 32.25 Example 8-bit A/D Timing Waveforms and Requirements

## Table 32-32: Example 8-bit A/D Converter Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>A01</td>
<td>NR</td>
<td>Resolution</td>
<td>—</td>
<td>—</td>
<td>8</td>
<td>bits</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td>(2)</td>
<td>—</td>
<td>—</td>
<td>8</td>
<td>bits</td>
<td>VREF = VDD = 2.5V</td>
</tr>
<tr>
<td>A02</td>
<td>EABS</td>
<td>Total Absolute error (2)</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSB</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>VREF = VDD = 2.5V</td>
</tr>
<tr>
<td>A03</td>
<td>EIL</td>
<td>Integral linearity error (2)</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSB</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>VREF = VDD = 2.5V</td>
</tr>
<tr>
<td>A04</td>
<td>EDL</td>
<td>Differential linearity error (2)</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSB</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>VREF = VDD = 2.5V</td>
</tr>
<tr>
<td>A05</td>
<td>EFS</td>
<td>Full scale error (2)</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSB</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>VREF = VDD = 2.5V</td>
</tr>
<tr>
<td>A06</td>
<td>EOFF</td>
<td>Offset error (2)</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSB</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>VREF = VDD = 2.5V</td>
</tr>
<tr>
<td>A10</td>
<td>—</td>
<td>Monotonicity</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td>VSS ≤ VAIN ≤ VREF</td>
</tr>
<tr>
<td>A20</td>
<td>VREF</td>
<td>Reference voltage</td>
<td>0V</td>
<td>—</td>
<td>AVDD</td>
<td>V</td>
<td>VREF delta when changing voltage levels on VREF inputs</td>
</tr>
<tr>
<td>A20A</td>
<td>(VREFH - VREFL)</td>
<td>3.0V</td>
<td>—</td>
<td>AVDD</td>
<td>V</td>
<td>Absolute minimum electrical specification to ensure 10-bit accuracy</td>
<td></td>
</tr>
<tr>
<td>A21</td>
<td>VREF+</td>
<td>Reference voltage high</td>
<td>AVSS + 3.0</td>
<td>AVDD + 0.3</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A22</td>
<td>VREF−</td>
<td>Reference voltage low</td>
<td>AVSS - 0.3</td>
<td>AVDD - 3.0</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A25</td>
<td>VAIN</td>
<td>Analog input voltage</td>
<td>AVSS - 0.3</td>
<td>AVREF + 0.3</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A30</td>
<td>ZAIN</td>
<td>Impedance of analog voltage source</td>
<td>—</td>
<td>—</td>
<td>10.0</td>
<td>kΩ</td>
<td></td>
</tr>
<tr>
<td>A40</td>
<td>IAD</td>
<td>A/D conversion current (VDD)</td>
<td>—</td>
<td>180</td>
<td>—</td>
<td>μA</td>
<td>Average current consumption when A/D is on (1)</td>
</tr>
<tr>
<td></td>
<td>PIC18CXXX</td>
<td>—</td>
<td>180</td>
<td>—</td>
<td>μA</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>PIC18LCXXX</td>
<td>—</td>
<td>90</td>
<td>—</td>
<td>μA</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A50</td>
<td>IREF</td>
<td>VREF input current (2)</td>
<td>10</td>
<td>—</td>
<td>1000</td>
<td>μA</td>
<td>Average current consumption when VAIN acquisition.</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>—</td>
<td>—</td>
<td>10</td>
<td>μA</td>
<td>During VAIN acquisition.</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>Based on differential of VHOLD to VAIN to charge CHOLD. See the A/D Converter section.</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>During A/D Conversion cycle</td>
</tr>
</tbody>
</table>

**Note 1:** When A/D is off, it will not consume any current other than minor leakage current. The power-down current spec includes any such leakage from the A/D module.

**Note 2:** If RA3 pin or VDD pin, whichever is selected as reference input.

**Note 3:** VSS ≤ VAIN ≤ VREF

**Note 3:** The A/D conversion result either increases or remains constant as the analog input increases.
### Figure 32-23: Example 8-bit A/D Conversion Timing Waveforms

![Waveform Diagram]

### Table 32-33: Example 8-bit A/D Conversion Requirements

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>PIC18CXXX</th>
<th>PIC18LCXXX</th>
<th>PIC18CXXX</th>
<th>PIC18LCXXX</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>A/D clock period</td>
<td>1.6</td>
<td>2.0</td>
<td>2.0</td>
<td>3.0</td>
</tr>
<tr>
<td></td>
<td></td>
<td>T AD</td>
<td>—</td>
<td>—</td>
<td>6.0</td>
<td>9.0</td>
</tr>
<tr>
<td></td>
<td></td>
<td>T CNV</td>
<td>11</td>
<td>11</td>
<td>TAD</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>T ACQ</td>
<td>Note 2</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>T GO</td>
<td>2T Osc</td>
<td>2T Osc</td>
<td>—</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>T SWC</td>
<td>1</td>
<td>1</td>
<td>TAD</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>T AMP</td>
<td>1 (5)</td>
<td>—</td>
<td>µs</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** If the A/D clock source is selected as RC, a time of Tcy is added before the A/D clock starts. This allows the SLEEP instruction to be executed.

**Note 2:** See the A/D Converter section for minimum requirements.
### 32.26 Example 10-bit A/D Timing Waveforms and Requirements

#### Table 32-34: Example 10-bit A/D Converter Characteristics

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>A01</td>
<td>NA</td>
<td>Resolution</td>
<td>—</td>
<td>—</td>
<td>10</td>
<td>bit</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td>A03</td>
<td>EIL</td>
<td>Integral linearity error</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSb</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td>A04</td>
<td>EDL</td>
<td>Differential linearity error</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSb</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td>A05</td>
<td>EFS</td>
<td>Full scale error</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSb</td>
<td>VREF = VDD ≥ 3.0V</td>
</tr>
<tr>
<td>A06</td>
<td>EOF</td>
<td>Offset error</td>
<td>—</td>
<td>—</td>
<td>&lt; ±1</td>
<td>LSb</td>
<td>VREF = VDD &lt; 3.0V</td>
</tr>
<tr>
<td>A10</td>
<td>—</td>
<td>Monotonicity</td>
<td>guaranteed (3)</td>
<td>—</td>
<td>VREF</td>
<td>V ≤ VAIN ≤ VREF</td>
<td></td>
</tr>
<tr>
<td>A20</td>
<td>VREF</td>
<td>Reference voltage</td>
<td>VREFH - VREFL</td>
<td>3V</td>
<td>—</td>
<td>V</td>
<td>For 10-bit resolution</td>
</tr>
<tr>
<td>A20A</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>V</td>
<td>—</td>
</tr>
<tr>
<td>A21</td>
<td>VREFH</td>
<td>Reference voltage High</td>
<td>AVSS</td>
<td>AVDD + 0.3V</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A22</td>
<td>VREFL</td>
<td>Reference voltage Low</td>
<td>AVSS - 0.3V</td>
<td>AVDD</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A25</td>
<td>VAIN</td>
<td>Analog input voltage</td>
<td>AVSS - 0.3V</td>
<td>VREF + 0.3V</td>
<td>V</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A30</td>
<td>ZAIN</td>
<td>Recommended impedance of analog voltage source</td>
<td>—</td>
<td>—</td>
<td>10.0</td>
<td>kΩ</td>
<td></td>
</tr>
<tr>
<td>A40</td>
<td>IAD</td>
<td>A/D conversion current (VDD)</td>
<td>—</td>
<td>180</td>
<td>—</td>
<td>μA</td>
<td>Average current consumption when A/D is on (1)</td>
</tr>
<tr>
<td>A50</td>
<td>IREF</td>
<td>VREF input current (Note 2)</td>
<td>10</td>
<td>—</td>
<td>1000</td>
<td>μA</td>
<td>During VAIN acquisition. Based on differential of VHOLD to VAIN. To charge CHOLD see the “10-bit A/D Converter” section. During A/D conversion cycle.</td>
</tr>
</tbody>
</table>

#### Note 1:
When A/D is off, it will not consume any current other than minor leakage current. The power-down current spec includes any such leakage from the A/D module.

#### Note 2:
VREF current is from RG0 and RG1 pins or AVDD and AVSS pins, whichever is selected as reference input.

#### Note 3:
VSS ≤ VAIN ≤ VREF.
# Section 32. Electrical Specifications

## Figure 32-24: Example 10-bit A/D Conversion Timing Waveforms

![Timing Waveforms Diagram](Note 2)

## Table 32-35: Example 10-bit A/D Conversion Requirements

<table>
<thead>
<tr>
<th>Param No.</th>
<th>Symbol</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>130</td>
<td>TAD</td>
<td>A/D clock period</td>
<td>1.6</td>
<td>20 (5)</td>
<td>µs</td>
<td>TOSC based, VREF ≥ 3.0V</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18CXXX</td>
<td>3.0</td>
<td>20 (5)</td>
<td>µs</td>
<td>TOSC based, VREF full range</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18LCXXX</td>
<td>2.0</td>
<td>6.0</td>
<td>µs</td>
<td>A/D RC Mode</td>
</tr>
<tr>
<td></td>
<td></td>
<td>PIC18CXXX</td>
<td>3.0</td>
<td>9.0</td>
<td>µs</td>
<td>A/D RC Mode</td>
</tr>
</tbody>
</table>

| 131       | TCNV   | Conversion time (not including acquisition time) (1) | 11 § | 12 § | TAD |
| 132       | TACQ   | Acquisition time (3) | 15  | —    | µs | -40°C ≤ Temp ≤ 125°C |
|           |        | 10  | —    | µs | 0°C ≤ Temp ≤ 125°C |
| 135       | TSWC   | Switching Time from convert → sample | —  | Note 4 |
| 136       | TAMP   | Amplifier settling time (2) | 1  | —  | µs |

**Note 1:** If the A/D clock source is selected as RC, a time of Tcy is added before the A/D clock starts. This allows the SLEEP instruction to be executed.

**Note 2:** This is a minimal RC delay (typically 100 nS), which also disconnects the holding capacitor from the analog input.

**Note 3:** This may be used if the "new" input voltage has not changed by more than 1LSb (i.e., 5 mV @ 5.12V) from the last sampled voltage (as stated on CHOLD).

**Note 4:** On the next Q4 cycle of the device clock.

**Note 5:** The time of the A/D clock period is dependent on the device frequency and the TAD clock divider.
32.27 Design Tips

No related design tips at this time.
Section 32. Electrical Specifications

32.28 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is, they may be written for the Base-Line, the Mid-Range, or High-End families), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to the Electrical Specifications are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related Application Notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

Note: Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/qa/qaex/codeex/
32.29 Revision History

Revision A

This is the initial released revision of the Electrical Specifications description.
Section 33. Device Characteristics

HIGHLIGHTS

33.1 Introduction ............................................................................................................... 33-2
33.2 Characterization vs. Electrical Specification .......................................................... 33-2
33.3 DC and AC Characteristics Graphs and Tables ......................................................... 33-2
33.4 Revision History ........................................................................................................ 33-26
33.1 Introduction

Microchip Technology Inc. provides characterization information on the devices that it manufactures. This information becomes available after the devices have undergone a complete characterization and the data has been analyzed. This data is taken on both device testers and on bench setups. The characterization data gives the designer a better understanding of the device characteristics, to better judge the acceptability of the device to the application.

33.2 Characterization vs. Electrical Specification

The difference between this information and the Electrical specifications can be classified as what the user should expect the devices to do vs. what Microchip tests the devices to do. The characterization graphs and tables provided are for design guidance and are not tested nor guaranteed.

There may be differences between what the characterization shows as the limits vs. that which is tested, as shown in the Electrical Specification section. This results from capabilities of the production tester equipment, plus whatever guard band that may be necessary.

33.3 DC and AC Characteristics Graphs and Tables

Each table gives specific information that may be useful design information. These values are taken under fixed circumstances. Measurements taken in your application may not lead to the same values if your circumstances are not the same.

In some graphs or tables the data presented are outside specified operating range (i.e., outside specified VDD range). This is for information only and devices will operate properly only within the specified range.

**Note:** The data presented in the device Data Sheet Characterization section is a statistical summary of data collected on units from different lots over a period of time and matrix samples. ‘Typical’ represents the mean of the distribution at, 25°C, while ‘max’ or ‘min’ represents (mean +3σ) and (mean -3σ) respectively, where σ is standard deviation.
Section 33. Device Characteristics

33.3.1 IPD vs. VDD

IPD is the current (I) that the device consumes when the device is in SLEEP mode (power-down), referred to as Power-down Current. These tests are taken with all I/O as inputs, either pulled high or low. That is, there are no floating inputs, nor are any pins driving an output (with a load).

The characterization shows graphs for both the Watchdog Timer (WDT) disabled and enabled. This is required since the WDT requires an on-chip RC oscillator which consumes additional current.

The device may have certain features and modules that can operate while the device is in SLEEP mode. Some of these modules are:

- Watchdog Timer (WDT)
- Brown-out Reset (BOR) circuitry
- Timer1
- Analog to Digital converter
- LCD module
- Comparators
- Voltage Reference

If these features are operating while the device is in SLEEP mode, a higher current will be consumed. When all features are disabled, the device will consume the lowest possible current (the leakage current). If more then one feature is enabled, then the expected current can easily be calculated as the base current (everything disabled and in SLEEP mode), plus all delta currents. Example 33-1 shows an example of calculating the typical currents for a device at 5V, with the WDT and Timer1 oscillator enabled.

Example 33-1: IPD Calculations with WDT and Timer1 Oscillator Enabled (@ 5V)

<table>
<thead>
<tr>
<th>Current Type</th>
<th>Value</th>
</tr>
</thead>
<tbody>
<tr>
<td>Base Current</td>
<td>14 nA</td>
</tr>
<tr>
<td>WDT Delta Current</td>
<td>14 µA</td>
</tr>
<tr>
<td>Timer1 Delta Current</td>
<td>22 µA</td>
</tr>
<tr>
<td>Total SLEEP Current</td>
<td>36 µA</td>
</tr>
</tbody>
</table>
Figure 33-1: Example Typical $I_{PD}$ vs. $V_{DD}$ (WDT Disabled, RC Mode)

Figure 33-2: Example Maximum $I_{PD}$ vs. $V_{DD}$ (WDT Disabled, RC Mode)
Section 33. Device Characteristics

Figure 33-3: Example Typical IPD vs. VDD @ 25°C (WDT Enabled, RC Mode)

![Graph showing typical IPD vs. VDD @ 25°C](image)

Figure 33-4: Example Maximum IPD vs. VDD (WDT Enabled, RC Mode)

![Graph showing maximum IPD vs. VDD with different temperatures](image)
Figure 33-5: Example Typical $I_D$ vs. $V_{DD}$ Brown-out Detect Enabled (RC Mode)

The shaded region represents the built-in hysteresis of the Brown-out Reset circuitry.

Figure 33-6: Example Maximum $I_D$ vs. $V_{DD}$ Brown-out Detect Enabled (85°C to -40°C, RC Mode)

The shaded region represents the built-in hysteresis of the Brown-out Reset circuitry.
Section 33. Device Characteristics

Figure 33-7: Example Typical IPD vs. Timer1 Enabled
(32 kHz, RC0/RC1 = 33 pF/33 pF, RC Mode)

Figure 33-8: Example Maximum IPD vs. Timer1 Enabled
(32 kHz, RC0/RC1 = 33 pF/33 pF, -40°C to 85°C, RC Mode)
33.3.2 I\textsubscript{DD} vs. Frequency

I\textsubscript{DD} is the current (I) that the device consumes when the device is in operating mode. This test is taken with all I/O as inputs, either pulled high or low. That is, there are no floating inputs, nor are any pins driving an output (with a load).

The I\textsubscript{DD} vs. Frequency charts measure the results on a Microchip automated bench setup, called the DCS (Data Collection System). The DCS accurately reflects the device and specified component values, that is, it does not add stray capacitance or current.

33.3.2.1 RC Measurements

For the RC measurement, the DCS selects a resistor and capacitor value and then, varies the voltage over the specified range. As the voltage is changed, the frequency of operation changes. For a fixed RC, as V\textsubscript{DD} increases, the frequency increases. After the measurement at this RC has been taken, the RC value is changed and the measurements are taken again. Each point on the graph corresponds to a device voltage, resistor value (R), and capacitor value (C).

Figure 33-9: Example Typical I\textsubscript{DD} vs. Frequency (RC Mode @ 22 pF, 25°C)
Section 33. Device Characteristics

Figure 33-10: Example Maximum \( I_{DD} \) vs. Frequency (RC Mode @ 22 pF, -40\(^\circ\)C to 85\(^\circ\)C)

Figure 33-11: Example Typical \( I_{DD} \) vs. Frequency (RC Mode @ 100 pF, 25\(^\circ\)C)

Shaded area is beyond recommended range.
Figure 33-12: Example Maximum \( I_{DD} \) vs. Frequency (RC Mode @ 100 pF, -40°C to 85°C)

Shaded area is beyond recommended range.

Figure 33-13: Example Typical \( I_{DD} \) vs. Frequency (RC Mode @ 300 pF, 25°C)
Section 33. Device Characteristics

Figure 33-14: Example Maximum \( I_{DO} \) vs. Frequency (RC Mode @ 300 pF, -40°C to 85°C)

Figure 33-15: Example Typical \( I_{DO} \) vs. Capacitance @ 500 kHz (RC Mode)
33.3.2.2 Crystal Oscillator Measurements

On the Data Collection System, there are several crystals. For this test, a crystal is multiplexed into the device circuit and the crystal’s capacitance values can be varied. The capacitance and voltage values are varied to determine the best characteristics (current, oscillator waveform and oscillator start-up), and then the currents are measured over voltage. The next crystal oscillator is then switched in and the procedure is repeated.

Figure 33-16: Example Typical Idd vs. Frequency (LP Mode, 25°C)

![Graph showing typical Idd vs. Frequency](image-url)

Figure 33-17: Example Maximum Idd vs. Frequency (LP Mode, -40°C to 85°C)

![Graph showing maximum Idd vs. Frequency](image-url)
Section 33. Device Characteristics

Figure 33-18: Example Typical I\textsubscript{DD} vs. Frequency (XT Mode, 25°C)

Figure 33-19: Example Maximum I\textsubscript{DD} vs. Frequency (XT Mode, -40°C to 85°C)
Figure 33-20: Example Typical I\(D\text{DD}\) vs. Frequency (HS Mode, 25°C)

Figure 33-21: Example Maximum I\(D\text{DD}\) vs. Frequency (HS Mode, -40°C to 85°C)
Section 33. Device Characteristics

33.3.3 RC Oscillator Frequency

These tables show the effects of the RC oscillator frequency as the device voltage varies. In these measurements, a capacitor and resistor value are selected and then, the frequency of the RC is measured, as the device voltage varies. The table shows the typical frequency for a R and C value at 5V, as well as the variation from this frequency that can be expected, due to device processing.

Figure 33-22: Example Typical RC Oscillator Frequency vs. Vdd

![Graph showing typical RC oscillator frequency vs. Vdd with shaded area indicating beyond recommended range.]

Shaded area is beyond recommended range.

Figure 33-23: Example Typical RC Oscillator Frequency vs. Vdd

![Graph showing typical RC oscillator frequency vs. Vdd with different R values.]

© 2000 Microchip Technology Inc.
Figure 33-24: Example Typical RC Oscillator Frequency vs. VDD

Table 33-1: Example RC Oscillator Frequencies

<table>
<thead>
<tr>
<th>CEXT</th>
<th>REXT</th>
<th>Average Fosc @ 5V, 25°C</th>
</tr>
</thead>
<tbody>
<tr>
<td>22 pF</td>
<td>5 k</td>
<td>4.12 MHz ± 1.4%</td>
</tr>
<tr>
<td></td>
<td>10 k</td>
<td>2.35 MHz ± 1.4%</td>
</tr>
<tr>
<td></td>
<td>100 k</td>
<td>268 kHz ± 1.1%</td>
</tr>
<tr>
<td>100 pF</td>
<td>3.3 k</td>
<td>1.80 MHz ± 1.0%</td>
</tr>
<tr>
<td></td>
<td>5 k</td>
<td>1.27 MHz ± 1.0%</td>
</tr>
<tr>
<td></td>
<td>10 k</td>
<td>888 kHz ± 1.2%</td>
</tr>
<tr>
<td></td>
<td>100 k</td>
<td>77.2 kHz ± 1.0%</td>
</tr>
<tr>
<td>300 pF</td>
<td>3.3 k</td>
<td>707 kHz ± 1.4%</td>
</tr>
<tr>
<td></td>
<td>5 k</td>
<td>501 kHz ± 1.2%</td>
</tr>
<tr>
<td></td>
<td>10 k</td>
<td>269 kHz ± 1.6%</td>
</tr>
<tr>
<td></td>
<td>100 k</td>
<td>28.3 kHz ± 1.1%</td>
</tr>
</tbody>
</table>

The percentage variation indicated here is part to part variation due to normal process distribution. The variation indicated is ±3 standard deviation from average value for VDD = 5V.
33.3.4 Oscillator Transconductance

Transconductance of the oscillator indicates the gain of the oscillator. As the transconductance increases, the gain of the oscillator circuit increases, which causes the current consumption of the oscillator circuit to increase. Also, as the transconductance increases, the maximum frequency that the oscillator circuit can support also increases, or the start-up time of the oscillator decreases.

Figure 33-25: Example Transconductance (gm) of HS Oscillator vs. VDD

Figure 33-26: Example Transconductance (gm) of LP Oscillator vs. VDD
Figure 33-27: Example Transconductance (gm) of XT Oscillator vs. VDD

Shaded area is beyond recommended range.
Section 33. Device Characteristics

33.3.5 Crystal Start-up Time

These graphs show the start-up time that one should expect to see at the specified voltage level, for a given crystal/capacitor combination.

Figure 33-28: Example Typical XTAL Start-up Time vs. VDD (LP Mode, 25°C)

Figure 33-29: Example Typical XTAL Start-up Time vs. VDD (HS Mode, 25°C)
Figure 33-30: Example Typical XTAL Start-up Time vs. VDD (XT Mode, 25°C)
Section 33. Device Characteristics

Figure 33-31: Example $V_{TH}$ (Input Threshold Trip Point Voltage) of I/O Pins vs. $V_{DD}$

![Graph showing $V_{TH}$ vs. $V_{DD}$]

Figure 33-32: Example $V_{IH}$, $V_{IL}$ of MCLR, T0CKI and OSC1 (in RC Mode) vs. $V_{DD}$

![Graph showing $V_{IH}$, $V_{IL}$ vs. $V_{DD}$]

Note: These input pins have Schmitt Trigger input buffers.
Figure 33-33: Example $V_{\text{TH}}$ (Input Threshold Trip Point Voltage) of OSC1 Input (in XT, HS, and LP modes) vs. $V_{\text{DD}}$

Figure 33-34: Example WDT Timer Time-out Period vs. $V_{\text{DD}}$
Section 33. Device Characteristics

Figure 33-35: Example $I_{OH}$ vs. $V_{OH}$, $V_{DD} = 3$ V

Figure 33-36: Example $I_{OH}$ vs. $V_{OH}$, $V_{DD} = 5$ V
Figure 33-37: Example I_{OL} vs. V_{OL}, V_{DD} = 3 V

Figure 33-38: Example I_{OL} vs. V_{OL}, V_{DD} = 5 V
33.3.6 Tested Crystals and Their Capacitor Values

This table shows the crystal frequency and manufacturer that was used for every test in this section, as well as the capacitor values/ranges that exhibited the best characteristics.

<table>
<thead>
<tr>
<th>Osc Type</th>
<th>Crystal Frequency</th>
<th>Capacitor Range C1</th>
<th>Capacitor Range C2</th>
</tr>
</thead>
<tbody>
<tr>
<td>LP</td>
<td>32 kHz</td>
<td>33 pF</td>
<td>33 pF</td>
</tr>
<tr>
<td></td>
<td>200 kHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
<tr>
<td>XT</td>
<td>200 kHz</td>
<td>47-68 pF</td>
<td>47-68 pF</td>
</tr>
<tr>
<td></td>
<td>1 MHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
<tr>
<td></td>
<td>4 MHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
<tr>
<td>HS</td>
<td>4 MHz</td>
<td>15 pF</td>
<td>15 pF</td>
</tr>
<tr>
<td></td>
<td>8 MHz</td>
<td>15-33 pF</td>
<td>15-33 pF</td>
</tr>
<tr>
<td></td>
<td>20 MHz</td>
<td>15-33 pF</td>
<td>15-33 pF</td>
</tr>
</tbody>
</table>

Note: Higher capacitance increases the stability of the oscillator but also increases the start-up time. These values are for design guidance only. Rs may be required in HS mode, as well as XT mode, to avoid overdriiving crystals with low drive level specification. Since each crystal has its own characteristics, the user should consult the crystal manufacturer for appropriate values of external components or verify oscillator performance.

<table>
<thead>
<tr>
<th>Crystals Used:</th>
</tr>
</thead>
<tbody>
<tr>
<td>32 kHz</td>
</tr>
<tr>
<td>200 kHz</td>
</tr>
<tr>
<td>1 MHz</td>
</tr>
<tr>
<td>4 MHz</td>
</tr>
<tr>
<td>8 MHz</td>
</tr>
<tr>
<td>20 MHz</td>
</tr>
</tbody>
</table>

33.3.7 Example EPROM Memory Erase Times

The UV erase time of an EPROM cell depends on the geometry size of the EPROM cell and the manufacturing technology. Table 33-3 shows some of the expected erase times for each different device.

<table>
<thead>
<tr>
<th>Example Device</th>
<th>Wavelength (Angstroms)</th>
<th>Intensity (µW/cm²)</th>
<th>Distance from UV Lamp (inches)</th>
<th>Typical Time (1) (minutes)</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>2537</td>
<td>12,000</td>
<td>1</td>
<td>60</td>
</tr>
</tbody>
</table>

Note 1: If these criteria are not met, the erase times will be different.

2: Refer to the device data sheet for the typical erase times for a device.
33.4 Revision History

Revision A

This is the initial released revision of the Device Characteristics description.
Section 34. Development Tools

HIGHLIGHTS

This section of the manual contains the following major topics:

34.1 Introduction ................................................................. 34-2
34.2 The Integrated Development Environment (IDE) .............. 34-3
34.3 MPLAB® Software Language Support .............................. 34-6
34.4 MPLAB-SIM Simulator Software ..................................... 34-8
34.5 MPLAB® Emulator Hardware Support ............................... 34-9
34.8 MPLAB Programmer Support ........................................... 34-10
34.9 Supplemental Tools ....................................................... 34-11
34.10 Development Boards ..................................................... 34-12
34.11 Development Tools for Other Microchip Products .......... 34-14
34.12 Related Application Notes .............................................. 34-15
34.13 Revision History .......................................................... 34-16
34.1 Introduction

Microchip offers a wide range of integrated development tools to ease the application development process. These tools can be broken down into the core development tools and the supplemental tools.

The core tools are as follows:

- MPLAB Integrated Development Environment, including full featured editor
- Language Products
  - MPASM Assembler
  - MPLAB-CXX C Compiler
  - MPLAB-C17
  - MPLAB-C18
  - MPLINK/MPLIB Linker Librarian
- MPLAB-SIM Software Simulator
- Real-Time In-Circuit Emulators
  - MPLAB-ICE Real-Time Emulator In-Circuit
  - ICEPIC Low-Cost Emulator with Breakpoint debug capabilities
- In-Circuit Debugger
  - MPLAB-ICD for 16F877
- Device Programmers
  - PROMATE® II Universal Programmer
  - PICSTART® Plus Entry-Level Development Programmer
- Development Boards
  - PICDEM-1 Low-Cost Demonstration Board
  - PICDEM-2 Low-Cost Demonstration Board
  - PICDEM-3 Low-Cost Demonstration Board
  - PICDEM-17 Low-Cost Demonstration Board
  - PICDEM-14A Low-Cost Demonstration Board

The minimum configuration of MPLAB is the Integrated Development Environment (IDE), the assembler (MPASM), and the software simulator (MPLAB-SIM). Other tools are added to MPLAB as they are installed. This gives a common platform for the design activity, from the writing and assembling of the source code, through the simulation/emulation, to the programming of prototype devices.

| Note: | The most current version may be downloaded from Microchip's web site for free. |

In addition to Microchip, there are many third party vendors. Microchip’s Third Party Handbook gives an overview of the manufactures and their tools.
Section 34. Development Tools

34.2 The Integrated Development Environment (IDE)

The core set of development tools operate under the IDE umbrella. The IDE is called MPLAB. This gives a consistent look and feel to all the development tools so that minimal learning of the new tool interface is required. The MPLAB IDE integrates all the following aspects of development:

- Source code editing
- Project management
- Machine code generation (from assembly or "C")
- Device simulation
- Device emulation
- Device programming

MPLAB is a PC based Windows® application. It has been extensively tested using Windows 95 and recommended in either of these operating environments:

- Windows 2000
- Windows NT 4.0
- Windows 98
- Windows 95
- Windows 3.X

This comprehensive tool suite allows the complete development of a project without leaving the MPLAB environment.
34.2.1 MPLAB

The MPLAB IDE Software brings an ease of software development previously unseen in the 8-bit microcontroller market. MPLAB is a Windows based application that contains:

- A full featured editor
- Four operating modes
  - editor
  - simulator
  - emulator
  - programmer
- A project manager
- Customizable tool bar and key mapping
- A status bar with project information
- Extensive on-line help

MPLAB allows you to:

- Edit your source files. This includes:
  - MPASM assembly language
  - MPLAB-CXX 'C' language
- One touch assemble (or compile) and download to PIC16/17 tools (automatically updates all project information)
- Debug using:
  - source files
  - absolute listing file
  - program memory
- Run up to four MPLAB-ICE emulators on the same PC
- Run or Single-step
  - program memory
  - source file
  - absolute listing

Microchip's simulator, MPLAB-SIM, operates under the same platform as the MPLAB-ICE emulator. This allows the user to learn a single tool set which functions equivalently for both the simulator and the full featured emulator.
Section 34. Development Tools

Figure 34-1 shows a typical MPLAB desktop with an open project. Some of the highlights are:

- Tool bars, multiple choices and user configurable
- Status, mode information, and button help on footer bar
- Multiple windows, such as
  - Source code
  - Source listing (most useful for 'C' programs)
  - Register file window (RAM)
  - Watch windows (to look at specific register)
  - Stop watch window for time/cycle calculations
- Programmer support (in this case PRO MATE pull down menu)

Note: This screen shot may not look exactly like the currently released MPLAB version.
34.3  **MPLAB ® Software Language Support**

To make the PICmicro device operate as desired in the application, a software program needs to be written for the microcontroller. This software program needs to be written in one of the programming languages for the device. Currently MPLAB supports two of Microchip's language products:

- Microchip Assembler (MPASM)
- Microchip 'C' Compiler (MPLAB-CXX)
- Other language products that support Common Object Description (COD) may also work with MPLAB

34.3.1  **Assembler (MPASM)**

The MPASM Universal Macro Assembler is a PC-hosted symbolic assembler. It supports all Microchip microcontroller families.

MPASM offers full featured Macro capabilities, conditional assembly, and several source and listing formats. It generates various object code formats to support Microchip's development tools as well as third party programmers.

MPASM allow full symbolic debugging from the Microchip Universal Emulator System (MPLAB-ICE).

MPASM has the following features to assist in developing software for specific use applications.

- Provides translation of Assembler source code to object code for all Microchip microcontrollers.
- Macro assembly capability.
- Produces all the files (Object, Listing, Symbol, and Special) required for symbolic debug with Microchip's emulator systems.
- Supports Hex (default), Decimal and Octal source and listing formats.

MPASM provides a rich directive language to support programming of the PICmicro. Directives are helpful in making the development of your assemble source code shorter and more maintainable.

34.3.2  **C Compilers**

The MPLAB-CXX is a complete 'C' compiler for Microchip's PICmicro family of microcontrollers. The compiler provides powerful integration capabilities and ease of use not found with other compilers.

For easier source level debugging, the compiler provides symbol information that is compatible with the MPLAB IDE memory display, Watch windows, and File register windows.

34.3.2.1  **MPLAB-C17 C Compiler**

The MPLAB-C17 Code Development System is a complete ANSI 'C' compiler and integrated development environment for Microchip's PIC17CXXX family of microcontrollers. This compiler provides powerful integration capabilities and ease of use not found with other compilers.

For easier source level debugging, the compiler provides symbol information that is compatible with the MPLAB IDE memory display.

34.3.2.2  **MPLAB-C18 C Compiler**

The MPLAB-C18 Code Development System is a complete ANSI 'C' compiler and integrated development environment for Microchip's PIC18CXXX family of microcontrollers. This compiler provides powerful integration capabilities and ease of use not found with other compilers.

For easier source level debugging, the compiler provides symbol information that is compatible with the MPLAB IDE memory display.
Section 34. Development Tools

34.3.3 MPLINK Linker

MPLINK is a linker for the Microchip C compiler, MPLAB-CXX, and the Microchip relocatable assembler, MPASM. MPLINK is a relocatable linker for MPASM and MPLAB-C17 and MPLAB-C18. It can link relocatable objects from assembly or C source files along with precompiled libraries using directives from a linker script.

MPLINK allows you to produce modular, re-usable code with MPLAB-CXX and MPASM. Control over the linking process is accomplished through a linker "script" file and with command line options. MPLINK ensures that all symbolic references are resolved and that code and data fit into the available PICmicro device.

MPLINK combines multiple input object modules generated by MPLAB-CXX or MPASM, into a single executable file. The actual addresses of data and the location of functions will be assigned when MPLINK is executed. This means that you will instruct MPLINK to place code and data somewhere within the named regions of memory, not to specific physical locations.

Once the linker knows about the ROM and RAM memory regions available in the target PICmicro device and it analyzes all the input files, it will try to fit the application’s routines into ROM and assign its data variables into available RAM. If there is too much code or too many variables to fit, MPLINK will give an error message.

MPLINK also provides flexibility for specifying that certain blocks of data memory are re-usable, so that different routines (which never call each other and don’t depend on this data to be retained between execution) can share limited RAM space.

MPLINK features include:

- MPLINK works with MPASM and MPLAB-C17 and MPLAB-C18.
- MPLINK allows all memory areas to be defined as sections to provide link-time flexibility.

34.3.4 MPLIB Librarian

MPLIB is a librarian for use with COFF object modules, created using either MPASM, MPASMWIN, or MPLAB-CXX or later. MPLIB is a librarian for pre-compiled code to be used with MPLINK. When a routine from a library is called from another source file, only the modules that contain that routine will be linked in with the application. This allows large libraries to be used efficiently in many different applications.

MPLIB manages the creation and modification of library files. A library file is a collection of object modules that are stored in a single file. There are several reasons for creating library files:

- Libraries make linking easier. Since library files can contain many object files, the name of a library file can be used instead of the names of many separate object files when linking.
- Libraries help keep code small. Since a linker only uses the required object files contained in a library, not all object files which are contained in the library necessarily wind up in the linker’s output module.
- Libraries make projects more maintainable. If a library is included in a project, the addition or removal of calls to that library will not require a change to the link process.
- Libraries help convey the purpose of a group of object modules. Since libraries can group together several related object modules, the purpose of a library file is usually more understandable than the purpose of its individual object modules. For example, the purpose of a file named “math.lib” is more apparent than the purpose of ‘power.o’, ‘ceiling.o’, and ‘floor.o’.
34.4 MPLAB-SIM Simulator Software

The software simulator is a no-cost tool with which to evaluate Microchip’s products and designs. The use of the simulator greatly helps debug software, particularly algorithms. Depending on the complexity of a design project, a time/cost benefit should be looked at comparing the simulator with an emulator.

For projects that have multiple engineers in the development, the simulator in conjunction with an emulator, can keep costs down and will allow speedy debug of the tough problems.

MPLAB-SIM Simulator simulates the PICmicro series microcontrollers on an instruction level. On any given instruction, the user may examine or modify any of the data areas or provide external stimulus to any of the pins. The input/output radix can be set by the user and the execution can be performed in; single step, execute until break, or in a trace mode.

MPLAB-SIM supports symbolic debugging using MPLAB-CXX, and MPASM. The Software Simulator offers the low cost flexibility to develop and debug code outside of the laboratory environment making it an excellent multi-project software development tool.
Section 34. Development Tools

34.5 MPLAB® Emulator Hardware Support

Microchip offers two emulators, a high-end version (MPLAB-ICE) and a low-cost version (ICEPIC). Both versions offer a very good price/feature value, and the selection of which emulator should depend on the feature set that you wish. For people looking at doing several projects with Microchip devices (or using the high-end devices), the use of MPLAB-ICE may offset the additional investment, through time savings achieved with the sophisticated breakpoint and trace capabilities.

34.6 MPLAB® High Performance Universal In-Circuit Emulator with MPLAB IDE

The MPLAB-ICE Universal In-Circuit Emulator is intended to provide the product development engineer with a complete microcontroller design tool set for PICmicro microcontrollers (MCUs). Software control of MPLAB-ICE is provided by the MPLAB Integrated Development Environment (IDE), which allows editing, "make" and download, and source debugging from a single environment.

Interchangeable target probes allow the system to be easily re-configured for emulation of different processors. The universal architecture of the MPLAB-ICE allows expansion to support all new Microchip microcontrollers.

The MPLAB-ICE Emulator System has been designed as a real-time emulation system with advanced features that are generally found on more expensive development tools.

A CE compliant version of MPLAB-ICE is available for European Union (EU) countries.

34.6.1 ICEPIC: Low-Cost PIC16CXXX In-Circuit Emulator

ICEPIC is a low-cost in-circuit emulator solution for the Microchip Base-line and Mid-Range families of 8-bit OTP microcontrollers.

ICEPIC features real-time emulation. ICEPIC is available under the MPLAB environment.

ICEPIC is designed by Neosoft Inc. and is manufactured under license by RF Solutions. Other emulator solutions may be available directly from RF Solutions.

34.7 MPLAB-ICD In-Circuit Debugger

Microchip’s In-Circuit Debugger, MPLAB-ICD, is a powerful, low-cost run-time development tool. This tool is based on the FLASH PIC16F877 and can be used to develop for this and other PICmicro microcontrollers from the PIC16CXXX family (see TB033 for more information). MPLAB-ICD utilizes the In-Circuit Debugging capability built into the PIC16F87X. This feature, along with Microchip’s In-Circuit Serial Programming protocol, offers cost-effective in-circuit FLASH programming and debugging from the graphical user interface of the MPLAB Integrated Development Environment. This enables a designer to develop and debug source code by watching variables, single-stepping and setting break points. Running at full speed enables testing hardware in real-time. The MPLAB-ICD is also a programmer for the flash PIC16F87X family.
34.8 MPLAB Programmer Support

Microchip offers two levels of device programmer support. For most bench setups the PICSTART Plus is sufficient. When true system qualification is done, the PRO MATE II should be the minimum used, due to the validation of program memory at VDD min and VDD max for maximum reliability.

34.8.1 PRO MATE® II: Universal Device Programmer

The PRO MATE II Universal Programmer is a full-featured programmer capable of operating in stand-alone mode as well as PC-hosted mode. PRO MATE II operates under MPLAB or as a DOS command driven program.

The PRO MATE II has programmable VDD and VPP supplies which allows it to verify programmed memory at VDD min and VDD max for maximum reliability. It has an LCD display for error messages, keys to enter commands and a modular detachable socket assembly to support various package types. In stand-alone mode, the PRO MATE II can read, verify, or program Base-Line, Mid-Range, and High-End devices. It can also set configuration and code-protect bits in this mode. The PRO MATE II programmer also supports Microchip’s Serial EEPROM and KEELog® Security devices.

A separate In-Circuit Serial Programming (ICSP) module is available for volume programming in a manufacturing environment. See the Programming module documentation for specific application requirements.

34.8.2 PICSTART® Plus Low-Cost Development Kit

The PICSTART Plus programmer is an easy-to-use, low-cost development programmer. It connects to the PC via one of the COM (RS-232) ports. MPLAB Integrated Development Environment software makes using the programmer simple and efficient. PICSTART Plus is not recommended for production programming, since it does not perform memory verification at VDDMIN and VDDMAX.

PICSTART Plus supports all Base-Line, Mid-Range, and High-End devices. For devices with up to more than 40 pins, an adapter socket is required. DIP packages are the form factor that are directly supported. Other package types may be supported with adapter sockets.

Note: The use of a PICSTART Plus Programmer is not recommended for ICSP. If ICSP is required, the use of the PRO MATE II Universal Programmer with the associated socket module is the recommended Microchip solution.
Section 34. Development Tools

34.9 Supplemental Tools

Microchip endeavors to provide a broad range of solutions to our customers. These tools are considered supplemental tools and may be available directly from Microchip or from another vendor. A comprehensive listing of alternate tool providers is contained in the Third Party Guide.

34.9.1 Third Party Guide

Looking for something else? Microchip strongly encourages and supports its Third Parties. Microchip publishes the “Third Party Guide”. It is an extensive volume that provides:

- Company
- Product
- Contact Information
- Consultants

For over 100 companies and 200 products. These products include Emulators, Device Programmers, Gang Programmers, Language Products, and other tool solutions.
34.10 Development Boards

Development boards give a quick start on a circuit that demonstrates the capabilities of a particular device. The device program can then be modified for your own evaluation of the device functionality and operation.

34.10.1 PICDEM-1 Low-Cost PIC16/17 Demonstration Board

The PICDEM-1 is a simple board which demonstrates the capabilities of several of Microchip’s microcontrollers. The microcontrollers supported are:

- PIC16C5X (PIC16C54 to PIC16C58A)
- PIC16C61
- PIC16C62X
- PIC16C71
- PIC16C710
- PIC16C711
- PIC16C8X
- PIC17C42A
- PIC17C43
- PIC17C44

All necessary hardware and software is included to run basic demo programs. The users can program the sample microcontrollers provided with the PICDEM-1 board, on a PRO MATE II or PICSTART Plus programmer, and easily test firmware. The user can also connect the PICDEM-1 board to the MPLAB-ICE emulator and download the firmware to the emulator for testing. Additional prototype area is available to build additional hardware. Some of the features include an RS-232 interface, a potentiometer for simulated analog input, push-button switches and eight LEDs connected to PORTB.

34.10.2 PICDEM-2 Low-Cost PIC16CXXX Demonstration Board

The PICDEM-2 is a simple demonstration board that supports the following microcontrollers:

- PIC16C62
- PIC16C63
- PIC16C64
- PIC16C65
- PIC16C66
- PIC16C67
- PIC16C72
- PIC16C73
- PIC16C74
- PIC16C76
- PIC16C77

All the necessary hardware and software is included to run the basic demonstration programs. The user can program the sample microcontrollers provided with the PICDEM-2 board, on a PRO MATE II programmer or PICSTART Plus, and easily test firmware. The MPLAB-ICE emulator may also be used with the PICDEM-2 board to test firmware. Additional prototype area has been provided for additional hardware. Some of the features include a RS-232 interface, push-button switches, a potentiometer for simulated analog input, a Serial EEPROM to demonstrate usage of the I2C bus and separate headers for connection to an LCD module and a keypad.
Section 34. Development Tools

34.10.3 PICDEM-3 Low-Cost PIC16CXXX Demonstration Board

The PICDEM-3 is a simple demonstration board that supports the PIC16C923 and PIC16C924 in the PLCC package. It will also support future 44-pin PLCC microcontrollers that have an LCD Module. All the necessary hardware and software is included to run the basic demonstration programs. The user can program the sample microcontrollers, provided with the PICDEM-3 board, on a PRO MATE II programmer or PICSTART Plus with an adapter socket, and easily test firmware. The MPLAB-ICE emulator may also be used with the PICDEM-3 board to test firmware. Additional prototype area has been provided for adding hardware. Some of the features include an RS-232 interface, push-button switches, a potentiometer for simulated analog input, a thermistor and separate headers for connection to an external LCD module and a keypad. Also provided on the PICDEM-3 board is an LCD panel, with 4 commons and 12 segments, that is capable of displaying time, temperature and day of the week. The PICDEM-3 provides an additional RS-232 interface and Windows 3.1 software for showing the de-multiplexed LCD signals on a PC. A simple serial interface allows the user to construct a hardware de-multiplexer for the LCD signals.

34.10.4 PICDEM-14A Low-Cost PIC14C000 Demonstration Board

The PICDEM-14A demo board is a general purpose platform which is provided to help evaluate the PIC14C000 mixed signal microcontroller. The board runs a PIC14C000 measuring the voltage of a potentiometer and the on-chip temperature sensor. The voltages are then calibrated to the internal bandgap voltage reference. The voltage and temperature data are then transmitted to the RS-232 port. This data can be displayed using a terminal emulation program, such as Windows Terminal. This demo board also includes peripherals that allow users to display data on an LCD panel, read from and write to a serial EEPROM, and prototype custom circuitry to interface to the microcontroller.

34.10.5 PICDEM-17

The PICDEM-17 is an evaluation board that demonstrates the capabilities of several Microchip microcontrollers, including PIC17C752, PIC17C756, PIC17C762, and PIC17C766. All necessary hardware is included to run basic demo programs, which are supplied on a 3.5-inch disk. A programmed sample is included, and the user may erase it and program it with the other sample programs using the PRO MATE II or PICSTART Plus device programmers and easily debug and test sample code. In addition, PICDEM-17 supports down-loading of programs to and executing out of external FLASH memory on board. The PICDEM-17 is also usable with the MPLAB-ICE emulator, and all of the sample programs can be run and modified using that emulator. Additionally, a generous prototype area is available for user hardware.
34.11 Development Tools for Other Microchip Products

34.11.1 SEEVAL® Evaluation and Programming System

The SEEVAL Serial EEPROM Designer’s Kit supports all Microchip 2-wire and 3-wire Serial EEPROMs. The kit includes everything necessary to read, write, erase or program special features of any Microchip SEEPROM product including Smart Serials™ and secure serials. The Total Endurance™ Disk is included to aid in trade-off analysis and reliability calculations. The total endurance kit can significantly reduce time-to-market and results in a more optimized system.

34.11.2 KeeLOQ® Evaluation and Programming Tools

KeeLOQ® evaluation and programming tools supports Microchip’s HCS Secure Data Products. The HCS evaluation kit includes an LCD display to show changing codes, a decoder to decode transmissions, and a programming interface to program test transmitters.
Section 34. Development Tools

34.12 Related Application Notes

This section lists application notes that are related to this section of the manual. These application notes may not be written specifically for the Enhanced MCU family (that is they may be written for the Base-Line, Mid-Range, or the High-End), but the concepts are pertinent, and could be used (with modification and possible limitations). The current application notes related to Microchip’s development tools are:

<table>
<thead>
<tr>
<th>Title</th>
<th>Application Note #</th>
</tr>
</thead>
<tbody>
<tr>
<td>No related application notes at this time.</td>
<td></td>
</tr>
</tbody>
</table>

**Note:** Several of the tools have tutorials which may be helpful in learning the tools. Please refer to the specific tool’s documentation for tutorial availability.

**Note:** Please visit the Microchip Web site for additional software code examples. These code examples are stand alone examples to assist in the understanding of the PIC18CXXX. The web address for these examples is:

http://www.microchip.com/10/faqs/codeex/
34.13 Revision History

Revision A

This is the initial released revision of Microchip's development tools description.
Section 35. Code Development

HIGHLIGHTS

This section of the manual contains the following major topics:

35.1 Overview ................................................................. 35-2
35.2 Good Practice ............................................................ 35-3
35.3 Diagnostic Code Techniques ...................................... 35-5
35.4 Example Scenario and Implementation ...................... 35-6
35.5 Implications of Using a High Level Language (HLL) .... 35-7
35.6 Revision History ........................................................ 35-8
35.1 Overview

This section covers some good programming practice as well as some diagnostic techniques that can be used in both the development stage, and the production release to help diagnose unexpected operation in the field. The advantage of including diagnostic code is that the device continually displays its status and the operational flow of the program.

These suggestions only scratch the surface of the possible techniques and good programming practices. Future revisions of this section will add additional suggestions on good programming practice and diagnostic techniques.
Section 35. Code Development

35.2 Good Practice

The following are some recommended programming practices. These will help ensure consistent program operation based on the stimulus supplied to the application software.

35.2.1 Use of Symbolic Code

Microchip supplies header files which use the register and bit name symbols specified in the device data sheets. The use of these symbols aids in several facets of software development. First, symbolic code makes the source file easier to read since the names relate to a function (as opposed to register addresses and bit positions). Example 35-1 shows two implementations. The first, though technically correct, is more difficult to follow than the second implementation. Also, the second allows for easier migration between devices (and PICmicro families) since any remapping of bit positions and register locations can automatically be handled by the software tool.

Example 35-1: Hard Coding vs. Symbolic Code

| BCF | 0xDB, 0, 0 | ; Clear the carry bit |
| :  | :         | :                   |
| BCF | STATUS, C  | ; Clear the carry bit |

35.2.2 Initialization of Data Memory

All Data Memory locations (SFRs and GPRs) should be initialized after a RESET. This will ensure that they are at a known state when the application code accesses each location, and ensures consistent operation.

When all data memory is not initialized, there is a possibility that the application software will read a location that is an indeterminate value. This may cause unexpected results from the application software. This issue is sometimes highlighted when development moves from a windowed device (where the window was not covered) to an OTP package.
35.2.3 Trap for Unexpected Conditions

Sometimes the application software will test only for the expected conditions. This is typically seen in the Interrupt Service Routine where only the expected sources for an interrupt are tested for. Example 35-2 shows a typical way that the conditions are tested, while Example 35-3 shows a more robust version. The TrapRoutine may do more then just loop waiting for a RESET. It could output a code on an I/O pin to indicate that this unexpected condition occurred.

Example 35-2: Typical ISR Code

```
ORG ISRVectorAddress ; Address for Interrupt Service Routine
ISRH BTFSS PIR1, ADIF ; A/D Interrupt?
    GOTO ADRoutine ; Yes, do A/D stuff
BTFSS PIR1, RCIF ; AUSART Receive Interrupt?
    GOTO ReceiveRoutine ; Yes, do Receive stuff
CCPRoutine ; Since not other interrupt sources
```

Example 35-3: Recommended ISR Code (with Trap)

```
ORG ISRVectorAddress ; Address for Interrupt Service Routine
ISRH BTFSS PIR1, ADIF ; A/D Interrupt?
    GOTO ADRoutine ; Yes, do A/D stuff
BTFSS PIR1, RCIF ; AUSART Receive Interrupt?
    GOTO ReceiveRoutine ; Yes, do Receive stuff
BTFSS PIR1, CCPIF ; CCP Interrupt?
    GOTO CCPRoutine ; Yes, do CCP stuff
TrapRoutine ; Should NEVER get here
    GOTO TrapRoutine ; If we do, loop forever and wait for WDT reset
```

35.2.4 Filling Unused Locations

In most application programs, not all program memory locations are used. These unprogrammed locations (0xFFFF) execute as a NOP instruction. Since program execution is never intended to go to one of these locations, the program should trap any occurrence of this. A good way to handle this is to “fill” all unprogrammed locations with a branch which goes to its own program memory location. With this, the WDT must be enabled to cause a device RESET and restart program execution.

Note: Remember that the CLRWDT instruction should have a minimum number of occurrences in the program and that the maximum time between the CLRWDT instructions must NOT be greater than the minimum WDT time-out period, multiplied by the selected prescaler.

The Microchip Assembler supplies a directive to do this. This directive is the “fill” directive.
Section 35. Code Development

35.3 Diagnostic Code Techniques
This section describes diagnostic code that can be embedded into the application code to help track the operation of the application software. Typically some mechanism is required to make the information available to the external world. This can be from a serial port (such as the Addressable USART module) or from extra I/O pins not used by the application.

35.3.1 Function Sequence
Have the diagnostic code output a unique code (value) for each function that the program enters. This allows the flow of the program to be monitored from outside the device (with a logic analyzer) and assist in determining if there is some unexpected condition that is causing code to execute in the observed sequence.

35.3.2 Stack Depth
A counter can be implemented that is incremented in each function that is called and each time that the program counter is at an interrupt vector address. The counter is then decremented at the end of the function or interrupt service routine. The value of the stack can then be output to indicate if the stack depth is out of its normal range (minimum and maximum).

35.3.3 A/D Operation
Visibility into the operation of the A/D can help validate the operation. Monitoring if the acquisition time is the expected time delay, as well as the result generated by the module. Achieving the A/D result accuracy depends on may factors, some internal and many external. The major internal factor is that there is a sufficient amount of time once the input channel is selected, until the conversion is started. This is called the acquisition time (TACQ).

35.3.3.1 A/D Acquisition Time
Use an I/O pin to indicate when the acquisition of the channel starts (toggle high) until the conversion is started (toggle high). If the input channel is changed force the I/O to toggle high. The last two I/O toggles before the A/D result is available, is acquisition time that the A/D had for that selected channel.

35.3.3.2 A/D Result
Output the result so that you can determine if the program execution flow can be attributed to the result from the A/D converter.
35.4 Example Scenario and Implementation

In this example application, one I/O port is not implemented. This means that there are eight I/O pins available to indicate the operational status of the PICmicro device and the flow of the application software. Let's say that the code implements 57 functions, the A/D is used and Addressable USART are used.

Six bits (Code5:Code0) are required to uniquely specify the 57 functions. After each function gets a code, an additional 7 codes are available. Table 35-1 shows how these codes could be defined, while Table 35-2 shows how the diagnostic output port could be defined.

<table>
<thead>
<tr>
<th>6-bit Code</th>
<th>Definition</th>
</tr>
</thead>
<tbody>
<tr>
<td>1111 11</td>
<td>Next Byte out is stack depth</td>
</tr>
<tr>
<td>1111 10</td>
<td>Next Byte out is A/D Result High Byte</td>
</tr>
<tr>
<td>1111 01</td>
<td>Next Byte out is Addressable USART received byte</td>
</tr>
<tr>
<td>1111 00</td>
<td>Next Byte out is A/D Result Low Byte</td>
</tr>
<tr>
<td>0000 00</td>
<td>Device has exited a device RESET. Rx1:Rx0 can be used to indicate the source of the RESET.</td>
</tr>
</tbody>
</table>

Table 35-2: PORT Function assignment

<table>
<thead>
<tr>
<th>Pin</th>
<th>Normal Mode</th>
<th>Mode after “Additional Codes” displayed</th>
<th>RESET</th>
</tr>
</thead>
<tbody>
<tr>
<td>Rx7</td>
<td>Code5</td>
<td>Byte value for the indicated function</td>
<td>0</td>
</tr>
<tr>
<td>Rx6</td>
<td>Code4</td>
<td></td>
<td>0</td>
</tr>
<tr>
<td>Rx5</td>
<td>Code3</td>
<td></td>
<td>0</td>
</tr>
<tr>
<td>Rx4</td>
<td>Code2</td>
<td></td>
<td>0</td>
</tr>
<tr>
<td>Rx3</td>
<td>Code1</td>
<td></td>
<td>0</td>
</tr>
<tr>
<td>Rx2</td>
<td>Code0</td>
<td></td>
<td>0</td>
</tr>
<tr>
<td>Rx1</td>
<td>Toggle for Start of acquisition</td>
<td>4 codes available to indicate source of RESET</td>
<td></td>
</tr>
<tr>
<td>Rx0</td>
<td>Toggle for Stop of acquisition</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
35.5 Implications of Using a High Level Language (HLL)

The use of a High Level Language, such as a C compiler, speeds the development cycle but can increase the difficulty in the use of diagnostic code. When writing at the high level, functions may be defined, but how the C compilers will implement those functions is hidden from the user. If the function is used only once, the C compiler may put that code in-line or use a `GOTO` to branch to the function and a `GOTO` to return from the function. In both these implementations, the stack is not affected. Also, if the function is called many times, but is small, it may be put inline. This again will not affect the stack used. These techniques used by the compiler may lead to efficient code generation, but may add to the difficulty in adding diagnostic code.
35.6 Revision History

Revision A

This is the initial released revision for the Code Development with a PICmicro device description.
APPENDIX A: \( \text{i}^2\text{C} \) \( \text{TM} \) OVERVIEW

This appendix provides an overview of the Inter-Integrated Circuit (\( \text{i}^2\text{C} \) \( \text{TM} \)) bus, with Subsection A.2 “Addressing \( \text{i}^2\text{C} \) Devices” discussing the operation of the SSP modules in \( \text{i}^2\text{C} \) mode.

The \( \text{i}^2\text{C} \) bus is a two-wire serial interface. The original specification, or standard mode, is for data transfers of up to 100 Kbps. An enhanced specification, or fast mode (400 Kbps) is supported. Standard and Fast mode devices will operate when attached to the same bus, if the bus operates at the speed of the slower device.

The \( \text{i}^2\text{C} \) interface employs a comprehensive protocol to ensure reliable transmission and reception of data. When transmitting data, one device is the “master” which initiates transfer on the bus and generates the clock signals to permit that transfer, while the other device(s) acts as the “slave.” All portions of the slave protocol are implemented in the SSP module’s hardware, except general call support, while portions of the master protocol need to be addressed in the PIC16CXX software. The MSSP module supports the full implementation of the \( \text{i}^2\text{C} \) master protocol, the general call address, and data transfers up to 1 Mbps. The 1 Mbps data transfers are supported by some of Microchips Serial EEPROMs. Table A-1 defines some of the \( \text{i}^2\text{C} \) bus terminology.

In the \( \text{i}^2\text{C} \) interface protocol each device has an address. When a master wishes to initiate a data transfer, it first transmits the address of the device that it wishes to “talk” to. All devices “listen” to see if this is their address. Within this address, a bit specifies if the master wishes to read-from/write-to the slave device. The master and slave are always in opposite modes (transmitter/receiver) of operation during a data transfer. That is, they can be thought of as operating in either of these two relations:

- Master-transmitter and Slave-receiver
- Slave-transmitter and Master-receiver

In both cases the master generates the clock signal.

The output stages of the clock (SCL) and data (SDA) lines must have an open-drain or open-collector in order to perform the wired-AND function of the bus. External pull-up resistors are used to ensure a high level when no device is pulling the line down. The number of devices that may be attached to the \( \text{i}^2\text{C} \) bus is limited only by the maximum bus loading specification of 400 pF and addressing capability.
A.1 Initiating and Terminating Data Transfer

During times of no data transfer (idle time), both the clock line (SCL) and the data line (SDA) are pulled high through the external pull-up resistors. The START and STOP conditions determine the start and stop of data transmission. The START condition is defined as a high to low transition of the SDA when the SCL is high. The STOP condition is defined as a low to high transition of the SDA when the SCL is high. Figure A-1 shows the START and STOP conditions. The master generates these conditions for starting and terminating data transfer. Due to the definition of the START and STOP conditions, when data is being transmitted, the SDA line can only change state when the SCL line is low.

Figure A-1: Start and Stop Conditions
### Section 36. Appendix

#### Table A-1: I^2^C Bus Terminology

<table>
<thead>
<tr>
<th>Term</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>Transmitter</td>
<td>The device that sends the data to the bus.</td>
</tr>
<tr>
<td>Receiver</td>
<td>The device that receives the data from the bus.</td>
</tr>
<tr>
<td>Master</td>
<td>The device which initiates the transfer, generates the clock and terminates the transfer.</td>
</tr>
<tr>
<td>Slave</td>
<td>The device addressed by a master.</td>
</tr>
<tr>
<td>Multi-master</td>
<td>More than one master device in a system.</td>
</tr>
<tr>
<td>Arbitration</td>
<td>Procedure that ensures that only one of the master devices will control the bus. This ensure that the transfer data does not get corrupted.</td>
</tr>
<tr>
<td>Synchronization</td>
<td>Procedure where the clock signals of two or more devices are synchronized.</td>
</tr>
</tbody>
</table>
A.2 Addressing I^2C Devices

There are two address formats. The simplest is the 7-bit address format with a R/W bit (Figure A-2). The more complex is the 10-bit address with a R/W bit (Figure A-3). For 10-bit address format, two bytes must be transmitted. The first five bits specify this to be a 10-bit address format. The 1st transmitted byte has 5 bits which specify a 10-bit address, the two MSbs of the address, and the R/W bit. The second byte is the remaining 8 bits of the address.

Figure A-2: 7-bit Address Format

Figure A-3: I^2C 10-bit Address Format
A.3 Transfer Acknowledge

All data must be transmitted per byte, with no limit to the number of bytes transmitted per data transfer. After each byte, the slave-receiver generates an acknowledge bit (ACK) (Figure A-4). When a slave-receiver doesn’t acknowledge the slave address or received data, the master must abort the transfer. The slave must leave SDA high so that the master can generate the STOP condition (Figure A-1).

Figure A-4: Slave-Receiver Acknowledge

If the master is receiving the data (master-receiver), it generates an acknowledge signal for each received byte of data, except for the last byte. To signal the end of data to the slave-transmitter, the master does not generate an acknowledge (not acknowledge). The slave then releases the SDA line so the master can generate the STOP condition. If the master also generates the STOP condition during the acknowledge pulse for valid termination of data transfer.

If the slave needs to delay the transmission of the next byte, holding the SCL line low will force the master into a wait state. Data transfer continues when the slave releases the SCL line. This allows the slave to move the received data or fetch the data it needs to transfer before allowing the clock to start. This wait state technique can also be implemented at the bit level, Figure A-5.

Figure A-5: Data Transfer Wait State
Figure A-6 and Figure A-7 show Master-transmitter and Master-receiver data transfer sequences.

**Figure A-6: Master-Transmitter Sequence**

For 7-bit address:

<table>
<thead>
<tr>
<th>S</th>
<th>Slave Address</th>
<th>R/W</th>
<th>Data</th>
<th>A</th>
<th>Data</th>
<th>P</th>
</tr>
</thead>
<tbody>
<tr>
<td>0' (write)</td>
<td>data transferred</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

A master transmitter addresses a slave receiver with a 7-bit address. The transfer direction is not changed.

For 10-bit address:

<table>
<thead>
<tr>
<th>S</th>
<th>Slave Address (Code + A9:A8)</th>
<th>R/W</th>
<th>A</th>
<th>Data</th>
<th>A</th>
<th>Data</th>
<th>A P</th>
</tr>
</thead>
<tbody>
<tr>
<td>(write)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

A master transmitter addresses a slave receiver with a 10-bit address.

- From master to slave
  - A = acknowledge (SDA low)
  - R = not acknowledge (SDA high)
  - S = START Condition
  - P = STOP Condition
- From slave to master
  - A = acknowledge (SDA low)
  - R = not acknowledge (SDA high)
  - S = START Condition
  - P = STOP Condition

**Figure A-7: Master-Receiver Sequence**

For 7-bit address:

<table>
<thead>
<tr>
<th>S</th>
<th>Slave Address</th>
<th>R/W</th>
<th>Data</th>
<th>A</th>
<th>Data</th>
<th>A P</th>
</tr>
</thead>
<tbody>
<tr>
<td>1' (read)</td>
<td>data transferred</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

A master reads a slave immediately after the first byte.

For 10-bit address:

<table>
<thead>
<tr>
<th>S</th>
<th>Slave Address (Code + A9:A8)</th>
<th>R/W</th>
<th>A</th>
<th>Data</th>
<th>A</th>
<th>Data</th>
<th>A P</th>
</tr>
</thead>
<tbody>
<tr>
<td>(write)</td>
<td>(read)</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

A master transmitter addresses a slave receiver with a 10-bit address.

- From master to slave
  - A = acknowledge (SDA low)
  - R = not acknowledge (SDA high)
  - S = START Condition
  - P = STOP Condition
- From slave to master
  - A = acknowledge (SDA low)
  - R = not acknowledge (SDA high)
  - S = START Condition
  - P = STOP Condition
When a master does not wish to relinquish the bus (which occurs by generating a STOP condition), a repeated START condition (Sr) must be generated. This condition is identical to the START condition (SDA goes high-to-low while SCL is high), but occurs after a data transfer acknowledge pulse (not the bus-free state). This allows a master to send “commands” to the slave and then receive the requested information or to address a different slave device. This sequence is shown in Figure A-8.

Figure A-8: Combined Format

Transfer direction of data and acknowledgment bits depends on R/W bits.

Combined format:

- Sr - repeated START Condition
- Direction of transfer may change at this point

Combined format - A master addresses a slave with a 10-bit address, then transmits data to this slave and reads data from this slave.

- A - acknowledge (SDA low)
- X - not acknowledge (SDA high)
- P - STOP Condition

From master to slave
- A - acknowledge (SDA low)
- X - not acknowledge (SDA high)
From slave to master
A.4 Multi-master

The \(^2\text{C}\) protocol allows a system to have more than one master. This is called a multi-master system. When two or more masters try to transfer data at the same time, arbitration and synchronization occur.

A.4.1 Arbitration

Arbitration takes place on the SDA line, while the SCL line is high. The master which transmits a high when the other master transmits a low, loses arbitration (Figure A-9) and turns off its data output stage. A master which lost arbitration can generate clock pulses until the end of the data byte where it lost arbitration. When the master devices are addressing the same device, arbitration continues into the data.

Figure A-9: Multi-Master Arbitration (Two Masters)

Masters that also incorporate the slave function, and have lost arbitration must immediately switch over to slave-receiver mode. This is because the winning master-transmitter may be addressing it.

Arbitration is not allowed between:

- A repeated START condition
- A STOP condition and a data bit
- A repeated START condition and a STOP condition

Care needs to be taken to ensure that these conditions do not occur.
Section 36. Appendix

A.4.2 Clock Synchronization

Clock synchronization occurs after the devices have started arbitration. This is performed using a wired-AND connection to the SCL line. A high to low transition on the SCL line causes the concerned devices to start counting off their low period. Once a device clock has gone low, it will hold the SCL line low until its SCL high state is reached. The low to high transition of this clock may not change the state of the SCL line, if another device clock is still within its low period. The SCL line is held low by the device with the longest low period. Devices with shorter low periods enter a high wait-state, until the SCL line comes high. When the SCL line comes high, all devices start counting off their high periods. The first device to complete its high period will pull the SCL line low. The SCL line high time is determined by the device with the shortest high period, Figure A-10.

Figure A-10: Clock Synchronization
Table A-2 and Table A-3 show the specifications of a compliant I²C bus. The column titled Microchip Parameter No. is provided to ease the user’s correlation to the corresponding parameter in the device data sheet. Figure A-11 and Figure A-12 show these times on the appropriate waveforms.

Figure A-11: I²C Bus Start/Stop Bits Timing Specification

Table A-2: I²C Bus Start/Stop Bits Timing Specification

<table>
<thead>
<tr>
<th>Microchip Parameter No.</th>
<th>Sym</th>
<th>Characteristic</th>
<th>Min</th>
<th>Typ</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>90</td>
<td>TSU:STA</td>
<td>START condition</td>
<td>100</td>
<td>4700</td>
<td></td>
<td>ns</td>
<td>Only relevant for repeated START condition</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Setup time</td>
<td>400</td>
<td>600</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>91</td>
<td>THD:STA</td>
<td>START condition</td>
<td>100</td>
<td>4000</td>
<td></td>
<td>ns</td>
<td>After this period the first clock pulse is generated</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Hold time</td>
<td>400</td>
<td>600</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>92</td>
<td>TSU:STO</td>
<td>STOP condition</td>
<td>100</td>
<td>4700</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Setup time</td>
<td>400</td>
<td>600</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>93</td>
<td>THD:STO</td>
<td>STOP condition</td>
<td>100</td>
<td>4000</td>
<td></td>
<td>ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>Hold time</td>
<td>400</td>
<td>600</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Figure A-12: I²C Bus Data Timing Specification
### Table A-3: I²C Bus Data Timing Specification

<table>
<thead>
<tr>
<th>Microchip Parameter No.</th>
<th>Sym</th>
<th>Characteristic</th>
<th>Min</th>
<th>Max</th>
<th>Units</th>
<th>Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>100</td>
<td>THIGH</td>
<td>Clock high time</td>
<td>100 kHz mode</td>
<td>4.0</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>101</td>
<td>TLOW</td>
<td>Clock low time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>1.3</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>102</td>
<td>TR</td>
<td>SDA and SCL rise time</td>
<td>100 kHz mode</td>
<td>—</td>
<td>1000 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>103</td>
<td>TF</td>
<td>SDA and SCL fall time</td>
<td>100 kHz mode</td>
<td>—</td>
<td>300 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>90</td>
<td>TSU/STA</td>
<td>START condition setup time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>91</td>
<td>THD/STA</td>
<td>START condition hold time</td>
<td>100 kHz mode</td>
<td>4.0</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>106</td>
<td>THD/DAT</td>
<td>Data input hold time</td>
<td>100 kHz mode</td>
<td>—</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0</td>
<td>0.9</td>
<td>µs</td>
</tr>
<tr>
<td>107</td>
<td>TSU/DAT</td>
<td>Data input setup time</td>
<td>100 kHz mode</td>
<td>—</td>
<td>250</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>100</td>
<td>—</td>
<td>ns</td>
</tr>
<tr>
<td>92</td>
<td>TSU/STO</td>
<td>STOP condition setup time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>0.6</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>109</td>
<td>TAA</td>
<td>Output valid from clock</td>
<td>100 kHz mode</td>
<td>—</td>
<td>3500 ns</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>—</td>
<td>1000 ns</td>
<td></td>
</tr>
<tr>
<td>110</td>
<td>TBUF</td>
<td>Bus free time</td>
<td>100 kHz mode</td>
<td>4.7</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>400 kHz mode</td>
<td>1.3</td>
<td>—</td>
<td>µs</td>
</tr>
<tr>
<td>D102</td>
<td>Cb</td>
<td>Bus capacitive loading</td>
<td>—</td>
<td>—</td>
<td>400 pF</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** As a transmitter, the device must provide this internal minimum delay time to bridge the undefined region (min. 300 ns) of the falling edge of SCL to avoid unintended generation of START or STOP conditions.

**Note 2:** A fast-mode I²C-bus device can be used in a standard-mode I²C-bus system, but the requirement $T_{SU/DAT} \geq 250$ ns must then be met. This will automatically be the case if the device does not stretch the LOW period of the SCL signal. If such a device does stretch the LOW period of the SCL signal, it must output the next data bit to the SDA line with a time delay of $T_{SU/DAT} + T_{SU/DAT} = 1000 + 250 = 1250$ ns (according to the standard-mode I²C bus specification) before the SCL line is released.
APPENDIX B: CAN OVERVIEW

This appendix provides an overview of the Controller Area Network (CAN) bus. The CAN Section of this reference manual discusses the implementation of the CAN protocol in that hardware module.

Not available at this time.
APPENDIX C: MODULE BLOCK DIAGRAMS AND REGISTERS

This appendix summarizes the block diagrams of the major circuits.

Not available at this time.
APPENDIX D: REGISTER DEFINITIONS

This appendix summarizes the register definitions.

Not available at this time.
APPENDIX E: MIGRATION TIPS

This appendix gives an overview of some things that need to be taken into account when migrating code from other PICmicro families to the PIC18CXXX. For additional information, please also refer to the following application notes:

- AN716
- AN726

Three major changes Timer0 has from the mid-range family are:

1. The Timer0 no longer shares the prescaler with the Watchdog Timer.
2. Timer0 has 16-bit as well as 8-bit capability.
3. Timer0 can now be turned off.

The default state for Timer0 is an 8-bit counter for downward compatibility with the mid-range Timer0.

The prescaler is no longer shared between the Timer0 module and the Watchdog Timer. Each have their own separate prescalers.

Note: The Timer0 prescaler is not shared with the watchdog as a postscaler. The Watchdog Timer has its own dedicated postscaler. The control bits and prescaler select bits are implemented as EPROM configuration bits.

Thus, a prescaler assignment for the Timer0 has no effect on the Watchdog Timer, and vice-versa.

T1OSCEN bit is now OSCEN.

Note: To achieve a 1:1 prescaler assignment for the TMR0 register, turn off the TMR0 prescaler (PSA is cleared).

The program space is implemented as a single contiguous block, as compared to Microchip’s mid-range and high-end controllers that divided the program memory into pages.

If interrupt priority is not used, all interrupts are treated as high priority, and function the same way as in mid-range and high-end controllers.

If interrupt priority is not used (all interrupt priority bits are set), GIE and PEIE/GIEL function the same as mid-range and high-end controllers.

Look-up tables are implemented two ways in the 18CXXX devices. The Computed Goto is compatible with the PIC16CXXX and PIC17CXXX parts. Code written for those devices will run on the PIC18CXXX devices with little or no change.

Table reads are implemented on the PIC17CXXX and PIC18CXXX devices. However, table operations on the PIC18CXXX work differently than on the PIC17CXXX.
E.1 Differences to the Mid-Range CCP module

The CCP1 and CCP2 modules function exactly as the modules of the mid-range devices with two exceptions:
1. The time base used for capture and compare of the CCP module can come from either Timer1 or Timer3. The default time base for both CCP modules when configured as a capture or compare mode is Timer1. Timer3 can be selected by configuring the Timer3 T3CCPx control bits in the Timer3 control register.
2. A toggle on compare mode has been added to the CCP modules.

E.1.1 Compatibility to PIC16CXX Interrupts

When the IPE bit in the RCON register is clear, the GIE/GIEH bit (INTCON<7>) is the global interrupt enable for all interrupts. The PEIE/GIEL bit (INTCON<6>) is still used to enable peripheral interrupts. The high priority interrupt vector is the same as the PIC16CXX interrupt vector and returns will work the same way. ISR code written for the PIC16CXX devices run unchanged as long as IPE=0.
Appendix

## E.2 Instruction Set Comparison

Table E-1 shows all the PIC18CXXX instructions and if that instruction is available in the other families (PIC17CXXX, PIC16CXX, or PIC16C5XX). Some instructions may be available, but have slightly different characteristics. Attached are notes regarding any major differences.

### Table E-1: PIC18CXXX Instruction Set Comparison

<table>
<thead>
<tr>
<th>BYTE-ORIENTED FILE REGISTER OPERATIONS</th>
<th>PIC18CXXX Instruction Set</th>
<th>PIC17CXX</th>
<th>PIC16CXX</th>
<th>PIC16C5X</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADDWF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>ADDWFCE</td>
<td>Yes</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ANDWF</td>
<td>Yes (Note 1)</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>COMF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>CPFSEQ</td>
<td>Yes</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CPFSGT</td>
<td>Yes</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CPFSLT</td>
<td>Yes</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>DAW</td>
<td>Yes (Note 5)</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>DCFSNZ</td>
<td>Yes</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>DECF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>DECFSZ</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>INCF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>INCFSZ</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>INFSNZ</td>
<td>Yes</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>IORWF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>MOVF</td>
<td>—</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>MOVFF</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>MOVWF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>MULWF</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>NEGF</td>
<td>Yes (Note 6)</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>NOP</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
<td></td>
</tr>
</tbody>
</table>

**Note 1:** CLRF and SETF instructions do not have 's' bit that exists in the PIC17CXX instructions.

2: CLRF does not exist as an instruction, but since the WREG is mapped in address space, WREG can be cleared using the CLRF instruction.

3: The RLCF and RRCF instructions are functionally identical to the PIC16CXX instructions RLF and RR. 

4: CALL and GOTO instructions are now 2-word instructions.

5: DAW always has the WREG as the destination. The PIC17CXX can also have a file as the destination.

6: NEGW is replaced by NEGF with the file register always being the destination.

7: The mnemonics for 17CXX instructions TABLRD and TABLWT are changed to TBLRD and TBLWT, respectively, and these instructions are special instructions which only exchange data between the TABLAT and the program memory.

8: The lower nibble of the Bank Select Register (BSR) now specifies which bank of RAM is selected.
## BYTE-ORIENTED FILE REGISTER OPERATIONS (Continued)

<table>
<thead>
<tr>
<th>Instruction Set</th>
<th>PIC17CXX</th>
<th>PIC16CXX</th>
<th>PIC16C5X</th>
</tr>
</thead>
<tbody>
<tr>
<td>RLCF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes (Note 3)</td>
</tr>
<tr>
<td>RLCNF</td>
<td>Yes</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>RLF</td>
<td>Yes</td>
<td>Yes (Note 3)</td>
<td>Yes (Note 3)</td>
</tr>
<tr>
<td>RRNF</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>SETF</td>
<td>Yes (Note 1)</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>SUBFB</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>SUBWF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>SUBWFB</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>SWAPF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>TABLERD</td>
<td>Yes (Note 7)</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>TBLWT</td>
<td>Yes (Note 7)</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>TSTFSZ</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>XORWF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
</tbody>
</table>

## BIT-ORIENTED FILE REGISTER OPERATIONS

<table>
<thead>
<tr>
<th>Instruction Set</th>
<th>PIC17CXX</th>
<th>PIC16CXX</th>
<th>PIC16C5X</th>
</tr>
</thead>
<tbody>
<tr>
<td>BCF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>BSF</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>BTFSC</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>BTFSS</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>BTG</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
</tbody>
</table>

### Note 1:
- CLRF and SETF instructions do not have 's' bit that exists in the PIC17CXX instructions.
- CLRW does not exist as an instruction, but since the WREG is mapped in address space, WREG can be cleared using the CLRF instruction.
- The RLCF and RRNF instructions are functionally identical to the PIC16CXX instructions RLF and RRF.
- CALL and GOTO instructions are now 2-word instructions.
- DAW always has the WREG as the destination. The PIC17CXX can also have a file as the destination.
- NEGW is replaced by NEGFW with the file register always being the destination.
- The mnemonics for 17CXX instructions TABLERD and TBLWT are changed to TBLRD and TBLWT, respectively, and these instructions are special instructions which only exchange data between the TABLAT and the program memory.
- The lower nibble of the Bank Select Register (BSR) now specifies which bank of RAM is selected.
## Section 36. Appendix

### CONTROL OPERATIONS

<table>
<thead>
<tr>
<th>Instruction Set</th>
<th>PIC17CXX</th>
<th>PIC16CXX</th>
<th>PIC16C5X</th>
</tr>
</thead>
<tbody>
<tr>
<td>BC</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BN</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BNC</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BNN</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BNV</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BNZ</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BRA</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BV</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>BZ</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>CALL (Note 4)</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>CLRWDT</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>GOTO (Note 4)</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>POP</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>PUSH</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>RCALL</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>RESET</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>RETFIE</td>
<td>Yes</td>
<td>Yes</td>
<td>—</td>
</tr>
<tr>
<td>RETURN</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>SLEEP</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
</tbody>
</table>

### LITERAL OPERATIONS

<table>
<thead>
<tr>
<th>Instruction Set</th>
<th>PIC17CXX</th>
<th>PIC16CXX</th>
<th>PIC16C5X</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADDLW</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>ANDLW</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>IORLW</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>LFSR</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>MOVLB</td>
<td>Yes (Note 8)</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>MOVLW</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>MULLW</td>
<td>Yes</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>RETLW</td>
<td>Yes</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>SUBLW</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
<tr>
<td>XORLW</td>
<td>Yes</td>
<td>Yes</td>
<td>Yes</td>
</tr>
</tbody>
</table>

Note 1: CLRF and SETF instructions do not have 's' bit that exists in the PIC17CXX instructions.

2: CLRF does not exist as an instruction, but since the WREG is mapped in address space, WREG can be cleared using the CLRF instruction.

3: The RLCF and RRCF instructions are functionally identical to the PIC16CXX instructions RLF and RRF.

4: CALL and GOTO instructions are now 2-word instructions.

5: DAW always has the WREG as the destination. The PIC17CXX can also have a file as the destination.

6: NEGW is replaced by NEGF with the file register always being the destination.

7: The mnemonics for 17CXX instructions TABLDRD and TABLWRT are changed to TBLRD and TBLWT, respectively, and these instructions are special instructions which only exchange data between the TABLAT and the program memory.

8: The lower nibble of the Bank Select Register (BSR) now specifies which bank of RAM is selected.
Table E-2 shows the instructions that are used in the other PICmicro families, but are not used in the PIC18CXXX instruction set.

Table E-2: Instructions Not Implemented or Modified From PIC18CXXX

<table>
<thead>
<tr>
<th>BYTE-ORIENTED FILE REGISTER OPERATIONS</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>Instruction</td>
<td>PIC17CXX</td>
</tr>
<tr>
<td>CLRW</td>
<td>—</td>
</tr>
<tr>
<td>DAW</td>
<td>Yes (Note 6)</td>
</tr>
<tr>
<td>MOVFP</td>
<td>Yes (Note 7)</td>
</tr>
<tr>
<td>MOVPF</td>
<td>Yes (Note 7)</td>
</tr>
<tr>
<td>NEGW</td>
<td>Yes (Note 3)</td>
</tr>
<tr>
<td>RLF</td>
<td>—</td>
</tr>
<tr>
<td>RRF</td>
<td>—</td>
</tr>
<tr>
<td>TLRD</td>
<td>Yes</td>
</tr>
<tr>
<td>TLWT</td>
<td>Yes</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>CONTROL OPERATIONS</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>Instruction</td>
<td>PIC17CXX</td>
</tr>
<tr>
<td>CALL</td>
<td>Yes (Note 2)</td>
</tr>
<tr>
<td>GOTO</td>
<td>Yes (Note 2)</td>
</tr>
<tr>
<td>LCALL</td>
<td>Yes</td>
</tr>
<tr>
<td>MOVLR</td>
<td>Yes</td>
</tr>
<tr>
<td>OPTION</td>
<td>—</td>
</tr>
<tr>
<td>TRIS</td>
<td>—</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>LITERAL OPERATIONS</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>Instruction</td>
<td>PIC17CXX</td>
</tr>
<tr>
<td>MOVLR</td>
<td>Yes</td>
</tr>
</tbody>
</table>

Note 1: CLRW does not exist as an instruction, but since the WREG is mapped in address space, WREG can be cleared using the CLRF instruction.

2: CALL and GOTO instructions are now 2-word instructions.

3: NEGW is replaced by NEGF with the file register always being the destination.

4: The mnemonics for RLF and RRF have been changed to RLCF and RRCF, respectively, but the functionality is identical.

5: This instruction is not recommended in PIC16CXX devices.

6: The PIC17CXX may also specify the file as the destination.

7: When migrating software the MOVFF instruction can be used.
APPENDIX F: ASCII CHARACTER SET

The PICmicro assembler recognizes the ASCII characters shown in Table F-1.

Table F-1: ASCII Character Set (7-bit Code)

<table>
<thead>
<tr>
<th>Least Significant nibble (LSn)</th>
<th>0 (0)</th>
<th>1 (16)</th>
<th>2 (32)</th>
<th>3 (48)</th>
<th>4 (64)</th>
<th>5 (80)</th>
<th>6 (96)</th>
<th>7 (112)</th>
</tr>
</thead>
<tbody>
<tr>
<td>0 NUL DLE SP 0</td>
<td>@ P</td>
<td>' p</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 SOH DC1 !</td>
<td>A Q a q</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>2 STX DC2</td>
<td>2 B R b r</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>3 ETX DC3</td>
<td># 3 C S c s</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>4 EOT DC4</td>
<td>$ 4 D T d t</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5 ENQ NAK</td>
<td>% 5 E U e u</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6 ACK SYN &amp;</td>
<td>6 F V f v</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7 BEL</td>
<td>^ 7 G W g w</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8 BS CAN</td>
<td>( 8 H X h x</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>9 HT</td>
<td>) 9 I Y i y</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A LF SUB</td>
<td>^ : J Z j z</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B VT ESC +</td>
<td>; K [ k {</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C FF FS</td>
<td>. &lt; L \</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D CR GS</td>
<td>- = M ) m</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E SO RS</td>
<td>. &gt; N ^ n</td>
<td>~</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F SI US</td>
<td>/ ? O ~ DEL</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Note: To obtain the decimal value, add the decimal most significant nibble (MSn) to the decimal least significant nibble (LSn).
APPENDIX G: COMMON SOCKET PINOUTS

This appendix shows the pinouts for some of the standard sockets that are commonly used in prototype and production applications. You may use these pinouts when you wiretap your breadboard with a socket. These diagrams will make constructing, debugging, and troubleshooting with the Picmicro families quicker and easier.

These figures shown in this appendix are for sockets that correspond to the following package types:

PLCC-to-PGA sockets
- 28-pin PLCC/CLCC
- 44-pin PLCC/CLCC
- 68-pin PLCC/CLCC

DIP sockets
- 18-pin
- 28-pin
- 40-pin

SOIC sockets
- 18-pin
- 28-pin
Figure G-1: DIP Packages Pinouts (Bottom View)

18-pin

28-pin

40-pin

Bottom View

Bottom View

Bottom View
Figure G-2: PGA Package Pinout (Bottom View) for LCC Packages
Figure G-2: PGA Package Pinout (Bottom View) for LCC Packages (Continued)

68-pin

<table>
<thead>
<tr>
<th>61</th>
<th>63</th>
<th>65</th>
<th>67</th>
<th>1</th>
<th>3</th>
<th>5</th>
<th>7</th>
<th>9</th>
</tr>
</thead>
<tbody>
<tr>
<td>60</td>
<td>62</td>
<td>64</td>
<td>66</td>
<td>68</td>
<td>2</td>
<td>4</td>
<td>6</td>
<td>8</td>
</tr>
<tr>
<td>58</td>
<td>59</td>
<td></td>
<td></td>
<td></td>
<td>13</td>
<td>12</td>
<td></td>
<td></td>
</tr>
<tr>
<td>56</td>
<td>57</td>
<td></td>
<td></td>
<td></td>
<td>15</td>
<td>14</td>
<td></td>
<td></td>
</tr>
<tr>
<td>54</td>
<td>55</td>
<td></td>
<td></td>
<td></td>
<td>17</td>
<td>16</td>
<td></td>
<td></td>
</tr>
<tr>
<td>52</td>
<td>53</td>
<td></td>
<td></td>
<td></td>
<td>19</td>
<td>18</td>
<td></td>
<td></td>
</tr>
<tr>
<td>50</td>
<td>51</td>
<td></td>
<td></td>
<td></td>
<td>21</td>
<td>20</td>
<td></td>
<td></td>
</tr>
<tr>
<td>48</td>
<td>49</td>
<td></td>
<td></td>
<td></td>
<td>23</td>
<td>22</td>
<td></td>
<td></td>
</tr>
<tr>
<td>46</td>
<td>47</td>
<td></td>
<td></td>
<td></td>
<td>25</td>
<td>24</td>
<td></td>
<td></td>
</tr>
<tr>
<td>44</td>
<td>45</td>
<td>42</td>
<td>40</td>
<td>38</td>
<td>36</td>
<td>34</td>
<td>32</td>
<td>30</td>
</tr>
<tr>
<td>43</td>
<td>41</td>
<td>39</td>
<td>37</td>
<td>35</td>
<td>33</td>
<td>31</td>
<td>29</td>
<td>27</td>
</tr>
</tbody>
</table>
Figure G-2: PGA Package Pinout (Bottom View) for LCC Packages (Continued)

84-pin

```
  75  77  79  81  83  1  3  5  7  9  11
  74  76  78  80  82  84  2  4  6  8  10  13  12
  72  73  
  70  71  17  16
  68  69  19  18
  66  67  21  20
  64  65  
  62  63  25  24
  60  61  27  26
  58  59  29  28
  56  57  31  30
  54  55  52  50  48  46  44  42  40  38  36  34  32
  53  51  49  47  45  43  41  39  37  35  33
```

Bottom View
Figure G-3: SOIC Socket Pinouts (Bottom View)

18-pin

- 18
- 17
- 16
- 15
- Bottom View
- 14
- 13
- 12
- 11
- 10

28-pin

- 28
- 27
- 26
- 25
- 24
- 23
- 22
- Bottom View
- 21
- 20
- 19
- 18
- 17
- 16
- 15

© 2000 Microchip Technology Inc.
APPENDIX H: REVISION HISTORY

Revision A

This is the initial released revision of the Enhanced MCU Reference Guide Appendix.
Section 37. Glossary

A

A/D
See description under "Analog to Digital (A/D)".

Access RAM
This is a region of data memory RAM that can be accessed regardless of the currently selected bank. This allows special function registers to be accessed by the instruction without changing the currently selected bank. Access RAM also contains some General Purpose Registers (GPRs). This is useful for the saving of required variables during context switching (such as during an interrupt).

Acquisition Time (TACQ)
This is related to Analog to Digital (A/D) converters. This is the time that the PIC18CXXX A/D's holding capacitor acquires the analog input voltage level connected to it. When the GO bit is set, the analog input is disconnected from the holding capacitor and the A/D conversion is started.

ALU
Arithmetical Logical Unit. Device logic that is responsible for the mathematical (add, subtract, ...), logical (and, or, ...), and shifting operation.

Analog to Digital (A/D)
The conversion of an analog input voltage to a ratiometric digital equivalent value.

Assembly Language
A symbolic language that describes the binary machine code in a readable form.

AUSART
Addressable Universal Synchronous Asynchronous Receiver Transmitter. This module can either operate as a full duplex asynchronous communications port, or a half duplex synchronous communications port. When operating in the asynchronous mode, the USART can be interfaced to a PC's serial port.
Bank
This is a method of addressing Data Memory. Since enhanced devices have 8-bits for direct addressing, instructions can address up to 256 bytes. To allow more data memory to be present on a device, data memory is partitioned into contiguous banks of 256 bytes each. To select the desired bank, the bank selection register (BSR) needs to be appropriately configured. 16 banks can be implemented.

Baud
Generally this is how the communication speed of serial ports is described. Equivalent to bits per second (bps).

BCD
See description under "Binary Coded Decimal (BCD)".

Binary Coded Decimal (BCD)
Each 4-bit nibble expresses a digit from 0-9. Usually two digits are contained in a byte yielding a range of 0 - 99.

BOR
See description under "Brown-out Reset (BOR)".

Brown-out
A condition where the supply voltage of the device temporarily falls below the specified minimum operation point. This can occur when a load is switched on and causes the system/device voltage to drop.

Brown-out Reset (BOR)
Circuitry which will force the device to the RESET state if the device's power supply voltage falls below a specified voltage level. Some devices have an internal BOR circuit, while other devices would require an external circuit to be created.

Bus width
This is the number of bits of information that a bus carries. For the Data Memory, the bus width is 8-bits. For enhanced devices the Program Memory bus width is 16-bits.
Section 37. Glossary

C

Capture
A function of the CCP module in which the value of a timer/counter is "captured" into a holding
register module when a predetermined event occurs.

CCP
Capture, Compare, Pulse Width Modulation (PWM). The CCP module can be configured to operate
as an input capture, or a timer compare, or a PWM output.

Compare
A function of the CCP module in which the device will perform an action when a timer's register
value matches the value in the compare register.

Compare Register
A 16-bit register that contains a value that is compared to the 16-bit TMR1 register. The compare
function triggers when the counter matches the contents of the compare register.

Capture Register
A 16-bit register that is loaded with the value of the 16-bit TMR1 register when a capture event
occurs.

Configuration Word
This is a non-volatile memory location that specifies the characteristics that the device will have
for operation (such as oscillator mode, WDT enable, start-up timer enables). These characteristics
can be specified at the time of device programming. For EPROM memory devices, as long
as the bit is a '1', it may at a later time be programmed as a '0'. The device must be erased for a
'0' to be returned to a '1'.

Conversion Time (Tconv)
This is related to Analog to Digital (A/D) converters. This is the time that the PIC18CXXX A/D's
converter requires to convert the analog voltage level on the holding capacitor to a digital value.

CPU
Central Processing Unit. Decodes the instructions, and determines the operands and operations
that are needed for program execution. Arithmetic, logical, or shift operations are passed to the
ALU.
D

**D/A**
See description under "Digital to Analog".

**DAC**
Digital to analog converter.

**Data Bus**
The bus which is used to transfer data to and from the data memory.

**Data EEPROM**
Data Electrically Erasable Programmable Read Only Memory. This memory is capable of being programmed and re-programmed by the CPU to ensure that in the case of a power loss, critical values/variables are retained in the non-volatile memory.

**Data Memory**
The memory that is on the Data Bus. This memory is volatile (SRAM) and contains both the Special Function Registers and General Purpose Registers.

**Direct Addressing**
When the Data Memory Address is contained in the Instruction. The execution of this type of instruction will always access the data at the embedded address.

**Digital to Analog**
The conversion of a digital value to an equivalent ratiometric analog voltage.

E

**EEPROM**
Electrically Erasable Programmable Read Only Memory. This memory has the capability to be programmed and erased in-circuit.

**EPROM**
Electrically Programmable Read Only Memory. This memory has the capability to be programmed in-circuit. Erasing requires that the program memory be exposed to UV light.

**EXTRC**
External Resistor-Capacitor (RC). Some devices have a device oscillator option that allows the clock to come from an external RC. This is the same as RC mode on some devices.

F

**FLASH Memory**
This memory has the capability to be programmed and erased in-circuit. Program Memory technology that is almost functionally equivalent to Program EEPROM Memory.

**Fosc**
Frequency of the device oscillator.
Section 37. Glossary

G

GIO
General Input/Output.

GPIO
General Purpose Input/Output.

GPR
General Purpose Register (RAM). A portion of the data memory that can be used to store the program’s dynamic variables.

H

Harvard Architecture
In this architecture, the Program Memory and Data Memory buses are separated. This allows concurrent accesses to Data Memory and Program Memory, which increases the performance of the device. All PICmicro devices implement a Harvard Architecture.

Holding Capacitor
This is a capacitor in the Analog to Digital (A/D) module which “holds” an analog input level once a conversion is started. During acquisition, the holding capacitor is charged/discharged by the voltage level on the analog input pin. Once the conversion is started, the holding capacitor is disconnected from the analog input and “holds” this voltage for the A/D conversion.

HS
High Speed. One of the device oscillator modes. The oscillator circuit is tuned to support the high frequency operation. Currently this allows for operation from 4 MHz to 25 MHz.
I²C
Inter-Integrated Circuit. This is a two wire communication interface. This feature is one of the modes of the "SSP" and "MSSP" modules.

Indirect Addressing
When the Data Memory Address is not contained in the Instruction, the instruction operates on the INDF address, which causes the Data Memory Address to be the value in the FSR register. The execution of the instruction will always access the data at the address pointed to by the FSR register.

Instruction Bus
The bus which is used to transfer instruction words from the program memory to the CPU.

Instruction Fetch
Due to the Harvard architecture, when one instruction is to be executed, the next location in program memory is "fetched" and ready to be decoded as soon as the currently executing instruction is completed.

Instruction Cycle
The events for an instruction to execute. There are four events which can generally be described as: Decode, Read, Execute, and Write. Not all events will be done by all instructions. To see the operations during the instruction cycle, please look at the description of each instruction. Four external clocks (Tosc) make one instruction cycle (Tcy).

Interrupt
A signal to the CPU that causes the program flow to be forced to the Interrupt Vector Address (04h in program memory). Before the program flow is changed, the contents of the Program Counter (PC) are forced onto the hardware stack, so that program execution may return to the interrupted point.

INTRC
Internal Resistor-Capacitor (RC). Some devices have a device oscillator option that allows the clock to come from an internal RC combination.
Section 37. Glossary

L

LCD
Liquid Crystal Display. Useful for giving visual status of a system. This may require the specification of custom LCD glass.

LED
Light Emitting Diode. Useful for giving visual status of a system.

Literal
This is a constant value that is embedded in an instruction word.

Long Word Instruction
An instruction word that embeds all the required information (opcode and data) into a single word. This ensures that every instruction is accessed and executed in a single instruction cycle.

LP
One of the device oscillator modes. Used for low frequency operation which allows the oscillator to be tuned for low power consumption. Operation is up to 200 kHz.

LSb
Least Significant Bit.

LSB
Least Significant Byte.

M

Machine cycle
This is a concept where the device clock is divided down to a unit time. For PICmicro devices, this unit time is 4 times the device oscillator (4Tosc), also known as Tcy.

MSb
Most Significant Bit.

MSB
Most Significant Byte.

MSSP
Master Synchronous Serial Port. The MSSP has two operational functions. The first is a "Serial Peripheral Interface (SPI)" and the second is the Inter-Integrated Circuit ("I²C"). The I²C function supports both master and slave functions in hardware.
Non-Return to Zero (NRZ)
Two-level encoding used to transmit data over a communications medium. A bit value of ‘1’ indicates a high voltage signal. A bit value of ‘0’ indicates a low voltage signal. The data line defaults to a high level.

NRZ
See description under "Non-Return to Zero (NRZ)".

Opcde
The portion of the 16-bit instruction word that specifies the operation that needs to occur. The opcode is of variable length depending on the instruction that needs to be executed. The opcode varies from 4-bits to 8-bits. The remainder of the instruction word contains program or data memory information.

Oscillator Start-up Timer (OST)
This timer counts 1024 crystal/resonator oscillator clock cycles before releasing the internal RESET signal.

OST
See description under "Oscillator Start-up Timer (OST)".
Section 37. Glossary

P

Pages
Method of addressing the Program Memory. Mid-range devices have 11-bit addressing for CALL and GOTO instructions, which gives these instructions a 2-Kword reach. To allow more program memory to be present on a device, program memory is partitioned into contiguous pages, where each page is 2-Kwords. To select the desired page, the page selection bits (PCLATCH<5:4>) need to be appropriately configured. Since there are presently 2 page selection bits, 4 pages can be implemented. The enhanced devices do not have paging. PIC16CXXX code migrates to the PIC18CXXX without modification (with respect to paging). Optimization may be implemented.

Parallel Slave Port (PSP)
A parallel communication port which is used to interface to a microprocessor's 8-bit data bus.

POP
A term used to refer to the action of restoring information from a stack (software and/or hardware). See "Serial Peripheral Interface (SPI)".

Postscaler
A circuit that slows the rate of the interrupt generation (or WDT Reset) from a counter/timer by dividing it down.

Power-on Reset (POR)
Circuitry which determines if the device power supply voltage rose from a powered down level (0V). If the device power supply voltage is rising from ground, a device RESET occurs and the PWRT is started.

Power-up Timer (PWRT)
A timer which holds the internal RESET signal low for a timed delay to allow the device voltage to reach the valid operating voltage range. Once the timer times out, the OST circuitry is enabled (for all crystal/resonator device oscillator modes).

Prescaler
A circuit that slows the rate of a clocking source to a counter/timer.

Program Bus
The bus used to transfer instruction words from the program memory to the CPU.

Program Counter
A register which specifies the address in program memory that contains the next instruction to execute.

Program Memory
Any memory that is on the program memory bus. Static variables may be contained in program memory, such as tables.

PSP
See description under "Parallel Slave Port (PSP)".

Pulse Width Modulation (PWM)
A serial signal in which the information is contained in the width of a (high) pulse of a constant frequency signal. A PWM output, from the CCP module, of the same duty cycle requires no software overhead.

PUSH
A term used to refer to the action of saving information onto a stack (software and/or hardware). See "Serial Peripheral Interface (SPI)".

PWM
See description under "Pulse Width Modulation (PWM)".
Q

Q-cycles
This is the same as a device oscillator cycle. There are 4 Q-cycles for each instruction cycle.

R

RC
Resistor-Capacitor. The default configuration for the device oscillator. This allows a "Real-Cheap" implementation for the device clock source. This clock source does not supply an accurate time-base.

Read-Modify-Write
This is where a register is read, then modified, and then written back to the original register. This may be done in one instruction cycle or multiple instruction cycles.

Register File
This is the Data Memory. Contains the SFRs and GPRs.

ROM
Read Only Memory. Memory that is fixed and cannot be modified.
Section 37. Glossary

S

Sampling Time
Sampling time is the complete time to get an A/D result. It includes the acquisition time and the conversion time.

Serial Peripheral Interface (SPI)
This is one of the modes of the "SSP" and "MSSP" modules. This is typically a 3-wire interface, with a data out line, a data in line, and a clock line. Since the clock is present, this is a synchronous interface.

SFR
Special Function Register. These registers contain the control bits and status information for the device.

Single Cycle Instruction
An instruction that executes in a "single" machine cycle (T CY).

SLEEP
This is a low power mode of the device, where the device’s oscillator circuitry is disabled. This reduces the current the device consumes. Certain peripherals may be placed into modes where they continue to operate.

Special Function Registers (SFR)
These registers contain the control bits and status information for the device.

SPI
See description under "Serial Peripheral Interface (SPI)".

SSP
Synchronous Serial Port. The SSP has two operational functions. The first is a "Serial Peripheral Interface (SPI)" and the second is the Inter-Integrated Circuit ("I\(^2\)C"). The I\(^2\)C function supports the slave function in hardware and has additional status information to support a software implemented master.

Stack
A portion of the CPU that retains the return address for program execution. The stack gets loaded with the value in the Program Counter when a CALL instruction is executed or if an interrupt occurs.
PIC18C Reference Manual

T

TAD
In the A/D Converter, the time for a single bit of the analog voltage to be converted to a digital value.

TCY
The time for an instruction to complete. This time is equal to Fosc/4 and is divided into four Q-cycles.

TOSC
The time for the single period of the device oscillator.

U

USART
Universal Synchronous Asynchronous Receiver Transmitter. This module can either operate as a full duplex asynchronous communications port, or a half duplex synchronous communications port. When operating in the asynchronous mode, the USART can be interfaced to a PC’s serial port.
Section 37. Glossary

V

Voltage Reference (VREF)
A voltage level that can be used as a reference point for A/D conversions (AVDD and AVSS) or the trip point for comparators.

von Neumann Architecture
In this architecture the Program Memory and Data Memory are contained in the same area and use the same bus. This means that accesses to the program memory and data memory must occur sequentially, which affects the performance of the device.

W

W Register
See description under "Working Register (WREG).

Watchdog Timer (WDT)
Used to increase the robustness of a design by recovering from software flows that were not expected in the design of the product or from other system related issues. The Watchdog Timer causes a RESET if it is not cleared prior to overflow. The clock source for a PICmicro device is an on-chip RC oscillator which enhances system reliability.

WDT
Watchdog Timer.

Working Register (WREG)
Can also be thought of as the accumulator of the device. Also used as an operand in conjunction with the ALU during two operand instructions.

X

XT
One of the device oscillator modes. Used for operation from 100 kHz to 4 MHz.
37.1 Revision History

Revision A

This is the initial released revision of the Glossary.
APPENDIX I:  SOURCE CODE

The software supplied herewith by Microchip Technology Incorporated (the “Company”) for its PICmicro® Microcontroller is intended and supplied to you, the Company’s customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved. Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

THIS SOFTWARE IS PROVIDED IN AN “AS IS” CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

******************************************************************************
PIC18CXX2 EXAMPLE CODE FOR PICDEM-2
******************************************************************************

; Software License Agreement

; The software supplied herewith by Microchip Technology Incorporated
; (the "Company") for its PICmicro® Microcontroller is intended and
; supplied to you, the Company’s customer, for use solely and
; exclusively on Microchip PICmicro Microcontroller products. The
; software is owned by the Company and/or its supplier, and is
; protected under applicable copyright laws. All rights are reserved.
; Any use in violation of the foregoing restrictions may subject the
; user to criminal sanctions under applicable laws, as well as to
; civil liability for the breach of the terms and conditions of this
; license.
;
; THIS SOFTWARE IS PROVIDED IN AN “AS IS” CONDITION. NO WARRANTIES,
; WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED
; TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
; PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT,
; IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR
; CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

;******************************************************************************
; PIC18CXX2 EXAMPLE CODE FOR PICDEM-2
;******************************************************************************

; TITLE: USART Demo Demonstration
; FILENAME: usart.asm
; REVISION HISTORY: A 5/13/00 jb format change
; HARDWARE: PICDEM-2 board
; FREQUENCY: 4MHz

;******************************************************************************
This program demonstrates basic functionality of the USART.

Port B is connected to 8 LEDs.
When the PIC18C452 receives a word of data from the USART, the value is displayed on the LEDs and is retransmitted to the host computer.

Set terminal program to 9600 baud, 1 stop bit, no parity

```c
list p=18c452; set processor type
list n=0; supress page breaks in list file
include <P18c452.INC>
```

; Reset and Interrupt Vectors
org 00000h; Reset Vector
gotoStart

org 00008h; Interrupt vector
gotoIntVector

; Program begins here
org 00020h; Beginning of program EPROM
Start
clrLATB; Clear PORTB output latches
clrTRISB; Config PORTB as all outputs
bcf TRISC,6; Make RC6 an output

movlw 19h; 9600 baud @4MHz
movwf SPBRG

bsf TXSTA,TXEN; Enable transmit
bsf TXSTA, BRGH; Select high baud rate

bsf RCSTA, SPEN; Enable Serial Port
bsf RCSTA, CREN; Enable continuous reception

bcf PIR1, RCIF; Clear RCIF Interrupt Flag
bsf PIE1, RCIE; Set RCIE Interrupt Enable
bsf INTCON, PEIE; Enable peripheral interrupts
bsf INTCON, GIE; Enable global interrupts

; Main loop
Main
  gotoMain; loop to self doing nothing

;******************************************************************************
; Interrupt Service Routine

IntVector
  ; save context (WREG and STATUS registers) if needed.
  btfssPIR1,RCIF; Did USART cause interrupt?
  gotoOtherInt; No, some other interrupt

  movlw06h; Mask out unwanted bits
  andwfRCSTA,W; Check for errors
  btfssSTATUS,Z; Was either error status bit set?
  gotoRcvError; Found error, flag it

  movfRCREG,W; Get input data
  movwfLATB; Display on LEDs
  movfTXREG; Echo character back
  gotoISREnd; go to end of ISR, restore context, return

RcvError
  bcf RCSTA,CREN; Clear receiver status
  bsf RCSTA,CREN
  movlwFFh; Light all LEDs
  movwfPORTB
  gotoISREnd; go to end of ISR, restore context, return

OtherInt
  goto$; Find cause of interrupt and service it before returning from
  ; interrupt. If not, the same interrupt will re-occur as soon
  ; as execution returns to interrupted program.

ISREnd
  ; Restore context if needed.
  retfie

end
SOFTWARE LICENSE AGREEMENT

The software supplied herewith by Microchip Technology Incorporated (the "Company") for its PICmicro® Microcontroller is intended and supplied to you, the Company’s customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved. Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

***************************************************************
PIC18CXX2 EXAMPLE CODE FOR PICDEM-2
***************************************************************
|
TITLE: Button Press Demonstration
FILENAME: btnn.asm
REVISION HISTORY: A 5/13/00 jb format change
HARDWARE: PICDEM-2 board
FREQUENCY: 4MHz

This program demonstrates how to read a push-button and control LED’s.
Port B is connected to 8 LEDs.
RA4 is connected to a switch (S2).
This program increments a file register count every time S2 is pressed.
The value of count is displayed on the LEDs connected to Port B.
The LEDs should increment in a binary manner each time S2 is pressed.

list p=18c452
#include<P18C452.INC>

***************************************************************
variables
Countequ0x000
; reset vectors
org 00000h; Reset Vector
gotoStart

;****
program code starts here
org 00020h; Beginning of program EPROM
Start
clrLATB; Clear PORTB output latch
clrTRISB; Make PORTB pins all outputs
clrCount; Clear Count
Loop
  btfscPORTA,4; Has S2 been pressed? (Normally high, goes low when pressed.)
  gotoLoop; No, check again
  IncCount
  incfCount,F; Increment Count
  movffCount,LATB; move Count to PORTB
Debounce
  btfssPORTA,4; Has key been released?
  gotoDebounce; No, wait some more
  gotoLoop; yes, wait for next key press

END ; directive indicates end of code
Software License Agreement

The software supplied herewith by Microchip Technology Incorporated (the "Company") for its PICmicro® Microcontroller is intended and supplied to you, the Company’s customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved.

Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

************************************************************
PIC18CXX2 EXAMPLE CODE FOR PICDEM-2
************************************************************

This program uses the advanced features of the PIC18C452, specifically full hardware support for master mode I2C.

This program loads EEPROM address 0x00 to 0xFF with 0x00 to 0xFF (each location contains its own address).

Each location is then read out, and compared to what is expected. The data is displayed on the LEDs. If the data is wrong, the TX LED will flash briefly before proceeding to the next address.

Revised Version(05-05-99).

Note: 1) All timing is based on a reference crystal frequency of 4MHz which is equivalent to an instruction cycle time of 1 usec.
2) Address and literal values are read in hexadecimal unless otherwise specified.
3) The PIC18C452 MSSP module offers full hardware support for...
; master mode I2C.
;******************************************************************************
LIST P=18C452
#include <P18C452.INC>
listn=0 ; suppress list file page breaks
listST=off; suppress list file symbol table
;******************************************************************************
; Register File Assignment
EEADDRequ0x000; Address register
EEDATAequ0x001; data to read/write
EESLAVEeqs 0x002; Device address (1010xxxy)
DelayCtr1equ0x003; delay routine counter
DelayCtr2equ0x004; delay routine counter
SlaveAddrequ0xA0; slave address literal
;******************************************************************************
; Vector Assignment
ORG 0x00000
gotoStart; Reset Vector
;******************************************************************************
; Main Program
ORG 0x00020; Start of Program space
Start
; initialize PORTB
clrLATB; Clear PORTB output latch
clrTRISB; Set PORTB as all outputs
; configure SSP for hardware master mode I2C
bsf SSPSTAT,SM; I2C slew rate control disabled
bsf SSPCON1,SSPM3; I2C master mode in hardware
bsf SSPCON1,SSPEN; enable SSP module
movlw0x09; set I2C clock rate to 100kHz
movfSSPADD
bsf TRISC,3; I2C SCL pin is input
bsf PORTC,3; (will be controlled by SSP)
bsf TRISC,4; I2C SDA pin is input
bsf PORTC,4; (will be controlled by SSP)
movlwSlaveAddr; KEYPROM I2C address
movwf EESLAVE

Main
  clrf PORTB; initialize variables
  clrf EEDATA
  clrf EEADDR

WrEEPROM
  bsf PORTB,7; indicate write, light TX LED
  rcallDelay
  bcf EESLAVE,0; write mode
  rcallWakeSlave; gets slave attention
  rcall1WrADDR; sends EEPROM address
  rcall1WrDATA; sends data to slave
  rcall1Stop; send stop bit
  incf PORTB,F; increment count
  incf EEDATA,F; increment data
  incf EEADDR,F; Point to next address
  btfss EEADDR,7; at end of EEPROM?
  goto WrEEPROM; no, write more data

RdLoop
  clrf PORTB; initialize variables
  clrf EEDATA
  clrf EEADDR

RdEEPROM
  rcallDelay
  bcf EESLAVE,0; write mode
  rcall1WakeSlave; gets slave attention
  rcall1WrADDR; sends EEPROM address
  rcall1Stop; send stop bit
  bsf EESLAVE,0; read mode
  rcall1WakeSlave; gets slave attention
  rcall1RdDATA; receive one data byte, leaves idle
  movf EEDATA,W; get data
  movwf PORTB; move received data to PORTB
  xorwf EEADDR,W; compare data with address
  bZ GoodData; branch if DATA = ADDR
  rcall Errorloop; DATA is wrong, indicate error
GoodData

Source Code

GoodData

Source Code
btfsc SSPCON2, RSEN; has repeated start been sent yet?
goto -2; no, loop back to test
bra rWakeSlave; send slave address again

;***************************************************************
; writes EEPROM memory address, hangs if no ACK
WrADDR
bcf PIR1, SSPIF; clear interrupt flag
movff EEADDR, SSPBUF; move EEPROM address to SSPBUF
btfss PIR1, SSPIF; has SSP completed sending EEPROM Address?
goto -2; no, loop back to test
btfsc SSPCON2, ACKSTAT; has slave sent ACK?
goto -2; no, try again
return

;***************************************************************
; Sends one byte of data to slave, hangs if noACK
WrDATA
bcf PIR1, SSPIF; clear interrupt flag
movff EEDATA, SSPBUF; move data to SSPBUF
btfss PIR1, SSPIF; has SSP completed sending data to EEPROM?
goto -2; no, loop back to test
btfsc SSPCON2, ACKSTAT; has slave sent ACK?
goto -2; no, try again
return

;***************************************************************
; receive one byte from slave
; do not send ACK, send stop bit instead
RdDATA
bcf PIR1, SSPIF; clear interrupt flag
bsf SSPCON2, RCEN; enable receive mode
btfss PIR1, SSPIF; has SSP received a data byte?
goto -2; no, loop back to test
bsf SSPCON2, ACKDT; no ACK
bsf SSPCON2,ACKEN; send ACKDT bit
btfsc SSPCON2,ACKEN; has ACKDT bit been sent yet?
goto $-2; no, loop back to test
bsf SSPCON2,PEN; send stop bit
btfsc SSPCON2,PEN; has stop bit been sent?
goto $-2; no, loop back to test
movff SSPBUF,EE DATA; save data to RAM
bsf SSPCON2,RCEN; disable receive mode
return

;*******************************************************
; Sends stop bit, waits until sent
Stop
bsf SSPCON2, PEN; send stop bit
btfsc SSPCON2, PEN; has stop bit been sent?
goto $-2; no, loop back to test
return

;*******************************************************
; a delay of 98.57mS
Delay
movlw 0x80
movwf DelayCtr2; preset
clr DelayCtr1; clear counter
Delay1
decsz DelayCtr1; decrement counter
bra Delay1; back to top of loop
decsz DelayCtr2; decrement counter
bra Delay1; back to top of loop
return

END
Software License Agreement

The software supplied herewith by Microchip Technology Incorporated (the "Company") for its PICmicro® Microcontroller is intended and supplied to you, the Company’s customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved. Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

PIC18CXX2 EXAMPLE CODE FOR PICDEM-2

TITLE: Interrupt Priority Demonstration
FILENAME: intrp_ex.asm
REVISION HISTORY: A 5/13/00 jb format change
HARDWARE: PICDEM-2 board
FREQUENCY: 4MHz

This program uses Timer1 and Timer3 to demonstrate the use of interrupt priority.

Timer1 is configured for high-priority interrupts and Timer3 is configured for low-priority interrupts. By writing to the PORTB LEDS, it is shown that a high-priority interrupts override low-priority interrupts.

list p=18c452, n=48, t=ON, st=OFF
#include "p18c452.inc"

------------------BIT DEFINITIONS-----------------------
F EQU 0x0001

------------------VECTORS-----------------------------

EQU Sx0001
org 0x000000; reset vector
bra START

org 0x000008; high priority interrupt vector
bra TMR1_ISR

org 0x000018; low priority interrupt vector
bra TMR3_ISR

;--------------------PROGRAM-----------------------------------
START
rcall INIT

;Set up priority interrupts.
bsf RCON,IPEN; enable priority interrupts.
bsf IPR1,TMR1IP;set Timer1 as a high priority interrupt source
bcf IPR2,TMR3IP;set Timer3 as a low priority interrupt source
bcf PIR1,TMR1IF;clear the Timer1 interrupt flag
bcf PIR2,TMR3IF;clear the Timer3 interrupt flag
bsf PIE1,TMR1IE; enable Timer1 interrupts
bsf PIE2,TMR3IE; enable Timer3 interrupts
bsf INTCON,GIEH;set the global interrupt enable bits
bsf INTCON,GIEL;

;Timer1 setup
clrFT1CON
clrf TMR1H; clear Timer1 high
clrf TMR1L; clear Timer1 low
bsf T1CON,TMR1ON; turn on Timer1

;Timer3 setup
clrFT3CON
movlw 0xF0
movwf TMR3H; write 0xF000 to Timer3
clrf TMR3L
bsf T3CON,TMR3ON; turn on Timer3

MLOOP
goto MLOOP

;------------------------SUBROUTINES-------------------------------
TMR1_ISR ; high priority isr

bcf PIR1,TMR1IF; clear the Timer1 interrupt flag.
bcf PORTB,0; turn off PORTB<0> to indicate high priority
; interrupt has overridden low priority.
bsf PORTB,7; Turn on PORTB<7> to indicate high priority
; interrupt is occurring.
T1POLL
btfssPIR1,TMR1IF; Poll TMR1 interrupt flag to wait for another
; TMR1 overflow.
bra T1POLL
bcf PIR1,TMR1IF; Clear the Timer1 interrupt flag again.
bcf PORTB,7; Turn off PORTB<7> to indicate the
; high-priority ISR is over.
retfie

TMR3_ISR ; low priority isr
bcf PIR2,TMR3IF; Clear the TMR3 interrupt flag.
movlw0xF0; Load TMR3 with the value 0xF000
movwfTMR3H
clrfTMR3L
bsf PORTB,0; Turn on PORTB<0> to indicate low priority
; interrupt is occurring.
T3POLL
btfssPIR2,TMR3IF; Poll TMR3 interrupt flag to wait for another TMR3 overflow.
bra T3POLL
movlw0xF0; Load TMR3 with the value 0xF000 again.
movwfTMR3H
clrfTMR3L
bcf PIR2,TMR3IF; Clear the Timer3 interrupt flag again.
bcf PORTB,0; Turn off PORTB<0> to indicate the low-priority ISR is over.
retfie

INIT
clrfPORTB; setup portb for outputs
clrfDDRB

return

END
; Software License Agreement
;
; The software supplied herewith by Microchip Technology Incorporated
; (the "Company") for its PICmicro® Microcontroller is intended and
; supplied to you, the Company’s customer, for use solely and
; exclusively on Microchip PICmicro Microcontroller products. The
; software is owned by the Company and/or its supplier, and is
; protected under applicable copyright laws. All rights are reserved.
; Any use in violation of the foregoing restrictions may subject the
; user to criminal sanctions under applicable laws, as well as to
; civil liability for the breach of the terms and conditions of this
; license.
;
; THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES,
; WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED
; TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
; PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT,
; IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR
; CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
;
;************************************************************
; PIC18CKX2 EXAMPLE CODE FOR PICDEM-2
; TITLE: Oscillator Switching Demonstration
; FILENAME: osc.asm
; REVISION HISTORY: A 5/13/00 jb format change
; HARDWARE: PICDEM-2 board
; FREQUENCY: 4MHz
;
; This program demonstrates the use of oscillator switching.
; The data held in MYDAT is periodically flashed on the PORTB
; LEDs. Each time a keypress is detected on RA4, the data is
; incremented and the oscillator source is changed.

list p=18c452, n=48, t=ON, st=OFF
#include "p18c452.inc"

KEY EQU 4

;-------------------18C452 RAM LOCATIONS-----------------------
COUNT0EQU 0x0000 ; used for software timing loop
COUNT1EQU 0x0001
COUNT2EQU 0x0002
COUNT3EQU 0x0003
COUNT4EQU 0x0004
COUNT5EQU 0x0005
COUNT6EQU 0x0006
COUNT7EQU 0x0007

MYDAT EQU 0x0002 ; data storage register

;------------------BIT DEFINITIONS------------------------------------
F EQU 0x0001

;------------------VECTORS--------------------------------------------
ORG 0x000000; reset vector
BRA START

;-------------------PROGRAM-----------------------------------
START
rcall INIT; setup ports, etc.
bsf T1CON, T1OSCEN; setup the LP oscillator

MLOOP
btfss PORTA, KEY
rcall KEYPRESS; call keypress routine if
; button is pressed
movff MYDAT, PORTB; move data to port b
rcall WAIT; wait a while
clrf PORTB; clear the port
rcall WAIT; wait a while
bra MLOOP

;-----------------------SUBROUTINES-----------------------------------
KEYPRESS
btfss PORTA, KEY
bra KEYPRESS
incf MYDAT

; Oscillator source changes every time the subroutine is called.
btg OSCCON, SCS
return

INIT
clrf PORTA
clrf PORTB
bsf DDRA, 4
clrf DDRB

return

WAIT ; software time delay

clrf COUNT0
movlw 0x08
movwf COUNT1

WLOOP
decfsz COUNT0, F
bra WLOOP
decfsz COUNT1, F
bra WLOOP

return

end
; Software License Agreement
;
; The software supplied herewith by Microchip Technology Incorporated
; (the "Company") for its PICmicro® Microcontroller is intended and
; supplied to you, the Company’s customer, for use solely and
; exclusively on Microchip PICmicro Microcontroller products. The
; software is owned by the Company and/or its supplier, and is
; protected under applicable copyright laws. All rights are reserved.
; Any use in violation of the foregoing restrictions may subject the
; user to criminal sanctions under applicable laws, as well as to
; civil liability for the breach of the terms and conditions of this
; license.
;
; THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES,
; WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED
; TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
; PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT,
; IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR
; CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
;
;**************************************************************************
; PIC18CXX2 EXAMPLE CODE FOR PICDEM-2
;**************************************************************************
; TITLE: Pulse Width Modulation Demonstration
; FILENAME: pwm_ex.asm
; REVISION HISTORY: A 5/13/00 jb format change
; HARDWARE: PICDEM-2 board
; FREQUENCY: 4MHz
;
;**************************************************************************

This program demonstrates pulse width modulation using CCP2.

The PWM period is fixed and then the duty cycle is varied by
looking up a new value in a table. When the final value of
the table is read, the table is read in reverse. The PWM output
is set to RC1. Parts Y3, C6 & C7 are not installed the
on the PICDEM 2 board. PWM signal can be observed on RC1 pin.

list p=18c452, n=48, t=ON, st=OFF
#include "p18c452.inc"

**************************************************************************

; bit definitions
F EQU 0x0001
; ************************************************************
; 18C452 RAM LOCATIONS
DIRFLAG EQU 0x0000
; ************************************************************
; 18C452 ROM LOCATIONS
TABLADDR EQU 0x0003000
; ************************************************************
; vectors
org 0x000000; reset vector
bra START
org 0x000008; high priority interrupt vector
bra TMR1_ISR
; ************************************************************
; program
START
; Set up PWM module
; Set PWM period by writing to PR2
; Set PWM duty cycle by writing to the CCPR2L register
; and the CCP2CON<5:4> bits
; Make the CCP2 pin an output by clearing the TRISC2<2> bit.
    clrf CCP2CON; CCP module is off
    bsf CCP2CON, CCP2M3; select PWM mode
    bsf CCP2CON, CCP2M2; select PWM mode
    movlw 0x3F; Set PWM frequency to 78.12kHz
    movwf PR2;
    bcf TRISC, 1; make channel 1 an output
    movlw 0x00
    movwf CCPR2L
; Set the TMR2 prescale value and enable Timer2 by writing to T2CON
; Configure the CCP2 module for PWM operation
    clrf T2CON; clear T2CON
    clrf TM2; clear Timer2
    bsf T2CON, TM2CON; turn on Timer2
; initialize direction flag for table
    clrf DIRFLAG
; Initialize the table pointer registers
; to the first location of the data stored in program memory.

movlw UPPER(TABLADDR)
movwf TBLPTRU
movlw HIGH(TABLADDR)
movwf TBLPTRH
movlw LOW(TABLADDR)
movwf TBLPTRL

; setup interrupt
bsf RCON, IPEN; enable priority interrupts.
bsf IP1H, TMRI1IF; set Timer1 as a high priority interrupt source
bsf PIE1, TMRI1IE; clear the Timer1 interrupt flag
bsf PIE1, TMRI1IE; enable Timer1 interrupts
bsf INTCON, GIEH; set the global interrupt enable bits
bsf INTCON, GIEL;"

; Timer1 setup
clr T1CON
clr TMRI1H; clear Timer1 high
clr TMRI1L; clear Timer1 low
bsf T1CON, TMRI1ON; turn on Timer1

MLOOP
  goto MLOOP

;******************************************************************************
; subroutines

; TMR1_ISR ; high priority ISR

  bcf TMRI1IF; Clear the Timer1 interrupt flag.
  rcall CHECK_ADDR
  btfsc DIRFLAG, 0
  bra RD_DOWN

; RD_UP

; Code here does a table read, post-increments, and writes the
; value to the CCFR2L PWM duty-cycle register.

  tblrd*+
  movfttabl, CCFR2L
bra T1POLL

RD_DOWN

; Code here does a table read, post-decrements, and writes the
; value to the CCPR2L PWM duty-cycle register.

  tblrd*=
  movf TABLAT,CCPR2L
  bra T1POLL

T1POLL
  btfssPIR1,TMR1IF; Poll TMR1 interrupt flag to wait for another
  ; TMR1 overflow.
  bra T1POLL
  bcf PIR1,TMR1IF; Clear the Timer1 interrupt flag again.
  retfie

CHECK_ADDR
  movlw LOW(TABEND) - 1
  subwf TBLPTRL,W
  bnz CHECK_LOW
  setf DIRFLAG
  return

CHECK_LOW
  movlw LOW(TABADDR)
  subwf TBLPTRL,W
  bnz DONE_CHECK
  clrf DIRFLAG
  DONE_CHECK
  return

;-------------------------------DATA---------------------------------------
org TABLADDR
DB 0x00,0x06,0x0C,0x12,0x18,0x1E,0x23,0x28
DB 0x2D,0x31,0x35,0x38,0x3A,0x3D,0x3E,0x3F
TABEND
END
SOFTWARE LICENSE AGREEMENT

The software supplied herewith by Microchip Technology Incorporated (the "Company") for its PICmicro® Microcontroller is intended and supplied to you, the Company's customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved. Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

PIC18CXX2 EXAMPLE CODE FOR PICDEM-2

FILE: table.asm

REVISED HISTORY: 5/13/00 jbd format change

HARDWARE: PICDEM-2 board

FREQUENCY: 4MHz

This program demonstrates the use of the TBLRD instruction to read program memory.

A pre-defined sequence of data bytes is included in program memory. Each time the button on pin RA4 is pressed, the next byte of data is accessed and displayed on the PORTB LEDs. When the beginning or end of the data is reached, the direction of access is reversed.
;-------------------18C452 ROM LOCATIONS-----------------------------

TABLADDR EQU 0x0003000

;------------------BIT DEFINITIONS------------------------------------

KEY EQU 4

;------------------VECTORS--------------------------------------------

ORG 0x000000; rest vector
GOTOSTART

ORG 0x000008; high priority interrupt vector
GOTOSTART

ORG 0x000018; low priority interrupt vector
GOTOSTART

;--------------------PROGRAM-----------------------------------

START

bsfDDRA,4; porta button input
clrf PORTB; setup portb for outputs
clrf DDRB
clrf DIRFLAG

; Code should be written here to initialize the table pointer registers
; to the first location of the data stored in program memory. Use
; the appropriate assembler directives to accomplish this.
; (Refer to the 'DATA' statements in this source code.)

movlwUPPER(TABLADDR)
movwTBLPTRU
movlwHIGH(TABLADDR)
movwTBLPTRH
movlwLOW(TABLADDR)
movwTBLPTRL

MLOOP

btfsc PORTA,KEY; keypress routine
bra $ - 2; decrement PC by 2 because
btfss PORTA, KEY; of byte addressing!
br $ - 2
rcall CHECK_ADDR
btsc DIRFLAG, 0
bra RD_DOWN

RD_UP
; Code here does a table read, post-increments, and writes the
; value to the PORTB LEDS
tblrdr+
movff TABLAT, PORTB
bra MLOOP

RD_DOWN
; Code here does a table read, post-decrements, and writes the
; value to the PORTB LEDs
tblrdr-
movff TABLAT, PORTB
bra MLOOP

;-----------------------------SUBROUTINES-------------------------------

CHECK_ADDR
movlw LOW(TABEND) - 1
subwf TBLPTRL, W
bnz CHECK_LOW
setf DIRFLAG
return

CHECK_LOW
movlw LOW(TABLADDR)
subwf TBLPTRL, W
bnz DONE_CHECK
clf DIRFLAG
DONE_CHECK
return

;-----------------------------------DATA---------------------------------
org TABLADDR

DATA 0x0201,0x0804,0x2010,0x8040 ; 0x003000 - 0x003007
DATA 0x4281,0x1824,0x2400,0x8142 ; 0x003008 - 0x00300F
DATA 0x1211,0x1814,0x2818,0x8848 ; 0x003010 - 0x003017
DATA 0xaa55,0xaa55,0x0100,0x0703 ; 0x003018 - 0x00301F
DATA 0x1F0F,0x7F3F,0xFFFF,0xFFFF ; 0x003020 - 0x003027

TABEND

END
Software License Agreement

The software supplied herewith by Microchip Technology Incorporated (the "Company") for its PICmicro® Microcontroller is intended and supplied to you, the Company’s customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved. Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

This SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

PIC18CXX2 EXAMPLE CODE FOR PICDEM-2

TITLE: Timer Read/Write Demonstration
FILENAME: tmrrw.asm
REVISION HISTORY: A 5/13/00 jb format change
HARDWARE: PICDEM-2 board
FREQUENCY: 4MHz

This program uses Timer1 and Timer3 to demonstrate the use of 8 and 16 bit write modes.

The counters are used to maintain overflow count registers (similar to a RTCC). The 16 bit write to Timer3 will introduce an error between the overflow registers.

This error is calculated in the main program loop and displayed on the PORTB LEDS.

list p=18c452, n=48, t=ON, st=OFF
#include "p18c452.inc"

----------18C452 RAM LOCATIONS-----------------------
T1COUNT EQU 0x0000
T3COUNT EQU 0x0001
;------------------BIT DEFINITIONS------------------------------------
F EQU 0x0001

;------------------VECTORS--------------------------------------------
org 0x000000; reset vector
bra START
org 0x000008; high priority interrupt vector
bra TMR_ISR
org 0x000018; low priority interrupt vector
bra START

;--------------------PROGRAM-----------------------------------
START
rcall INIT

; Setup Timer1
clr T1CON
bsf T1CON,T1CKPS1; set prescaler 1:8
bsf T1CON,T1CKPS0; set prescaler 1:8
movlw 0x80; initialize TMR1 with 8000
movwf TMR1H;
clf TMR1L;
bsf T1CON,TMR1ON; turn on TMR1

; Setup Timer3
clr T3CON
bsf T3CON,TMR3CS
bsf T3CON,RD16
bsf T3CON,T3CKPS1; set prescaler 1:8
bsf T3CON,T3CKPS0; set prescaler 1:8
movlw 0x80; initialize TMR3 with 8000
movwf TMR3H;
clf TMR3L;
bsf T3CON,TMR3ON;
MLOOP

; Subtract T3COUNT from T1COUNT
; and write the result to PORTB LEDS.

movf T3COUNT, W
subwf T1COUNT, W
movwf PORTB
bra MLOOP

; ------------------------ SUBROUTINES --------------------------

TMR_ISR

btfsc PIR1, TMR1IF; check which timer caused the interrupt
bra T1_HANDLER

btfsc PIR2, TMR3IF
bra T3_HANDLER

retfie

T1_HANDLER

; Load Timer1 with 0x8000 by writing to the high byte only.
; Increment the T1COUNT register.
; Clear the Timer1 interrupt flag.

movlw 0x80
movwf TMR1H
incf T1COUNT
bcf PIR1, TMR1IF
retfie

T3_HANDLER

; Load Timer3 with 0x8000 by performing a 16-bit timer write.
; Increment the T3COUNT register.
; Clear the Timer3 interrupt flag.

movlw 0x80
movwf TMR3H
clrf TMR3L
incf T3COUNT
bcf   PIR2,TMR3IF
retfie

;SUBROUTINES

INIT
clrf T1COUNT
clrf T3COUNT
bsf   DDRA,4; porta
clrf PORTB; setup portb for outputs
clrf DDRB
bsf   PIE1,TMR1IE; enable interrupts
bsf   PIE2,TMR3IE
bsf   INTCON,PEIE; enable peripheral interrupts
bsf   INTCON,GIE; enable global interrupts
return

END
Software License Agreement

The software supplied herewith by Microchip Technology Incorporated (the "Company") for its PICmicro® Microcontroller is intended and supplied to you, the Company’s customer, for use solely and exclusively on Microchip PICmicro Microcontroller products. The software is owned by the Company and/or its supplier, and is protected under applicable copyright laws. All rights are reserved.

Any use in violation of the foregoing restrictions may subject the user to criminal sanctions under applicable laws, as well as to civil liability for the breach of the terms and conditions of this license.

THIS SOFTWARE IS PROVIDED IN AN "AS IS" CONDITION. NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. THE COMPANY SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.

************************************************************
PIC18CXX2 EXAMPLE CODE FOR PICDEM-2
************************************************************

This program is a simple implementation of the PIC18C452’s A/D.

One Channel is selected (AN0).

The hardware for this program is the PICDEM-2 board. The program converts the potentiometer value on RA0 and displays it as an 8 bit binary value on Port B.

The A/D is configured as follows:

Vref = +5V internal
A/D Osc. = internal RC
A/D Channel = AN0 (RA0)

LIST F=18C452
#include <P18C452.INC>; File contains addresses for register and bit names
; reset and interrupt vectors
org 0x00000; Reset Vector Address
gotoStart

org 0x00008; Interrupt Vector Address
gotoISR; goto Interrupt Service Routine

;******************************************************************************
; program code starts here
org 0x00020
Start
clrfPORTB; clear all bits of PORTB
clfTRISB; Set PORTB as outputs
callInitializeAD; configure A/D module
callSetupDelay; delay for 15 instruction cycles
bsf ADCON0,GO; Start first A/D conversion
MaingotoMain; do nothing loop

;******************************************************************************
; Service A/D interrupt
; Get value and display on LEDs

ISR
; Save context (WREG and STATUS) if required.
btfssPIR1,ADIF; Did A/D cause interrupt?
gotootherInt; No, check other sources

movfADRESH,W; Get A/D value
movfLATB,W; Display on LEDs
bcf PIR1,ADIF; Reset A/D int flag
callSetupDelay; Delay for 15 cycles
bsf ADCON0,GO; Start A/D conversion
gotoEndISR; return from ISR

OtherInt
; This would be replaced by code to check and service other interrupt sources
; trap here, loops to self
EndISR
    ; Restore context if saved.
    retfie; Return, enables GIE

;******************************************************************************
; InitializeAD - initializes and sets up the A/D hardware.
; Select AN0 to AN3 as analog inputs, RC clock, and read AN0.
InitializeAD
    movlwB'00000100'; Make RA0,RA1,RA4 analog inputs
    movwfADCON1
    movlwB'11000001'; Select RC osc, AN0 selected,
    movwfADCON0; A/D enabled
    bcf PIR1,ADIF; Clear A/D interrupt flag
    bsf PIE1,ADIE; Enable A/D interrupt
    bsf INTCON,PEIE; Enable peripheral interrupts
    bsf INTCON,GIE; Enable Global interrupts
    return

;******************************************************************************
; This is used to allow the A/D time to sample the input
; (acquisition time).
; This routine requires 11 cycles to complete.
; The call and return add another 4 cycles.
; 15 cycles with Fosc=4MHz means this delay consumes 15us.
SetupDelay
    movlw.3 ; Load Temp with decimal 3
    movwfTEMP
    SD
    decfszTEMP, F; Delay loop
    gotoSD
    return
    END
**Index**

### A/D
- A/D Characteristics .......................................................... 8-3
- Accuray/Errror ................................................................. 25-17, 26-17
- ADCON0 Register .............................................................. 25-2, 26-2
- ADCON1 Register .............................................................. 25-6
- ADIF bit ........................................................................ 25-7, 26-7
- Analog Input Model Block Diagram............................. 25-9, 26-9
- Configuring Analog Port Pins ........................................ 25-11, 26-11
- Configuring the Interrupt ................................................ 25-7, 26-7
- Configuring the Module .................................................... 25-7, 26-7
- Connection Considerations ............................................. 25-18, 26-18
- Conversion Clock ............................................................. 25-10, 26-10
- Conversions ................................................................ 25-12, 26-12
- Converter Characteristics ............................................... 32-36
- converter characteristics ................................................... 32-14
- Delays ............................................................................ 25-9, 26-9
- Effects of a Reset ......................................................... 25-16, 26-16, 27-6
- Equations ...................................................................... 25-8, 26-8
- Flowchart of A/D Operation ........................................... 25-13, 26-13
- GO/DONE bit ................................................................ 25-7, 26-7
- Internal Sampling Switch (Rss) Impedence .......... 25-8, 26-8
- Operation During Sleep ................................................. 25-16, 26-16, 27-6
- Sampling Requirements ................................................. 25-8, 26-8
- Sampling Time ............................................................... 25-8, 26-8
- Source Impedence ........................................................... 25-8, 26-8
- Time Delays .................................................................... 25-9, 26-9
- Transfer Function ............................................................ 25-18, 26-18
- ACK ............................................................................. 20-19
- Acknowledge Pulse ......................................................... 20-19
- ADDLW Instruction .......................................................... 31-14
- ADDWF Instruction .......................................................... 31-14
- ADDRESS Register .......................................................... 25-2, 26-2
- AKS ................................................................................ 20-37
- ANDLW Instruction ......................................................... 31-22
- Assembler .................................................................... 34-6

### B
- Baud Rate Generator ......................................................... 20-30
- BCF Instruction ............................................................... 31-28
- Block Diagrams ............................................................... 25-9, 26-9
- Analog Input Model .......................................................... 25-9, 26-9
- Baud Rate Generator ......................................................... 20-30
- External Brown-out Protection Circuit (Case1) .............. 3-13
- I2C Master Mode .............................................................. 20-27
- I2C Module ................................................................. 20-17
- SSP (I2C Mode) ............................................................. 20-17
- SSP (SPI Mode) .............................................................. 20-9
- SSP Module (I2C Master Mode) ....................................... 20-3
- SSP Module (SPI Mode) ................................................... 20-2
- Timer1 ........................................................................... 16-2
- BODEN ....................................................................... 3-10
- BRG ............................................................................. 20-30
- BRGH bit ..................................................................... 21-5
- Baud Rate Generator ......................................................... 20-30
- Brown-out Protection ...................................................... 3-13
- BWF Instruction ............................................................... 31-44, 31-47
- BTFSC Instruction ............................................................ 31-45
- BTFSS Instruction ............................................................ 31-45
- Buffered Full bit, BF ..................................................... 20-19
- Bus Arbitration ............................................................... 20-48
- Bus Collision ................................................................. 20-48
- Bus Collision During a RESTART Condition .............. 20-51
- Bus Collision During a Start Condition ....................... 20-49
- Bus Collision During a Stop Condition ....................... 20-52

### C
- C Compiler (MP-C) ............................................................... 34-6
- CALL Instruction ............................................................... 31-50
- Clock/Instruction Cycle (Figure) ........................................ 4-5
- Clocking Scheme/Instruction Cycle ......................... 4-5
- CRLF Instruction .............................................................. 31-52, 31-114
- CLRWD T Instruction ...................................................... 31-53
- Code Examples ............................................................... 31-54
- Loading the SPBUF register .......................................... 19-9, 20-10
- Code Protection ............................................................... 29-10
- COMF Instruction ............................................................. 31-54
- DC Characteristics ........................................................... 32-6
- PIC16C73 ................................................................. 32-6
- PIC16C74 ................................................................. 32-6
- DECF Instruction .............................................................. 31-62, 31-64
- DECFSZ Instruction .......................................................... 31-66, 31-68
- Development Support ..................................................... 34-2
- Filter/Mask Truth Table .................................................... 22-54
- Flowcharts ...................................................................... 20-35
- Restart Condition ............................................................. 20-35
- Start Condition ................................................................. 20-32
- Stop Condition ................................................................. 20-46
- FS0 ................................................................................... 8-3
- FS1 ................................................................................... 8-3
- FS2 ................................................................................... 8-3
- FS3 ................................................................................... 8-3
- General Call Address Sequence ..................................... 20-25
- General Call Address Support ......................................... 20-25
- GOTO Instructions ............................................................ 31-70
- I2C ..................................................................................... 20-17
- BF ................................................................................. 19-20
- CKP ............................................................................... 19-24
- I2C Overview ................................................................. 36-1
- Initiating and Terminating Data Transfer ....................... 36-1
- START .......................................................................... 36-2
- STOP ............................................................................. 36-2
- Master Mode Receiver Flowchart ..................................... 36-2
- Master Mode Reception .................................................... 20-41
- Master Mode Reception .................................................... 36-2
- Master Mode Restart Condition ...................................... 20-40
- Master Mode Restart Condition ...................................... 20-33
- Master Mode Reset ......................................................... 20-18
- Master Module ............................................................... 20-18
- 10-bit Address mode ....................................................... 20-20
- Acknowledge Flowchart .................................................. 20-44
- Acknowledge Sequence Timing ......................................... 20-43
- Addressing ................................................................... 20-20
- Baud Rate Generator ......................................................... 20-30
- Block Diagram ............................................................... 20-27
- BRG Block Diagram ......................................................... 20-30
- BRG Reset due to SDA Collision ....................................... 20-50
- BRG Timing ................................................................. 20-30
- Bus Arbitration ............................................................... 20-48
- Bus Collision ................................................................. 20-48
- Acknowledge ................................................................. 20-48
- Restart Condition ............................................................. 20-51
- Restart Condition Timing (Case1) .................................... 20-51
PIC18C Reference Manual

Restart Condition Timing (Case 2) ................. 20-51
Start Condition ........................................ 20-49
Start Condition Timing .......................... 20-49, 20-50
Stop Condition ................................. 20-52
Stop Condition Timing (Case 1) ............. 20-52
Stop Condition Timing (Case 2) ........... 20-52
Transmit Timing ................................. 20-48
Bus Collision timing ........................... 20-48
Clock Arbitration .................................... 20-47
Clock Arbitration Timing (Master Transmit) .... 20-47
Conditions to not give ACK Pulse .. 20-19
General Call Address Support .............. 20-25
Master Mode ........................................ 20-27
Master Mode 7-bit Reception timing ....... 20-42
Master Mode Operation ...................... 20-29
Master Mode Start Condition ............ 20-31
Master Mode Transmission ................. 20-37
Master Mode Transmission Sequence .... 20-29
Master Transmit Flowchart ................. 20-38
Multi-Master Communication .......... 20-48
Multi-master Mode ............................. 20-28
Operation ........................................ 20-17
Repeat Start Condition timing ............ 20-34
Restart Condition Flowchart ............... 20-35
Slave Mode ........................................ 20-19
Slave Reception ................................... 20-21
Slave Transmission ............................. 20-22
SSPBUF ........................................ 20-18
Start Condition Flowchart .................. 20-32
Stop Condition Flowchart .................... 20-46
Stop Condition Receive or Transmit timing .. 20-45
Stop Condition Timing ......................... 20-45
Waves for 7-bit Reception ................. 20-22
Waves for 7-bit Transmission ............. 20-22
I²C Module Address Register, SSPADD ........ 20-18
I²C Slave Mode ..................................... 20-19
INCFSZ Instruction ............................. 31-72
INCF Instruction .................................. 31-74, 31-76
Instruction Flow/Pipelining ................. 4-6
Instruction Set
ADDWF ........................................ 31-14
ADDWF ........................................ 31-16, 31-20
ANDWF .......................................... 31-22
ANDWF .......................................... 31-24, 31-26, 31-30, 31-32, 31-34,
.................................................. 31-36, 31-38, 31-40, 31-42, 31-48
BCF ............................................ 31-28
BF ............................................ 31-44, 31-47
BF ............................................ 31-45
BF ............................................ 31-46, 31-56, 31-58, 31-60
BF ............................................ 31-52
BF ............................................ 31-52, 31-114
BF ............................................ 31-53
BF ............................................ 31-54
BF ............................................ 31-62, 31-64
BF ............................................ 31-66, 31-68
BF ............................................ 31-70
BF ............................................ 31-72
BF ............................................ 31-74, 31-76
BF ............................................ 31-78
BF ............................................ 31-80
BF ............................................ 31-82, 31-84, 31-86, 31-87, 31-88
BF ............................................ 31-89, 31-90, 31-91, 31-92, 31-94,
.................................................. 31-95, 31-96
BF ............................................ 31-93, 31-98
BF ............................................ 31-100
BF ............................................ 31-102
BF ............................................ 31-104
RLF ........................................... 31-106, 31-108
RRF ........................................... 31-110, 31-112
SLEEP ........................................... 31-115
SUBLV ........................................... 31-116, 31-118
SUBW ........................................... 31-120, 31-122
SWAPF ........................................... 31-124, 31-126, 31-128
XORLW ........................................... 31-130, 31-132
XORWF ........................................... 31-134
Summary Table .................................. 5-3, 5-6
Inter-Integrated Circuit (I²C) ............ 20-2
Internal Sampling Switch (Rss) Impedance .. 25-8, 25-8
Interrupts
Flag bits
TRI1E .......................................... 10-4
TRI1IF .......................................... 10-4
TRI2IE .......................................... 10-4
TRI2IF .......................................... 10-4
TRI3IE .......................................... 10-4
TRI3IF .......................................... 10-4
Logic ............................................ 10-4
Introduction ...................................... 31-2
IORLW Instruction ......................... 31-78
IORWF Instruction ............................. 31-80
Loading of PC .................................. 31-7
M
MOVCL Instruction .... 31-82, 31-84, 31-86, 31-87, 31-88
MOVWF Instruction .... 31-89, 31-90, 31-91, 31-92,
.................................................. 31-94, 31-95, 31-96
MPASM Assembler ......................... 34-2, 34-6
MP-C C Compiler .......................... 34-6
MPSIM Software Simulator .............. 34-3
Multi-Master Communication .......... 20-48
Multi-Master Mode ...................... 20-28
Multiply Examples
16 x 16 Routine ...................... 6-4
16 x 16 Signed Routine ................. 6-5
8 x 8 Routine ...................... 6-3
8 x 8 Signed Routine ................. 6-3
N
NOR Instruction ...................... 31-93, 31-98
O
OSCCON ...................................... 2-3
OSCCON Register ......................... 2-3
Oscillator Start-up Time (Figure) ........ 3-5
CPO ........................................ 8-3
P
PICDEM-1 Low-Cost PIC16/17 Demo Board .... 34-2, 34-12
PICDEM-2 Low-Cost PIC16CXX Demo Board ... 34-2, 34-12
PICDEM-3 Low-Cost PIC16C9XX Demo Board ... 34-13
PICSTARTLite Low-Cost Development System .... 34-2, 34-10
Pin Functions
PCLATH Vpp ................................ 4-9
PCLATH CLKIN .......................... 4-8, 4-9
PCLATH CLKOUT ......................... 4-8, 4-9
RA0/AN0 .................................. 4-9
RA1/AN1 .................................. 4-9
RA2/AN2 .................................. 4-9
RA3/AN3/VREF ......................... 4-9
RA4/TOCKI ................................ 4-9
RA5/AN4/SS ................................ 4-9
RB0/INT .................................. 4-9
RB1 .......................................... 4-9
RB2 .......................................... 4-9
RB3 .......................................... 4-9
RB4 .......................................... 4-9