你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32CubeMX |42-使用DMA2D加速显存数据传输

[复制链接]
STMCU小助手 发布时间:2021-7-26 10:38
一、使用CPU搬运数据到显存
7 S1 ^0 R# {" k5 `2 X6 D# i5 c' M3 M* ~在上一篇文章中讲述了如何配置 LTDC 驱动 RGB 屏幕:
  @& w. ~9 V- s; U) m# L% B! a6 k0 \. `# ?  v6 P

% r' A: e6 z/ b( ]! f, qSTM32CubeMX | 41-使用LTDC驱动TFT-LCD屏幕(RGB屏)。/ {' T' o* M! P; K+ R

8 \3 t4 C* a8 d3 m, V

: S( s0 C8 W% Z+ `本节中我们接着上一节的实验,讲述如何使用 DMA2D 实现打点、画线、填充等函数,只需要单层全屏即可,修改LTDC层配置如下:5 j; S! v+ i& Q& F, F, _- B2 \6 k$ X
1.png
+ T  `* s. P, D) F+ a: ]2 Q; }. U1 W* h% g6 X5 z- y9 g

* n$ d! _# e/ G; `4 I1. 编写lcd驱动头文件7 T& _& i1 u- U; I) @1 r
创建lcd_rgb_ltdc_drv.h文件,存放关于操作LCD屏幕的一些宏定义配置和函数定义:
; H! F; ]% R' O) T. u4 I) ]
  1. <font face="微软雅黑" size="3">#ifndef _LCD_RGB_LTDC_DRV_H_
    & s8 O. Q, j; j, c- \
  2. #define _LCD_RGB_LTDC_DRV_H_
    1 |7 Q4 s  V9 p  n% x
  3. 4 l2 F% F9 W& i8 m% B8 f+ {. ?( J- }' x
  4. #include "ltdc.h"
    ' y% M7 B# d# v' ^6 H7 b, |

  5. 7 _) L" e* a9 a9 T( Q( m
  6. /**3 \; e# x9 g# s/ r4 J% x
  7. * @brief   Windows size on lcd.4 t  C# ?* R9 `; E
  8. */  Y3 `# W' i9 B& o/ z8 z
  9. #define LCD_WIDTH       1024
    . r( J* n/ b4 U/ F: W3 v/ ?/ C1 }
  10. #define LCD_HEIGHT      600
    3 Q% R4 h+ w' w0 x) d

  11. $ {+ T/ u$ x) ^( ~6 S/ ~, {
  12. /**" V) R. Q" z! @0 G
  13. * @brief   Backlight control pin of lcd.. k; N7 ~" X2 A/ f- }8 ?! x: j0 N% k
  14. */
    ) `2 ?" k& X; W5 l, s1 o6 {2 g
  15. #define LCD_BL_GPIO_PORT    GPIOB
    - S; Z& ^/ N4 f: g6 r" d1 M3 |2 S. e
  16. #define LCD_BL_GPIO_PIN     GPIO_PIN_5
    9 p+ n( y3 W7 o) e7 j/ l

  17. 0 J: p; a- L/ `$ e! ~$ l1 L
  18. /**
    1 v' Z) k! ~) a& S
  19. * @brief   start address of lcd framebuffer.
    " ?/ q8 ~' B# c& P& n3 {
  20. */7 q% N1 i+ D4 H5 ?
  21. #define LCD_FRAME_BUFFER    0xc0000000% B) ^$ E9 w2 N5 ~6 K

  22. ; B# `* Q" \2 H. a! R
  23. /**
    / C# L: u/ [$ e7 E/ d3 O
  24. * @brief   color3 v* Q, _1 X% u! S' l
  25. * @note    rgb565   , ~7 y0 C0 F" }6 D
  26. */2 ~7 ]: y: T2 p4 [* s) N7 F
  27. #define BLACK   0x0000
    4 b0 i6 n* G8 C
  28. #define BLUE    0x001F4 k- D* L% e) t' r/ Z+ g0 h5 \  J2 r
  29. #define GREEN   0x07E0  U* j0 ~6 [  t2 K: h6 I4 Y
  30. #define GBLUE   0X07FF
    3 y! Q* y) `0 B& Q- `, o
  31. #define GRAY    0X8430
    ) y" P! G7 u' O0 y1 k# G
  32. #define BROWN   0XBC40
    1 S3 D) `+ g7 w6 L* P
  33. #define RED     0xF800
    ( d6 |8 o) B; h
  34. #define PINK    0XF81F; E+ I8 K: e/ Q
  35. #define BRRED   0XFC07
    4 m' F$ h2 j: V  j9 L3 `5 y
  36. #define YELLOW  0xFFE0
    % F  [' \1 \4 G. G* ]* D7 ]( E
  37. #define WHITE   0xFFFF
    0 {- w! T) n+ }: W% p

  38. - q/ J* f, b8 y) s8 Z9 I. w" n3 D) s
  39. /**) ?) a4 J5 J. U  P. y$ V( {
  40. * @brief       Control the lcd backlight.) T# h7 o: \+ k# O; h
  41. * @param[in]   brightness  the value of lcd backlight.
    6 _+ e* b. ]. V
  42. * @return      None0 n! I& `' g7 C. N1 i& G
  43. */
    5 g, k9 D5 N9 q( ]9 @$ m3 c1 q
  44. void lcd_backlight_control(uint8_t bightness);
    2 G6 P  Y* i  J

  45. ( H4 m$ _6 Z" g5 Y& k
  46. /**0 i) X* p! J7 @
  47. * @brief       LCD initialization.
    ; ^- y- |) H* J
  48. * @param       None6 a- O4 h: g8 J/ ^
  49. * @return      None& p+ P5 G5 X# b) F9 K7 F: ^) g: W
  50. */. M! d3 Y3 r. c/ B( l
  51. void lcd_init(void);
    7 C" o! q9 N4 Q3 _

  52. 2 o  ~  _: P% Z: e# D' _/ P5 }0 z
  53. /**) z6 R9 f# m* }4 X% N
  54. * @brief       Clear lcd.
    ( A/ t- p! {: p) r$ z5 O3 g
  55. * @param[in]   color   rgb565.
    ( a, M  ?, N& R8 j& j
  56. * @return      None% H8 v% M! x; n0 ], y
  57. */
    2 @* X9 W. G. X0 w% V
  58. void lcd_clear(uint16_t color);, k  G& J9 G- r$ o6 g( t
  59. / o$ v) Y. o, {- s1 _* Q5 C- @6 E
  60. #endif /* _LCD_RGB_LTDC_DRV_H_ */</font>
复制代码
! F# S; L- O9 }7 R" _! Q, \

0 ^$ f3 y. m7 M6 X2. lcd驱动实现: H9 T3 m( L' p+ ^
创建lcd_rgb_ltdc_drv.c文件,存放关于操作LCD屏幕的函数实现。
* ~8 v3 A; K& i; k) Y% g) r7 r, R3 \& t% E% ~# b
8 l0 P+ }0 l" [& k" N, g- P
首先是背光控制实现,应该使用pwm实现背光调节,本文中为了方便直接使用GPIO控制:# v' D0 i+ Z2 d( Y
  1. <font face="微软雅黑" size="3">void lcd_backlight_control(uint8_t bightness)" |0 r/ y5 W% J& P& a; o8 b# ?& n
  2. {' ]* R( }0 T- u
  3.     // todo: use pwm to control backlight' [1 W1 O& S2 ]/ o
  4. 4 z; v. H7 u& o8 L4 t: q; ]
  5.     if (bightness) {   
    7 l' s6 a5 f& ^' u  ]( \
  6.         // turn on the backlight
    , p( R+ m3 J" i. F: ^( ~
  7.         HAL_GPIO_WritePin(LCD_BL_GPIO_PORT, LCD_BL_GPIO_PIN, GPIO_PIN_SET);
    ! ^+ Y% t7 H1 K  q
  8.     } else {            6 w9 y9 O# ^" B; P2 H" t7 M6 t
  9.         // turn off the backlight) M9 y3 S- c: V1 r, t$ Y
  10.         HAL_GPIO_WritePin(LCD_BL_GPIO_PORT, LCD_BL_GPIO_PIN, GPIO_PIN_RESET);* l8 E# G& R! i& V5 N* Y* G! x
  11.     }7 k$ a2 s' @$ [/ s! p  ]2 m/ m
  12. }</font>
复制代码
接着实现lcd清屏函数,使用CPU(循环)搬运数据到显存中:( F0 t' |* f* j" J& b: N
  1. <font face="微软雅黑" size="3">void lcd_clear(uint16_t color)
    4 S, Q6 Z: N7 }  |0 Q$ |, ~
  2. {
    + R: I$ }* U5 k& \% u
  3.     uint16_t *ptr = (uint16_t*)LCD_FRAME_BUFFER;
    7 u0 p% o% {8 [5 {
  4.     uint32_t i = 0;/ b) @1 C% a; Q6 |& E4 T
  5. % Z% D7 _3 R. f' ~2 T
  6.     while (i++ < LCD_WIDTH*LCD_HEIGHT) {
    $ \5 R) u" {% v! m& r3 T3 s
  7.         *(ptr+i) = color;- O! K7 C* B/ g6 \5 W  j' ]* H
  8.     }
    , h2 |8 j# {0 E- ~' k+ _( r
  9. }</font>
复制代码
最后实现lcd初始化函数,先刷显存,然后开背光,防止设备上电时屏幕闪烁:
  Z* U3 s7 F+ R! |2 y: f. \
  1. <font face="微软雅黑" size="3">void lcd_init()- d( u* d0 q5 o5 |4 h, I' Z
  2. {
    0 {) N+ |( ~6 k3 s2 |  a8 {
  3.     lcd_clear(BLACK);* k: a3 p( z( m% ]
  4.     lcd_backlight_control(255);
    ) y7 N0 w( b2 N- L" Q: Q2 U$ y
  5. }</font>
复制代码
! ?' }: G& V8 b$ N0 j! F/ d4 e
1 g, S4 W) A4 v" X: Q0 f$ P9 E
3. lcd单次清屏时间测试* A; y" O: i4 {
在main.c中包含驱动头文件:" I( r2 v0 `1 w3 Q5 A, O
  1. <font face="微软雅黑" size="3">#include "lcd_rgb_ltdc_drv.h"</font>
复制代码
在main函数中的sdram初始化函数之后,添加lcd初始化函数,并使用HAL库自带的systick时间戳测量一次清屏的时间:# d0 d" n3 d6 a8 N+ Y+ {
  1. <font face="微软雅黑" size="3">/* USER CODE BEGIN 2 */
    5 g% D3 L0 L1 K' F, ?
  2. printf("sdram test by mculover666\r\n");
    & M5 C- h! `* A6 z
  3. SDRAM_Init();
    7 }( R: e  T$ b1 I5 F" G2 r: n
  4. printf("sdram init success\r\n");
    ) \) h- l' J! q+ _% K

  5. 0 u* b9 ]! Q) D3 ]
  6. lcd_init();
    9 k' F$ Q5 I6 N6 `
  7. " ]% y; K: m" T3 ^& k) Q' E3 z
  8. start_time = HAL_GetTick();
      n" M" O8 ?: M
  9. lcd_clear(PINK);4 I# ~$ S( m& _5 k
  10. end_time = HAL_GetTick();1 X8 n6 T% U3 L, S1 ~

  11.   e* }/ |; a  M' p
  12. printf("lcd clear spend time:%ld ms\r\n", end_time - start_time);8 p! U" S$ p. u% P9 V
  13. /* USER CODE END 2 */</font>
复制代码
编译、运行,在串口助手可以看到使用CPU搬运数据到显存中,在-Og优化等级下单次清屏需要 155 ms左右,在-O0优化等级下单次清屏需要321ms左右:, M% I% l4 m0 p0 |
2.png + t3 _( V5 V+ i. S, Q" ^7 h

) p  ~/ `  U) i

; O+ K9 R5 W  v; ^二、使用DMA2D加速显存数据搬运
% C) l. G+ B, t5 i1. DMA2D) d& Q" H7 V& R6 A5 |8 s
在STM32中,DMA2D外设专门用来给LCD显示加速,有LTDC外设的型号中,通常也会配套有DMA2D。
# q0 k4 T/ J2 i2 G4 T. S! h' j
4 c! Z' G$ _1 M1 c3 n% o! v

$ s. N1 y3 @' H# cDMA2D外设主要提供了两个功能:
! F6 X! z1 G% u2 }
# b) c) d( g5 ]: O

- x' B+ O% W8 Q: C, oDMA数据搬运:常用从寄存器到存储器、存储器到存储器两种模式,快速高效,并且不占用cpu资源;
2 i8 |* \% ^9 V8 f2D图形加速:支持快速格式转换和混合;2 T1 a: S3 i) w+ R4 B6 c
本文中主要使用到DMA2D外设的数据搬运功能,使用起来也是比较简单。- u6 I$ _  m4 Z1 V0 c
0 n0 I1 U4 }* C  t1 ?# ]1 E
) P, a: ?! L; e+ w% R. O
2. 开启DMA2D( ]+ F+ H; ^  b' g
3.png
* T+ Y1 y# O9 s1 w重新生成工程,cubemx会自动生成并调用dma2d初始化函数,完成dma2d外设时钟使能以及dma2d传输模式配置。
  n+ y/ Z8 N9 [5 @* }4 @
# U  Z; F9 h* a9 P$ x5 v
) y+ X$ X  a- S
3. 使用DMA2D实现lcd清屏函数, D9 c0 n! V- Z
在lcd驱动头文件中添加一个宏定义,用于控制是否使能DMA2D:+ x3 R( v9 u9 W, B
  1. <font face="微软雅黑" size="3">/**
    " F3 E3 C2 V# P
  2. * @brief   whether use dma2d to transfer data to lcd framebuffer.. Z  Y7 P% z# y6 b! p' J1 ~9 l
  3. */$ c1 n8 K% m) `) q2 F
  4. #define USE_DMA2D_EN        1</font>
复制代码
接着对DMA2D传输操作进行封装,编写一个DMA2D传输函数:
  L% i1 h: y) f3 b. H  i
  1. <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)
    ( k1 I! B! f8 M6 b1 E
  2. {
    4 ~2 r8 x9 P7 o  h, E1 M
  3.     DMA2D->CR = DMA2D_R2M;   // dma2d mode: register to memory.
    ' P, O4 ?8 M! H8 J
  4.     DMA2D->OPFCCR = DMA2D_OUTPUT_RGB565;
    . ]2 B  U. ~) o

  5. & z3 ]& K1 l* {7 w' B- I
  6.     DMA2D->OCOLR = color;8 j) H, u( K& w; r! _/ H4 x
  7.     DMA2D->OMAR = (uint32_t)addr;* Z3 F( ?* W& n1 i# r
  8.     DMA2D->OOR = offsetLine;) ~2 n" ?  S! o6 n2 T! i
  9.     DMA2D->NLR = (uint32_t)(xSize << 16) | (uint16_t)ySize;/ ]# n& \- C5 W5 `$ L) l
  10.     5 M/ }" F2 F1 C* @6 X" `
  11.     DMA2D->CR |= DMA2D_CR_START;- J8 D5 C3 M5 ]6 {$ ?" ?- d) V
  12.     while (DMA2D->CR & DMA2D_CR_START);/ r6 Y* J, D. d8 k3 [$ y: u
  13. }</font>
复制代码
利用此DMA2D传输函数,重新添加清屏函数的实现:2 x$ v) m6 t9 x5 Z" U
  1. <font face="微软雅黑" size="3">void lcd_clear(uint16_t color)' J0 `; Y+ j. ?6 }
  2. {- [& r4 P: F, A% y
  3. #if USE_DMA2D_EN
    ! ?5 T+ N9 q1 _3 F& u
  4.     dma2d_transfer_data_r2m((uint32_t *)LCD_FRAME_BUFFER, LCD_WIDTH, LCD_HEIGHT, 0, color);7 P2 q7 y( d+ h- c, W  c' m% d
  5. #else2 ?! b& \+ k' o8 n( o2 z$ L
  6.     uint16_t *ptr = (uint16_t*)LCD_FRAME_BUFFER;
    * k$ w1 A; b8 ]& u! }
  7.     uint32_t i = 0;1 [4 F9 m6 |  m' `  D5 p
  8. ) a8 j% z* O  y# B4 u
  9.     while (i++ < LCD_WIDTH*LCD_HEIGHT) {
    2 ]9 J/ g  m$ D" {
  10.         *(ptr+i) = color;* C: U! k! o: A: g) q3 W
  11.     }
    : d+ e: D: P# }3 h5 f
  12. #endif /* USE_DMA2D_EN */
    " m" p! ~) i' u9 Y3 J6 ]
  13. }</font>
