DMA具备双缓冲机制,可以让数据在传输时候不断流,也就是我们所谓的PING-PONG-BUFFER,也就是说有AB两个BUFFER,DMA访问A时候,CPU访问B,DMA访问B时候,CPU访问A.这种实现导致总线矩阵相对复杂,所以,一般低端MCU也不会有这个机制.但是在做这个机制的试验时候,虽然可以做到PING-PONG BUFFER的传输,但是..却有点小意外.其实双缓冲,在配置过程并不难,比如提供一个配置参考:
% `4 o f- T. {8 j# i' ~+ G4 R* l4 y. M6 G* O0 C
- DMA_DeInit(DMA1_Stream4);3 Z8 ?& @$ q/ s# F5 f9 Z3 H4 \: h
- DMA_StructInit(&DMA_InitStructure);
! H% L' e( K7 c7 K+ ?% c2 O1 z; Z8 f - DMA_InitStructure.DMA_Channel = DMA_Channel_7;
9 e0 o, T) q) Q - DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART3->DR; //数据传输的外设首地址
& i% v" d1 ]& I9 C7 U+ b - DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Buffer0; //自己定义待发送数组的首地址,要强制转换为32位
( B9 t1 Z6 n5 A; P+ D0 h3 D - DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral; //数据传输方向选择为内存->外设
5 r: c2 m$ B! i/ Y8 u - DMA_InitStructure.DMA_BufferSize = 4; //传输数据大小为8,大小要配合定义的数组类型和外设数据类型,否则会丢失或补充数据- Y5 {" [ ]: y( B9 Y" W, I2 R
- DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址寄存器自动增加禁止,因为这里只用到了 DR 数据寄存器
3 m: h X! v4 `4 {2 I. L - DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址自增允许,因为要读取一个数组" w+ D% q! {) ^) P! }) m* M
- DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; //外设的数据大小,因为 USART3_DR 数据寄存器为8位,故选Byte) n5 X& l i. L# ^ s
- DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //这里也选 Byte5 ]9 T, t4 `0 x$ e5 x' z$ i4 n2 V1 D
- DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //DMA 传输模式为 DMA_Mode_Normal,如果为 DMA_Mode_Circular,将会循环传输, l, V2 g. F" L, i% d$ r
- DMA_InitStructure.DMA_Priority = DMA_Priority_Low; //优先级为 High8 n% G6 u8 H3 o. e: B
- DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;4 W3 @" I* j5 W4 P$ v
- DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_1QuarterFull;
9 R; T* Y8 Q' F! B0 x3 s* ? - DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
6 f$ w3 q8 S' l7 B1 K, v - DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
0 K' ^, E! N0 X$ b - DMA_DoubleBufferModeConfig(DMA1_Stream4, (uint32_t)Buffer1, DMA_Memory_0);. i" z6 I0 ~: O* }, @$ E+ I
- DMA_DoubleBufferModeCmd(DMA1_Stream4, ENABLE);
复制代码 1 z6 I5 Z) y* I1 y. H# _
我们用了DMA1中STREAM4的CH7,所以这么初始化,并填入双Buffer,每个Buffer长度是4.因为用到的是Stream4,所以查询DMA_IT_TCIF4,然后再通过DMA_GetCurrentMemoryTarget,就可以知道该换那个Buffer,查到是在用1的Buffer,我们就填0号数组,否则填1号数组,就这么简单.
- K8 g ?: I6 ]3 b ~+ s" V
- @3 q9 ?# H* g7 W7 p. V1 I- void DMA1_Stream4_IRQHandler(void)
5 |' j4 G( a: g - {% q8 a4 _2 k9 Y4 Z7 X
- uint16_t i = 0;
8 {9 I) z& m% B# e' W - GPIO_ResetBits(GPIOB, GPIO_Pin_12);
3 j' M b- s8 S) c( s* I# y& _% | - if(DMA_GetITStatus(DMA1_Stream4, DMA_IT_TCIF4) == SET)+ c$ ^3 }# T- ~# ^
- {/ {& }( c7 H& k2 d5 P X0 V7 a* k
- DMA_ClearITPendingBit(DMA1_Stream4, DMA_IT_TCIF4);; @0 j5 w; \& ]) ?$ \9 x
- ( t9 \# M! k# n; ?3 X" _+ Z" a
- if(DMA_GetCurrentMemoryTarget(DMA1_Stream4))
/ J t5 r3 z6 z0 ` - {2 J$ N2 A8 }1 r* l% U
- //现在访问是Buufer1,所以写Buffer0.
0 B" Z' }/ I( d% p$ w% H - for(i = 0; i < 4; i++). W3 Y y5 @8 Q/ ?
- {
! b9 _ O& `# N - Buffer0[i]++;3 C/ h8 m* ]6 g9 r* c9 X
- }! q5 X- l9 N- C8 C
- }9 ?9 j8 T0 f; S6 L
- else
/ L y8 N; T" i" v# L - {
2 o( C U {6 r( o1 n - for(i = 0; i < 4; i++) ^ [) {# n8 f2 @4 Y+ K6 o5 s% i% V
- {
9 Y" B0 J4 K) k% W8 G$ b1 ] -
3 F5 d0 E1 c3 ?1 f: P% p N3 y3 i - Buffer1[i]--;
* s9 k6 k* w7 Q+ I - }
5 `$ o+ U( ?! q& W3 J - }
- K8 I% n, z2 m6 D' z9 a - 2 N. H# G; T: _0 i4 }1 U
- }& k+ G; q; R+ Z% [) E/ v" ?* @
-
- G' l% X. P) b/ l - }
复制代码
4 u$ {, U U1 ?————————————————1 g% y2 }1 f1 d$ H; c4 t1 [7 t
版权声明:JawSoW
8 N8 i9 G( w2 g, P! ^* B. O: P8 B% k& X; h. |! t- O) V, v
|