目录
一、STM32F103系列芯片的地址映射和寄存器映射原理和GPIO端口的初始化设置三步骤
二、以 STM32最小系统核心板(STM32F103C8T6)+面板板+3只红黄绿LED 灯搭建电路,轮流闪烁,间隔时长1秒。
一、STM32F103系列芯片的地址映射和寄存器映射原理和GPIO端口的初始化设置三步骤
1.STM32F103的存储器映射&寄存器映射
存储器映射
在STM32内部的每个内存块存放不同的功能,而每个内存块都有地址。每个内存块中又分为好几个内存块,每个内存块有不同的寄存器,一个地址存放4个字节,在STM32内存中,1个地址存放32位的寄存器,有的寄存器有可能用不到32位(地址之间都相差4,比如第一个地址为0x00,那第二个地址为0x04),只用其中的低16位,那高16位将被保。外设内存块中,GPIO的内存块,在这个内存块中又分为GPIOA内存块,GPIOB内存块等等,内存块中存放的就是一些寄存器,每4个地址存放一个32位的寄存器。偏移地址是基于每个最小内存块(就是像GPIOA,GPIOB之类的内存块)的首地址。每个最小内存块都有地址范围,且GPIO内存块的首地址就是GPIOA内存块的首地址。
寄存器映射
寄存器映射:给具有特殊功能的内存块的首地址取一个具有特殊意义名称的过程就叫做寄存器映射。
学过C语言的同学都知道指针,我们要想取地址中的值必须利用指针,比如GPIOA_OTYPER寄存器的偏移地址为0x04,且GPIOA内存块的首地址为0X4000 0000,那么我们就可以利用下面的方式来取寄存器中的值。
首先我们知道GPIOA_OTYPER寄存器的地址为(0x04+0x4000 0000),这样写还不够,因为编译器并不知道这是地址,所以我们利用指针将其转成地址(unsigned int *)(0x04+0x4000 0000),最后我们再利用指针*(unsigned int *)(0x04+0x4000 0000)就可以读写寄存器了。
#define GPIOA_OTYPER *(unsigned int *)(0x04+0x4000 0000)这就是寄存器映射。
2.GPIO端口的初始化设置三步骤
GPIO端口初始化时,需要下面的步骤:
-
使能GPIO时钟,RCC_APB2PeriphClockCmd。
-
设置GPIO参数:输出OR输入,工作模式,端口翻转速率;
-
调用初始化函数:GPIO_Init
二、以 STM32最小系统核心板(STM32F103C8T6)+面板板+3只红黄绿LED 灯搭建电路,轮流闪烁,间隔时长1秒。
1.STM32核心板原理图
2.安装CH340-driver驱动
3.检查端口
右击我的电脑,点击管理,点击设备管理器查看端口
4.将核心板的boom调为1,0
5.搭建流水灯电路
对于USB转TTL模块和stm32f103c8t6连接
GND — GND,3v3 — 3v3,TXD — A10,RXD — A9
6.代码实现并进行烧录
1.新建项目
点击Project下的New uVision Project:,然后选择项目路径,填写文件名。
选择芯片:
在右边文件夹添加新文件:
选择一个.c文件进行创建。
之后将所需要的启动文件复制到项目目录下(f103c8t6启动文件为startup_stm32f10x_md.s:
然后将其添加到文件夹:
最终结果:
打开魔术棒,如下图所示勾选Create HEX File:
输入代码:
#define GPIOB_BASE 0x40010C00
#define GPIOC_BASE 0x40011000
#define GPIOA_BASE 0x40010800
#define RCC_APB2ENR (*(unsigned int *)0x40021018)
#define GPIOB_CRL (*(unsigned int *)0x40010C00)
#define GPIOC_CRH (*(unsigned int *)0x40011004)
#define GPIOA_CRL (*(unsigned int *)0x40010800)
#define GPIOB_ODR (*(unsigned int *)0x40010C0C)
#define GPIOC_ODR (*(unsigned int *)0x4001100C)
#define GPIOA_ODR (*(unsigned int *)0x4001080C)
void SystemInit(void);
void Delay_ms(volatile unsigned int);
void Delay_ms( volatile unsigned int t)
{
unsigned int i;
while(t--)
for (i=0;i<800;i++);
}
int main(){
// 开启时钟
RCC_APB2ENR |= (1<<3); // 开启 GPIOB 时钟
RCC_APB2ENR |= (1<<4); // 开启 GPIOC 时钟
RCC_APB2ENR |= (1<<2); // 开启 GPIOA 时钟
// 设置 GPIO 为推挽输出
// 设置 GPIOB 最后四位为 0001 (B0)
GPIOB_CRL |= (1<<0); // 最后一位设置为1
GPIOB_CRL &= ~(0xE); // 倒数二、三、四位设置为0
// 设置 GPIOC 前四位为 0001 (C15)
GPIOC_CRH |= (1<<28); // 第四位设置为1
GPIOC_CRH &= ~(0xE0000000); // 前三位设置为0
// 设置 GPIOA 最后四位为 0001 (A0)
GPIOA_CRL |= (1<<0); // 最后一位设置为1
GPIOA_CRL &= ~(0xE); // 倒数二、三、四位设置为0
// 3个LED初始化为不亮(即高点位)
GPIOB_ODR |= (1<<0); // 最后一位设置为1
GPIOC_ODR |= (1<<15); // 倒数第15位设置为1
GPIOA_ODR |= (1<<0); // 最后一位设置为1
while(1){
GPIOB_ODR &= ~(1<<0); // 点灯1
Delay_ms(1000000);
GPIOB_ODR |= (1<<0); // 灭灯1
Delay_ms(1000000);
GPIOC_ODR &= ~(1<<15); // 点灯2
Delay_ms(1000000);
GPIOC_ODR |= (1<<15); // 灭灯2
Delay_ms(1000000);
GPIOA_ODR &= ~(1<<0); // 点灯3
Delay_ms(1000000);
GPIOA_ODR |= (1<<0); // 灭灯3
Delay_ms(1000000);
}
}
void SystemInit(){
}
进行编译并生成了LED.hex文件
将面包板和USB连接到电脑,打开mcuisp,上传HEX文件到stm32f103c8t6上并烧录:
7.最终结果演示
三、总结
本次流水灯实验难度相对较大,对于我来说比较困难的几个点在于:电路的搭建和端口识别问题,但这些问题都在同学和论坛大佬的帮助下一一解决了。随着学的东西越高级,我就越能体会到基础的重要,不能害怕BUG,因为错,才会有经验。我应该学会和同学讨论,交流出新知,利用网上的教学视频和博客。通过本实验的学习,我对C语言调用函数参数的传递方式以及寄存器使用方法有了更加深刻的理解。
四、参考链接
STM32------寄存器映射和存储器映射 - crazyyang - 博客园
STM32最小核心板F103串口通信USART_vic_to_ry的博客-CSDN博客