完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
|
|
相关推荐
1个回答
|
|
摘录自:零死角玩转 STM32—基于野火 F429[挑战者]开发板
8.3 实验:构建库函数雏形 虽然库的优点多多,但很多人对库还是很忌惮,因为一开始用库的时候有很多代码, 很多文件,不知道如何入手。不知道您是否认同这么一句话:一切的恐惧都来源于认知的 空缺。我们对库忌惮那是因为我们不知道什么是库,不知道库是怎么实现的。零死角玩转 STM32 —基于野火 F429[挑战者]开发板 第 60 页 共 1046 接下来,我们在寄存器点亮 LED 的代码上继续完善,把代码一层层封装,实现库的最 初的雏形,相信经过这一步的学习后,您对库的运用会游刃有余。这里我们只讲如何实现 GPIO 函数库,其他外设的我们直接参考 ST 标准库学习即可,不必自己写。 下面请打开本章配套例程“构建库函数雏形”来阅读理解,该例程是在上一章的基础 上修改得来的。 8.3.1 修改寄存器地址封装 上一章中我们在操作寄存器的时候,操作的是都寄存器的绝对地址,如果每个外设寄 存器都这样操作,那将非常麻烦。我们考虑到外设寄存器的地址都是基于外设基地址的偏 移地址,都是在外设基地址上逐个连续递增的,每个寄存器占 32 个或者 16 个字节,这种 方式跟结构体里面的成员类似。所以我们可以定义一种外设结构体,结构体的地址等于外 设的基地址,结构体的成员等于寄存器,成员的排列顺序跟寄存器的顺序一样。这样我们 操作寄存器的时候就不用每次都找到绝对地址,只要知道外设的基地址就可以操作外设的 全部寄存器,即操作结构体的成员即可。 在工程中的“stm32f4xx.h”文件中,我们使用结构体封装 GPIO 及 RCC 外设的的寄存 器,见代码清单 8-1。结构体成员的顺序按照寄存器的偏移地址从低到高排列,成员类型 跟寄存器类型一样。如不理解 C 语言对寄存器的封的语法原理,请参考《C 语言对寄存器 的封装》 小节。 代码清单 8-1 封装寄存器列表 1 //volatile 表示易变的变量,防止编译器优化 2 #define __IO volatile 3 typedef unsigned int uint32_t; 4 typedef unsigned short uint16_t; 5 6 /* GPIO 寄存器列表 */ 7 typedef struct { 8 __IO uint32_t MODER; /*GPIO 模式寄存器 地址偏移: 0x00 */ 9 __IO uint32_t OTYPER; /*GPIO 输出类型寄存器 地址偏移: 0x04 */ 10 __IO uint32_t OSPEEDR; /*GPIO 输出速度寄存器 地址偏移: 0x08 */ 11 __IO uint32_t PUPDR; /*GPIO 上拉/下拉寄存器 地址偏移: 0x0C */ 12 __IO uint32_t IDR; /*GPIO 输入数据寄存器 地址偏移: 0x10 */ 13 __IO uint32_t ODR; /*GPIO 输出数据寄存器 地址偏移: 0x14 */ 14 __IO uint16_t BSRRL; /*GPIO 置位/复位寄存器低 16 位部分 地址偏移: 0x18 */ 15 __IO uint16_t BSRRH; /*GPIO 置位/复位寄存器 高 16 位部分地址偏移: 0x1A / 16 __IO uint32_t LCKR; /GPIO 配置锁定寄存器 地址偏移: 0x1C / 17 __IO uint32_t AFR[2]; /GPIO 复用功能配置寄存器 地址偏移: 0x20-0x24 / 18 } GPIO_TypeDef; 19 20 /RCC 寄存器列表/ 21 typedef struct { 22 __IO uint32_t CR; /!< RCC 时钟控制寄存器,地址偏移: 0x00 / 23 __IO uint32_t PLLCFGR; /!< RCC PLL 配置寄存器,地址偏移: 0x04 / 24 __IO uint32_t CFGR; /!< RCC 时钟配置寄存器,地址偏移: 0x08 / 25 __IO uint32_t CIR; /!< RCC 时钟中断寄存器,地址偏移: 0x0C / 26 __IO uint32_t AHB1RSTR; /!< RCC AHB1 外设复位寄存器,地址偏移: 0x10 / 27 __IO uint32_t AHB2RSTR; /!< RCC AHB2 外设复位寄存器,地址偏移: 0x14 / 28 __IO uint32_t AHB3RSTR; /!< RCC AHB3 外设复位寄存器,地址偏移: 0x18 / 29 __IO uint32_t RESERVED0; /!< 保留, 地址偏移: 0x1C / 30 __IO uint32_t APB1RSTR; /!< RCC APB1 外设复位寄存器,地址偏移: 0x20 /零死角玩转 STM32 —基于野火 F429[挑战者]开发板 第 61 页 共 1046 31 __IO uint32_t APB2RSTR; /!< RCC APB2 外设复位寄存器,地址偏移: 0x24/ 32 __IO uint32_t RESERVED1[2]; /!< 保留,地址偏移: 0x28-0x2C/ 33 __IO uint32_t AHB1ENR; /!< RCC AHB1 外设时钟寄存器,地址偏移: 0x30 / 34 __IO uint32_t AHB2ENR; /!< RCC AHB2 外设时钟寄存器,地址偏移: 0x34 / 35 __IO uint32_t AHB3ENR; /!< RCC AHB3 外设时钟寄存器,地址偏移: 0x38 */ 36 /RCC 后面还有很多寄存器,此处省略/ 37 } RCC_TypeDef; 这段代码在每个结构体成员前增加了一个“__IO”前缀,它的原型在这段代码的第一 行,代表了 C 语言中的关键字“volatile”,在 C 语言中该关键字用于表示变量是易变的, 要求编译器不要优化。这些结构体内的成员,都代表着寄存器,而寄存器很多时候是由外 设或 STM32 芯片状态修改的,也就是说即使 CPU 不执行代码修改这些变量,变量的值也 有可能被外设修改、更新,所以每次使用这些变量的时候,我们都要求 CPU 去该变量的地 址重新访问。若没有这个关键字修饰,在某些情况下,编译器认为没有代码修改该变量, 就直接从 CPU 的某个缓存获取该变量值,这时可以加快执行速度,但该缓存中的是陈旧数 据,与我们要求的寄存器最新状态可能会有出入。 |
|
|
|
只有小组成员才能发言,加入小组>>
调试STM32H750的FMC总线读写PSRAM遇到的问题求解?
1678 浏览 1 评论
X-NUCLEO-IHM08M1板文档中输出电流为15Arms,15Arms是怎么得出来的呢?
1580 浏览 1 评论
1012 浏览 2 评论
STM32F030F4 HSI时钟温度测试过不去是怎么回事?
703 浏览 2 评论
ST25R3916能否对ISO15693的标签芯片进行分区域写密码?
1626 浏览 2 评论
1892浏览 9评论
STM32仿真器是选择ST-LINK还是选择J-LINK?各有什么优势啊?
674浏览 4评论
STM32F0_TIM2输出pwm2后OLED变暗或者系统重启是怎么回事?
537浏览 3评论
556浏览 3评论
stm32cubemx生成mdk-arm v4项目文件无法打开是什么原因导致的?
524浏览 3评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-12-3 06:46 , Processed in 1.292088 second(s), Total 78, Slave 62 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号