在 STM32F1 系列中,内存到内存传输模式允许 DMA 在两个内存地址之间传输数据,而不涉及外设。为了实现内存到内存的传输,需要启用 DMA_M2M
配置项,而不是 DMA_DIR_PeripheralDST
或 DMA_DIR_PeripheralSRC
。内存到内存模式会忽略 DMA_DIR
设置,因为数据源和数据目标都在内存中。
内存到内存传输的配置步骤
1. 设置 DMA_M2M_Enable
以启用内存到内存传输
要实现内存到内存传输,需要将 DMA_M2M
设置为 DMA_M2M_Enable
。此外,还需要指定源地址和目标地址(都为内存地址),并启用内存地址自增。
2. DMA 配置示例代码
以下代码展示了如何配置 DMA 在两个内存缓冲区之间传输数据:
#include "stm32f10x.h"
#define BUFFER_SIZE 10
uint8_t src_buffer[BUFFER_SIZE] = {'D', 'M', 'A', ' ', 'T', 'e', 's', 't', '!', '\0'};
uint8_t dest_buffer[BUFFER_SIZE] = {0}; // 用于接收 DMA 传输的数据
void DMA_MemoryToMemory_Init(void);
int main(void) {
DMA_MemoryToMemory_Init();
// 启动 DMA 传输
DMA_Cmd(DMA1_Channel1, ENABLE);
// 等待传输完成
while (!DMA_GetFlagStatus(DMA1_FLAG_TC1));
// 清除传输完成标志
DMA_ClearFlag(DMA1_FLAG_TC1);
while (1) {
// 主循环 - 传输完成后 dest_buffer 中应包含 src_buffer 的内容
}
}
void DMA_MemoryToMemory_Init(void) {
// 启用 DMA1 的时钟
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
DMA_InitTypeDef DMA_InitStructure;
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)src_buffer; // 源内存地址
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)dest_buffer; // 目标内存地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; // 在内存到内存模式下此字段会被忽略
DMA_InitStructure.DMA_BufferSize = BUFFER_SIZE; // 传输的数据量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable; // 源地址递增
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; // 目标地址递增
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; // 数据大小为 8 位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; // 单次传输模式
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Enable; // 启用内存到内存模式
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
}
代码解释
- DMA_PeripheralBaseAddr:指定数据源的内存地址,这里为
src_buffer
的地址。 - DMA_MemoryBaseAddr:指定数据目标的内存地址,这里为
dest_buffer
的地址。 - DMA_DIR:虽然设置为
DMA_DIR_PeripheralSRC
,但在内存到内存模式 (DMA_M2M_Enable
) 下,这一参数会被忽略。 - DMA_M2M:设置为
DMA_M2M_Enable
,表示启用内存到内存传输模式。 - DMA_Mode:设置为
DMA_Mode_Normal
,表示 DMA 完成单次传输后停止。
注意事项
- 数据传输方向 (
DMA_DIR
) 被忽略:在内存到内存模式下,DMA_DIR
设置不会生效,因为数据传输不会涉及外设。 - 源和目标地址递增:通常设置
DMA_PeripheralInc
和DMA_MemoryInc
为Enable
,这样 DMA 在传输过程中会逐步递增地址。 - 传输完成标志:传输完成后,可以通过检查
DMA1_FLAG_TC1
标志位确认数据已传输完毕,并清除此标志。
适用场景
内存到内存传输适用于以下情况:
- 数据缓冲区复制:快速将数据从一个内存缓冲区复制到另一个缓冲区,适用于数据预处理或双缓冲模式。
- 处理大量数据:DMA 在后台自动完成数据传输,CPU 可以执行其他任务,提高效率。
通过正确配置 DMA_M2M
,可以让 DMA 在 STM32 中实现高效的内存到内存传输,从而减轻 CPU 的负担。