一、背景
9 J9 x4 c* _3 { Z m g: J在引脚充足、外设没有复用的情况下,基本上是使用连续连接去驱动段码屏的,如图一;# d3 G! |4 f. e2 |4 w
[, c9 ]) Z. @" v而在引脚不够,外设又复用的情况下,就必须得使用非连续的引脚驱动段码屏,如图二,因此本文主要是介绍如何使用非连续引脚驱动断码屏。. O+ G7 a1 T J" }; [
+ ~% z4 ~5 e, Z) |: L二、环境准备( |2 r0 }% T- p3 w) o8 q
1、基于4COM- 8段段码屏;3 F( \+ d: } M
段码屏的段码如下:! P; ~) N+ I8 ]+ A* F
" Z, Q2 n- U& g0 u* x
2、基于STM32L053开发板+ t+ j# z" G2 _! c* h
3、基于STM32CubeMX软件生成代码
# b. k9 }0 @. N9 b. y三、驱动段码屏原理/ V1 V2 z0 ~3 ~" t2 {
段码屏其实就是点亮一个有固定形状的LED灯,将这些LED灯按照一定数字或者文字或者符号的外形放置就组成了要点亮的一个符号。如果按照上述原理去驱动4位数字,8个段,一共得需要32个的IO口,这显然是不合理,因此需要使用STM32单片机自带的LCD驱动功能。拿我现在使用的单片机来说,它能够驱动4COM、64个段的段码屏,本次使用的是4COM-8SEG段码屏;那么点亮一个数字,就可以让这个数字的公共端处于低电平,另外,要显示数字,就需要对应的段处于高电平(实际上是交流信号,非高低电平驱动),这样所在COM位置就显示想要的数字,那么要显示4位数字,就需要利用人眼的“暂留”效应进行动态扫描。) y( p. ^6 J/ g: `1 j. F7 o/ }
四、配置工程& Y* N4 R1 z' I4 K
1、新建工程,使能时钟/ J/ U4 L# C3 F7 i$ t& h
5 Y/ Y6 |& m z" c2 E6 H, {/ U* h9 u5 {9 `- _$ J4 e
7 H# x) J" s; k3 X1 Y' K6 C2、使能LCD相应的段' H4 b# B& ^2 ]( `# n2 a
1)连续段" H; R5 ^( g0 k% A' g: l4 [
2 d5 C- r0 _5 z# z: e' b5 F6 w) I$ j# q' q. n; ?6 V! a
2)非连续段0 T6 ~/ B1 Y; E! a
1 X7 L1 e3 ~$ i& v
9 o: g1 l W9 y& ~- ?5 m3、编写LCD驱动代码
7 \. y$ P/ j! P0 a% I$ a1)必备知识1 S- \4 f# s( a) W+ u
点亮不同数字,显示对应段的表
1 R+ M r7 |0 W$ g% V
2 o$ i7 D3 ~; j7 a" G: p要让数字对应段点亮,必须使得COM端和SEG段形成电压差,如图
$ ? ]& K; w" k
& A0 X2 M/ A* z2 y' BSTM32L052是使用双缓存区进行数据动态刷新的; X$ {$ s0 Y7 W5 D/ U# E/ J7 Y
2)下面详细介绍驱动原理
/ _4 [% C( u _6 R+ ja、连续引脚 c* M) f0 k8 u Z) E7 U$ g
原理与代码实现/ K" s1 z! ^. B8 X
如需要第一个数字显示数字0,则需要使得A-F,这6个段点亮,那么需要动态点亮这个几段,最简单的方法就是HAL_LCD_Write这个函数,分别写入4个COM段和SEG0-SEG1对应的缓存区。
) a' K1 \ b: g- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER0, 0,0x03);//将COM0 SEG0-1点亮; [! A o/ o' i9 t5 o+ N# i
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER2, 0,0x01 << 2);//将COM1 SEG0-1点亮;9 h/ \/ \/ r- R7 s
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER4, 0,0x03 << 4);//将COM2 SEG0-1点亮;. n1 O6 \! \; B, n6 n, n
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER6, 0,0x02 << 6);//将COM3 SEG0-1点亮;
* p) R* N, [) @( J" X' n# Z; ]& S( V - HAL_LCD_UpdateDisplayRequest(&hlcd);
复制代码 这样可以把0-9的数字按两个SEG组成一个数字,形成如下表- uint8_t NumArr[10][4] =2 s; _0 M0 \9 v
- {+ c7 A7 d9 @4 F# O. J& A
- /*{0,*/{0x3,0x1,0x3,0x2},
, d: P0 |' m8 J& a! T* J. ] - /*{1,*/{0x2,0x0,0x2,0x0},
0 @" p" Z3 c. ]+ } - /*{2,*/{0x3,0x2,0x1,0x2},
, P3 n1 G, V. W: n/ v" ^4 `8 k - /*{3,*/{0x3,0x2,0x2,0x2},
+ p& e( U8 q8 b4 {) ^1 n: l - /*{4,*/{0x2,0x3,0x2,0x0},) G* F/ L1 g7 w9 b- H
- /*{5,*/{0x1,0x3,0x2,0x2},
" _3 |1 m: d2 E0 G% t1 u7 C" x - /*{6,*/{0x1,0x3,0x3,0x2},
~" P7 q/ B! V' O5 @ - /*{7,*/{0x3,0x0,0x2,0x0},) }. N/ W: d- G1 L
- /*{8,*/{0x3,0x3,0x3,0x2},
" p+ i0 y- c t, v0 x% e' n( G8 P - /*{9,*/{0x3,0x3,0x2,0x2},$ s3 O+ S( x, R0 U, `3 x
- 7 O+ D; x, A9 K, M6 L
- };
复制代码 因为要使4位LCD显示4个数字,需要先对四个数字见处理,如下* ~3 w4 N% ^- f1 G b; r6 o
- uint8_t SEG1,SEG2,SEG3,SEG4;
! C/ a2 `1 b: G1 w
4 i; W! X7 r! c. K1 \( [/ u( {- void showTimer(uint16_t mm,uint16_t ss); g, V5 U8 j# i4 Q
- {1 N' t* B3 I7 e( X1 N
- SEG1 = SEG2 = SEG3 = 0;
0 W7 q1 m* Q# e7 d - SEG4 = 0x01;
) E/ d$ U% s F i, N6 S - SEG1|=NumArr[mm/10][0];
* v& g. ]8 W7 D; Q - SEG2|=NumArr[mm/10][1];+ D7 p7 H0 o; K4 [
- SEG3|=NumArr[mm/10][2];" b- S9 v9 a) `7 B
- SEG4|=NumArr[mm/10][3];
2 Y/ l1 F ], j [) V( y
: s7 R. G1 e0 E. c" V' A7 {
r) p4 ^4 N9 I O. u* l' ?! D' Y0 J- 2 Y2 g# q! u' u# O- o% A# [
- SEG1|=(NumArr[mm%10][0]<<2);$ b" k, y# s# u1 m) \# T
- SEG2|=(NumArr[mm%10][1]<<2);
, S9 e9 q) O" T; U - SEG3|=(NumArr[mm%10][2]<<2);7 V# h% X6 S9 c
- SEG4|=(NumArr[mm%10][3]<<2);8 t4 G- A, B$ b; ^
6 i8 i7 |) B; Q- s- SEG1|=(NumArr[ss/10][0]<<4);
7 m% e/ ^/ G/ Q8 [ l# I - SEG2|=(NumArr[ss/10][1]<<4);# _ K5 \. m; O
- SEG3|=(NumArr[ss/10][2]<<4); c% ~# X* G9 R4 E! U
- SEG4|=(NumArr[ss/10][3]<<4);2 n4 a; T# h* x/ b
! Y2 O$ Y r; W8 y0 i+ {3 V- SEG1|=(NumArr[ss%10][0]<<6);+ `6 i# I' [, P( a, F
- SEG2|=(NumArr[ss%10][1]<<6);
" v! F6 w2 A! N2 @ - SEG3|=(NumArr[ss%10][2]<<6);
% q* @* J3 F+ i - SEG4|=(NumArr[ss%10][3]<<6);( s2 G7 u) P3 A9 N) m3 L
- # n6 U6 h( s X+ k
- 5 K3 u2 F ^1 V$ l$ w7 J1 M6 \
- }
复制代码 $ [4 @! b K: z6 u2 K0 ~( f# @; O
数据刷新功能函数如下:- void ReFreshLCD(void)
7 n+ z" W$ C% K2 ~; m - {( {+ E$ ?4 o4 L9 u
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER0, 0,SEG1);
, _$ P3 w4 G. c0 G4 @; i, ^& Q - HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER2, 0,SEG2);
m' W- j/ \, g8 `8 b. F - HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER4, 0,SEG3);4 V- F5 C" J: A" w# t [/ C; ^
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER6, 0,SEG4); / ?* n( y: S0 E- R
9 e! @8 x6 |! w7 R7 {1 I- HAL_LCD_UpdateDisplayRequest(&hlcd);
B- D* k. j1 C4 m - }
复制代码
o+ V1 E. G' R: z8 t/ t5 Zb、非连续引脚
4 \' F) O( {: H' {1 ^; V 这种情况不能使用上述方法,需要使用直接法,就是直接对A-G这几个段是否点亮移位。. j$ O# B0 z, D/ T
原理于代码实现
$ e$ N5 ^8 g; p1 A% ?: Q) D$ i8 s首先,需要数字点亮段的对应关系,建立如下一个表,也就是某个段是否显示,用0和1表示" N8 I# I2 |0 [# T9 ?
- uint8_t DisplayCode[SIGN_NUM][SEG_NUM] = {
1 k( k6 d* F" R9 I, Y3 y1 x - /*A B F G E C P D */) E7 o4 N' x6 O, K
- /*{0,*/{0x1,0x1,0x1,0x0,0x1,0x1,0x0,0x1},
; {7 X3 W. o8 w5 y - /*{1,*/{0x0,0x1,0x0,0x0,0x0,0x1,0x0,0x0}, g; [( a+ X' d' Y/ G
- /*{2,*/{0x1,0x1,0x0,0x1,0x1,0x0,0x0,0x1},
' S- W& X9 _; {% {* T$ r - /*{3,*/{0x1,0x1,0x0,0x1,0x0,0x1,0x0,0x1},
& ~' \. X7 ^2 \/ x( D - /*{4,*/{0x0,0x1,0x1,0x1,0x0,0x1,0x0,0x0},( c4 a$ `: h" m c
- /*{5,*/{0x1,0x0,0x1,0x1,0x0,0x1,0x0,0x1},
( l, f7 O& q* _3 }- b5 w- D" {- W - /*{6,*/{0x1,0x0,0x1,0x1,0x1,0x1,0x0,0x1},
1 v6 ^% {7 L: n+ `" i) ?) R( l - /*{7,*/{0x1,0x1,0x0,0x0,0x0,0x1,0x0,0x0},# L3 x" i& {% g9 ]1 N0 h
- /*{8,*/{0x1,0x1,0x1,0x1,0x1,0x1,0x0,0x1},
0 H! P; z S2 E/ N - /*{9,*/{0x1,0x1,0x1,0x1,0x0,0x1,0x0,0x1}
5 _' M, }( G- }2 s2 N8 U - };
复制代码 接着,需要定义使用了哪些段以及建立段表
6 t/ ?8 k l @- /段定义
, }# j9 l y2 \ - #define SEG_Q 0& m: R N6 Y/ C7 Q/ B% F: k1 ~
- #define SEG_Q_COM1 12 t3 L! U7 \+ k
- #define SEG_B 29 |% W* N1 e( B, ^5 p2 ]1 W
- #define SEG_B_COM2 3
" z1 y8 a( S( u! S1 z1 b$ W - #define SEG_S 4; P+ x9 }0 `' ], q
- #define SEG_S_COM3 5
/ S" ]$ `6 C2 t+ B. R7 t - #define SEG_G 68 _+ e2 N* v( ^; e9 h0 E
- #define SEG_G_COM4 8& f7 {3 Q3 d/ X* ~* b
" A3 y$ y# u; H) ]- R- uint8_t useSegNo [SEG_NUM] = {: q$ ~! o8 A6 e! z, n& `
- SEG_Q,SEG_Q_COM1,SEG_B,SEG_B_COM2,SEG_S,SEG_S_COM3,SEG_G,SEG_G_COM4,
5 D! _8 X$ ]0 g! ]( t, {* z6 a0 R - };
复制代码 其次,需要进行每个COM要显示哪些内容进行处理。原理就是针对将所用4位数字对应COM0 即,AB是否显示一起形成一个要显示数字,写入RAM0寄存器中。如图这样组成一个数字操作。代码实现如下
1 b* p: U+ A, ]
4 {# ^- `( [! b8 g+ }! t
- void ConvertSign(uint8_t numFour,uint8_t numThree,uint8_t numTwo,uint8_t numOne) {
. }2 I# o( `8 p
7 |3 K- G# K. ?! e |: P- uint8_t i = 0; r. v7 ?( l# p; @1 u
- uint8_t j = 0;
: `9 z, E2 N" s& F) s$ g' n - display[0] = numFour;
5 g9 }0 j& [& n- f - display[1] = numThree;7 M$ d n v" v' X+ e# c4 A, G
- display[2] = numTwo;/ o% q7 s& C' x& m q. J
- display[3] = numOne;
( i( ~5 a( Z9 U! t0 v9 T - for(i = 0; i < NUM_CNT; i++) {
% v/ M" P+ Z+ p# P @ - result[i] = 0;
5 c4 T8 Q% X8 d, N$ z3 o$ x7 q - }* w2 O5 j7 b |) z2 F) c; p+ ^2 e
- for(j = 0;j < NUM_CNT;j++) {( r0 X9 w1 k G' P
- for(i = 0; i < SEG_NUM / 2; i++) {
8 `# p) z% B F7 H - result[j] |= (DisplayCode[display[i]][2*j] << useSegNo[2*i] | DisplayCode[display[i]][2*j+1] << useSegNo[2*i+1]);
0 P) y" X) G1 { - //result[j] |= NumArr_ONE[display[i]][j];// 0x03 | 0x02 | 0x03 | 0x03
( G9 b& @8 e2 z - }# x) b# A4 Q" T* t+ n
- }5 D& A1 t, Z# w Y" W
- - n8 t6 [4 o+ C9 S
- //result[i] |= (DisplayCode[display[i]][2*j] << useSegNo[2*j] | DisplayCode[display[i]][2*j+1] << useSegNo[2*j+1]);3、2、3、3( j7 p. m) J1 A- g" y) n" ?
- }
复制代码
: A1 G3 Q" [9 G% v A/ t 最后就是数据刷新函数
" u# o9 K' m9 a0 {5 M; p- void ReFreshLCD(void)0 E) z7 l, K7 l; E
- {
: V- x# E# f, u8 t% b
3 E( U: G' Y4 [/ X6 I+ e- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER0, 0,result[0]);1 \( M3 _# e* X9 a6 a
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER2, 0,result[1]);) K& ?. J' ~; G5 X; n9 a
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER4, 0,result[2]);* m' z$ g4 ?* M- a+ k# a& O$ i
- HAL_LCD_Write(&hlcd, LCD_RAM_REGISTER6, 0,result[3] | 0x01); 2 ?' r- D. Z8 e0 q% A
- HAL_LCD_UpdateDisplayRequest(&hlcd);6 u2 T2 h6 A9 C
- }
复制代码 ; L- w- |: d9 ]
, ^3 [- a1 o E/ k( M9 a3、下载验证! g1 d$ o' S* ? Z% f p i
连续段 [% i- P7 d @
9 }. U0 Z: p1 l* a$ O' S# J. u( V3 C# b非连续段. v( W8 q5 z8 U% ^8 Q
短接PB3核PB4。3 l9 j; l# {& W
9 L- K( j3 ~% ^, R7 t
' X% ]) [7 k F" i q4、源代码
- N: }( o) j* Q3 K- M8 [
STM32L053-LCD-非连续.zip
(6.27 MB, 下载次数: 3)
|