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

【经验分享】STM32之DMA

[复制链接]
STMCU小助手 发布时间:2022-1-12 20:14
一、DMA简介
1、DMA简介
  DMA(Direct Memory Access:直接内存存取)是一种可以大大减轻CPU工作量的数据转移方式。
  CPU有转移数据、计算、控制程序转移等很多功能,但其实转移数据(尤其是转移大量数据)是可以不需要CPU参与。比如希望外设A的数据拷贝到外设B,只要给两种外设提供一条数据通路,再加上一些控制转移的部件就可以完成数据的拷贝。
  DMA就是基于以上设想设计的,它的作用就是解决大量数据转移过度消耗CPU资源的问题。有了DMA使CPU更专注于更加实用的操作--计算、控制等。
2、DMA的工作原理
  DMA的作用就是实现数据的直接传输,而去掉了传统数据传输需要CPU寄存器参与的环节,主要涉及四种情况的数据传输,但本质上是一样的,都是从内存的某一区域传输到内存的另一区域(外设的数据寄存器本质上就是内存的一个存储单元)。四种情况的数据传输如下:
  • 外设到内存
  • 内存到外设
  • 内存到内存
  • 外设到外设
    5 F. J* p3 z$ V: n4 ]) k$ W

    0 m6 V' e0 ~9 ]: y: @
  当用户将参数设置好,主要涉及源地址、目标地址、传输数据量这三个,DMA控制器就会启动数据传输,传输的终点就是剩余传输数据量为0(循环传输不是这样的)。换句话说只要剩余传输数据量不是0,而且DMA是启动状态,那么就会发生数据传输。  
3、DMA是否影响CPU的运行
  在X86架构系统中,当DMA运作时(假设我们从磁盘拷贝一个文件到U盘),DMA实际上会占用系统总线周期中的一部分时间。也就是说,在DMA未开启前,系统总线可能完全被CPU使用;当DMA开启后,系统总线要为DMA分配一定的时间,以保证DMA和CPU同时运作。那么显然,DMA会降低CPU的运行速度。
  在STM32控制器中,芯片采用Cortex-M3架构,总线结构有了很大的优化,DMA占用另外的总线,并不会与CPU的系统总线发生冲突。也就是说,DMA的使用不会影响CPU的运行速度。