复制代码
上层测试代码不变,确保控制dma2d的宏使能:( A. M) w; ], q) Y
  1. <font face="微软雅黑" size="3">
    9 o3 z0 b/ e% ]7 V+ k
  2. #define USE_DMA2D_EN        1</font>
复制代码
编译、下载,在串口助手中查看清屏一次所需时间:: o, X% W/ e2 l8 \
4.png ; |) q$ n' H9 ^$ A
可以看到,刷屏一次只需31ms即可,并且在使用dma2d传输数据的情况下,数据传输时间和编译优化等级无关。
, O/ z" S+ Z( M2 c- a
/ f7 @! g# D$ T. Y" x' P+ `; ^) f2 _

& f& _! E: q% m三、LCD基本功能实现
# v/ p+ Q" X8 n. VLCD基本功能包括打点、读点、画线、绘图等函数。* ]; c* ~7 i3 U' {. v

6 r( J0 ]4 ?7 a
: z% n4 u$ c$ R# v6 c! A$ P. R
1. 打点函数
  ?- H% t  G! M; |  J$ y+ C打点函数的核心是计算当前用户给出的坐标位置在显存中的位置,两种实现如下:
; u+ x$ g, X3 g7 |) n( Q) V" l7 m4 H  g# n5 A0 Q0 V. H
  1. <font face="微软雅黑" size="3">void lcd_draw_point(uint16_t x, uint16_t y, uint16_t color)
    ; e& n# `+ _  x$ e$ X4 f5 ]& x- h' Y
  2. {
    0 V4 Y1 a) ^  D. E: F& |6 O  Y
  3.     uint32_t pos;
    . p  E' J0 h. _1 _
  4.     uint16_t *ptr;# I2 X7 S8 Q5 s/ P& }

  5. 9 [5 O: u$ [$ n7 y6 P
  6.     // check position.
    1 [1 R0 l; m* Z
  7.     if (x > LCD_WIDTH || y > LCD_HEIGHT) {
    ( I8 o4 G' r1 f2 ~/ ]4 T, N9 d
  8.         return;4 C0 R/ L! l2 t! W1 f) n1 n; D
  9.     }
    # u$ T! C' w/ k' R
  10. / `5 v9 f" s3 I
  11.     // calculate the position offset in framebuffer.
    ' z7 a6 @$ ~/ \; `$ D7 H& t
  12.     pos = x + y*LCD_WIDTH;
    0 m9 L- t% }1 e0 k- \
  13.     ptr = (uint16_t*)LCD_FRAME_BUFFER;' M9 p3 e* Y& ^% L$ z6 R
  14. 6 r1 C# ^4 |4 c' q- z# x, O' J
  15.     // modify the framebuffer.0 m9 {3 A& B/ m
  16. #if USE_DMA2D_EN
    6 i* F3 \# P' m( c
  17.     dma2d_transfer_data_r2m((uint32_t *)(ptr+pos), 1, 1, 0, color);) v2 O, ?; f5 z* A( |% k2 r) T
  18. #else
    8 `" w. x! y) a7 y7 V
  19.     *(ptr+pos) = color;
    , F# O5 W$ C- Y
  20. #endif /* USE_DMA2D_EN */
    ' ^" o$ c7 B: I. I
  21. }</font>
复制代码
2. 读点函数实现
% y8 Q* F- ?4 A! U0 q读点函数实现的核心也是计算出用户给出的坐标位置在显存中的位置:
+ m: j9 ~8 q: y% n) g
$ ]% \! @* I7 U  m3 y" H
  1. <font face="微软雅黑" size="3">uint16_t lcd_read_point(uint16_t x, uint16_t y)9 ]1 A( y0 _/ }4 E9 f. f6 B
  2. {: r9 W) }% \8 D% ~; ?  T0 D
  3.     uint32_t pos;2 V  y+ R5 T2 F- S* ]
  4.     uint16_t *ptr, data;
    * L6 |8 i+ c7 u& z
  5. & l9 R8 h- a2 r$ M
  6.     // check position.
    6 b. @/ [( ^  a$ g" c# Z! l5 V
  7.     if (x > LCD_WIDTH || y > LCD_HEIGHT) {
    # a9 R) i/ L7 V0 q6 E
  8.         return 0;
    0 T. Q  E" @2 `0 D
  9.     }
    ( m5 Z* R- |( e* o4 t

  10. & ]4 X0 ^6 h( c4 u# ^1 d& K
  11.     // calculate the position offset in framebuffer./ t% L3 }/ ^) u5 g7 e; H
  12.     pos = x + y*LCD_WIDTH;0 V, }3 k3 Z) Z, G
  13.     ptr = (uint16_t*)LCD_FRAME_BUFFER;' w) Y- h1 X$ w' \, _/ [; I
  14. , h/ G; P# }# u* w- v
  15.     // read the framebuffer.0 M" Q4 J* m$ o" D, s) G
  16.     data = *(ptr+pos);
    9 }3 k& F2 D; A4 J2 N: L0 R, Y

  17. ) I/ c+ T/ ?% W6 n
  18.     return data;
    : k- U5 f* f. n8 V. G- \& A
  19. }</font>
