[stm32] 一個簡單的stm32vet6驅動2.4寸240X320的8位並口tft屏DEMO


 

書接上文:

最近在研究用低速、低RAM的單片機來驅動小LCD或TFT彩屏實現動畫效果

首先我用一個16MHz晶振的m0內核的8位單片機nRF51822嘗試驅動一個1.77寸的4線SPI屏(128X160),

發現,刷一屏大約要0.8s左右的時間,

具體收錄在《1、一個簡單的nRF51822驅動的天馬4線SPI-1.77寸LCD彩屏DEMO》中

覺得,如果用72MHz的STM32也許效果會好很多

於是在stm32上做了個類似的版本,

具體收錄在《一個簡單的stm32vet6驅動的天馬4線SPI-1.77寸LCD彩屏DEMO》中

發現刷一屏0.2s左右,

效果是有的,但是還不能達到支持播放流暢動畫的效果!

於是,決定將串行數據改成並行數據傳輸

本節將帶來一個用stm32驅動的2.4寸240X320的8位並口tft屏的刷屏效果

 

工程結構

 

main.c

 1 /* Includes ------------------------------------------------------------------*/
 2 #include "stm32f10x.h"
 3 #include "LCD2.h"
 4 
 5 
 6 void RCC_Configuration(void);
 7 /****************************************************************************
 8 * 名    稱:int main(void)
 9 * 功    能:主函數
10 * 入口參數:無
11 * 出口參數:無
12 * 說    明:
13 * 調用方法:無 
14 ****************************************************************************/ 
15 int main(void)
16 {
17     RCC_Configuration();                   //系統時鍾配置
18     LCD2_GPIO_Init();
19     LCD2_Init();
20     while (1)
21     {
22         Show_RGB(0,240,0,320,0xff0f);
23         DELAY_MS(1000);
24         Show_RGB(0,240,0,320,0x00fe);
25         DELAY_MS(1000);
26     }
27 }
28 
29 /****************************************************************************
30 * 名    稱:void RCC_Configuration(void)
31 * 功    能:系統時鍾配置為72MHZ
32 * 入口參數:無
33 * 出口參數:無
34 * 說    明:
35 * 調用方法:無 
36 ****************************************************************************/ 
37 void RCC_Configuration(void)
38 {   
39   SystemInit();
40 }

 

