DMA具备双缓冲机制,可以让数据在传输时候不断流,也就是我们所谓的PING-PONG-BUFFER,也就是说有AB两个BUFFER,DMA访问A时候,CPU访问B,DMA访问B时候,CPU访问A.这种实现导致总线矩阵相对复杂,所以,一般低端MCU也不会有这个机制.但是在做这个机制的试验时候,虽然可以做到PING-PONG BUFFER的传输,但是..却有点小意外.其实双缓冲,在配置过程并不难,比如提供一个配置参考:
( T( r0 g7 o D" q+ @5 n( ^- m% T% A; b& a' ^& S9 w- j! U3 a
- DMA_DeInit(DMA1_Stream4);0 g5 q$ p+ l- ^ O
- DMA_StructInit(&DMA_InitStructure);' h8 Y5 Q6 H+ F1 x7 M9 P, G! E
- DMA_InitStructure.DMA_Channel = DMA_Channel_7;' O$ T/ G6 k7 t7 } w- Q
- DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART3->DR; //数据传输的外设首地址
' d* t4 r9 C4 i! ~$ H+ n+ q5 M - DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Buffer0; //自己定义待发送数组的首地址,要强制转换为32位3 U+ q |! c; O s2 G2 D* l, H
- DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral; //数据传输方向选择为内存->外设2 |2 e! f. }0 _0 }, d2 J
- DMA_InitStructure.DMA_BufferSize = 4; //传输数据大小为8,大小要配合定义的数组类型和外设数据类型,否则会丢失或补充数据1 o+ u$ V" N) r4 P2 K- `3 }. |
- DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址寄存器自动增加禁止,因为这里只用到了 DR 数据寄存器8 k, ]' Y" p1 X
- DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址自增允许,因为要读取一个数组
! g! `; W7 ]( ?; W7 S - DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外设的数据大小,因为 USART3_DR 数据寄存器为8位,故选Byte# l. q8 a, }9 N, F
- DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //这里也选 Byte
& j7 q1 U0 s4 Q: J1 q) Y0 v - DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //DMA 传输模式为 DMA_Mode_Normal,如果为 DMA_Mode_Circular,将会循环传输6 y0 C4 s7 |% z, z
- DMA_InitStructure.DMA_Priority = DMA_Priority_Low; //优先级为 High8 t# k0 r+ s ^% h4 f
- DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
+ H# x7 }2 T. F; ?% N( T - DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_1QuarterFull;
$ Y; Q$ Q( O# n% A- c q - DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;7 y' W/ ~$ _6 @" z' G
- DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
* v; Y: z; j7 E - DMA_DoubleBufferModeConfig(DMA1_Stream4, (uint32_t)Buffer1, DMA_Memory_0);$ X/ ?; Y6 O3 |3 n* N8 \& L
- DMA_DoubleBufferModeCmd(DMA1_Stream4, ENABLE);
复制代码
- m: U, X) d; S/ ~+ ~( d- ]. H4 C我们用了DMA1中STREAM4的CH7,所以这么初始化,并填入双Buffer,每个Buffer长度是4.因为用到的是Stream4,所以查询DMA_IT_TCIF4,然后再通过DMA_GetCurrentMemoryTarget,就可以知道该换那个Buffer,查到是在用1的Buffer,我们就填0号数组,否则填1号数组,就这么简单.( f; o( H! B! K3 ~ O
. g& j! ]: b( z$ h. ]/ j
- void DMA1_Stream4_IRQHandler(void)
# C+ R8 ?9 |: k- K5 E8 \ - {
: r( U8 S% T" d - uint16_t i = 0;
4 U% E7 x* b2 U6 d& p1 L1 J - GPIO_ResetBits(GPIOB, GPIO_Pin_12);: D6 B, ^- H4 L$ g/ l; P
- if(DMA_GetITStatus(DMA1_Stream4, DMA_IT_TCIF4) == SET)
3 c7 m& n% n1 @# p* c - {7 U" Z5 ^3 }' |( p" @
- DMA_ClearITPendingBit(DMA1_Stream4, DMA_IT_TCIF4);3 U) _3 q# F$ z. _( x' e! m" v* f
-
& f! F: S n' N+ L+ U - if(DMA_GetCurrentMemoryTarget(DMA1_Stream4))1 g" Z4 [7 [9 H
- {
L; U# Q) y' ?% i* c: t - //现在访问是Buufer1,所以写Buffer0.( g: R, \2 t% m5 ]+ J, I
- for(i = 0; i < 4; i++)9 K- F3 M5 T ^% E
- {
9 E" |5 B1 A r; c9 @9 e - Buffer0[i]++;4 M2 z4 ]% ^4 F, B5 B$ t4 K+ b9 O
- }
2 k/ G7 G% {( y& ^% e; B* Z, d - }
" E! ]: O. H* }# c. l( V2 p0 c7 K - else% A! F) F" h4 i6 k
- {1 J8 O' u% T: o. ]' C8 s+ d# X4 y. a
- for(i = 0; i < 4; i++): L# K: o' |9 |2 _ z
- {
9 q9 p" @" T& L' V -
, i _" K7 K+ C - Buffer1[i]--;" T" Z8 [% f o2 D; A
- }4 ?/ W5 x) l! j) ]
- }1 _6 ~5 `7 R. N$ n4 b; ]9 Z- d
-
3 `7 u& @5 v; q9 ] - }
' o& a' X, Q: \/ L2 q - # N# [# ^4 V9 X8 d9 o7 i) w
- }
复制代码
. ?: y0 f) b+ p7 e7 _————————————————0 h! P$ c* h5 B# s' ] U! x
版权声明:JawSoW' j1 \5 L7 [- X# K! d) a
% m' R2 V* f* O# O8 Q |