單片機(jī)直接驅(qū)動(dòng)段式液晶
0X00
最近幾天需要用到段式液晶,同事打樣回來,驚奇發(fā)現(xiàn)驅(qū)動(dòng)不了,放狗一頓搜,才發(fā)現(xiàn)硬件連接錯(cuò)誤,同事直接把seg com直接連接到單片機(jī)IO上了,直接用IO來驅(qū)動(dòng)段式液晶不是不可以,但是1/2偏壓的液晶還好,再往上1/3,1/4...etc就異常費(fèi)勁,通用的做法是使用ht1621類似的驅(qū)動(dòng)芯片。這涉及到LCD的相關(guān)知識,就記錄一下吧。
參考資料:
液晶原理百度文庫
silicon labs 公司AN202關(guān)于直接驅(qū)動(dòng)的方案
0X01 LCD的顯示原理
LCD的結(jié)構(gòu)從底到上依次為偏振片 玻璃基板 液晶分子 玻璃基板 偏振片上下兩個(gè)偏振片呈90度,簡而言之,基板之間不加交流電時(shí)液晶分子會(huì)導(dǎo)致光線轉(zhuǎn)向從而穿過兩個(gè)呈90度的偏振片,而加上超過閾值交流電以后液晶不再對光線有轉(zhuǎn)向作用,導(dǎo)致光線不能穿過偏振片,從而產(chǎn)生明暗變化(裝作很懂的樣子),至于為什么需要加交流電,因?yàn)槿绻挥弥绷麟姇?huì)導(dǎo)致液晶壽命變短、重影。。。。
0X02 控制方式
在LCD的參數(shù)中有幾個(gè)重要參數(shù),duty,bias,driver voltage。duty表示掃描周期一般為與com引腳個(gè)數(shù)有關(guān),bias表示偏壓,偏壓1/3表示把VDD分三份,電壓差越大越亮,bias與com引腳個(gè)數(shù)也有關(guān)系,因?yàn)閏om腳越多周期越短,為了保持亮度一致,需要提高選通和不選通的電壓差。需要分更多份電壓。。。(更不懂了,敬請參考文首文庫資料)一圖勝萬言。
0X03 驅(qū)動(dòng)芯片驅(qū)動(dòng)段式液晶
使用ht1621等驅(qū)動(dòng)芯片比較容易驅(qū)動(dòng),根據(jù)RAM映射表填入數(shù)據(jù)即可,驅(qū)動(dòng)芯片會(huì)自動(dòng)更新數(shù)據(jù)。
0X04 單片機(jī)直接驅(qū)動(dòng)
剛開始用硬件直接驅(qū)動(dòng)我是拒絕的,想想都覺得麻煩,經(jīng)過兩天的學(xué)習(xí)(恩,真的花了兩天時(shí)間。。。),詳細(xì)了解了驅(qū)動(dòng)過程,仔細(xì)思考一番,覺得直接驅(qū)動(dòng)也不是那么麻煩,為了紀(jì)念兩天時(shí)光,用了一下午寫了個(gè)驅(qū)動(dòng),驗(yàn)證了自己思考的時(shí)序,我這份驅(qū)動(dòng)沒有使用定時(shí)器,如果使用的話最好在全狀態(tài)機(jī)環(huán)境下使用,避免阻塞,當(dāng)然你也可以自己加到定時(shí)器中斷里。僅僅為驗(yàn)證想法,不多說上代碼。
1 //芯片:stm8s007
2 //開發(fā)環(huán)境:windows IAR
3 //液晶模塊與單片機(jī)連接比較亂,所以自己重定義了一下,
4 //液晶參數(shù):1/4duty,1/3bias,5V driver voltage
5 //com口采用上下拉電阻產(chǎn)生一個(gè)中間電壓
6
7 #include "display.h"
8
9 #define SEG_ALL_ON (0xffffffff)
10 #define SEG_ALL_OFF (0x00000000)
11
12 //PA_ODR_ODR0
13 #define PIN(__port,__pin) (P##__port##_ODR_ODR##__pin)
14 //PE_CR1_C10
15 #define PINOD(__port,__pin) (P##__port##_CR1_C1##__pin)
16
17 #define SEG1(__val) (PIN(B,7)=(__val))
18 #define SEG2(__val) (PIN(A,6)=(__val))
19 #define SEG3(__val) (PIN(A,3)=(__val))
20 #define SEG4(__val) (PIN(D,3)=(__val))
21 #define SEG5(__val) (PIN(D,2)=(__val))
22 #define SEG6(__val) (PIN(D,0)=(__val))
23 #define SEG7(__val) (PIN(E,0)=(__val))
24 #define SEG8(__val) (PIN(E,1)=(__val))
25 #define SEG9(__val) (PIN(E,2)=(__val))
26 #define SEG10(__val) (PIN(E,3)=(__val))
27 #define SEG11(__val) (PIN(G,0)=(__val))
28 #define SEG12(__val) (PIN(G,1)=(__val))
29 #define SEG13(__val) (PIN(C,7)=(__val))
30 #define SEG14(__val) (PIN(C,6)=(__val))
31 #define SEG15(__val) (PIN(C,5)=(__val))
32 #define SEG16(__val) (PIN(C,4)=(__val))
33 #define SEG17(__val) (PIN(C,3)=(__val))
34 #define SEG18(__val) (PIN(C,2)=(__val))
35 #define SEG19(__val) (PIN(C,1)=(__val))
36 #define SEG20(__val) (PIN(E,5)=(__val))
37
38 //PE_CR1 |= BIN(0,0,0,0, 0,0,0,0);//0 open drain 1 push-pull
39 #define COM4_ODR(__val) (PIN(E,6)=(__val))
40 #define COM4_DIR(__val) (PINOD(E,6)=(__val))
41 #define COM3_ODR(__val) (PIN(E,7)=(__val))
42 #define COM3_DIR(__val) (PINOD(E,7)=(__val))
43 #define COM2_ODR(__val) (PIN(B,0)=(__val))
44 #define COM2_DIR(__val) (PINOD(B,0)=(__val))
45 #define COM1_ODR(__val) (PIN(B,1)=(__val))
46 #define COM1_DIR(__val) (PINOD(B,1)=(__val))
47
48 #define COM1(__val) (COM1_DIR(1),COM1_ODR(__val))
49 #define COM1_OD() (COM1_DIR(0),COM1_ODR(1))
50 #define COM2(__val) (COM2_DIR(1),COM2_ODR(__val))
51 #define COM2_OD() (COM2_DIR(0),COM2_ODR(1))
52 #define COM3(__val) (COM3_DIR(1),COM3_ODR(__val))
53 #define COM3_OD() (COM3_DIR(0),COM3_ODR(1))
54 #define COM4(__val) (COM4_DIR(1),COM4_ODR(__val))
55 #define COM4_OD() (COM4_DIR(0),COM4_ODR(1))
56
57
58 static void seg_load(uint32_t val)
59 {
60 SEG1(val&0x0001);
61 val >>= 1;
62 SEG2(val&0x0001);
63 val >>= 1;
64 SEG3(val&0x0001);
65 val >>= 1;
66 SEG4(val&0x0001);
67 val >>= 1;
68 SEG5(val&0x0001);
69 val >>= 1;
70 SEG6(val&0x0001);
71 val >>= 1;
72 SEG7(val&0x0001);
73 val >>= 1;
74 SEG8(val&0x0001);
75 val >>= 1;
76 SEG9(val&0x0001);
77 val >>= 1;
78 SEG10(val&0x0001);
79 val >>= 1;
80 SEG11(val&0x0001);
81 val >>= 1;
82 SEG12(val&0x0001);
83 val >>= 1;
84 SEG13(val&0x0001);
85 val >>= 1;
86 SEG14(val&0x0001);
87 val >>= 1;
88 SEG15(val&0x0001);
89 val >>= 1;
90 SEG16(val&0x0001);
91 val >>= 1;
92 SEG17(val&0x0001);
93 val >>= 1;
94 SEG18(val&0x0001);
95 val >>= 1;
96 SEG19(val&0x0001);
97 val >>= 1;
98 SEG20(val&0x0001);
99 val >>= 1;
100 }
101
102 void display_init()
103 {
104 PE_DDR |= BIN(1,1,0,0, 0,0,0,0);//0 in 1 out
105 PE_CR1 |= BIN(0,0,0,0, 0,0,0,0);//0 open drain 1 push-pull
106 PE_CR2 |= BIN(1,1,0,0, 0,0,0,0);//0 2M 1 10M
107
108 PB_DDR |= BIN(0,0,0,0, 0,0,1,1);//0 in 1 out
109 PB_CR1 |= BIN(0,0,0,0, 0,0,0,0);//0 open drain 1 push-pull
110 PB_CR2 |= BIN(0,0,0,0, 0,0,1,1);//0 2M 1 10M
111
112
113 PA_DDR |= BIN(0,1,0,0, 1,0,0,0);//0 in 1 out
114 PA_CR1 |= BIN(0,1,0,0, 1,0,0,0);//0 open drain 1 push-pull
115 PA_CR2 |= BIN(0,1,0,0, 1,0,0,0);//0 2M 1 10M
116
117 PB_DDR |= BIN(1,0,0,0, 0,0,0,0);//0 in 1 out
118 PB_CR1 |= BIN(1,0,0,0, 0,0,0,0);//0 open drain 1 push-pull
119 PB_CR2 |= BIN(1,0,0,0, 0,0,0,0);//0 2M 1 10M
120
121 PC_DDR |= BIN(1,1,1,1, 1,1,1,0);//0 in 1 out
122 PC_CR1 |= BIN(1,1,1,1, 1,1,1,0);//0 open drain 1 push-pull
123 PC_CR2 |= BIN(1,1,1,1, 1,1,1,0);//0 2M 1 10M
124
125 PD_DDR |= BIN(0,0,0,0, 1,1,0,1);//0 in 1 out
126 PD_CR1 |= BIN(0,0,0,0, 1,1,0,1);//0 open drain 1 push-pull
127 PD_CR2 |= BIN(0,0,0,0, 1,1,0,1);//0 2M 1 10M
128
129 PE_DDR |= BIN(0,0,1,0, 1,1,1,1);//0 in 1 out
130 PE_CR1 |= BIN(0,0,1,0, 1,1,1,1);//0 open drain 1 push-pull
131 PE_CR2 |= BIN(0,0,1,0, 1,1,1,1);//0 2M 1 10M
132
133 PG_DDR |= BIN(0,0,0,0, 0,0,1,1);//0 in 1 out
134 PG_CR1 |= BIN(0,0,0,0, 0,0,1,1);//0 open drain 1 push-pull
135 PG_CR2 |= BIN(0,0,0,0, 0,0,1,1);//0 2M 1 10M
136
137 COM1_OD();
138 COM2_OD();
139 COM3_OD();
140 COM4_OD();
141 seg_load(SEG_ALL_ON);
142 }
143
144 uint32_t g_wTestValue=1;
145 static void dispaly_com()
146 {
147 static enum{
148 DISPLAY_COM_START = 0,
149 DISPLAY_COM_COM1,
150 DISPLAY_COM_COM2,
151 DISPLAY_COM_COM3,
152 DISPLAY_COM_COM4,
153 }s_emState = DISPLAY_COM_START;
154 static uint8_t IsTrue=1;
155
156 switch(s_emState) {
157 case DISPLAY_COM_START:
158 s_emState = DISPLAY_COM_COM1;
159 //break;
160 case DISPLAY_COM_COM1:
161 COM1_OD();
162 COM2_OD();
163 COM3_OD();
164 COM4_OD();
165 if(IsTrue) {
166 COM1(1);
167 //seg_load(~g_wTestValue);
168 //seg_load(~4096);
169 seg_load(SEG_ALL_ON);
170 }else {
171