LCD2.c

  1 #include "LCD2.h"
  2 
  3 
  4 
  5 void LCD2_GPIO_Init()
  6 {
  7     GPIO_InitTypeDef GPIO_InitStructure;
  8 
  9     RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOD | RCC_APB2Periph_AFIO, ENABLE);
 10     
 11     
 12     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8 | GPIO_Pin_9;
 13     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
 14     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;             //口線翻轉速度為50MHz
 15     GPIO_Init(GPIOB, &GPIO_InitStructure);
 16     
 17     //8位數據輸出
 18     GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7;
 19     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
 20     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;             //口線翻轉速度為50MHz
 21     GPIO_Init(GPIOD, &GPIO_InitStructure);            
 22 }
 23 
 24 //////////////////////////////////////////////////////////////////
 25 //最底層數據傳輸函數
 26 //////////////////////////////////////////////////////////////////
 27 //寫命令
 28 void Write_Cmd(unsigned char DH,unsigned char DL)
 29 {
 30     LCD2_CS=0;
 31     LCD2_RS=0;
 32 
 33     DataPort=DH;
 34     LCD2_RW=0;
 35     LCD2_RW=1;
 36 
 37     DataPort=DL;
 38     
 39     LCD2_RW=0;
 40     LCD2_RW=1;
 41     LCD2_CS=1;
 42 }
 43 //寫數據 雙8位
 44 void Write_Data(unsigned char DH,unsigned char DL)
 45 {
 46     LCD2_CS=0;
 47     
 48     LCD2_RS=1;
 49     DataPort=DH;
 50     LCD2_RW=0;
 51     LCD2_RW=1;
 52 
 53     DataPort=DL;    
 54     LCD2_RW=0;
 55     LCD2_RW=1;
 56     LCD2_CS=1;
 57 }
 58 
 59 //寫數據 雙8位
 60 void Write_Data2(unsigned char DH,unsigned char DL)
 61 {
 62     DataPort=DH;
 63     LCD2_RW=0;
 64     LCD2_RW=1;
 65 
 66     DataPort=DL;    
 67     LCD2_RW=0;
 68     LCD2_RW=1;
 69 }
 70 
 71 //////////////////////////////////////////////////////////////////
 72 //調用上面最底層實現稍高層寫命令和數據函數
 73 //////////////////////////////////////////////////////////////////
 74 /*----------------------------------------------------------------
 75                          寫命令、寫數據
 76 輸入參數:x 需要輸入的命令 16位
 77           y 需要輸入的數據 16位
 78 ----------------------------------------------------------------*/
 79 void  Write_Cmd_Data (unsigned char x,unsigned int y)
 80 {
 81     unsigned char m,n;
 82     m=y>>8;
 83     n=y;
 84     Write_Cmd(0x00,x);
 85     Write_Data(m,n);
 86 }
 87 /*----------------------------------------------------------------
 88                          寫16位數據
 89 ----------------------------------------------------------------*/
 90 void  Write_Data_U16(unsigned int y)
 91 {
 92     unsigned char m,n;
 93     m=y>>8;
 94     n=y;
 95     Write_Data2(m,n);
 96 }
 97 
 98 /*----------------------------------------------------------------
 99                             液晶初始化
100 ----------------------------------------------------------------*/
101 void LCD2_Init(void)
102 { 
103     LCD2_CS=1;
104     DELAY_MS(5);
105     LCD2_RES=0;
106     DELAY_MS(5);
107     LCD2_RES=1;
108     DELAY_MS(50);
109     Write_Cmd_Data(0x0001,0x0100); 
110     Write_Cmd_Data(0x0002,0x0700); 
111     Write_Cmd_Data(0x0003,0x1030); 
112     Write_Cmd_Data(0x0004,0x0000); 
113     Write_Cmd_Data(0x0008,0x0207);  
114     Write_Cmd_Data(0x0009,0x0000);
115     Write_Cmd_Data(0x000A,0x0000); 
116     Write_Cmd_Data(0x000C,0x0000); 
117     Write_Cmd_Data(0x000D,0x0000);
118     Write_Cmd_Data(0x000F,0x0000);
119     //power on sequence VGHVGL
120     Write_Cmd_Data(0x0010,0x0000);   
121     Write_Cmd_Data(0x0011,0x0007);  
122     Write_Cmd_Data(0x0012,0x0000);  
123     Write_Cmd_Data(0x0013,0x0000); 
124     //vgh 
125     Write_Cmd_Data(0x0010,0x1290);   
126     Write_Cmd_Data(0x0011,0x0227);
127     //DELAY_MS(100);
128     //vregiout 
129     Write_Cmd_Data(0x0012,0x001d); //0x001b
130     //DELAY_MS(100); 
131     //vom amplitude
132     Write_Cmd_Data(0x0013,0x1500);
133     //DELAY_MS(100); 
134     //vom H
135     Write_Cmd_Data(0x0029,0x0018); 
136     Write_Cmd_Data(0x002B,0x000D); 
137 
138     //gamma
139     Write_Cmd_Data(0x0030,0x0004);
140     Write_Cmd_Data(0x0031,0x0307);
141     Write_Cmd_Data(0x0032,0x0002);// 0006
142     Write_Cmd_Data(0x0035,0x0206);
143     Write_Cmd_Data(0x0036,0x0408);
144     Write_Cmd_Data(0x0037,0x0507); 
145     Write_Cmd_Data(0x0038,0x0204);//0200
146     Write_Cmd_Data(0x0039,0x0707); 
147     Write_Cmd_Data(0x003C,0x0405);// 0504
148     Write_Cmd_Data(0x003D,0x0F02); 
149     //ram
150     Write_Cmd_Data(0x0050,0x0000); 
151     Write_Cmd_Data(0x0051,0x00EF);
152     Write_Cmd_Data(0x0052,0x0000); 
153     Write_Cmd_Data(0x0053,0x013F);  
154     Write_Cmd_Data(0x0060,0xA700); 
155     Write_Cmd_Data(0x0061,0x0001); 
156     Write_Cmd_Data(0x006A,0x0000); 
157     //
158     Write_Cmd_Data(0x0080,0x0000); 
159     Write_Cmd_Data(0x0081,0x0000); 
160     Write_Cmd_Data(0x0082,0x0000); 
161     Write_Cmd_Data(0x0083,0x0000); 
162     Write_Cmd_Data(0x0084,0x0000); 
163     Write_Cmd_Data(0x0085,0x0000); 
164     //
165     Write_Cmd_Data(0x0090,0x0010); 
166     Write_Cmd_Data(0x0092,0x0600); 
167     Write_Cmd_Data(0x0093,0x0003); 
168     Write_Cmd_Data(0x0095,0x0110); 
169     Write_Cmd_Data(0x0097,0x0000); 
170     Write_Cmd_Data(0x0098,0x0000);
171     Write_Cmd_Data(0x0007,0x0133);
172     
173     //    Write_Cmd_Data(0x0022);//        
174 }
175 
176 /*----------------------------------------------------------------
177                          設置坐標
178 ----------------------------------------------------------------*/
179 /*----------------------------------------------------------------
180                             全局變量
181 ----------------------------------------------------------------*/
182 #define WINDOW_XADDR_START    0x0050 // Horizontal Start Address Set
183 #define WINDOW_XADDR_END    0x0051 // Horizontal End Address Set
184 #define WINDOW_YADDR_START    0x0052 // Vertical Start Address Set
185 #define WINDOW_YADDR_END    0x0053 // Vertical End Address Set
186 #define GRAM_XADDR            0x0020 // GRAM Horizontal Address Set
187 #define GRAM_YADDR            0x0021 // GRAM Vertical Address Set
188 #define GRAMWR                 0x0022 // memory write
189 void LCD_SetPos(unsigned int x0,unsigned int x1,unsigned int y0,unsigned int y1)
190 {
191     Write_Cmd_Data(WINDOW_XADDR_START,x0);
192     Write_Cmd_Data(WINDOW_XADDR_END,x1);
193     Write_Cmd_Data(WINDOW_YADDR_START,y0);
194     Write_Cmd_Data(WINDOW_YADDR_END,y1);
195     Write_Cmd_Data(GRAM_XADDR,x0);
196     Write_Cmd_Data(GRAM_YADDR,y0);
197     Write_Cmd (0x00,0x22);//LCD_WriteCMD(GRAMWR);
198 }
199 
200 /*----------------------------------------------------------------
201                             顯示RGB顏色
202 輸入參數:x0,y0 起始坐標
203           x1,y1 結束坐標
204           Color  背景顏色
205 ----------------------------------------------------------------*/
206 void Show_RGB (unsigned int x0,unsigned int x1,unsigned int y0,unsigned int y1,unsigned int Color)
207 {
208     unsigned int i,j;
209     LCD_SetPos(x0,x1,y0,y1);
210     LCD2_CS=0;
211     LCD2_RS=1;
212 //    for (i=y0;i<=y1;i++)
213 //    {
214 //       for (j=x0;j<=x1;j++)
215 //           Write_Data_U16(Color);
216 //    }
217     
218     for (i=0;i<=(y1-y0+1)*(x1-x0+1);i+=32)
219     {
220          Write_Data_U16(Color);
221          Write_Data_U16(Color);
222         Write_Data_U16(Color);
223          Write_Data_U16(Color);
224          Write_Data_U16(Color);
225          Write_Data_U16(Color);
226         Write_Data_U16(Color);
227          Write_Data_U16(Color);
228         Write_Data_U16(Color);
229          Write_Data_U16(Color);
230         Write_Data_U16(Color);
231          Write_Data_U16(Color);
232          Write_Data_U16(Color);
233          Write_Data_U16(Color);
234         Write_Data_U16(Color);
235          Write_Data_U16(Color);
236          Write_Data_U16(Color);
237          Write_Data_U16(Color);
238         Write_Data_U16(Color);
239          Write_Data_U16(Color);
240          Write_Data_U16(Color);
241          Write_Data_U16(Color);
242         Write_Data_U16(Color);
243          Write_Data_U16(Color);
244         Write_Data_U16(Color);
245          Write_Data_U16(Color);
246         Write_Data_U16(Color);
247          Write_Data_U16(Color);
248          Write_Data_U16(Color);
249          Write_Data_U16(Color);
250         Write_Data_U16(Color);
251          Write_Data_U16(Color);
252     }
253         LCD2_CS=1;
254 }
255 
256 
257 void Delay_ms(u16 time)
258 {
259     u16 i=0;  
260     while(time--)
261     {
262         i=12000; 
263         while(i--);    
264     }
265 }

