# AP-415 APPLICATION NOTE ## 83C51FA/FB PCA Cookbook **BETSY JONES**ECO APPLICATIONS ENGINEER February 1994 Order Number: 270609-002 Information in this document is provided in connection with Intel products. Intel assumes no liability whatsoever, including infringement of any patent or copyright, for sale and use of Intel products except as provided in Intel's Terms and Conditions of Sale for such products. Intel retains the right to make changes to these specifications at any time, without notice. Microcomputer Products may have minor variations to this specification known as errata. \*Other brands and names are the property of their respective owners. †Since publication of documents referenced in this document, registration of the Pentium, OverDrive and iCOMP trademarks has been issued to Intel Corporation. Contact your local Intel sales office or your distributor to obtain the latest specifications before placing your product order. Copies of documents which have an ordering number and are referenced in this document, or other Intel literature, may be obtained from: Intel Corporation P.O. Box 7641 Mt. Prospect, IL 60056-7641 or call 1-800-879-4683 COPYRIGHT © INTEL CORPORATION, 1996 ### 83C51FA/FB PCA COOKBOOK | CONTENTS | PAGE | |-----------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------| | PCA OVERVIEW | 1 | | PCA TIMER/COUNTER | 1 | | COMPARE/CAPTURE MODULES | 3 | | CAPTURE MODE Measuring Pulse Widths Measuring Periods Measuring Frequencies Measuring Duty Cycles Measuring Phase Differences Reading the PCA Timer | 5<br>7<br>9<br>10 | | COMPARE MODE | 13 | | SOFTWARE TIMER | 13 | | HIGH SPEED OUTPUT | 15 | | WATCHDOG TIMER | 18 | | PULSE WIDTH MODULATOR | 19 | | CONCLUSION | 21 | | APPENDICES | | | A. Test Routines | | | B. Duty Cycle Calculation | B-1 | | C. Special Function Registers | C-1 | | 1. PCA Timer/Counter and Compare/ Capture Modules | FIGURES | PAGE | LISTINGS | PAGE | |--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------|------|---------------------------------|-----------| | 2. PCA Interrupt 4 3. Measuring Duty Cycle 3. Measuring Phase Differences 4. Measuring Period 5. Measuring Period 6. High Speed Output (Without Interrupt) 7. Measuring Duty Cycle 9 7. High Speed Output (With Interrupt) 8. Measuring Phase Differences 10 8. High Speed Output (Single Pulse) 9. Software Timer Mode 13 9. Watchdog Timer 10. High Speed Output Mode 15 10. PWM 11. Watchdog Timer Mode 18 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 2. CMOD Values 3. Compare/Capture Mode Values 3. Compare/Capture Mode Values | 1. PCA Timer/Counter and Compare/ | | 1. Measuring Pulse Widths | 6 | | 3. PCA Capture Mode 4. Measuring Pulse Width 5 5. Measuring Period 6. Measuring Frequency 7. Measuring Duty Cycle 8. Measuring Phase Differences 10 9. Software Timer Mode 13 10. High Speed Output (With Interrupt) 9. Watchdog Timer Mode 14. Measuring Phase Differences 15. Software Timer 16. High Speed Output (With Interrupt) 17. High Speed Output (With Interrupt) 18. High Speed Output (Single Pulse) 19. Watchdog Timer 10. PWM 11. Watchdog Timer Mode 15 12. PWM Mode 16 18 19 19 10. CCAPnH Varies Duty Cycle 19 10. PWM 11. PCA Timer/Counter Inputs 12. CMOD Values 13. Compare/Capture Mode Values 14. Measuring Phase Differences 15. Software Timer 16. High Speed Output (Without Interrupt) 10. High Speed Output (Single Pulse) 11. PWM 12. PWM 13. CCAPnH Varies Duty Cycle 14. Measuring Phase Differences 15. Software Timer 16. High Speed Output (Without Interrupt) 17. High Speed Output (Single Pulse) 18. High Speed Output (Single Pulse) 19. Watchdog Timer 10. PWM 11. Watchdog Timer Mode 12 12. PWM Mode 13 13. CCAPnH Varies Duty Cycle 14. Measuring Phase Differences 15. Software Timer 16. High Speed Output (Without Interrupt) 17. High Speed Output (Single Pulse) 19. Watchdog Timer 10. PWM 11. Watchdog Timer Mode 12 12. PWM Mode 13 13. CCAPnH Varies Duty Cycle 14. Measuring Phase Differences 15. Software Timer 16. High Speed Output (Without Interrupt) 17. High Speed Output (Without Interrupt) 18. High Speed Output (Without Interrupt) 19. Watchdog Timer 10. PWM 10. PWM 11. Watchdog Timer Mode 12 12. CMOD Values 13 13. CCAPnH Varies Duty Cycle 14. Measuring Phase Differences 15 16. High Speed Output (Without Interrupt) 17. High Speed Output (Without Interrupt) 18. High Speed Output (Without Interrupt) 19. Watchdog Timer Wat | | | 2. Measuring Frequencies | 8 | | 4. Measuring Pulse Width 5 5. Measuring Period 7 6. Measuring Frequency 7 7. Measuring Duty Cycle 9 8. Measuring Phase Differences 10 9. Software Timer Mode 13 10. High Speed Output (With Interrupt) 9 11. Watchdog Timer Mode 15 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 14. Measuring Thase Differences 5 5. Software Timer 6 6. High Speed Output (With Interrupt) 9 7. High Speed Output (Single Pulse) 9 9. Watchdog Timer 10 10. PWM 10 11. PWM 11 12. PWM Mode 15 13. CCAPnH Varies Duty Cycle 20 14. Measuring Thase Differences 5 15. Software Timer 10 16. High Speed Output (Without Interrupt) 10 17. High Speed Output (Without Interrupt) 10 18. High Speed Output (Single Pulse) 10 19. Watchdog Timer 10 10. PWM 11 11. Watchdog Timer Mode 15 12. PCA Timer/Counter Inputs 12 13. CCAPnH Varies Duty Cycle 20 14. Measuring Thase Differences 15 15. Software Timer 10 15. Software Timer 10 16. High Speed Output (Without Interrupt) 10 17. High Speed Output (Without Interrupt) 10 18. High Speed Output (Single Pulse) 10 19. Watchdog Timer 10 10. PWM 11 10. PWM 12 11. Watchdog Timer Mode 11 12. PCA Timer/Counter Inputs 12 13. CCAPnH Varies Duty Cycle 20 14. Measuring Thase Differences 10 15. Software Timer 10 15. High Speed Output (Without Interrupt) 10 15. CHICAL TIMER 10 16. High Speed Output (With Interrupt) 10 16. High Speed Output (With Interrupt) 10 17. High Speed Output (Without Interrupt) 10 18. High Speed Output (Without Interrupt) 10 18. High Speed Output (Without Interrupt) 10 19. Watchdog Timer 19 | | | 3. Measuring Duty Cycle | 9 | | 4. Measuring Pulse Width 5 5. Measuring Period 6. High Speed Output (Without Interrupt) 7 7. Measuring Duty Cycle 9 7. High Speed Output (With Interrupt) 8. High Speed Output (Single Pulse) 9 9. Software Timer Mode 13 10. High Speed Output Mode 15 11. Watchdog Timer Mode 18 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 5. Software Timer 6 6. High Speed Output (Without Interrupt) 9 7. High Speed Output (With Interrupt) 10 8. High Speed Output (Single Pulse) 10 9. Watchdog Timer 10 10. PWM 11 11. Watchdog Timer Mode 11 12. PWM Mode 12 13. CCAPnH Varies Duty Cycle 20 14. PCA Timer/Counter Inputs 15 15. Software Timer 16 16. High Speed Output (Without Interrupt) 11 17. High Speed Output (Without Interrupt) 11 18. High Speed Output (Without Interrupt) 11 19. High Speed Output (Without Interrupt) 12 19. Watchdog Timer 11 10. PWM 11 10. PWM 12 11. Watchdog Timer Mode 11 12. PWM Mode 12 13. CCAPnH Varies Duty Cycle 20 14. CMOD Values 20 15. Software Timer 16 16. High Speed Output (Without Interrupt) 12 17. High Speed Output (Without Interrupt) 12 18. High Speed Output (Single Pulse) 12 19. Watchdog Timer 12 10. PWM 11 10. PWM 12 11. Watchdog Timer Mode 18 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 14. CMOD Values 20 15. Software Timer 16 16. High Speed Output (With Interrupt) 12 16. High Speed Output (Without Interrupt) 12 17. High Speed Output (With Interrupt) 12 18. High Speed Output (Single Pulse) 12 19. Watchdog Timer 12 10. PWM 11 10. PWM 12 11. Watchdog Timer Mode 18 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 14. PCA Timer/Counter Inputs 22 15. CMOD Values 23 16. CMOD Values 23 16. High Speed Output (Without Interrupt) 24 17. PCA Timer/Counter Inputs 24 18. High Speed Output (Without Interrupt) 25 18. High Speed Output (Without Interrupt) 27 18. High Speed Output (Without Interrupt) 27 18. High Speed Output (Without Interrupt) 27 18. High Speed Output (Without Interrupt) 27 18. High Speed Output (Without Interrupt) 27 18. High Speed Output (Without Interrupt) 27 18. High Speed Output (Without Interrup | 3. PCA Capture Mode | 5 | 4. Measuring Phase Differences | 11 | | 6. Measuring Frequency 7 7. Measuring Duty Cycle 9 8. Measuring Phase Differences 10 9. Software Timer Mode 13 10. High Speed Output (Single Pulse) 9 11. Watchdog Timer Mode 15 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 15. High Speed Output (With Interrupt) 8 16. High Speed Output (With Interrupt) 10 17. High Speed Output (Single Pulse) 10 18. High Speed Output (Single Pulse) 10 19. Watchdog Timer 10 10. PWM 10 11. PWM 11 12. PWM Mode 12 13. CCAPnH Varies Duty Cycle 20 14. PCA Timer/Counter Inputs 12 15. CMOD Values 13 16. Compare/Capture Mode Values 15 17. Compare/Capture Mode Values 15 18. High Speed Output (With Interrupt) 10 19. Watchdog Timer 10 10. PWM 10 10. PWM 10 11. Watchdog Timer Mode 18 12. PCA Timer/Counter Inputs 12 13. CCAPnH Varies Duty Cycle 20 14. CAPnH Varies Duty Cycle 20 15. High Speed Output (With Interrupt) 10 16. High Speed Output (With Interrupt) 10 17. High Speed Output (With Interrupt) 10 18. High Speed Output (With Interrupt) 10 19. Watchdog Timer 10 19. Watchdog Timer 10 19. PWM 1 | _ | | _ | | | 6. Measuring Frequency 7 7. Measuring Duty Cycle 9 8. Measuring Phase Differences 10 9. Software Timer Mode 13 10. High Speed Output (Single Pulse) 9. Watchdog Timer 10. PWM 11. Watchdog Timer Mode 18 12. PWM Mode 19 13. CCAPnH Varies Duty Cycle 20 14. Measuring Phase Differences 10 15. High Speed Output (Single Pulse) 9. Watchdog Timer 10. PWM 10. PWM 10. PWM 10. PWM 10. PWM 10. PWM 10. PCA Timer/Counter Inputs 10. PCA Timer/Counter Inputs 10. PCA Timer/Counter Inputs 10. PCA Timer/Capture Mode Values 10. Compare/Capture Mode Values 10. PCA Timer/Capture Ti | 5. Measuring Period | 7 | 6. High Speed Output (Without | | | 8. Measuring Phase Differences 10 8. High Speed Output (Single Pulse) 9. Software Timer Mode 13 9. Watchdog Timer 10. High Speed Output Mode 15 10. PWM 11. Watchdog Timer Mode 18 12. PWM Mode 19 1. PCA Timer/Counter Inputs 13. CCAPnH Varies Duty Cycle 20 1. CMOD Values 3. Compare/Capture Mode Values | 6. Measuring Frequency | 7 | Interrupt) | 15 | | 9. Software Timer Mode 13 9. Watchdog Timer 10. High Speed Output Mode 15 10. PWM 11. Watchdog Timer Mode 18 TABLES PA 12. PWM Mode 19 1. PCA Timer/Counter Inputs 2. CMOD Values 13. CCAPnH Varies Duty Cycle 20 2. CMOD Values 3. Compare/Capture Mode Values | 7. Measuring Duty Cycle | 9 | 7. High Speed Output (With Inte | rrupt) 16 | | 10. High Speed Output Mode 15 10. PWM 11. Watchdog Timer Mode 18 12. PWM Mode 19 1. PCA Timer/Counter Inputs 1. PCA Timer/Counter Inputs 2. CMOD Values 2. CMOD Values 3. Compare/Capture Mode Val | 8. Measuring Phase Differences | 10 | 8. High Speed Output (Single Pu | ılse) 17 | | 11. Watchdog Timer Mode | 9. Software Timer Mode | 13 | 9. Watchdog Timer | 19 | | 12. PWM Mode | 10. High Speed Output Mode | 15 | 10. PWM | 21 | | 12. PWM Mode | 11. Watchdog Timer Mode | 18 | TABLES | PAGE | | 13. CCAPnH Varies Duty Cycle | 12. PWM Mode | 19 | <del>-</del> | | | 3. Compare/Capture Mode Values | 13. CCAPnH Varies Duty Cycle | 20 | · · | | | · | | | | | | 4. PWM Frequencies | | | 3. Compare/Capture Mode Valu | es 3 | | | | | 4. PWM Frequencies | 20 | This application note illustrates the different functions of the Programmable Counter Array (PCA) which are available on the 83C51FA and 83C51FB. Included are cookbook samples of code in typical applications to simplify the use of the PCA. Since all the examples are written in assembly language, it is assumed the reader is familiar with ASM51. For further information on these products or ASM51 refer to the Embedded Controller Handbook (Vol. I). #### **PCA OVERVIEW** The major new feature on the 83C51FA and 83C51FB is the Programmable Counter Array. The PCA provides more timing capabilities with less CPU intervention than the standard timer/counters. Its advantages include reduced software overhead and improved accuracy. The PCA consists of a dedicated timer/counter which serves as the time base for an array of five compare/capture modules. Figure 1 shows a block diagram of the PCA. Notice that the PCA timer and modules are all 16-bits. If an external event is associated with a module, that function is shared with the corresponding Port 1 pin. If the module is not using the port pin, the pin can still be used for standard I/O. Each of the five modules can be programmed in any one of the following modes: - Rising and/or Falling Edge Capture - Software Timer - High Speed Output - Watchdog Timer (Module 4 only) - Pulse Width Modulator. All of these modes will be discussed later in detail. However, let's first look at how to set up the PCA timer and modules. #### PCA TIMER/COUNTER The timer/counter for the PCA is a free-running 16-bit timer consisting of registers CH and CL (the high and low bytes of the count values). It is the only timer which can service the PCA. The clock input can be selected from the following four modes: - oscillator frequency ÷ 12 (Mode 0) - oscillator frequency ÷ 4 (Mode 1) - Timer 0 overflows (Mode 2) - external input on P1.2 (Mode 3) Figure 1. PCA Timer/Counter and Compare/Capture Modules 1 The table below summarizes the various clock inputs for each mode at two common frequencies. In Mode 0, the clock input is simply a machine cycle count, whereas in Mode 1 the input is clocked three times faster. In Mode 2, Timer 0 overflows are counted allowing for a range of slower inputs to the timer. And finally, if the input is external the PCA timer counts 1-to-0 transitions with the maximum clock frequency equal to $\frac{1}{8}$ x oscillator frequency. Table 1. PCA Timer/Counter Inputs | PCA Timer/Counter Mode | Clock | ncrements | |--------------------------------------------------------------------------------------------------------|--------------------------------------|-----------------------------------------| | TOA TIME! Journal Mode | 12 MHz | 16 MHz | | Mode 0: fosc / 12 | 1 μsec | 0.75 μsec | | Mode 1: fosc / 4 | 330 nsec | 250 nsec | | Mode 2*: Timer 0 Overflows<br>Timer 0 programmed in:<br>8-bit mode<br>16-bit mdoe<br>8-bit auto-reload | 256 μsec<br>65 msec<br>1 to 255 μsec | 192 µsec<br>49 msec<br>0.75 to 191 µsec | | Mode 3: External Input MAX | 0.66 μsec | 0.50 µsec | <sup>\*</sup>In Mode 2, the overflow interrupt for Timer 0 does not need to be enabled. Special Function Register CMOD contains the Count Pulse Select bits (CPS1 and CPS0) to specify the PCA timer input. This register also contains the ECF bit which enables an interrupt when the counter overflows. In addition, the user has the option of turning off the PCA timer during Idle Mode by setting the Counter Idle bit (CIDL). This can further reduce power consumption by an additional 30%. #### **CMOD:** Counter Mode Register | CIDL | WDTE | _ | _ | _ | CPS1 | CPS0 | ECF | |--------------|-------------------------------------|---|---|---|------|------|------------| | Address = 0I | ddress = 0D9H Reset Value = 00XX X0 | | | | | | 00XX X000B | Not Bit Addressable #### NOTE: The user should write 0s to unimplemented bits. These bits may be used in future MCS-51 products to invoke new features, and in that case the inactive value of the new bit will be 0. When read, these bits must be treated as don't-cares. Table 2 lists the values for CMOD in the four possible timer modes with and without the overflow interrupt enabled. This list assumes that the PCA will be left running during Idle Mode. **Table 2. CMOD Values** | PCA Count Pulse Selected | CMOD value | | | | | | |--------------------------|---------------------------|------------------------|--|--|--|--| | TOA Obunt Tuise beledied | without interrupt enabled | with interrupt enabled | | | | | | Internal clock, Fosc/12 | 00 H | 01 H | | | | | | Internal clock, Fosc/ 4 | 02 H | 03 H | | | | | | Timer 0 overflow | 04H | 05 H | | | | | | External clock at P1.2 | 06 H | 07 H | | | | | The CCON register shown below contains the Counter Run bit (CR) which turns the timer on or off. When the PCA timer overflows, the Counter Overflow bit (CF) gets set. CCON also contains the five event flags for the PCA modules. The purpose of these flags will be discussed in the next section. #### **CCON:** Counter Control Register | | CF0 | | |--|-----|--| |--|-----|--| Address = 0D8H Reset Value = 00X0 0000B Bit Addressable The PCA timer registers (CH and CL) can be read and written to at any time. However, to read the full 16-bit timer value simultaneously requires using one of the PCA modules in the capture mode and toggling a port pin in software. More information on reading the PCA timer is provided in the section on the Capture Mode. #### **COMPARE/CAPTURE MODULES** Each of the five compare/capture modules has a mode register called CCAPMn (n = 0,1,2,3,or 4) to select which function it will perform. Note the ECCFn bit which enables an interrupt to occur when a module's event flag is set. #### **CCAPMn:** Compare/Capture Mode Register | _ | ECOMn | CAPPn | CAPNn | MATn | TOGn | PWMn | ECCFn | |-------------|-------------|-------|-------|------|------|---------------|--------------| | Address = 0 | DAH (n=0) | | | | | Reset Value = | = X000 0000B | | 01 | DBH (n = 1) | | | | | | | | 01 | DCH (n=2) | | | | | | | | 01 | DDH (n=3) | | | | | | | | 01 | DEH (n=4) | | | | | | | Table 3 lists the CCAPMn values for each different mode with and without the PCA interrupt enabled; that is, the interrupt is optional for all modes. However, some of the PCA modes require software servicing. For example, the Capture modes need an interrupt so that back-to-back events can be recognized. Also, in most applications the purpose of the Software Timer mode is to generate interrupts in software so it would be useless not to have the interrupt enabled. The PWM mode, on the other hand, does not require CPU intervention so the interrupt is normally not enabled. Table 3. Compare/Capture Mode Values | tanto et compano, captano meno tanaco | | | | | | | | |---------------------------------------|---------------------------|------------------------|--|--|--|--|--| | Module Function | CCAPMn Value | | | | | | | | module i dilottori | without interrupt enabled | with interrupt enabled | | | | | | | Capture Positive only | 20H | 21 H | | | | | | | Capture Negative only | 10H | 11 H | | | | | | | Capture Pos. or Neg. | 30H | 31 H | | | | | | | Software Timer | 48H | 49 H | | | | | | | High Speed Output | 4C H | 4D H | | | | | | | Watchdog Timer | 48 or 4C H | _ | | | | | | | Pulse Width Modulator | 42 H | 43H | | | | | | It should be mentioned that a particular module can change modes within the program. For example, a module might be used to sample incoming data. Initially it could be set up to capture a falling edge transition. Then the same module can be reconfigured as a software timer to interrupt the CPU at regular intervals and sample the pin. Each module also has a pair of 8-bit compare/capture registers (CCAPnH, CCAPnL) associated with it. These registers are used to store the time when a capture event occurred or when a compare event should occur. Remember, event times are based on the free-running PCA timer (CH and CL). For the PWM mode, the high byte register CCAPnH controls the duty cycle of the waveform. When an event occurs, a flag in CCON is set for the appropriate module. This register is bit addressable so that event flags can be checked individually. #### **CCON:** Counter Control Register | CF | CR | _ | CCF4 | CCF3 | CCF2 | CCF1 | CCF0 | |----|----|---|------|------|------|------|------| | | | | | | _ | | | Address = 0D8H Bit Addressable Reset Value = 00X0 0000B These five event flags plus the PCA timer overflow flag share an interrupt vector as shown below. These flags are not cleared when the hardware vectors to the PCA interrupt address (0033H) so that the user can determine which event caused the interrupt. This also allows the user to define the priority of servicing each module. Figure 2. PCA Interrupt An additional bit was added to the Interrupt Enable (IE) register for the PCA interrupt. Similarly, a high priority bit was added to the Interrupt Priority (IP) register. #### IE: Interrupt Enable Register | EA | EC | ET2 | ES | ET1 | EX1 | ET0 | EX0 | | |---------------------------------------------------------|-----|-----|----|-----|-----|-----|-----|--| | Address = 0A8H Reset Value = 0000 0000B Bit Addressable | | | | | | | | | | IP: Interrupt Priority Register | | | | | | | | | | _ | PPC | PT2 | PS | PT1 | PX1 | PT0 | PX0 | | | Address = 0001 | | | | | | | | | Address = 0B8H Bit Addressable Reset Value = X000 0000B Remember, each of the six possible sources for the PCA interrupt must be individually enabled as well—in the CCAPMn register for the modules and in the CCON register for the timer. #### **CAPTURE MODE** Both positive and negative transitions can trigger a capture with the PCA. This allows the PCA flexibility to measure periods, pulse widths, duty cycles, and phase differences on up to five separate inputs. This section gives examples of all these different applications. Figure 3 shows how the PCA handles a capture event. Using Module 0 for this example, the signal is input to P1.3. When a transition is detected on that pin, the 16-bit value of the PCA timer (CH,CL) is loaded into the capture registers (CCAP0H,CCAP0L). Module 0's event flag is set and an interrupt is flagged. The interrupt will then be generated if it has been properly enabled. In the interrupt service routine, the 16-bit capture value must be saved in RAM before the next event capture occurs; a subsequent capture will write over the first capture value. Also, since the hardware does not clear the event flag, it must be cleared in software. The time it takes to service this interrupt routine determines the resolution of back-to-back events with the same PCA module. To store two 8-bit registers and clear the event flag takes at least 9 machine cycles. That includes the call to the interrupt routine. At 12 MHz, this routine would take less than 10 microseconds. However, depending on the frequency and interrupt latency, the resolution will vary with each application. #### **Measuring Pulse Widths** To measure the pulse width of a signal, the PCA module must capture both rising and falling edges (see Figure 4). The module can be programmed to capture either edge if it is known which edge will occur first. However, if this is not known, the user can select which edge will trigger the first capture by choosing the proper mode for the module. Listing 1 shows an example of measuring pulse widths. (It's assumed the incoming signal matches the one in Figure 4.) In the interrupt routine the first set of capture values are stored in RAM. After the second capture, a subtraction routine calculates the pulse width in units of PCA timer ticks. Note that the subtraction does not have to be completed in the interrupt service routine. Also, this example assumes that the two capture events will occur within 2<sup>16</sup> counts of the PCA timer, i.e. rollovers of the PCA timer are not counted. Figure 4. Measuring Pulse Width Figure 3. PCA Capture Mode (Module 0) #### Listing 1. Measuring Pulse Widths ``` ; RAM locations to store capture values PULSE_WIDTH 30H 32H FLAG BIT 20H.0 ORG OOOOH JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT PCA_INIT: ; Initialize PCA timer MOV CMOD, #OOH ; Input to timer = 1/12 X Fosc MOV CH, #OOH MOV CL, #OOH Initialize Module O in capture mode MOV CCAPMO, #21H ; Capture positive edge first ; for measuring pulse width SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn PCA timer on CLR FLAG ; clear test flag ********************** Main program goes here This example assumes Module O is the only PCA module being used. If other modules are used, software must check which module's event caused the interrupt. PCA_INTERRUPT: ; Clear Module O's event flag CLR CCFO JB FLAG, SECOND_CAPTURE ; Check if this is the first ; capture or second FIRST_CAPTURE: ; Save 16-bit capture value MOV CAPTURE, CCAPOL ; in RAM MOV CAPTURE+1, CCAPOH MOV CCAPMO, #11H ; Change module to now capture ; falling edges SETB FLAG ; Signify 1st capture complete RETI SECOND_CAPTURE: PUSH ACC PUSH PSW CLR C MOV A, CCAPOL ; 16-bit subtract SUBB A, CAPTURE MOV PULSE_WIDTH, A ; 16-bit result stored in MOV A, CCAPOH SUBB A, CAPTURE+1 ; two 8-bit RAM locations MOV PULSE_WIDTH+1, A MOV CCAPMO, #21H ; Optional -- needed if user wants to ; measure next pulse width CLR FLAG POP PSW POP ACC RETT ``` #### **Measuring Periods** Measuring the period of a signal with the PCA is similar to measuring the pulse width. The only difference will be the trigger source for the capture mode. In Figure 5, rising edges are captured to calculate the period. The code is identical to Listing 1 except that the capture mode should not be changed in the interrupt routine. The result of the subtraction will be the period. Figure 5. Measuring Period #### **Measuring Frequencies** Measuring a frequency with the PCA capture mode involves calculating a sample time for a known number of samples. In Figure 6, the time between the first capture and the "Nth" capture equals the sample time T. Listing 2 shows the code for N=10 samples. It's assumed that the sample time is less than $2^{16}$ counts of the PCA timer. Figure 6. Measuring Frequency 7 #### **Listing 2. Measuring Frequencies** ``` ; RAM locations to store capture values 30H CAPTURE DATA PERIOD DATA 32H SAMPLE_COUNT DATA 34H FLAG BIT 20H.0 ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT PCA_INIT: ; Initialization of PCA timer, Module O, and interrupt is the ; same as in Listing 1. Also need to initialize the sample MOV SAMPLE_COUNT, #10D ; N = 10 for this example Main program goes here This code assumes only Module 0 is being used. PCA_INTERRUPT: CLR CCFO ; Clear module 0's event flag JB FLAG, NEXT_CAPTURE FIRST_CAPTURE: MOV CAPTURE, CCAPOL MOV CAPTURE+1, CCAPOH SETB FLAG ; Signify first capture complete RETI NEXT_CAPTURE: DJNZ SAMPLE_COUNT, EXIT PUSH ACC PUSH PSW CLR C MOV A, CCAPOL ; 16-bit subtraction SUBB A, CAPTURE MOV PERIOD, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PERIOD+1, A MOV SAMPLE_COUNT, #10D ; Reload for next period CLR FLAG POP PSW POP ACC EXIT: RETI ``` The user may instead want to measure frequency by counting pulses for a known sample time. In this case, one module is programmed in the capture mode to count edges (either rising or falling), and a second module is programmed as a software timer to mark the sample time. An example of a software timer is given later. For information on resolution in measuring frequencies, refer to Article Reprint AR-517, "Using the 8051 Microcontroller with Resonant Transducers," in the Embedded Controller Handbook. #### **Measuring Duty Cycles** To measure the duty cycle of an incoming signal, both rising and falling edges need to be captured. Then the duty cycle must be calculated based on three capture values as seen in Figure 7. The same initialization routine is used from the previous example. Only the PCA interrupt service routine is given in Listing 3. Figure 7. Measuring Duty Cycle #### **Listing 3. Measuring Duty Cycle** ``` ; RAM locations to store capture values CAPTURE DATA 30H PULSE_WIDTH DATA 32H PERIOD DATA 34H FLAG_1 BIT 20H.0 FLAG_2 BIT 20H.1 ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT ; Initialization for PCA timer, module, and interrupt the same as in Listing 1. Capture positive edge first, then either ; edge. Main program goes here This code assumes only Module 0 is being used. PCA_INTERRUPT: ; Clear Module O's event flag CLR CCFO JB FLAG_1, SECOND_CAPTURE FIRST_CAPTURE: MOV CAPTURE, CCAPOL MOV CAPTURE+1, CCAPOH SETB FLAG_1 ; Signify first capture complete MOV CCAPMO, #31H ; Capture either edge now RETI ``` 9 #### Listing 3. Measuring Duty Cycle (Continued) ``` SECOND_CAPTURE: PUSH ACC PUSH PSW JB FLAG_2, THIRD_CAPTURE ; Calculate pulse width CLR C MOV A, CCAPOL ; 16-bit subtract SUBB A, CAPTURE MOV PULSE_WIDTH, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PULSE_WIDTH+1, A SETB FLAG_2 ; Signify second capture complete POP PSW POP ACC RETI THIRD_CAPTURE: CLR C ; Calculate period MOV A, CCAPOL ; 16-bit subtract SUBB A, CAPTURE MOV PERIOD, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PERIOD+1, A MOV CCAPMO, #21H ; Optional - reconfigure module to CLR FLAG_1 ; capture positive edges for next CLR FLAG_2 ; cycle POP PSW POP ACC RETI ``` After the third capture, a 16-bit by 16-bit divide routine needs to be executed. This routine is located in Appendix B. Due to its length, it's up to the user whether the divide routine should be completed in the interrupt routine or be called as a subroutine from the main program. between two or more signals. For this example, two signals are input to Modules 0 and 1 as seen in Figure 8. Both modules are programmed to capture rising edges only. Listing 4 shows the code needed to measure the difference between these two signals. This code does not assume one signal is leading or lagging the other. #### **Measuring Phase Differences** Because the PCA modules share the same time base, the PCA is useful for measuring the phase difference Figure 8. Measuring Phase Differences #### **Listing 4. Measuring Phase Differences** ``` ; RAM locations to store capture values CAPTURE_O DATA 30H CAPTURE_1 DATA 32H PHASE DATA 34H BIT 20H.0 FLAG_O FLAG_1 BIT 20H.1 ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT PCA_INIT: ; Same initialization for PCA timer, and interrupt as ; in Listing 1. Initialize two PCA modules as follows: MOV CCAPMO, #21H ; Module O capture rising edges MOV CCAPMO, #21H ; module 0 captu MOV CCAPMI, #21H ; Module 1 same Main program goes here ; This code assumes only Modules O and 1 are being used. PCA_INTERRUPT: JB CCFO, MODULE_0 ; Determine which module's JB CCF1, MODULE_1 ; event caused the interrupt MODULE_0: ; Clear Module O's event flag CLR CCFO MOV CAPTURE_O, CCAPOL ; Save 16-bit capture value MOV CAPTURE_0+1, CCAPOH JB FLAG_1, CALCULATE_PHASE ; If capture complete on ; Module 1, go to calculation SETB FLAG_O ; Signify capture on Module 0 RETI ``` #### Listing 4. Measuring Phase Differences (Continued) ``` MODULE_1: CLR CCF_1 ; Clear Module 1's event flag MOV CAPTURE_1, CCAP1L MOV CAPTURE_1+1, CCAP1H JB FLAG_O, CALCULATE_PHASE ; If capture complete on ; Module O, go to calculation SETB FLAG_1 ; Signify capture on Module 1 RETI CALCULATE_PHASE: ; This calculation does not PUSH ACC PUSH PSW ; have to be completed in the CLR C ; interrupt Service routine JB FLAG_O, MODO_LEADING JB FLAG_1, MOD1_LEADING MODO_LEADING: MOV A, CAPTURE_1 SUBB A, CAPTURE_0 MOV PHASE, A MOV A, CAPTURE_1+1 SUBB A, CAPTURE_0+1 MOV PHASE+1, A CLR FLAG_0 JMP EXIT MOD1_LEADING: MOV A, CAPTURE_O SUBB A, CAPTURE_1 MOV PHASE, A MOV A, CAPTURE_0+1 SUBB A, CAPTURE_1+1 MOV PHASE+1, A CLR FLAG_1 EXIT: POP PSW POP ACC RETI ``` #### Reading the PCA Timer Some applications may require that the PCA timer be read instantaneously as a real-time event. Since the timer consists of two 8-bit registers (CH,CL), it would normally take two MOV instructions to read the whole timer. An invalid read could occur if the registers rolled over in the middle of the two MOVs. However, with the capture mode a 16-bit timer value can be loaded into the capture registers by toggling a port pin. For example, configure Module 0 to capture falling edges and initialize P1.3 to be high. Then when the user wants to read the PCA timer, clear P1.3 and the full 16-bit timer value will be saved in the capture registers. It's still optional whether the user wants to generate an interrupt with the capture. #### **COMPARE MODE** In this mode, the 16-bit value of the PCA timer is compared with a 16-bit value pre-loaded in the module's compare registers. The comparison occurs three times per machine cycle in order to recognize the fastest possible clock input, i.e. $\frac{1}{4}$ x oscillator frequency. When there is a match, one of three events can happen: (1) an interrupt — Software Timer mode (2) toggle of a port pin — High Speed Output mode (3) a reset — Watchdog Timer mode. Examples of each compare mode will follow. #### SOFTWARE TIMER In most applications a software timer is used to trigger interrupt routines which must occur at periodic intervals. Figure 9 shows the sequence of events for the Software Timer mode. The user preloads a 16-bit value in a module's compare registers. When a match occurs between this compare value and the PCA timer, an event flag is set and an interrupt is flagged. An interrupt is then generated if it has been enabled. If necessary, a new 16-bit compare value can be loaded into (CCAP0H, CCAP0L) during the interrupt routine. The user should be aware that the hardware temporarily disables the comparator function while these registers are being updated so that an invalid match will not occur. That is, a write to the low byte (CCAPn0) disables the comparator while a write to the high byte (CCAP0H) re-enables the comparator. For this reason, user software must write to CCAP0L first, then CCAP0H. The user may also want to hold off any interrupts from occurring while these registers are being updated. This can easily be done by clearing the EA bit. See the code example in Listing 5. Figure 9. Software Timer Mode (Module 0) #### Listing 5. Software Timer ``` ; Generate an interrupt in software every 20 msec ; Frequency = 12 MHz ; PCA clock input = 1/12 x Fosc \rightarrow 1 \mu sec ; Calculate reload value for compare registers: 20 msec ----- = 20,000 counts 1 µsec/count ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT PCA_INIT: ; Initialize PCA timer same as in Listing 1 ; MOV CCAPMO, #49H ; Module 0 in Software Timer mode MOV CCAPOL, #LOW(20000) ; Write to low byte first MOV CCAPOH, #HIGH(20000) SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn on PCA timer Main program goes here PCA_INTERRUPT: ; Clear Module O's event flag CLR CCFO PUSH ACC PUSH PSW ; Hold off interrupts ; 16-Bit Add ; Next match will occur CLR EA MOV A, #LOW(20000) ADD A, CCAPOL ; 20,000 counts later MOV CCAPOL, A MOV A, #HIGH(20000) ADDC A, CCAPOH MOV CCAPOH, A SETB EA ; Continue with routine • POP PSW POP ACC RETT ``` #### **HIGH SPEED OUTPUT** The High Speed Output (HSO) mode toggles a port pin when a match occurs between the PCA timer and the preloaded value in the compare registers (see Figure 10). The HSO mode is more accurate than toggling pins in software because the toggle occurs *before* branching to an interrupt, i.e. interrupt latency will not effect the accuracy of the output. In fact, the interrupt is optional. Only if the user wants to change the time for the next toggle is it necessary to update the compare registers. Otherwise, the next toggle will occur when the PCA timer rolls over and matches the last compare value. Examples of both are shown. Figure 10. High Speed Output Mode (Module 0) Without any CPU intervention, the fastest waveform the PCA can generate with the HSO mode is a 30.5 Hz signal at 16 MHz. Refer to Listing 6. By changing the PCA clock input, slower waveforms can also be generated. #### Listing 6. High Speed Output (Without Interrupt) ``` ; Maximum output with HSO mode without interrupts = 30.5 \text{ Hz} signal = 16 MHz Frequency ; PCA clock input = 1/4 x Fosc \rightarrow 250 nsec MOV CMOD, #02H MOV CL, #OOH MOV CH, #OOH MOV CCAPMO, #4CH ; HSO mode without interrupt enabled ; Write to low byte first MOV CCAPOL, #OFFH ; Pl.3 will toggle every 2^{16} counts MOV CCAPOH, #OFFH ; or 16.4 msec ; Period = 30.5 \text{ Hz} SETB CR ; Turn on PCA timer ``` In this next example, the PCA interrupt is used to change the compare value for each toggle. This way a variable frequency output can be generated. Listing 7 shows an output of 1 KHz at 16 Mhz. **Listing 7. High Speed Output (With Interrupt)** ``` → 500 μsec → t + 2000 counts 500~\mu sec = 2000 counts 250 nsec/count 270609-11 ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT PCA_INIT: MOV CMOD, #02H ; Clock input = 250 nsec MOV CL, #00H MOV CH, #00H ; at 16 MHz MOV CCAPMO, #4DH ; Module O in HSO mode with ; PCA interrupt enabled MOV CCAPOL, #LOW(1000) MOV CCAPOH, #HIGH(1000) ; t = 1000 (arbitrary) CLR P1.3 SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn on PCA timer Main program goes here This code assumes only Module 0 is being used. PCA_INTERRUPT: CLR CCFO ; Clear Module O's event flag PUSH ACC PUSH PSW CLR EA ; Hold off interrupts ; 16-bit add ; 2000 counts later, Pl.3 MOV A, #LOW(2000) ADD A, CCAPOL MOV CCAPOL, A ; will toggle MOV A, #HIGH(2000) ADDC A, CCAPOH MOV CCAPOH, A SETB EA POP PSW POP ACC RETI ``` Another option with the HSO mode is to generate a single pulse. Listing 8 shows the code for an output with a pulse width of 20 $\mu$ sec. As in the previous example, the PCA interrupt will be used to change the time for the toggle. The first toggle will occur at time "t". After 80 counts of the PCA timer, 20 $\mu$ sec will have expired, and the next toggle will occur. Then the HSO mode will be disabled. Listing 8. High Speed Output (Single Pulse) #### WATCHDOG TIMER An on-board watchdog timer is available with the PCA to improve the reliability of the system without increasing chip count. Watchdog timers are useful for systems which are susceptible to noise, power glitches, or electrostatic discharge. Module 4 is the only PCA module which can be programmed as a watchdog. However, this module can still be used for other modes if the watchdog is not needed. Figure 11 shows a diagram of how the watchdog works. The user pre-loads a 16-bit value in the compare registers. Just like the other compare modes, this 16-bit value is compared to the PCA timer value. If a match is allowed to occur, an internal reset will be generated. This will not cause the RST pin to be driven high. In order to hold off the reset, the user has three options: - (1) periodically change the compare value so it will never match the PCA timer, - (2) periodically change the PCA timer value so it will never match the compare value, or - (3) disable the watchdog by clearing the WDTE bit before a match occurs and then re-enable it. The first two options are more reliable because the watchdog timer is never disabled as in option #3. If the program counter ever goes astray, a match will eventually occur and cause an internal reset. The second option is also not recommended if other PCA modules are being used. Remember, the PCA timer is the time base for all modules; changing the time base for other modules would not be a good idea. Thus, in most applications the first solution is the best option. Listing 9 shows the code for initializing the watchdog timer. Module 4 can be configured in either compare mode, and the WDTE bit in CMOD must also be set. The user's software then must periodically change (CCAP4H,CCAP4L) to keep a match from occurring with the PCA timer (CH,CL). This code is given in the WATCHDOG routine. This routine should not be part of an interrupt service routine. Why? Because if the program counter goes astray and gets stuck in an infinite loop, interrupts will still be serviced and the watchdog will keep getting reset. Thus, the purpose of the watchdog would be defeated. Instead call this subroutine from the main program within $2^{16}$ count of the PCA timer. Figure 11. Watchdog Timer Mode (Module 4) #### Listing 9. Watchdog Timer ``` INIT_WATCHDOG: MOV CCAPM4, #4CH ; Module 4 in compare mode ; Write to low byte first MOV CCAP4L, #OFFH MOV CCAP4H, #OFFH ; Before PCA timer counts up to ; FFFF Hex, these compare values ; must be changed ORL CMOD, #40H ; Set the WDTE bit to enable the ; watchdog timer without changing ; the other bits in CMOD ******************** Main program goes here, but CALL WATCHDOG periodically. ****** WATCHDOG: ; Hold off interrupts CLR EA MOV CCAP4L, #00 ; Next compare value is within MOV CCAP4H, CH ; 255 counts of the current PCA SETB EA ; timer value RET ``` #### **PULSE WIDTH MODULATOR** The PCA can generate 8-bit PWMs by comparing the low byte of the PCA timer (CL) with the low byte of the compare registers (CCAPnL). When CL < CCAPnL the output is low. When CL $\geq$ CCAPnL the output is high. To control the duty cycle of the output, the user actually loads a value into the high byte CCAPnH (see Figure 12). Since a write to this register is asynchronous, a new value is not shifted into CCAPnL for comparison until the next period of the output: that is, when CL rolls over from 255 to 00. This mechanism provides "glitch-free" writes to CCAPnH when the duty cycle of the output is changed. CCAPnH can contain any integer from 0 to 255, but Figure 13 shows a few common duty cycles and the corresponding values for CCAPnH. Note that a 0% duty cycle can be obtained by writing to the port pin directly with the CLR bit instruction. To calculate the CCAPnH value for a given duty cycle, use the following equation: ``` CCAPnH = 256 (1 - Duty Cycle) ``` where CCAPnH is an 8-bit integer and Duty Cycle is expressed as a fraction. Figure 12. PWM Mode (Module 0) Figure 13. CCAPnH Varies Duty Cycle Table 4. PWM Frequencies. | Table III Will requestions | | | | | | | | |-----------------------------------------------------------|------------------------------------------|------------------------------------------|--|--|--|--|--| | PCA Timer Mode | PWM Frequency | | | | | | | | TOA TIME! MOUE | 12 MHz | 16 MHz | | | | | | | 1/12 Osc. Frequency | 3.9 KHz | 5.2 KHz | | | | | | | 1/4 Osc. Frequency | 11.8 KHz | 15.6 KHz | | | | | | | Timer 0 Overflow:<br>8-bit<br>16-bit<br>8-bit Auto-Reload | 15.5 Hz<br>0.06 Hz<br>3.9 KHz to 15.3 Hz | 20.3 Hz<br>0.08 Hz<br>5.2 KHz to 20.3 Hz | | | | | | | External Input (Max) | 5.9 KHz | 7.8 KHz | | | | | | #### Listing 10. PWM ``` INIT-PWM: SETB Pl.3 ; For alternate function MOV CMOD, #02H ; Clock input = 250 nsec at 16 MHz MOV CL, #00H ; Frequency of output = 15.6 KHz MOV CH, #00H MOV CCAPMO, #42H ; Module 0 in PWM mode MOV CCAPOL, #00H MOV CCAPOH, #128D ; 50 percent duty cycle ; SETB CR ; Turn on PCA timer ``` The frequency of the PWM output will depend on which of the four inputs is chosen for the PCA timer. The maximum frequency is 15.6 KHz at 16 MHz. Refer to Table 4 for a summary of the different PWM frequencies possible with the PCA. Listing 10 shows how to initialize Module 0 for a PWM signal at 50% duty cycle. Notice that no PCA interrupt is needed to generate the PWM (i.e no software overhead!). To create a PWM output on the 8051 requires a hardware timer plus software overhead to toggle the port pin. The advantage of the PCA is obvious, not to mention it can support up to 5 PWM outputs with just one chip. #### CONCLUSION This list of examples with the PCA is by no means exhaustive. However, the advantages of the PCA can easily be seen from the given applications. For example, the PCA can provide better resolution than Timers 0, 1 and 2 because the PCA clock rate can be three times faster. The PCA can also perform many tasks that these hardware timers can not, i.e. measure phase differences between signals or generate PWMs. In a sense, the PCA provides the user with five more timer/counters in addition to Timers 0, 1 and 2 on the 8XC51FA/FB. Appendix A includes test routines for all the software examples in this application note. The divide routine for calculating duty cycles is in Appendix B. And finally, Appendix C is a table of the Special Function Registers for the 8XC51FA/FB with the new or modified registers **boldfaced**. ## APPENDIX A TEST ROUTINES ``` Listing 1a - Measuring Pulse Widths $nomod51 $nosymbols $nolist $include (reg252.pdf) $list Variables CAPTURE PULSE_WIDTH FLAG DATA DATA BIT 30H 32H ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT ; Initialize PCA timer PCA_INIT: MOV CMOD, #00H MOV CH, #00 ; Input to PCA timer = 1/12 x Fosc MOV CL, #00 Initialize Module 0 in capture mode MOV CCAPMO, #21H ; Capture positive edge first on P1.3 MOV CCAPOH, #00 MOV CCAPOL, #00 SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn PCA timer on CLR FLAG ; Clear test flag Test program only WAIT: JMP$ ; Wait for PCA interrupt JMP WAIT This code assumes Module 0 is the only module being used. If other PCA module's are being used, software must check which module's event flag caused the interrupt. PCA_INTERRUPT: CLR CCF0 ; Clear module 0's event flag JB FLAG, SECOND_CAPTURE ; FIRST_CAPTURE: MOV CAPTURE, CCAP0L MOV CAPTURE+1, CCAP0H 270609-16 ``` ``` MOV CCAPMO, #11H SETB FLAG RETI ; SECOND_CAPTURE: PUSH ACC PUSH PSW CLR C MOV A, CCAPOL SUBB A, CAPTURE MOV PULSE_WIDTH, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PULSE_WIDTH+1, A ; MOV CCAPMO, #21H CLR FLAG POP PSW POP ACC RETI ; END ; Change module to now capture ; falling edges ; Signify first capture complete ``` ``` Listing 1b - Measuring Periods $nomod51 $nosymbols $nolist $include (reg252.pdf) $list Variables CAPTURE PERIOD FLAG DATA 30H DATA BIT 32H 20H.0 ORG 0000H JMP PCA_INIT ; ORG 0033H JMP PCA_INTERRUPT Initialize PCA timer NIT: MOV CMOD, #00H MOV CH, #00H MOV CL, #00 PCA_INIT: ; Input to timer = 1/12 x Fosc Initialize Module 0 in capture mode MOV CCAPM0, #21H ; Capture rising edges on P1.3 MOV CCAPOH, #00 MOV CCAPOL, #00 SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn PCA timer on ; Clear test flag CLR FLAG Test program only WAIT: JMP $ ; Wait for PCA interrupt JMP WAIT ; This code assumes only Module 0 is being used. If other modules are being used, software must check which module's flag caused the interrupt. PCA_INTERRUPT: CLR CCF0 JB FLAG, SECOND_CAPTURE ; Clear module 0's event flag FIRST_CAPTURE: MOV CAPTURE, CCAP0L MOV CAPTURE+1, CCAP0H 270609-18 ``` ``` SETB FLAG RETI ; SECOND_CAPTURE: PUSH ACC PUSH PSW CLR C MOV A, CCAPOL SUBB A, CAPTURE MOV PERIOD, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PERIOD+1, A ; CLR FLAG POP PSW POP ACC RETI ; END ; Signify first capture complete ; 16-Bit subtraction ; 16-Bit subtraction ; 270609–19 ``` ``` Listing 2 - Measuring Frequencies $nomod51 $nosymbols $nolist $include (reg252.pdf) $list Variables ; CAPTURE PERIOD SAMPLE_COUNT FLAG DATA DATA DATA 30H 32H 34H 20H.0 ORG 0000H JMP PCA_INIT ; ORG 0033H JMP PCA_INTERRUPT Initialize PCA timer NIT: MOV CMOD, #00H MOV CH, #00 MOV CL, #00 PCA_INIT: ; Input to PCA timer = 1/12 x Fosc Initialize Module 0 in capture mode MOV CCAPM0, #21H ; Capture positive edges on P1.3 MOV CCAPOH, #00 MOV CCAPOL, #00 ; MOV SAMPLE_COUNT, #10D ; N = 10 for this example SETB EC ; Enable PCA interrupt SETB EA ; Turn PCA timer on SETB CR CLR FLAG ; Test flag Test program only WAIT: JMP $ ; Wait for PCA interrupt JMP WAIT ; This code assumes only Module 0 is being used. PCA_INTERRUPT: CLR CCF0 JB FLAG, NEXT_CAPTURE ; Clear module 0's event flag ; 270609-20 ``` ``` FIRST_CAPTURE: MOV CAPTURE, CCAPOL MOV CAPTURE+1, CCAPOH SETB FLAG RETI ; NEXT_CAPTURE: DJNZ SAMPLE_COUNT, EXIT PUSH ACC PUSH PSW CLR C MOV A, CCAPOL SUBB A, CAPTURE MOV PERIOD, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PERIOD+1, A ; MOV SAMPLE_COUNT, #10D CLR FLAG POP PSW POP ACC EXIT: RETI ; END ; Signify first capture complete ; 16-Bit subtraction ``` ``` Listing 3 - Measuring Duty Cycle $nomod51 $nosymbols $nolist $include (reg252.pdf) $list Variables CAPTURE PULSE_WIDTH PERIOD DATA DATA 30H 32H DATA 34H ; FLAG_1 FLAG_2 20H.0 BIT 20H.1 ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT Initialize PCA timer NIT: MOV CMOD, #00H MOV CH, #00 MOV CL, #00 PCA_INIT: ; Input to PCA timer = 1/12 x Fosc Initialize Module 0 in capture mode MOV CCAPM0, #21H ; Capture positive edge first on P1.3 MOV CCAPOH, #00 MOV CCAPOL, #00 CLR FLAG_1 CLR FLAG_2 ; Clear test flags ; Enable PCA interrupt SETB EC SETB EA SETB CR ; Turn PCA timer on ; Test program only , WAIT: JMP $ ; Wait for PCA interrupt JMP WAIT ; This code assumes Module 0 is the only PCA module being used. ; PCA_INTERRUPT: CLR CCF0 JB FLAG_1, SECOND_CAPTURE ; Clear module 0's event flag 270609-22 ``` ``` FIRST_CAPTURE: MOV CAPTURE, CCAPOL MOV CAPTURE+1, CCAPOH SETB FLAG 1 MOV CCAPMO, #31H PETI ; Signify first capture complete ; Capture either edge now RETI SECOND_CAPTURE: PUSH ACC PUSH PSW PUSH PSW JB FLAG_2, THIRD_CAPTURE CLR C MOV A, CCAPOL SUBB A, CAPTURE MOV PULSE_WIDTH, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PULSE_WIDTH+1, A ; Calculate pulse width ; 16-bit subtract SETB FLAG_2 POP PSW POP ACC ; Signify second capture complete RETI THIRD_CAPTURE: JRE: CLR C MOV A, CCAPOL SUBB A, CAPTURE MOV PERIOD, A MOV A, CCAPOH SUBB A, CAPTURE+1 MOV PERIOD+1, A ; Calculate period ; 16-bit subtract ; MOV CCAPM0, #21H CLR FLAG_1 CLR FLAG_2 POP PSW POP ACC ; Optional- reconfigure module to ; capture positive edges for ; next cycle RETI END 270609-23 ``` ``` Listing 4 - Measuring Phase Differences $nomod51 $nosymbols $nolist $include (reg252.pdf) $list Variables CAPTURE_0 CAPTURE_1 PHASE DATA 30H 32H DATA 34H FLAG_0 BIT 20H.0 FLAG_1 20H.1 ; ORG 0000H JMP PCA_INIT ; ORG 0033H JMP PCA_INTERRUPT ; Initialize PCA timer PCA_INIT: MOV CMOD, #00H MOV CH, #00 MOV CL, #00 ; Input to PCA timer = 1/12 x Fosc Initialize Modules 0 & 1 in capture mode MOV CCAPM0, #21H ; Capture positive edges on P1.3 MOV CCAPOH, #00 MOV CCAPOL, #00 MOV CCAPM1, #21H MOV CCAP1H, #00 MOV CCAP1L, #00 ; Capture positive edges on P1.4 ; MOV R0, #0FFH MOV R1, #0FFH ; Used for test program only CLR FLAG_0 CLR FLAG_1 ; Clear test flags SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn PCA timer on ; 270609-24 ``` ``` Test program only CALL TOG1 CALL DELAY2 CALL TOG2 JMP MAIN MAIN: ; Generate two waveforms ; with known phase difference ; These two waveforms are input to ; P1.3 and P1.4 TOG1: CPL P1.6 CALL DELAY1 RET TOG2: CPL P1.5 CALL DELAY1 DJNZ RO, $ DELAY1: DELAY2: DJNZ R1, $ RET This code assumes only Modules 0 and 1 are being used. PCA_INTERRUPT: JB CCF0, MODULE_0 ; Determine which module's event JB CCF1, MODULE_1 ; caused the interrupt , MODULE_0: CLR CCF0 MOV CAPTURE_0, CCAP0L MOV CAPTURE_0+1, CCAP0H JB FLAG_1, CALCULATE_PHASE ; Clear Module 0's event flag ; If capture is complete on Module 1, ; go to calculation ; Signify capture complete on ; Module 0 SETB FLAG_0 RETI MODULE_1: CLR CCF1 ; Clear Module 1's event flag MOV CAPTURE_1, CCAP1L MOV CAPTURE_1+1, CCAP1H JB FLAG_0, CALCULATE_PHASE ; If capture is complete on Module 0, ; go to calculation ; Signify capture complete ; Module 1 SETB FLAG_1 RETI CALCULATE_PHASE: PUSH ACC ; This calculation does not have to PUSH PSW ; be completed in the interrupt CLR C JB FLAG_0, MOD0_LEADING 270609-25 ``` ``` Listing 5. Software Timer $nomod51 $nosymbols $nolist $include (reg252.pdf) $list Software Timer mode which interrupts every 20 msec with Fosc = 12 MHz. ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT Initialize PCA timer MOV CMOD, #00H MOV CH, #00 MOV CL, #00 PCA_INIT: ; Input to PCA timer = 1/12 x Fosc MOV CCAPM0, #49H MOV CCAP0L, #LOW(20000) MOV CCAP0H, #HIGH(20000) ; Software Timer mode with interrupt ; Write to low byte first SETB EC SETB EA ; Enable PCA interrupt SETB CR ; Turn PCA timer on Test program only WAIT: JMP$ ; Wait for PCA interrupt JMP WAIT This code assumes Module 0 is the only module being used. If other PCA module's are being used, software must check which module's event flag caused the interrupt. PCA_INTERRUPT: CLR CCF0 PUSH ACC PUSH PSW ; Clear module 0's event flag PUSH PSW CLR EA MOV A, #LOW(20000) ADD A, CCAPOL MOV CCAPOL, A MOV A, #HIGH(20000) ADDC A, CCAPOH MOV CCAPOH, A SETR EA ; Hold off interrupts ; 16-bit add ; Next match will occur 20,000 ; counts later SETB EA POP PSW POP ACC RETI END 270609-27 ``` ``` ; Listing 6. High Speed Output (without interrupt) $nomod51 $nosymbols $nolist sinclude (reg252.pdf) $list ; HSO mode without PCA interrupt. Maximum frequency output = 30.5 Hz at Fosc = 16 MHz. ; ORG 0000H JMP PCA_INIT ; Initialize PCA timer PCA_INIT: MOV CMOD, #02H MOV CH, #00 MOV CL, #00 ; MOV CCAPOH, #0FFH MOV CCAPOH, #0FFH MOV CCAPOH, #0FFH SETB CR ; Period = 30.5 Hz ; Turn PCA timer on ; END $270609-28 ``` ``` Listing 7. High Speed Output (with interrupts) $nomod51 $nosymbols $include (reg252.pdf) $list HSO mode with variable frequency. This example outputs a 1KHz signal ORG 0000H JMP PCA INIT ; ORG 0033H JMP PCA_INTERRUPT Initialize PCA timer NIT: MOV CMOD, #02H MOV CH, #00 ; Input to PCA timer = 1/4 x Fosc PCA_INIT: MOV CL, #00 MOV CCAPM0, #4DH MOV CCAP0L, #LOW(1000) MOV CCAP0H, #HIGH(1000) CLR P1.3 ; HSO mode with interrupt enabled ; t = 1000 arbitrary SETB EC SETB EA ; Enable PCA interrupt SETB CR ; Turn PCA timer on Test program only JMP $ ; Wait for PCA interrupt WAIT: JMP WAIT: JMP WAIT This code assumes Module 0 is the only module being used. If other PCA module's are being used, software must check which module's event flag caused the interrupt. PCA INTERRUPT: CLR CCF0 PUSH ACC PUSH PSW ; Clear module 0's event flag CLR EA MOV A, #LOW(2000) ADD A, CCAPOL MOV CCAPOL, A ; Hold off interrupts ; 16-bit add ; 2000 counts later P1.3 ; will toggle MOV A, #HIGH(2000) ADDC A, CCAP0H MOV CCAP0H, A 270609-29 SETB EA POP PSW POP ACC RETI END 270609-30 ``` ``` Listing 8. High Speed Output (Single Pulse) $nomod51 $nosymbols $nolist $include (reg252.pdf) HSO mode generates a single pulse width of 20 usecs with Fosc = 16 MHz. ORG 0000H JMP PCA_INIT ORG 0033H JMP PCA_INTERRUPT Initialize PCA timer NIT: MOV CMOD, #02H MOV CH, #00 PCA_INIT: ; Input to PCA timer = 1/4 x Fosc MOV CL, #00 MOV CCAPMO, #4DH MOV CCAPOL, #LOW(1000) MOV CCAPOH, #HIGH(1000) ; HSO mode with interrupt enabled ; t = 1000 arbitrary CLR P1.3 SETB EC ; Enable PCA interrupt SETB EA SETB CR ; Turn PCA timer on Test program only WAIT: JMP $ ; Wait for PCA interrupt JMP WAIT ; WAIT: This code assumes Module 0 is the only module being used. If other PCA module's are being used, software must check which module's event flag caused the interrupt. PCA_INTERRUPT: CLR CCF0 JNB P1.3, DONE ; Clear module 0's event flag PUSH ACC PUSH PSW CLR EA MOV A, #LOW(80) ADD A, CCAPOL MOV CCAPOL, A ; Hold off interrupts ; 16-bit add ; 80 counts later P1.3 ; will toggle MOV A, #HIGH(80) 270609-31 ADDC A, CCAP0H MOV CCAP0H, A SETB EA POP PSW POP ACC RETI DONE: MOV CCAPMO, #00H ; Disable HSO mode RETI END 270609-32 ``` ``` Listing 9. Watchdog Timer $nomod51 $nosymbols $nolist $include (reg252.pdf) $list ORG 0000H JMP PCA_INIT Initialize PCA timer NIT: MOV CMOD, #00H MOV CH, #00 MOV CL, #00 PCA_INIT: ; Input to PCA timer = 1/12 x Fosc ; Module 4 in compare mode ; Write to low byte first ; Before PCA timer counts up to FFFF Hex, MOV CCAPM4, #4CH MOV CCAP4L, #0FFH MOV CCAP4H, #0FFH ; these compare values must be changed ; Set the WDTE bit to enable watchdog timer ORL CMOD, #40H SETB CR ; Turn PCA timer on Test program only MOV R1, #120D MOV R0, #0FFH START: ; Delay for approx. 60 msec DJNZ R0, $ DJNZ R1, MAIN CALL WATCHDOG JMP START MAIN: ; Check that watchdog never causes a reset WATCHDOG: CLR EA MOV CCAP4L, #00H MOV CCAP4H, CH ; Hold off interrupts ; Next compare value is within ; 255 counts of the current PCA ; timer value SETB EA RET END 270609-33 ``` ``` ; Listing 10. Pulse Width Modulator $nomod51 $nosymbols $nolist $include (reg252.pdf) $list ; PWM mode — Maximum frequency output = 15.6 KHz with Fosc = 16 Mhz. ; ORG 0000H JMP PCA_INIT ; Initialize PCA timer PCA_INIT: SETB P1.3 ; For alternate function MOV CMOD, #02H ; Input to PCA timer = 1/4 x Fosc MOV CH, #00 ; At 16 MHz, frequency = 15.6 KHz MOV CCAPOL, #00H MOV CCAPOL, #00H MOV CCAPOL, #128D ; PWM Mode ; Write to low byte first ; 50 percent duty cycle ; Turn PCA timer on $270609-34 ``` ## **APPENDIX B Duty Cycle Calculation** \$DEBUG SHORT\_DIVISION SEGMENT CODE EXTRN DATA(PULSE\_WIDTH, PERIOD, DUTY\_CYCLE) PUBLIC DUTY\_CYCLE\_CALCULATION RSEG SHORT\_DIVISION DUTY\_CYCLE\_CALCULATION CALCULATES DUTY\_CYCLE = PULSE\_WIDTH / PERIOD Inputs to this routine are 16-bit pulse width and period measurements of a rectangular waveform. The output is a 9-bit BCD number representing the duty cycle of the waveform. The low 8 bits of the result are returned in DUTY\_CYCLE. The 9th bit is the carry bit in the PSW. If the duty cycle is between 0 and 99 percent, the carry bit is 0 and DUTY\_CYCLE contains the two BCD digits representing the duty cycle as a percent. If the duty cycle is 100 percent, the carry bit is 1 and DUTY\_CYCLE INPUTS: PULSE\_WIDTH 2 bytes in externally defined DATA (low byte at PULSE\_WIDTH, high byte at PULSE\_WIDTH+1) > 2 bytes in externally defined DATA (low byte at PERIOD, high byte at PERIOD+1) OUTPUT: DUTY\_CYCLE 1 byte in externally defined DATA VARIABLES AND REGISTERS MODIFIED: PULSE\_WIDTH, DUTY\_CYCLE ACC, B, PSW, R2, R3 ERROR EXIT: Exit with OV = 1 indicates PULSE\_WIDTH > PERIOD. DUTY\_CYCLE\_CALCULATION: MOV A,PERIOD+1 CJNE A,PULSE\_WIDTH+1,NOT\_EQUAL MOV A,PERIOD CJNE A,PULSE\_WIDTH,NOT\_EQUAL 270609-35 ``` EQUAL: SETB C MOV DUTY_CYCLE,#0 CLR OV RET NOT_EQUAL: JNC SETB OV RET CONTINUE: MOV R2.#8 MOV DUTY_CYCLE,#0 MOV PULSE_WIDTH RLC A MOV APULSE_WIDTH+1,A MOV APULSE_WIDTH+1,A MOV APULSE_WIDTH+1,A MOV APULSE_WIDTH+1,A MOV APULSE_WIDTH+1,A MOV APULSE_WIDTH+1,C MOV PULSE_WIDTH+1,C MOV PULSE_WIDTH+1,C MOV PULSE_WIDTH+1,C MOV PULSE_WIDTH+1,C MOV R3.A COMPARE: CJNE R3.#0,DONE MOV APULSE_WIDTH+1 CJNE APERIOD-1,DONE MOV APULSE_WIDTH+1 CJNE APERIOD-1,DONE MOV APULSE_WIDTH+1 CJNE APERIOD-1,DONE SOPL CPL C BUILD_DUTY_CYCLE MOV ADUTY_CYCLE HCC MOV APULSE_WIDTH SUBB APERIOD MOV APULSE_WIDTH+1 APERIOD-1 MOV APULSE_WIDTH+1,A MOV APILSE_WIDTH+1,A APILSE ``` ``` FINAL_TIMES_TWO: MOV A_PULSE_WIDTH RLC A MOV PULSE_WIDTH+1 RLC A MOV PULSE_WIDTH+1,A MOV A_PULSE_WIDTH+1,A MOV A_R3 RLC A MOV PULSE_WIDTH+1,A MOV PULSE_WIDTH+1,A MOV R3,A RLC A MOV R3,A FINAL_COMPARE: CJNE R3,#0,FINAL_DONE MOV A_PULSE_WIDTH+1 CJNE A_PERIOD_FINAL_DONE MOV A_PULSE_WIDTH CJNE A_PERIOD_FINAL_DONE FINAL_DONE: JC CONVERT_TO_BCD MOV A_DUTY_CYCLE ADD A_#1 MOV DUTY_CYCLE,A JNC CONVERT_TO_BCD CLR OV RETO CONVERT_TO_BCD: MOV A_DUTY_CYCLE MOV B_#10 MUL BB XCA A_B XWAP A MOV A_#10 MUL A8 XCA A_B ``` ## **APPENDIX C** A map of the Special Function Register (SFR) space is shown in Table A1. Those registers which are new or have new bits added for the 83C51FA and 83C51FB have been boldfaced. Note that not all of the addresses are occupied. Unoccupied addresses are not implemented on the chip. Read accesses to these addresses will in general return random data, and write accesses will have no effect. User software should not write 1s to these unimplemented locations, since they may be used in future 8051 family products to invoke new features. In that case the reset or inactive values of the new bits will always be 0, and their active values will be 1. Table A1. Special Function Register Memory Map and Values After Reset | F8 | | СН | ССАРОН | CCAP1H | CCAP2H | ССАРЗН | CCAP4H | | FF | |------|--------------------|--------------------|--------------------|--------------------|--------------------|--------------------|--------------------|----------------------|---------| | | | 00000000 | XXXXXXX | XXXXXXX | XXXXXXX | XXXXXXX | XXXXXXX | | - | | F0 | * B | | | | | | | | F7 | | | 00000000 | | | | | | | | - | | E8 | | CL | CCAP0L | CCAP1L | CCAP2L | CCAP3L | CCAP4L | | EF | | | | 00000000 | **** | XXXXXXX | **** | **** | **** | | | | E0 | * ACC<br>00000000 | | | | | | | | E7 | | D8 | CCON<br>00X00000 | CMOD<br>00XXX000 | CCAPM0<br>X0000000 | CCAPM1<br>X0000000 | CCAPM2<br>X0000000 | CCAPM3<br>X0000000 | CCAPM4<br>X0000000 | | DF | | D0 | * PSW<br>00000000 | | | | | | | | D7 | | C8 | | T2MOD<br>XXXXXXX0 | RCAP2L<br>00000000 | RCAP2H<br>00000000 | TL2<br>00000000 | TH2<br>00000000 | | | CF | | C0 | | | | | | | | | C7 | | В8 | * IP<br>X0000000 | SADEN<br>00000000 | | | | | | | BF | | В0 | * P3 | | | | | | | | В7 | | | 11111111 | | | | | | | | | | A8 | * IE<br>00000000 | SADDR<br>00000000 | | | | | | | AF | | ΑO | * P2 | | | | | | | | A7 | | , 10 | 11111111 | | | | | | | | ļ · · · | | 98 | * SCON<br>00000000 | * SBUF<br>XXXXXXXX | | | | | | | 9F | | 90 | * P1<br>11111111 | | | | | | | | 97 | | 88 | * TCON<br>00000000 | * TMOD<br>00000000 | * TL0<br>00000000 | * TL1<br>00000000 | * TH0<br>00000000 | * TH1<br>00000000 | | | 8F | | 80 | * P0<br>11111111 | * SP<br>00000111 | * DPL<br>00000000 | * DPH<br>00000000 | | | | *PCON **<br>00XX0000 | 1 | <sup>\* =</sup> Found in the 8051 core (See 8051 Hardware Description in the Embedded Controller Handbook for explanations of these SFRs). \*\* = See description of PCON SFR. Bit PCON.4 is not affected by reset. X = Undefined.