终于比完了赛也收拾完了学校一切的事务,终于有时间来学习自己喜欢的东西了!!(搓手手)最开始我准备用开发板学习,我使用的开发板是STM32F429挑战者V2和V1,所以我迫不及待地下载了keil 5 和装上了pack包,然后就遇到了小问题:
由于这个问题会导致我下不进程序,会使我出现以下的错误:
Error: Flash Download failed - Target DLL has been cancelled
SWD/JTAG Communication Failure
....
终于花了半个晚上,找到了原因以及解决方法:
File : D:\keil4\ARM\PACK\Keil\STM32F4xx_DFP\2.12.0\Keil.STM32F4xx_DFP.pdsc
Context : Item #1: <control if="jep106id != 0x20"> Item #0: <block>::Line 2 Expression : " Message(2, "Not a genuine ST Device! Abort connection.");" ----------------------------^ E203 : Undefined identifier - function 'Message'
这是出现问题报错的语句,出现这个问题有可能是因为keil版本的问题也有可能是pack包版本过低所导致的,它的原意是STM32FXX_DEF文件中的Message函数不能正常识别,它也可以通过重装软件解决,可这有点繁琐,所以我上网找到了如下的方法:
-
复制报错的文件路径,找到pdsc文件。
-
更改pdsc的属性,右键点击此文件,将只读属性取消。
-
取消后,打开pdsc文件,找到对应message函数,将Message(2, "Not a genuine ST Device! Abort connection.");删除即可。
-
最后,将属性改回只读,就可以编译并且下载程序了!
在解决这个问题的时候也有一些发现,刚开始以外是芯片的问题,于是就研究了BOOT IO的启动方式,它的启动方式如下:
STM32芯片上有两个管脚BOOT0和BOOT1,这两个管脚在芯片复位时的电平状态决定了芯片复位后从哪个区域开始执行程序。
BOOT1 = X BOOT0 = 0 从用户闪存(flash)启动,这是正常模式 BOOT1 = 0 BOOT0 = 1 从内置SRAM(内存)启动,这是调试模式 BOOT1 = 1 BOOT0 = 1 从系统存储器启动,这种模式也可以用于调试
那么解决了基本的问题,那就开始学习吧🤦♀️首先是初识STM32。
STM32作为一个微控制器,内核为Cortex-M4,主频为180M,自带了各种常用通信接口,功能有如下:
-
串口(USART)-- Usb转串口模块、ESP8266、WIFI、GPS、GSM、串口屏、指纹识别...
-
内部集成电路(IIC) -- EEPROM 、 电容屏 、 陀螺仪MPU6050、OLED...
-
串行通信借口(SPI) -- 串行FLASH 、 以太网 W5500 、 音频模块VS1053 ...
-
SDIO \ FMC \ I2S \ SAI \ ADC \GPIO ...
存储器映射:存储器本身不具有地址信息,它的地址是由芯片厂商或用户分配,给存储器分配地址的过程就称为存储器映射。
序号 | 用途 | 地址范围 |
---|---|---|
Block 0 | SRAM(FLASH) | 0x0000 0000 ~ 0x1FFF FFFF(512MB) |
Block 1 | SRAM | 0x2000 0000 ~ 0x3FFF FFFF(512MB) |
Block 2 | 片上外设 | 0x4000 0000 ~ 0x5FFF FFFF(512MB) |
Block 3 | FMC的bank1 ~ bank2 | 0x6000 0000 ~ 0x7FFF FFFF(512MB) |
Block 4 | FMC的bank3 ~ bank4 | 0x8000 0000 ~ 0x9FFF FFFF(512MB) |
Block 5 | FMC | 0xA000 0000 ~ 0xCFFF FFFF(512MB) |
Block 6 | FMC | 0xD000 0000 ~ 0xDFFF FFFF(512MB) |
Block 7 | Cortex-M4内部外设 | 0xE000 0000 ~ 0xFFFF FFFF(512MB) |
寄存器映射(是我们最为熟悉的):在存储器Block2这块区域,设计的是片上外设,它们以四个字节为一个单位,共32bit,每一个单元对应不同的功能,当我们控制这些单元时就可以驱动外设工作。我们可以找到每个单元的起始地址,然后通过C语言指针的操作方式来访问这些单元,如果每次都是通过这种地址来访问,不仅不好记忆而且还容易出错,这时我们可以根据每个单元功能的不同,以功能为名给这个内存单元取一个别名,这个别名就是我们常说的寄存器。这个给已经分配好地址的有特定功能的内存单元取别名的过程就叫寄存器映射。
// GPIOH 端口全部输出 高电平
#define GPIOH_ODR (unsignedint*)(0x40021C14)
*GPIOH_ODR = 0xFF;
GPIO八种模式 typedef enum
{ GPIO_Mode_AIN = 0x0, /* 模拟输入 */ GPIO_Mode_IN_FLOATING = 0x04, /* 浮空输入,复位后的状态 */ GPIO_Mode_IPD = 0x28, /* 下拉输入 */ GPIO_Mode_IPU = 0x48, /* 上拉输入 */ GPIO_Mode_Out_OD = 0x14, /* 开漏输出 */ GPIO_Mode_Out_PP = 0x10, /* 推挽输出 */ GPIO_Mode_AF_OD = 0x1C, /* 复用开漏输出 */ GPIO_Mode_AF_PP = 0x18 /* 复用推挽输出 */
}GPIOMode_TypeDef;
端口配置低寄存器 (GPIOx_CRL)(x = A...E)
端口配置高寄存器(GPIOx_CRH)(x = A...E)
端口端口输入数据寄存器(GPIOx_IDR)(x = A...E):这些位为只读并只能以字(16位)的形式读出。读出的值为对应I/O口的状态。
端口输出数据寄存器(GPIOx_ODR)(x = A...E):只能以字(16bit)的形式操作,复位值全是0。写0即输出0,写1即输出1。
端口位设置/清除寄存器(GPIOx_BSRR)(x = A...E):高16bit写1用于清0,低16bit写1用于置位,同时写1的话低16bi有效。
端口位清除寄存器(GPIOx_BRR)(x = A...E):低16位写1用于置位。
端口配置锁定寄存器(GPIOx_LCKR)(x = A...E)