注:代碼比較容易理解,不做詳解

 

另外補上這三個小實驗的連線圖:

1、這個是本節的並行接口與屏幕的連接方式:

2、這是上兩節串行接口的連線,上面對應的引腳連接是與nRF51822的(第一次試驗),下面對應的連接是與stm32的(第二次試驗)

 

小結

從效果圖上看,即使采用stm32的8位並行來驅動屏幕速度還是達不到刷新動畫的效果~

之后我也傳輸數據的函數上做了些優化,可效果還是不明顯——

第一點:優化前RS等引腳的定義要通過宏展開,每次計算BitBand后面的式子~

1 #define BitBand(Addr, Bit) *((volatile int*)(((int)(Addr) & 0x60000000) + 0x02000000 + (int)(Addr) * 0x20 + (Bit) * 4))
2 #define LCD2_CS         BitBand(&GPIOB->ODR, 8)
3 #define LCD2_RES        BitBand(&GPIOB->ODR, 9)
4 #define LCD2_RS         BitBand(&GPIOB->ODR, 7)
5 #define LCD2_RW         BitBand(&GPIOB->ODR, 6)
6 #define DataPort         GPIOD->ODR

優化后采用直接把值賦給對應的引腳來減少運算量

1 #define BitBand(Addr, Bit) *((volatile int*)(((int)(Addr) & 0x60000000) + 0x02000000 + (int)(Addr) * 0x20 + (Bit) * 4))
2 #define LCD2_CS         (*((volatile int*)0x422181A0))   //BitBand(&GPIOB->ODR, 8)
3 #define LCD2_RES        (*((volatile int*)0x422181A4))   //BitBand(&GPIOB->ODR, 9)
4 #define LCD2_RS         (*((volatile int*)0x4221819C))   //BitBand(&GPIOB->ODR, 7)
5 #define LCD2_RW         (*((volatile int*)0x42218198))   //BitBand(&GPIOB->ODR, 6)
6 #define DataPort         (*((volatile int*)0x4001140C)) //GPIOD->ODR

第二點:為了減少Show_RGB函數中循環中的Write_Data_U16的調用,直接將Write_Data_U16計算拆到最細放到循環內

第三點:為了排除Write_Data_U16中4、5兩行移位運算和類型轉換所帶來的時間花銷,直接采用上圖中全局變量color1、color2來直接賦值,查看效果有沒有提升~

1 void  Write_Data_U16(unsigned int y)
2 {
3     unsigned char m,n;
4     m=y>>8;
5     n=y;
6     Write_Data2(m,n);
7 }

本篇中資源鏈接:http://pan.baidu.com/s/1bnjw1Fh

注:其中未優化版工程比較簡潔方便理解學習,優化測試版是為了提升傳輸速率做的幾點優化(效果不大,代碼稍亂)

 

 

 

@beautifulzzzz 

  2015-11-28 持續更新中~


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM