DMAC也可以和外设进行数据交互。之前我们曾使用PDC进行USART的数据回显,这次就使用DMAC完成相同的工作。而且由于DMAC有内部的缓冲区,实现起来更为简单。
一、 USART设置
因为之前已经做过相关的实验,这里不再重复。需要注意的是,要注意JP11的跳线,以选择正确的协议(RS232)。另外,如果使用硬件握手协议的话,注意设置PC端串口通信软件的线路控制信号。
另外,由于不再使用手动的缓冲区和PDC,所以不需要进行相关的设置。同时,也不用再使用USART的接收超时功能。
二、 DMAC设置
本次使用的通道依然为通道0:
#define DMAC_CH 0
-
启用DMAC:
// 代码略...
-
设置DSCR为0,以进行单次传输:
DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_DSCR = 0;
-
设置SADDR以及DADDR。
因为数据都在US_RHR和US_THR的低位上,所以将源地址和目标地址分别设为这两个寄存器的地址即可。
DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_SADDR = &(USART1->US_RHR); DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_DADDR = &(USART1->US_THR);
-
设置CTRLA和CTRLB。
在USART数据位为8位时,一次传输一个字节即可。
DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_CTRLA = DMAC_CTRLA_BTSIZE(16) // 进行16次传输 | DMAC_CTRLA_SRC_WIDTH_BYTE // 一次传输一个字节 | DMAC_CTRLA_DST_WIDTH_BYTE // 同上 ; DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_CTRLB = DMAC_CTRLB_FC_PER2PER_DMA_FC //外设至外设的传输 | DMAC_CTRLB_SRC_INCR_FIXED // 传输时源地址固定 | DMAC_CTRLB_DST_INCR_FIXED // 传输时目标地址固定 ;
-
设置CFG寄存器。
因为DMAC和USART1之间有硬件握手接口,所以这里使用硬件握手接口即可(否则需要使用软件握手接口手动触发传输)。USART1的发送接口号为5,接收接口号为6:
由于需要尽快将DMAC内部缓冲区的内容传输出去,所以一旦其数据量可以发送,就发送出去。
DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_CFG = DMAC_CFG_SRC_H2SEL_HW // 使用硬件握手 | DMAC_CFG_DST_H2SEL_HW | DMAC_CFG_SRC_PER(6) // 接口号 | DMAC_CFG_DST_PER(5) | DMAC_CFG_SOD_DISABLE | DMAC_CFG_FIFOCFG_ASAP_CFG // 尽快发送数据 ;
-
启用中断。
在传输任务完成后,需要重新启用通道,以重新开始任务。
DMAC->DMAC_CHER = DMAC_CHER_ENA0 << DMAC_CH; // NVIC中断设置的代码略...
-
中断处理。
在中断中重新设置CTRLA寄存器的BTSIZE字段,再启用通道即可。
void DMAC_Handler(void) { uint32_t status = DMAC->DMAC_EBCISR; // 判断是否为指定中断 if (status & (DMAC_EBCISR_CBTC0 << DMAC_CH)) { // 设置 CTRLA DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_CTRLA &= ~(uint32_t)DMAC_CTRLA_BTSIZE_Msk; DMAC->DMAC_CH_NUM[DMAC_CH].DMAC_CTRLA |= DMAC_CTRLA_BTSIZE(16); // 再次启用通道 DMAC->DMAC_CHER = DMAC_CHER_ENA0 << DMAC_CH; } }