本文参考自由串口DMA配置扯出来的一些问题
查看原文: 原文地址
A:使用串口DMA收发,在初始化的时候,写了如下程序
DMA_InitStruct.DMA_BufferSize = BufferSize;
DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralDST;
DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;
DMA_InitStruct.DMA_MemoryBaseAddr = (uint32_t)str_UsartTx.Buff;
DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)USART1->DR;
DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStruct.DMA_Priority = DMA_Priority_High;
DMA_Init(DMA1_Channel4,&DMA_InitStruct);
自认为对寄存器比较熟悉了,结果翻车了,于是开始了对于USART1->DR的深究
首先是USART1的定义
#define USART1 ((USART_TypeDef *) USART1_BASE)
可以看出来,USART1是一个USART_TypeDef类型的结构体指针了,USART1_BASE很明显是USART1对应的地址。
继续看USART_TypeDef的定义
typedef struct
{
__IO uint16_t SR;
uint16_t RESERVED0;
__IO uint16_t DR;
uint16_t RESERVED1;
__IO uint16_t BRR;
uint16_t RESERVED2;
__IO uint16_t CR1;
uint16_t RESERVED3;
__IO uint16_t CR2;
uint16_t RESERVED4;
__IO uint16_t CR3;
uint16_t RESERVED5;
__IO uint16_t GTPR;
uint16_t RESERVED6;
} USART_TypeDef;
这个如果看USART寄存器映射,你就明白为什么会是这种结构了。至于RESERVEDn,是因为USART对应寄存器的高16位没有用到的缘故。顺便放一个USART的寄存器映射图
至于__IO,cor_cm3.h中有定义
#define __O volatile /*!< Defines 'write only' permissions */
#define __IO volatile /*!< Defines 'read / write' permissions */
OK,这就明了了,USART1是一个结构体指针,指向USART1的物理地址,所以需要用->来访问结构体成员,所以USART1->DR就是USART1中的DR成员,因为USART1指向了USART1的物理地址,所以DR就指向了USART1的DR寄存器
同样的方式,就可以很好的理解32中其他的寄存器访问原理了
本文参考自由串口DMA配置扯出来的一些问题
查看原文: 原文地址