二、STM32的DMA结构
1、DMA的主要特性
● 12个 独立的可配置的通道(请求)DMA1有7个通道,DMA2 有5个通道
" |% x) i$ t3 j5 U' j# }● 每个通道都直接连接专用的硬件DMA请求,每个通道都同样支持软件触发。这些功能通过, n2 A+ b# e; E7 ]
软件来配置。
& o+ r& \7 v. P1 M: p0 k* _! c● 在七个请求间的优先权可以通过软件编程设置(共有四级:很高、高、中等和低),假如在相
2 u+ I' y1 O0 l# X2 B. K; J, H等优先权时由硬件决定(请求0优先于请求1,依此类推) 。
7 e6 u7 ]: S: C& D; c- b3 p● 独立的源和目标数据区的传输宽度(字节、半字、全字),模拟打包和拆包的过程。源和目标( k4 A- ^5 [( V7 b5 }; D% K2 k
地址必须按数据传输宽度对齐。2 Q' ~1 Q) `) i$ i& k% d
● 支持循环的缓冲器管理3 ?2 G$ Q, T; \0 b; p' m
● 每个通道都有3个事件标志(DMA 半传输,DMA传输完成和DMA传输出错),这3个事件标志
8 w9 N( ?1 B! `+ @7 H* b* a& t逻辑或成为一个单独的中断请求。% h# p/ \$ n/ c- m+ i% B
● 存储器和存储器间的传输' ?2 S* d5 h3 T3 {0 ~
● 外设和存储器,存储器和外设的传输
2 Q6 W! A( C8 ]- o" z2 s9 |● 闪存、SRAM 、外设的SRAM 、APB1 APB2和AHB外设均可作为访问的源和目标。
$ X# t3 l0 Q, ~& Q9 h0 U4 F6 I& \● 可编程的数据传输数目:最大为65536
下面为功能框图:
/ ^" p5 m. Z6 @  j+ w& r) f
211431529658805.png

* ?2 V5 ]/ w, G  c
2、两个DMA控制器结构
① DMA1 controller
) X" Q/ o! |$ m- o  n. [
211458074189733.png

0 G2 \4 G0 l* E0 |
② DMA2 controller
, q' ?: s, y, D
211458540128699.png

8 F" e9 d% S' c6 ^- J3 V
3、DMA寄存器列表
211442253242949.png
① 中断类
DMA_ISR:   DMA中断状态寄存器
DMA_IFCR:  DMA中断标志位清除寄存器
说明:  DMA1、DMA2分别有一组寄存器。
② 控制传输类
DMA_CCRx:   DMA通道x配置寄存器 
DMA_CNDTRx:  DMA通道x数据数量寄存器
DMA_CPARx:   DMA通道x外设地址寄存器
DMA_CMARx:  DMA通道x内存地址寄存器
说明:   
1> 每一个通道都有一组寄存器。
2> DMA_CPARx、DMA_CMARx是没有差别的,它们都可以存放外设的地址、内存的地址。DMA_CPARx、DMA_CMARx只不过起得名字有差别而已。
4、STM32的DMA工作特点
① DMA进行数据传输的必要条件
  • 剩余传输数据量大于0
  • DMA通道传输使能
  • 通道上DMA数据传输有事件请求
    ) Z% u1 a. U( |, L3 G4 X+ I: L

    6 p7 _; S  W8 r, b7 ^: A' K
前两者都好理解,对于第三点确实需要详细的解释,请看下边的三条。
② 外设到XX方向的传输
  假设是ADC到存储器的数据传输,显然ADC的DMA传输的源地址是ADC的数据寄存器。并不是说只要DMA通道传输使能后,就立即进行数据传输。只有当一次ADC转化完成,ADC的DMA通道的传输事件有效,DMA才会从ADC的数据寄存器读出数据,写入目的地址。当DMA在读取ADC的数据寄存器时,同时使ADC的DMA通道传输事件无效。显然,要等到下一次ADC转换完成后,才能启动再一次的数据传输。
③存储器对XX的DMA传输
  因为数据是准备好的,不像ADC还需要等待数据到位。所以,不需要对应通道的事件。只要使能DMA数据传输就一直传输,直到达到设定的传输量。
example:
1.内存到内存
  DMA传输请求一直有效
2.内存到串口
  DMA传输请求一直有效
一种解释:
  存储器对存储器的置位,就相当于相应通道的事件有效。 对应通道的事件有效和存储器对存储器的置位,就是传输的触发位。每次传输的事件置位一次,完成一次传输。如果是由外设引发的DMA传输,则传输完成后,相应传输事件会置为无效,而存储器对存储器的传输,则一次传输完成后,相应事件一直有效,直至完成设定的传输量。               
④外设以DMA方式工作时,能否再以软件方式进行操作?
  有一点是肯定的,当外设以DMA方式正在数据传输时,不可能再相应CPU的软件控制命令,否则这不符合逻辑。
  但是,倘若外设仅仅配置成DMA工作方式,但是DMA请求并未产生,数据传输并没有进行。此时,软件控制命令仍然能够对外设进行控制。这是笔者在串口以DMA方式发送数据情形下,所得到的测试结论。
三、STM32的DMA软件编程
1、“内存到内存”模式传输
初始化配置
  1. uint8_t SendBuff[SENDBUFF_SIZE];5 C* U7 q  A( O. l+ ?) m
  2. uint8_t ReceiveBuff[RXBUFF_SIZE];- o# a- o, p/ q- R2 s& t
  3. /**' }3 e1 s( G/ @" I5 W
  4.   * @brief  USART1 TX DMA 配置,内存到内存% x. I5 @/ y0 C9 x& d( _, s' B
  5.   * @param  无: H% w' I* V( p% W/ v
  6.   * @retval 无/ X  G# ?& G; v/ }% F; X
  7.   */
    5 i  l3 N. e" i3 |/ A
  8. void DMA_Mem2Mem_Config(void): G4 B. @; T2 m2 ~6 h1 ~, B1 {
  9. {- l# [4 H7 k4 M+ `% {( m. F! w
  10.         DMA_InitTypeDef DMA_InitStructure;* r' c. |! K5 J) X; [
  11.     & M/ E. j8 Y9 F6 _- h  c
  12.         /*开启DMA时钟*/- J0 G: t, b+ G6 ]3 L
  13.         RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);   
    8 Q+ _5 \; n; O8 R( B/ o3 l' P7 I
  14. : D" L, i! p# ~( ~) X2 J  O
  15.         /*设置DMA源地址*/4 c2 s8 u4 I7 o! T' C  T4 d
  16.         DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)SendBuff;4 I* \0 a& v4 A! a6 D
  17. ' ]: A1 j3 t+ |  ]
  18.         /*设置DMA目的地址*/
    ( n& v+ w% L. K: C% Y% G
  19.         DMA_InitStructure.DMA_PeripheralBaseAddr =  (uint32_t)ReceiveBuff;       ) L, I/ D) l! W" e5 {& j/ Z0 W% M, ]
  20. & G1 I( U4 Z/ D' b
  21.         /*方向:从内存SendBuff到内存ReceiveBuff*/        
      D8 `/ a5 o; O1 \% `9 O
  22.         DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;   
    : B. h4 |5 `1 E: @+ ]  e8 s$ t- g: [

  23. % R" R' W) d1 o* I; r! Y" Z) U
  24.         /*传输大小DMA_BufferSize=SENDBUFF_SIZE*/   
    . G. V8 b3 c: R! k) v' X6 m
  25.         DMA_InitStructure.DMA_BufferSize = SENDBUFF_SIZE;$ ~. b0 M: ]* p+ o4 D& i

  26. 4 ~& `; `0 A% F
  27.         /*ReceiveBuff地址自增*/      
    ( {: C7 I+ _+ l* ?4 F2 Y
  28.         DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;
    3 ]: a. @8 W4 G
  29. . {* T# N! ]( n- k: V
  30.         /*SENDBUFF_SIZE地址自增*/: G) a/ `7 Q' `1 x
  31.         DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;    8 I5 K- [5 s  H  G9 }7 r

  32. ) ?3 s5 t  I% P2 x0 O
  33.         /*ReceiveBuff数据单位*/    9 a5 g* R5 x6 w" z7 T! c
  34.         DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
    ' O% I! g9 O" }9 \+ N  H0 t

  35. 1 o& I8 R8 {/ i
  36.         /*SENDBUFF_SIZE数据单位*/
    : w, a" _/ f$ S. }$ b
  37.         DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;     ( s8 w2 u: e, U' Z9 U

  38. " v2 t4 }! a  p0 g* i
  39.         /*DMA模式:正常模式*/
    * F* C3 k* _$ ]4 L$ ~
  40.         DMA_InitStructure.DMA_Mode = DMA_Mode_Normal ;
    % P# V$ j6 \+ h2 z& K) ]9 D3 T+ i
  41. . s" i" O/ N, ^3 A% |# E
  42.         /*优先级:中*/    $ V5 B- Y3 j0 |7 ~; }- X7 T
  43.         DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;  
    . `' g  e" u2 P$ Y

  44. & Q, K8 t# P; l# S8 _' c
  45.         /*使能内存到内存的传输    */
    0 C3 A) x: i0 p, R
  46.         DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;; t: Z6 X  ^3 s; }+ f7 r' x$ s
  47. 9 J3 k% K/ T& W1 C- f' F
  48.         /*配置DMA1的4通道*/           8 m3 K4 h7 p! q. X
  49.         DMA_Init(DMA1_Channel4, &DMA_InitStructure);        
    ( O/ ^7 r4 k* c* U6 y4 ~
  50.         3 n: c; |% K: ~% x% p
  51.         /*失能DMA1的4通道,一旦使能就开始传输*/9 Q9 ]0 F4 q9 N2 m7 C0 Q+ w
  52.         DMA_Cmd (DMA1_Channel4,DISABLE);
    7 J4 b7 z( [/ w& g. O, h
  53. }
复制代码

0 N* N% y! N6 w4 [' y) E  W
DMA中断配置
  1. /**
    2 z: m! y; I* t* c" Z
  2.   * @brief  DMA 中断配置; C0 I. O7 R6 E5 a& e& ?
  3.   * @param  无# @1 D. g2 Z9 j' i6 q  T) d  {
  4.   * @retval 无9 |& M6 L' W: {$ t7 w9 Z
  5.   */
    . {0 J+ M! l) G8 q+ I
  6. void DMA_NVIC_Configuration(void)/ z: Q. e* F6 `8 w
  7. {                                                                      # `+ R$ n) F& F6 J- T" z
  8.     NVIC_InitTypeDef NVIC_InitStructure;            ! G4 b" N) T% H* S
  9.     $ Z2 I9 L/ B+ X" O# C
  10.     /* 配置中断源 */
    / l; S% g8 L0 ?5 l
  11.     NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_IRQn;
    , {" v6 ?8 T3 E
  12.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
    # S: z* Y6 w+ x, o  Z
  13.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;9 Q6 j. y: a; z( b: ^! K
  14.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    / a7 ?/ P# q+ T; \6 g4 |
  15.     NVIC_Init(&NVIC_InitStructure);   
    . T! a) R$ j* j, C

  16. 1 U5 ^5 G# C9 p! d
  17.     /* 配置DMA发送完成后产生中断 */            
    & J7 W: {6 d" V7 Z( M+ \/ ]+ h6 z
  18.     DMA_ITConfig(DMA1_Channel4,DMA_IT_TC,ENABLE);
    + b- A# q- J! D' ?; T( b' {
  19. }
复制代码

' u/ x# M6 C$ t* g
③启动传输
  1. DMA_Cmd (DMA1_Channel4,ENABLE);
复制代码

, i/ I; [' |: x/ Y4 p. i
2、利用DMA实现循环传输
方法1:单次传输模式
  当传输结束时,触发DMA中断,在中断程序中首先失能DMA通道,然后修改该通道的传输数据量。最后重新使能DMA通道,注意只有失能的DMA通道才能成功修改传输数据量。
方法2:循环传输模式
  当传输结束时,硬件自动会将传输数据量寄存器进行重装,进行下一轮的数据传输。
四、再谈STM32的DMA传输是否影响CPU的运行速度
  声明:经过笔者测试,当DMA工作在内存到外设的传输和内存到内存的传输时,都不会影响CPU的运行速度。为了给这种现象一个合理的解释,笔者做以下猜测:
1、S3C2440的DMA传输
  S3C2440的SDRAM是外置的,并且SDRAM的数据线、地址线、控制线总共只有一组。假设DMA传输的方向是内存到外设,当DMA运作时,需要占用SDRAM的三类线才才能实现传输;而与此同时CPU也需要通过这三类线来访问SDRAM来读取程序、读写数据。
  显然,DMA的运行与CPU的运行有交叉点,DMA就会影响到CPU的运行。
2、STM32的DMA传输
  STM32与S3C2440的区别是很大的,S3C2440是微处理器,RAM外置且空间很大;STM32是微控制器,RAM片内集成且空间较小。此时,ST公司就有可能提升DMA的运作效率,使DMA的工作不影响到CPU的运行。
  外设与外设之间的DMA传输,因为与CPU的运行没有交叉点(CPU的数据流注意是在Flash、内存、寄存器中传输),所以不会影响CPU的运行速度。唯一有可能影响的是外设与内存或者内存与内存之间的DMA传输。
  倘若ST公司的SRAM是一个双口RAM,也就是同时可以由两组接口对RAM进行访问,就可以很好的解决速度影响问题。倘若CPU恒定占有一组接口,而另一组接口留给DMA控制器。那么当外设与内存或者内存与内存之间的DMA传输时,由于不与CPU的访问SRAM接口冲突,所以可以解决速度影响问题。
  但其实偶尔还是会影响的,当CPU访问SRAM的空间和DMA访问SRAM的空间相同时,SRAM势必会对这种情况进行仲裁,这可能会影响到CPU的访问SRAM的速度。其实,这种情况的概率也是很小的,所以即使影响CPU的运行速度,也不会很大。

& n+ b- _/ D2 E8 v0 g9 E1 P) v
收藏 1 评论0 发布时间:2022-1-12 20:14

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版