一、使用CPU搬运数据到显存 l7 x/ R0 R! K7 r9 S+ Y( x
在上一篇文章中讲述了如何配置 LTDC 驱动 RGB 屏幕:
/ ^ w% S6 [8 b& O4 v! v, k. l/ P3 d' Z- a0 K; V
3 Y! H2 q$ |' G! L/ t& T0 TSTM32CubeMX | 41-使用LTDC驱动TFT-LCD屏幕(RGB屏)。- T; M! s2 {7 \% z
6 V$ [+ J6 m* w: y; {# G3 }! K
) v- C" G+ W$ Q
本节中我们接着上一节的实验,讲述如何使用 DMA2D 实现打点、画线、填充等函数,只需要单层全屏即可,修改LTDC层配置如下:
5 d4 {/ j$ j% @* g6 }$ {6 `
5 i7 E4 Z+ U% t5 L/ x0 ?+ C! H4 r- P: n6 H) h" }
4 Z5 z0 ?8 b/ g8 l& V5 c6 c6 O1. 编写lcd驱动头文件/ k4 U! J( M; J' J, e
创建lcd_rgb_ltdc_drv.h文件,存放关于操作LCD屏幕的一些宏定义配置和函数定义:& s$ T. r5 @4 a3 o- v8 Y
- <font face="微软雅黑" size="3">#ifndef _LCD_RGB_LTDC_DRV_H_
* [7 _5 ]2 _9 H9 b - #define _LCD_RGB_LTDC_DRV_H_: R; {" Y6 W/ Z! a O
- 5 o1 g2 n | z7 t
- #include "ltdc.h"2 [& Q/ E2 O0 n- I9 O1 |
' P" Z& C- ]2 @' I( u9 e5 ]- /**# r+ I8 u y( D, G$ M/ _
- * @brief Windows size on lcd.
& m( O. c0 B* ~- u - */
6 [% ?# e# {2 ] - #define LCD_WIDTH 10242 {& V% s- D, I, U. ~" N
- #define LCD_HEIGHT 600
3 P0 h% r) T) z& ~7 ?+ M1 ~3 [ - 9 F: ^5 V, U, E$ q" k/ W$ T
- /**
' R! d6 a- Y# [; ?- s9 E6 o - * @brief Backlight control pin of lcd.
1 @6 z- d5 {1 X% p( U9 Z - */
; e8 F6 p) s: h) y - #define LCD_BL_GPIO_PORT GPIOB
. }: `9 l {. z0 }& c - #define LCD_BL_GPIO_PIN GPIO_PIN_5
% \* U6 J) ?3 K6 r/ r& \' q - ! K3 a4 A( L6 A6 f6 R
- /**7 U5 v/ X" `3 B8 L
- * @brief start address of lcd framebuffer.' S- U; c; i S" w% Q
- */& ^- B' n, D) O$ G. q$ Z6 P% d& Z
- #define LCD_FRAME_BUFFER 0xc0000000; V# C, d9 d' u1 g
- & [" R0 T$ b$ X; s) s
- /**
3 i) n: A2 Z _- N - * @brief color8 _: x- \. Q- w+ o
- * @note rgb565
# {$ B: ]/ ^, B1 |* _( t- _' j - */# B! H! o' r/ X) R) M
- #define BLACK 0x0000+ G0 D! [0 g+ a9 D
- #define BLUE 0x001F
" b' s, o& [' W/ V- p - #define GREEN 0x07E0* @ F3 T2 c# r
- #define GBLUE 0X07FF( l+ V$ c# C) G) N) B
- #define GRAY 0X8430
6 q& J9 N7 X9 x' F' d - #define BROWN 0XBC406 w; B2 t9 _8 d: X0 C( ]5 y
- #define RED 0xF8009 w* o" b3 M5 s w" t- d! C
- #define PINK 0XF81F
9 s0 t, p5 L% }1 E ?1 V - #define BRRED 0XFC07
1 r+ e" y- X4 Q - #define YELLOW 0xFFE0$ `8 ?, y- m1 w, C& R' C9 m
- #define WHITE 0xFFFF! p3 N, w b" V( Y' C
- 4 p* K, A0 @5 C" W8 s! w3 y$ T
- /**, J# i! T. N) l. a4 z6 M
- * @brief Control the lcd backlight.6 G- N- K! h' z9 _- K
- * @param[in] brightness the value of lcd backlight. H5 C; N6 E! H; h6 Z- l( e! _
- * @return None* M& r9 j( K' V$ b" u5 F& U1 U* o
- */6 I) H1 t j. v: W
- void lcd_backlight_control(uint8_t bightness);
& ^6 G5 H8 G) w6 j
8 G8 a5 B/ o+ }! t3 G- /**8 X' Z7 G; V9 S# ~# I
- * @brief LCD initialization.
0 j$ o2 L) E1 H1 `. r- d - * @param None2 d' B" w0 n) k+ l2 b% f* S
- * @return None
1 u. x& d [- ` j% @ - */
, Y0 k+ `; V, g. f( o - void lcd_init(void);
4 G: Y5 D- {. T( f4 }' _! r, |; L - : m. ]& \. R H
- /**
0 L ~# d% r) {* P' _5 q - * @brief Clear lcd.( ~' Q, o: F: y: z) ?4 C8 g2 Q+ O
- * @param[in] color rgb565.) i7 ^6 s# [! @/ z
- * @return None1 P/ d( N4 G9 t
- */1 i1 ~7 p$ }* s& L, A& ?2 Z$ N1 @
- void lcd_clear(uint16_t color);
" F+ n+ `) `4 z& m* z' x$ |
2 k( N$ E- x+ q2 R& \! n& J1 {7 w- #endif /* _LCD_RGB_LTDC_DRV_H_ */</font>
复制代码
2 S' I" s5 Y2 u- j( R' e; t5 H$ P9 J8 t. L- R/ C/ C9 V3 o1 N
2. lcd驱动实现2 n( v8 H( R) R" F! j4 o
创建lcd_rgb_ltdc_drv.c文件,存放关于操作LCD屏幕的函数实现。' ?4 S3 |$ n3 `+ K
+ L4 E( n/ @$ p4 t F
1 k' e* I& I. f7 z9 O1 s8 ]( d
首先是背光控制实现,应该使用pwm实现背光调节,本文中为了方便直接使用GPIO控制:$ Y2 c( I( d1 t- N$ ?
- <font face="微软雅黑" size="3">void lcd_backlight_control(uint8_t bightness)
( N/ M3 O* Z" t - {
5 l) b" ?: R* ~9 n6 b - // todo: use pwm to control backlight
5 L* F( G* j9 Q' r$ w$ z
+ n2 q8 S- D( c- if (bightness) { " H& p/ i( C% Z, p
- // turn on the backlight
% d6 m/ n H/ V- r - HAL_GPIO_WritePin(LCD_BL_GPIO_PORT, LCD_BL_GPIO_PIN, GPIO_PIN_SET);
5 Q9 B8 o7 B2 j2 W0 O- N0 N - } else { . p2 v9 A$ W, W
- // turn off the backlight
0 {0 W7 \- W! w: p( ?3 y# q) {' C - HAL_GPIO_WritePin(LCD_BL_GPIO_PORT, LCD_BL_GPIO_PIN, GPIO_PIN_RESET);4 M- h! f; ~* D8 v# \# b; B* I
- }
% c8 U* m1 r% r1 x7 ^2 T - }</font>
复制代码 接着实现lcd清屏函数,使用CPU(循环)搬运数据到显存中:# A0 f. C0 _& Z5 d' T) o8 U
- <font face="微软雅黑" size="3">void lcd_clear(uint16_t color)5 l+ y1 I- O7 P) r
- {/ s+ t5 h4 J: ]6 E+ W3 H
- uint16_t *ptr = (uint16_t*)LCD_FRAME_BUFFER;! F, H* \+ w. X1 g4 S! x2 {
- uint32_t i = 0;
) N9 Y( C- u. a1 V
# \2 A8 g2 g+ x! \% O3 _6 G- while (i++ < LCD_WIDTH*LCD_HEIGHT) {
" D- E4 N3 [* o& q! X/ y - *(ptr+i) = color;& t7 U- f" v1 J& u- ]$ c' ?4 X6 Y
- }
7 M4 u# q# O, H! x- O) D; u - }</font>
复制代码 最后实现lcd初始化函数,先刷显存,然后开背光,防止设备上电时屏幕闪烁:) D6 l; |3 p. [
- <font face="微软雅黑" size="3">void lcd_init()
9 C! p2 S+ `8 n' C; A4 v - {2 E! S. c/ K4 s3 c
- lcd_clear(BLACK);8 b% q, ?& k- A. p- D
- lcd_backlight_control(255);7 z& z8 x- }" o
- }</font>
复制代码 , m# _# g- e) h+ Z) @% ~
* x' S! w7 G6 o# _
3. lcd单次清屏时间测试
" w. ?& @0 U& l1 |在main.c中包含驱动头文件:! v- a* h7 a& u
- <font face="微软雅黑" size="3">#include "lcd_rgb_ltdc_drv.h"</font>
复制代码 在main函数中的sdram初始化函数之后,添加lcd初始化函数,并使用HAL库自带的systick时间戳测量一次清屏的时间:
0 n: j4 Y% X1 g- <font face="微软雅黑" size="3">/* USER CODE BEGIN 2 */9 c' H: ]1 L0 A4 E# x7 G- \5 e+ k$ m
- printf("sdram test by mculover666\r\n");8 x$ K( ~5 a, U) ^6 P- u7 J
- SDRAM_Init();
+ C0 c1 w0 ]* {8 \6 M) U - printf("sdram init success\r\n");
' I6 c i; q# Y C) V+ p0 G
& \; E4 O6 J0 ]2 M- lcd_init();
% j5 r9 f0 e1 `7 a
. a/ U6 l4 T4 O y: B m) V- start_time = HAL_GetTick();
) V3 W7 F+ ]9 Q) \+ U) S% @8 `! m - lcd_clear(PINK);
4 x N4 C$ e2 w' J - end_time = HAL_GetTick();
8 G' i& ]1 K! c- S9 R: n
/ B# S4 U, z; N: H0 u& ?" M- printf("lcd clear spend time:%ld ms\r\n", end_time - start_time);
1 Q: P# R) T/ D - /* USER CODE END 2 */</font>
复制代码 编译、运行,在串口助手可以看到使用CPU搬运数据到显存中,在-Og优化等级下单次清屏需要 155 ms左右,在-O0优化等级下单次清屏需要321ms左右:
1 S# I( {3 J8 U7 A" `# V; X. I
2 _0 |( H. `* u
+ T# k6 f9 b1 d+ M! T% x% k8 @6 k
$ C9 X i# ?! w: W6 G( D6 e- c二、使用DMA2D加速显存数据搬运; o- R# _% h# ]
1. DMA2D$ M' ?: q3 }) e( V* _+ [
在STM32中,DMA2D外设专门用来给LCD显示加速,有LTDC外设的型号中,通常也会配套有DMA2D。' e& \; G) X8 X& ?5 D
- S% D, Y$ p' `9 p; X
, k# j* ?8 g/ G$ E) s2 Q7 ~
DMA2D外设主要提供了两个功能:/ x/ C+ P6 P* i; F
- e. p2 }) h( d- ?8 f% \. D* m! j( s( D4 R! q1 K" I( `) Y& U
DMA数据搬运:常用从寄存器到存储器、存储器到存储器两种模式,快速高效,并且不占用cpu资源;
# r' k1 E5 p7 J8 ~" W2D图形加速:支持快速格式转换和混合;4 p3 r* z0 c/ P2 W2 f" c
本文中主要使用到DMA2D外设的数据搬运功能,使用起来也是比较简单。
( x( J$ w9 ^$ q \3 h+ p( {
8 f: }# g, D+ O. w+ h- `. k
' ? z, Z0 J9 q3 T2. 开启DMA2D- G4 h b, p- o4 T- w
" N! s% h3 r9 {/ t( F重新生成工程,cubemx会自动生成并调用dma2d初始化函数,完成dma2d外设时钟使能以及dma2d传输模式配置。
; b/ `0 K! ]5 l. Q! w
: A( {3 G' R% j: r# {
+ M* S" K1 `# X) l* a# B6 B8 Z3. 使用DMA2D实现lcd清屏函数
" x: |. ?$ d& p* E' o( b! G在lcd驱动头文件中添加一个宏定义,用于控制是否使能DMA2D:& u8 X7 \# `; c3 @$ Y+ N. \9 C
- <font face="微软雅黑" size="3">/**' W5 C% K: T" U" W6 z2 z
- * @brief whether use dma2d to transfer data to lcd framebuffer.
1 M* ~/ _6 S" J+ B3 R - */
7 g3 h/ U% m# z) c0 f3 P% l - #define USE_DMA2D_EN 1</font>
复制代码 接着对DMA2D传输操作进行封装,编写一个DMA2D传输函数:
. ]$ L7 k3 J6 G7 v1 x* \- <font face="微软雅黑" size="3">static void dma2d_transfer_data_r2m(uint32_t *addr, uint32_t xSize, uint32_t ySize, uint32_t offsetLine, uint16_t color)' S' b, T) c0 r! S' u1 F5 c) n! K
- {
; N, }2 G; I" n0 l - DMA2D->CR = DMA2D_R2M; // dma2d mode: register to memory., P" t0 m- k9 p! a9 H. B
- DMA2D->OPFCCR = DMA2D_OUTPUT_RGB565;
0 T5 ~& e( K; j, j6 a& ], Q
0 g m* ?; V$ C$ z/ L- DMA2D->OCOLR = color;. `4 P; d% w7 o2 N' I. f) r9 e
- DMA2D->OMAR = (uint32_t)addr;
' J) a: f$ b, X1 m6 w - DMA2D->OOR = offsetLine;
8 E; D8 }% I* f6 u - DMA2D->NLR = (uint32_t)(xSize << 16) | (uint16_t)ySize;
6 x! w) U- p2 M( _ -
4 k% M+ A6 J/ r. U5 e% t- M - DMA2D->CR |= DMA2D_CR_START;7 ]- P' u/ V8 n6 L% z
- while (DMA2D->CR & DMA2D_CR_START);
; f3 \6 Q3 N2 ^* r% b8 U - }</font>
复制代码 利用此DMA2D传输函数,重新添加清屏函数的实现:1 L: p' [7 N5 ^. v" I# _0 N
- <font face="微软雅黑" size="3">void lcd_clear(uint16_t color)
: q) ]! ?$ p' g P8 k - {
1 l% x( w, `9 G+ m9 k! Z, m* y - #if USE_DMA2D_EN$ I- q1 \; {5 `
- dma2d_transfer_data_r2m((uint32_t *)LCD_FRAME_BUFFER, LCD_WIDTH, LCD_HEIGHT, 0, color);5 w' j3 B* x) a' T+ [3 |
- #else Q! W5 V) s1 M3 i3 ]$ q
- uint16_t *ptr = (uint16_t*)LCD_FRAME_BUFFER;! R7 Z7 m- [( |1 ^1 a
- uint32_t i = 0;
) K7 `; n( S: O% @4 }6 W
8 u- s h, q; [: K" I2 }7 E. t- while (i++ < LCD_WIDTH*LCD_HEIGHT) {
* R4 _3 u) e$ Z5 E8 f! u! J - *(ptr+i) = color;
' V$ o" t9 `' h! k- S9 a( c - }3 q$ F" W- Y! A' A
- #endif /* USE_DMA2D_EN */
& j5 Y% V) b& ^3 R - }</font>
复制代码 上层测试代码不变,确保控制dma2d的宏使能:2 b+ a* G5 U' {( {
- <font face="微软雅黑" size="3">
5 U9 U" M( a7 ?( M - #define USE_DMA2D_EN 1</font>
复制代码 编译、下载,在串口助手中查看清屏一次所需时间:
' O) _" U, H( [; j9 v& C
! [' `+ _ g4 A! r, a7 c& D
可以看到,刷屏一次只需31ms即可,并且在使用dma2d传输数据的情况下,数据传输时间和编译优化等级无关。9 a2 W8 h) r" j! _
3 G9 q: Z; f, g& M2 [+ r G1 u5 M. `# z7 ^ U/ U
三、LCD基本功能实现
' H; e3 @8 L" u6 ~5 }LCD基本功能包括打点、读点、画线、绘图等函数。
0 C0 a/ R- m/ w: I3 s& E( O, b j+ V& v
4 b1 [% ^' ]9 w6 w
1. 打点函数
, C9 @% |9 k- D5 `) j9 Z打点函数的核心是计算当前用户给出的坐标位置在显存中的位置,两种实现如下:$ H) i" Y. Y: x b
! m. C6 \( H: h/ t
- <font face="微软雅黑" size="3">void lcd_draw_point(uint16_t x, uint16_t y, uint16_t color)
" @( h9 D2 b0 V# n5 E5 B6 m - {
, p0 ^) O. m0 B" [; M% v1 [6 S5 N - uint32_t pos;" f$ r* n$ b+ X
- uint16_t *ptr;8 ^3 ~4 d2 ?) y/ M* z% ?
# v* V3 r0 j3 J# R% q4 G; ^5 _- // check position.
9 x0 W o* }% L: Q9 n - if (x > LCD_WIDTH || y > LCD_HEIGHT) {, d+ x2 e0 p' ]6 H! {3 m, }8 ~* ?
- return;
& W) [8 Q: H+ M; E( b- ~: b - }5 V( s0 N! N' b- z+ q
1 T0 m' q# \- i! A- // calculate the position offset in framebuffer.
# O6 O: s/ q/ Z2 K7 a% ^+ W - pos = x + y*LCD_WIDTH;/ @& J$ w/ J$ d- s$ X4 v7 g
- ptr = (uint16_t*)LCD_FRAME_BUFFER;' s1 x/ S! o9 G+ `; a
- ) c2 Y( q: m7 u; Z" v
- // modify the framebuffer.
3 D3 q, m" U1 F. m4 B - #if USE_DMA2D_EN9 t1 W$ W6 b0 z$ E
- dma2d_transfer_data_r2m((uint32_t *)(ptr+pos), 1, 1, 0, color);
* ~ f. j# o( x, G/ Z" b - #else
6 i2 i+ ]; r1 ~! j5 {1 U( Z8 L - *(ptr+pos) = color;' i1 I2 x* v! T+ x; V
- #endif /* USE_DMA2D_EN */8 y9 g: y+ C+ o7 M! x& T
- }</font>
复制代码 2. 读点函数实现$ D7 L1 r5 F: O6 t8 Y+ y
读点函数实现的核心也是计算出用户给出的坐标位置在显存中的位置:7 y- |& L! s; ^/ n4 F
6 N2 R5 W! S* _/ C f4 {- <font face="微软雅黑" size="3">uint16_t lcd_read_point(uint16_t x, uint16_t y)
0 V8 g7 F3 h. L' f" W" Z - {5 s6 ?8 s9 s# q: W" i2 c+ V- u8 ^3 V
- uint32_t pos;2 |6 |& S: V5 b) [% k
- uint16_t *ptr, data;* [3 o5 Z1 A2 Q- c( F1 ]
- " ]/ v9 ?1 q }, \3 i2 j9 u2 m
- // check position.1 |" c5 _2 ~$ D e6 y! I3 U1 S
- if (x > LCD_WIDTH || y > LCD_HEIGHT) {1 U' b, B, N- q. r
- return 0;
* S7 M! a0 R. h - }
# M2 d# }# u: S. ?" \8 r: f( D
+ m- }9 y3 y; H, D% G- // calculate the position offset in framebuffer.7 ] {: x: U& A# E
- pos = x + y*LCD_WIDTH;
* @$ @' h. d& | - ptr = (uint16_t*)LCD_FRAME_BUFFER;
- G# e0 t9 J' B: n6 H z y1 v$ H - 9 W" ?9 ?% @% S! [0 l& n9 x1 m/ c
- // read the framebuffer.
7 e. k0 U/ a2 ~4 t3 T" |' \ - data = *(ptr+pos);
4 _, k7 Y0 N) v. _/ b, d) |' X
8 {0 t. E' v2 [ e9 f; W' K* i0 P- return data;4 S! R3 d0 u& c6 b! l
- }</font>
复制代码 3. 画线、画矩形、画圆
2 G) u, {1 F5 m& U这三个功能都是基于打点函数,使用 Bresenham 算法,代码篇幅过多,如有兴趣可直接查看本篇源码。
$ p( b9 k! ^! @1 O( C0 B$ m, q( @4 e& A; O9 y
) m& q5 L Z' K+ i, X/ Z* v+ @; Q
4. 测试
( }; y, t0 S+ D! r在main函数中添加测试代码:. V0 E8 a3 ~8 o1 M6 q3 V
8 a. q5 X9 \$ l8 y( c
- <font face="微软雅黑" size="3">lcd_draw_line(0, 0, 1024, 600, GREEN);& ?# W4 w( E+ n ^
- lcd_draw_line(0, 300, 1024, 300, RED);
9 `6 ^2 W. C, F - lcd_draw_line(512, 0, 512, 600, BLUE);
8 |* h. l2 z& l" T `9 I, c - lcd_draw_line(1024, 0, 0, 600, YELLOW);9 T( r0 d+ s$ n+ r( z
- 9 w! b. L+ p/ u0 I; Q' w
- lcd_draw_rect(256, 150, 1024-256, 600-150, PINK);</font>
复制代码 编译,下载,结果如下图:2 B, M* E- w- [9 c; h8 e2 `
8 v6 a( i! g0 _ B( k$ s
$ r U- D) r0 t, V$ c$ }: Z
" D) Q0 p( w7 v$ V6 S4 U
|