复制代码
3. 画线、画矩形、画圆' }" A- P" b4 s* `) E
这三个功能都是基于打点函数,使用 Bresenham 算法,代码篇幅过多,如有兴趣可直接查看本篇源码。. K  i5 ?+ @, n0 b+ ?
- l) }9 r- |) z) W. h/ ~
  O& a/ @, v. U1 h
4. 测试
6 L* L; J3 n- {  N9 \在main函数中添加测试代码:
- Y+ o/ @8 w! D3 N8 G: `( i
. T% P# x4 ]9 {! l' W2 P
  1. <font face="微软雅黑" size="3">lcd_draw_line(0, 0, 1024, 600, GREEN);
    % H( O; q: y! a/ A
  2. lcd_draw_line(0, 300, 1024, 300, RED);: C2 c. E5 O* M& M- b" R
  3. lcd_draw_line(512, 0, 512, 600, BLUE);
    7 U# x. u0 U9 s
  4. lcd_draw_line(1024, 0, 0, 600, YELLOW);
    ; b4 V5 \8 j( j, a! x
  5. 3 ~' H! ~5 b* O( w" [
  6. lcd_draw_rect(256, 150, 1024-256, 600-150, PINK);</font>
复制代码
编译,下载,结果如下图:
# f6 j( F7 m' O! z4 e, j 5.png $ C/ {$ ]0 ~4 x. U1 L: t
1 R7 @2 O" b5 p
( j" |- i+ G5 H4 K3 I
3.png
2.png
1.png
4.png
5.png
收藏 评论0 发布时间:2021-7-26 10:38

举报

0个回答
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版