content:pic:pic18_7_segment_displays
Differences
This shows you the differences between two versions of the page.
content:pic:pic18_7_segment_displays [2022/07/02 11:15] – created admin | content:pic:pic18_7_segment_displays [2022/07/02 11:16] (current) – removed admin | ||
---|---|---|---|
Line 1: | Line 1: | ||
- | ====== PIC18: 7-segment displays ====== | ||
- | {{: | ||
- | |||
- | In fact they are quite simple, because they consist of 8 leds (decimal point included) connected in parallel and with the cathode (or anode) in common; by turning on specific leds (//' | ||
- | |||
- | Here we see the two types of displays: | ||
- | |||
- | {{: | ||
- | |||
- | When used with PICs the different anodes of a single 7-segment display are connected to different PORTx pins and the common cathodes (from now on we will talk about common cathode displays only) are grounded: | ||
- | |||
- | {{: | ||
- | |||
- | By making certain PORTD pins high the corrispondent display segments are lit and numbers or letters appear; for instance to display | ||
- | |||
- | {{: | ||
- | |||
- | ===== Multiplexing ===== | ||
- | |||
- | In case we have multiple displays PORT lines should be doubled or multiplied; even if a single PIC could have that many pins a better approach is to use a trick called **// | ||
- | |||
- | {{: | ||
- | |||
- | The common cathodes cannot be directly connected to RE0:RE2 to sink current, because a single PIC port can only sink a maximum of 25mA; with 330ohm resistors and considering the voltage drop of a LED, the current that passes through a single segment is 10mA, so even with only 3 segments lit (for instance to display the digit //' | ||
- | |||
- | Let's say we want to display the number //248//. We want to display the //units// and so we copy the binary value //' | ||
- | |||
- | * clear PORTE,0 and PORTE,2 | ||
- | * output //' | ||
- | * set PORTE,1 | ||
- | |||
- | The same process will be followed for the // | ||
- | |||
- | {{: | ||
- | |||
- | The smaller we keep the delay, the faster the switch between the digits will be: | ||
- | |||
- | {{: | ||
- | |||
- | If the **delay is small enough**, the human eye won't notice the switch and **all digits will appear on at the same time**: | ||
- | |||
- | {{: | ||
- | |||
- | Enough with the theory for now: let's display some real values on the digits! | ||
- | |||
- | ===== Assembly code Main routine ===== | ||
- | |||
- | For this example we are going to use **three digits** as the image below, **which will display the value of the A/D conversion from the analog trimmer** of Freedom II. The **main routine** will be very light because it will only wait for the following interrupts to occur: | ||
- | |||
- | * A/D conversion interrupt from trimmer | ||
- | * TMR0 overflow interrupt | ||
- | |||
- | The first one intercepts the operation of the trimmer and, if activated, calls the appropriate sub to do the A/D conversion. | ||
- | |||
- | The purpose of the second interrupt routine is to decide which digit has to be turned on; when the timer overflows **//1// is left shifted in //DIGITS// register**, from bit 0 to bit 2. What happens next is, if for instance DIGITS,0 is set: | ||
- | |||
- | * only PORTE,0 is set high, which lets pass current to the base of the transistor of the //units// digit (the rightmost one), in order to // | ||
- | * the value of //units// is transformed in the proper way to display them, as in the table above | ||
- | * PORTD is loaded with this value | ||
- | * only the //units// digit displays it, the others not being activated by the saturation of the corresponding transistor | ||
- | |||
- | As soon as TMR0 overflows, //DIGITS// file register is shifted left and now DIGITS,1 is set, which sets high PORTE,1 and saturates the //tenths// digit' | ||
- | When TMR0 overflows once more it's the turn of the // | ||
- | |||
- | {{: | ||
- | |||
- | Now that the overall behaviour of the program should be clear, let's dig a bit deeper into **the assembly code**. | ||
- | |||
- | ===== From bytes to units, tenths and hundreds ===== | ||
- | |||
- | When an A/D conversion occurs, the resulting byte value is passed to the sub // | ||
- | |||
- | LoopHundreds | ||
- | subwf number, | ||
- | bnc GetTenths | ||
- | incf hundreds, | ||
- | goto LoopHundreds | ||
- | |||
- | In the first instruction .100 is subtracted from _number_; if there is no borrow (that is //Carry=1// - see **[[pic18_branch_instructions|branch instructions]]**), | ||
- | Those single values can now be displayed onto the corresponding digits. | ||
- | |||
- | ===== Look-up tables ===== | ||
- | |||
- | In order to display a digit, //8 units// for instance, its binary value b' | ||
- | |||
- | (movf realValue, | ||
- | movwf PCL | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | |||
- | Here the //real// value is copied to W and then added to Program Counter Low to get the //display// value returned; cool, isn't it? Well, things are not so simple... | ||
- | |||
- | ==== WREG doubled ==== | ||
- | |||
- | As we have seen in a **[[pic18_guide_assembling_linking_programming_linux|previous article]]**, | ||
- | |||
- | Address | ||
- | ------- | ||
- | ... | ||
- | 000004 | ||
- | 000006 | ||
- | 000008 | ||
- | 00000a | ||
- | 00000c | ||
- | 00000e | ||
- | 000010 | ||
- | 000012 | ||
- | 000014 | ||
- | 000016 | ||
- | ... | ||
- | |||
- | If 8 (the real value) is added to PCL, this last one become address 0x00000c (0x000004 + 0x8) and the displayed value is ' | ||
- | |||
- | addwf WREG,W ; W+W = 2W (16bit program words); offset x2 | ||
- | movwf PCL | ||
- | retlw b' | ||
- | ... | ||
- | |||
- | That one above works...if we are lucky! There' | ||
- | |||
- | ==== PCLATH and PCLATU ==== | ||
- | |||
- | When we have to change PC it's important that all 21 bits be modified // | ||
- | |||
- | {{: | ||
- | |||
- | In the above way all 21 bits of PC are modified at once; so it's important, **before writing to PCL, to have PCLATH and PCLATU ready with the correct values**; secondly the code should be robust enough to take into account possible // | ||
- | |||
- | 0000f6 | ||
- | ; words) | ||
- | 0000f8 | ||
- | ; to location based on the | ||
- | ; value of W reg and then | ||
- | ; returns with literal | ||
- | 0000fa | ||
- | 0000fc | ||
- | 0000fe | ||
- | 000100 | ||
- | 000102 | ||
- | 000104 | ||
- | 000106 | ||
- | 000108 | ||
- | 00010a | ||
- | 00010c | ||
- | ... | ||
- | |||
- | In the above (not working) example PCL takes care of the lower bits (from 0xF6 to 0xFE and then 0x00,0x02 and so on), but PCLATH should handle the middle bits and PCLATU the leftmost higher bits; initially PCLATH should be 0x00, then it should change to 0x01. | ||
- | |||
- | _7SegDisplayTable | ||
- | andlw b' | ||
- | addwf WREG,W ; W+W = 2W (16bit program words); offset x2 | ||
- | addlw LOW Table_7 ; W = offset x2 + LOW PC [7:0] | ||
- | movwf TableTemp ; | ||
- | movlw HIGH Table_7 ; W = HIGH PC [15:8] | ||
- | movwf PCLATH ; copy W to PCLATH | ||
- | clrf WREG | ||
- | addwfc PCLATH, | ||
- | ; (clrf and mov* don't touch C flag) | ||
- | movlw UPPER Table_7 ; W = UPPER PC [20:16] | ||
- | movwf PCLATU ; copy W to PCLATU | ||
- | clrf WREG | ||
- | addwfc PCLATU, | ||
- | ; (clrf and mov* don't touch C flag) | ||
- | movf TableTemp, | ||
- | ; PCLATH and PCLATU | ||
- | movwf PCL | ||
- | ; counter by any operation that writes PCL." | ||
- | Table_7 | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | retlw b' | ||
- | |||
- | So, here is what we call a robust code! Let's see the relevant lines in depth. | ||
- | |||
- | ==== Low, High and Upper ==== | ||
- | |||
- | First of all W is //masked//, in order to avoid spurious values higher than 0xFF: | ||
- | |||
- | andlw b' | ||
- | |||
- | Then we have WREG doubled and we have it added to the //low byte// of the address located at the label Table_7, which marks the beginning of the look-up table: | ||
- | |||
- | addlw LOW Table_7 | ||
- | |||
- | PCL, PCLATH and PCLATU represent different bytes of a 21-bit address; a way to split the address into those parts is by using the **operands LOW, HIGH and UPPER**. Here's a definition taken from MPASM Assembler User's Guide: | ||
- | |||
- | {{: | ||
- | |||
- | Let's see this thing in practice; suppose we want to display //' | ||
- | |||
- | 0x0000fa | ||
- | ... | ||
- | 0x00010a | ||
- | 0x00010c | ||
- | |||
- | So first we have //8// doubled, that is //0x10//; we add this value to // | ||
- | |||
- | 0x fa + | ||
- | 0x 10 = | ||
- | --------- | ||
- | 0x(1)0a --> PCL | ||
- | | ||
- | |||
- | But since // | ||
- | |||
- | Let's deal with PCLATH now and so let's consider the [15-8] bytes of the address: | ||
- | |||
- | movlw HIGH Table_7 | ||
- | movwf PCLATH | ||
- | |||
- | Hey, but wait! We **got a Carry** from our previous sum (//addlw LOW Table_7//), so let's add it to PCLATH; as written in the comments CLRF and MOV* operations in between don't touch C flag. | ||
- | The instruction _addwfc_ is just what we need for this purpose (PCLATH + Carry); we just have to clear WREG, which in this case is not really needed: | ||
- | |||
- | clrf WREG | ||
- | addwfc PCLATH,F | ||
- | |||
- | Here's the sum of PCLATH and Carry: | ||
- | |||
- | 0x00 + | ||
- | 0x01 = | ||
- | ------- | ||
- | 0x01 --> PCLATH | ||
- | |||
- | The same process is used to get PCLATU, but in this case it's just 0x0; **as soon as PCL is written to (//movwf PCL//) at the same time PCLATH and PCLATU are transferred to their corrisponing positions in the 21-bit address** (see image above), that is // | ||
- | |||
- | And that pretty much completes the description of the code, which can be **downloaded in the .zip file below**. | ||
- | |||
- | Here's we can watch the final result: | ||
- | |||
- | {{youtube> | ||
- | |||
- | ===== Reference ===== | ||
- | |||
- | "The Essential PIC18 Microcontroller" |
content/pic/pic18_7_segment_displays.1656753307.txt.gz · Last modified: 2022/07/02 11:15 by admin