【STM32-MX_GPIO_Init分析】

MX_GPIO_Init分析源码如下:
在这里插入图片描述

__HAL_RCC_GPIOE_CLK_ENABLE源码如下:
在这里插入图片描述

#define RCC                 ((RCC_TypeDef *) RCC_BASE)
#define RCC_BASE              (AHB1PERIPH_BASE + 0x3800UL)
#define AHB1PERIPH_BASE       (PERIPH_BASE + 0x00020000UL)
#define PERIPH_BASE           0x40000000UL
// 可以看到最后RCC地址为	0x40000000UL + 0x00020000UL + 0x3800UL,即0x40023800

我们从数据手册的寄存器地址映射也可以看到RCC地址确实是0x40023800
在这里插入图片描述

RCC_TypeDef结构体描述如下:
在这里插入图片描述
我们查看参考文档
在这里插入图片描述
可以发现这里的寄存器是跟参考文档一样按照顺序排列,每个寄存器32位4字节。

在这里插入图片描述

其中RCC->AHB1ENR地址应该为0x40023800+0x30,而RCC_AHB1ENR_GPIOEEN0x1UL<<4U,即0x1<<4,在位4,通过查文档,可以看到GPIOEEN确实在位4。
在这里插入图片描述
这里SET_BIT为REG |= BIT,即*(0x40023800+0x30) |= (0x1<<4);,可以看到通过或运算将RCC->AHB1ENR寄存器的RCC_AHB1ENR_GPIOEEN位设置为1,而不影响其他位,即开启了对应寄存器的时钟功能;
这里READ_BIT为REG & BIT,即(0x40023800+0x30) & (0x1<<4);,可以看到通过与运算将RCC->AHB1ENR寄存器的RCC_AHB1ENR_GPIOEEN位读取出来,而不读取其他位,即读取对应寄存器的时钟开启状态功能;
在这里插入图片描述

UNUSED这里就做了强转,避免编译器提示变量未使用;
在这里插入图片描述

至此,MX_GPIO_Init函数分析完毕,即开启对应GPIO时钟,然后对GPIO做些初始化。最重要的在于通过或运算向寄存器指定位写入值,通过与运算读取寄存器指定位的值。

  • 9
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32F407单片机16bit_DAC_AD5689模拟量(-10V~10V)电压输出实验KEIL工程源码: int main(void) { uint16_t data=0xFFFF/2; double temp,opa; /* 复位所有外设,初始化Flash接口和系统滴答定时器 */ HAL_Init(); /* 配置系统时钟 */ SystemClock_Config(); /* 初始化串口并配置串口中断优先级 */ MX_DEBUG_USART_Init(); KEY_GPIO_Init(); printf("硬石DAC(AD5689)模块模拟量电压输出测试\n"); AD5689_Init(); AD5689_WriteUpdate_DACREG(DAC_A,data); AD5689_WriteUpdate_DACREG(DAC_B,0xFFFF-data); printf("data:%d\n",data); opa=OPA_RES_R2/OPA_RES_R1; while(1) { if(KEY1_StateRead()==KEY_DOWN) { if(data>(0xFFFF-1000)) data=(0xFFFF-1000); data +=1000; AD5689_WriteUpdate_DACREG(DAC_A,data); AD5689_WriteUpdate_DACREG(DAC_B,0xFFFF-data); temp=(double)(data*2-0xFFFF)*2500*opa/0xFFFF; //temp为目标电源值,这里先放大1000倍(方便计算而已),等后面显示再还原 //data是数字量DA值,当data取值为:0~0xFFFF对应AD5689输出为0~5V //本例程是输出-10V~10V,这个功能主要是靠运放实现,特殊的电路使得: //AD5689输出0V时对应运放输出-10V,AD5689输出2.5V对应运放输出0V,AD5689输出5V对应运放输出10V //(上面虽说是10V,实际上应该是 2.5V*opa(运放放大倍数),这里opa=40.2K/10K=4.02) //所以使得程序:data值为0时运放输出-10V, data为0xFFFF/2时输出运放输出0V,data为0xFFFF时输出运放输出10V //temp=(data-0xFFFF/2)/(0xFFFF/2)*2.5*1000*opa printf("data:%d->%0.3fV\n",data,temp/1000); } if(KEY2_StateRead()==KEY_DOWN) { if(data<1000) data=1000; data -=1000;

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值