第一种:指针形式
led.c文件:
#include "led.h"
#include "stm32f4xx.h"
void LED_Init(void)
{
//1左移五位既是将第五位置为1,查寄存器可知1是使能0失能
RCC->AHB1ENR|= 1<<5;//使能RCC的AHB1时钟
//PF9 的GPIO配置
//将(1 1)左移18位再取反清零:2代表两位控制一IO,因为IO是F9,所以最后是~(3<<2*9),达到清零效果
GPIOF->MODER &= ~(3<<2*9);//配置为0用&,配置为1用 |
GPIOF->MODER |= 1<<(2*9);//1既是0 1,代表输出模式(查寄存器)
GPIOF->OSPEEDR &= ~(3<<2*9);//同理,清零
GPIOF->OSPEEDR |= 2<<(2*9);//2是1 0,将F9设置为50MHZ的速度
GPIOF->PUPDR &= ~(3<<2*9);//清零
GPIOF->PUPDR |=1<<(2*9);//上拉
GPIOF->OTYPER &= ~(1<<9);//清零,该寄存器是低16位有效,1位控制1个IO
GPIOF->OTYPER |=0<<9;//置0,既是输出推挽(复位)
GPIOF->ODR|= 1<<9;//1 控制高低电平
//GPIOF->ODR&=~(1<<9);//
//PF10 同理配置PF10
GPIOF->MODER &= ~(3<<2*10);
GPIOF->MODER |= 1<<(2*10);
GPIOF->OSPEEDR &= ~(3<<2*10);
GPIOF->OSPEEDR |= 2<<(2*10);
GPIOF->PUPDR &= ~(3<<2*10);
GPIOF->PUPDR |=1<<(2*10);
GPIOF->OTYPER &= ~(1<<10);
GPIOF->OTYPER |=0<<10;
GPIOF->ODR|= 1<<10;
}
///
led.h文件:
#ifndef __LED_H
#define __LED_H
void LED_Init(void);
#endif
///
main.c文件:
#include "stm32f4xx.h"
#include "led.h"
#include "delay.h"
int main(void)
{
delay_init(168);
LED_Init();
while(1)
{
GPIOF->ODR&=~(1<<9);
GPIOF->ODR&=~(1<<10);
delay_ms(500);
GPIOF->ODR |= 1<<9;
GPIOF->ODR |=1<<10;
delay_ms(500);
}
}
------------------------------------------------------------------------------------------------------------------
第二种方法:配置基地址开始
// 设置寄存器宏定义
//1、时钟线
#define RCC_AHB1ENR *(volatile unsigned long *)(0x40023800 + 0x30) // volatile 防止编译器优化 AHB1外设使能时钟线
#define GPIOF_MODER *(volatile unsigned long *)(0x40021400 + 0x00) // GPIOF的模式寄存器
#define GPIOF_OTYPER *(volatile unsigned long *)(0x40021400 + 0x04) // GPIOF的输出类型寄存器
#define GPIOF_OSPEEDR *(volatile unsigned long *)(0x40021400 + 0x08) // GPIOF的输出速度寄存器PUPDR
#define GPIOF_PUPDR *(volatile unsigned long *)(0x40021400 + 0x0c) // GPIOF的上下拉寄存器
#define GPIOF_ODR *(volatile unsigned long *)(0x40021400 + 0x14) // GPIOF的输出数据寄存器
#define GPIOF_BSRR *(volatile unsigned long *)(0x40021400 + 0x18) // GPIOF的复位/置位寄存器
#define GPIOE_MODER *(volatile unsigned long *)(0x40021000 + 0x00) // GPIOF的模式寄存器
#define GPIOE_OTYPER *(volatile unsigned long *)(0x40021000 + 0x04) // GPIOF的输出类型寄存器
#define GPIOE_OSPEEDR *(volatile unsigned long *)(0x40021000 + 0x08) // GPIOF的输出速度寄存器PUPDR
#define GPIOE_PUPDR *(volatile unsigned long *)(0x40021000 + 0x0c) // GPIOF的上下拉寄存器
#define GPIOE_ODR *(volatile unsigned long *)(0x40021000 + 0x14) // GPIOF的输出数据寄存器
#define GPIOE_BSRR *(volatile unsigned long *)(0x40021000 + 0x18) // GPIOF的复位/置位寄存器
// 大概时间的延时函数
void delay(int n)
{
while(n--)
{
int i = 1000;
while(i--);
}
}
// 初始化LED
int init_led(void)
{
// 2、初始化时钟、寄存器 第5位置1就是使能
RCC_AHB1ENR |= 1<<5;
//3、配置寄存器各种参数,LED只是输出,所以选择每个参数里面的输出模式
// ①初始化GPIOF为输出模式
//---------------模式--移动位数
GPIOF_MODER &= ~(0x3<<18); // 清空第18位和19位 11
GPIOF_MODER |= 1<<18; // 配置PF9
//GPIOF_MODER &= ~(0xF<<18); // 清空第18位和19位、20位、21位 1111
//GPIOF_MODER |= 0x5<<18; // 配置PF9、PF10
GPIOF_MODER &= ~(0x3<<20); // 清空第20位、21位 11
GPIOF_MODER |= 1<<20;//配置PF10
// ②配置GPIOF的输出类型为推挽
GPIOF_OTYPER &= ~(0x1<<9); // 清空第9位
GPIOF_OTYPER &= ~(0x1<<10); // 清空第10位
// ③配置GPIOF的输出速度
GPIOF_OSPEEDR &= ~(0x3<<18); // 清空第18位和19位 11
GPIOF_OSPEEDR &= ~(0x3<<20); // 清空第20 21 11
GPIOF_OSPEEDR |= 0x2<<18; // 50M速度
GPIOF_OSPEEDR |= 0x2<<20; // 50M速度
// ④配置 GPIOF为上拉输出
GPIOF_PUPDR &= ~(0x3<<18); // 清空第18位和19位 11
GPIOF_PUPDR |= 1<<18; // 配置PF9为上拉输出
GPIOF_PUPDR &= ~(0x3<<20); // 清空第20位和21位 11
GPIOF_PUPDR |= 1<<20; // 配置PF10为上拉输出
// ⑤输出数据:高电平 1 低电平 0
GPIOF_ODR |= 1<<9; // PF的第9个引脚输出高电平
GPIOF_ODR |= 1<<10; // PF的第10个引脚输出高电平
/*********************************PE脚**************************************/
// 初始化时钟 寄存器第4位置1就是使能
RCC_AHB1ENR |= 1<<4; //PE脚
// 初始化GPIOF为输出模式
GPIOE_MODER &= ~(0x3<<26); // 清空第26位和272829位 11
GPIOE_MODER |= 1<<26; // 配置PF9, 1010即是5,可同时配置
GPIOE_MODER &= ~(0x3<<28); // 清空第20位、21位 11
GPIOE_MODER |= 1<<28;//配置PF10
// 配置GPIOF的输出类型为推挽
GPIOE_OTYPER &= ~(0x1<<13); // 清空第9位
GPIOE_OTYPER &= ~(0x1<<14); // 清空第10位
// 配置GPIOF的输出速度
GPIOE_OSPEEDR &= ~(0x3<<26); // 清空第18位和19位 11
GPIOE_OSPEEDR &= ~(0x3<<28); // 清空第20 21 11
GPIOE_OSPEEDR |= 0x2<<26; // 50M速度
GPIOE_OSPEEDR |= 0x2<<28; // 50M速度
// 配置 GPIOF为上拉输出
GPIOE_PUPDR &= ~(0x3<<26); // 清空第18位和19位 11
GPIOE_PUPDR |= 1<<26; // 配置PF9为上拉输出
GPIOE_PUPDR &= ~(0x3<<28); // 清空第20位和21位 11
GPIOE_PUPDR |= 1<<28 ;// 配置PF10为上拉输出
// 输出数据:高电平 1 低电平 0
GPIOE_ODR |= 1<<13; // PF的第9个引脚输出高电平
GPIOE_ODR |= 1<<14; // PF的第10个引脚输出高电平
return 0;
}
int main(void)
{
// 初始化LED
init_led();
while (1)
{
// 点亮LED0
GPIOF_ODR &= ~(1<<9); // PF的第9个引脚输出低电平
delay(2000);
GPIOF_ODR |= 1<<9;
delay(2000);
// 输出数据:高电平 1 低电平 0
GPIOF_ODR &= ~(1<<10);
delay(2000);
GPIOF_ODR |= 1<<10; // PF的第9个引脚输出高电平
delay(2000);
GPIOE_ODR &= ~(1<<13);
delay(2000);
GPIOE_ODR |= 1<<13; // PE的第13、14个引脚输出高电平
delay(2000);
GPIOE_ODR &= ~(1<<14);
delay(2000);
GPIOE_ODR |= 1<<14;
delay(2000);
}
}
两者的区别在于一个需要配置基地址,一个直接用指针对寄存器进行操作,那怎样配置基地址,下面解释一下:
1、根据STM32F4xx中文参考手册的2.3节存储器的映射找到对应总线、外设的边界基地址,我这里只截了部分图,仅供参考
2、找到对应寄存器的偏移地址,在STM32F4xx中文参考手册7.4节
配置后如下所示:
2、最后对应寄存器进行对其位移操作即可(寄存器映射),可参考上面的代码
总结:这里以一个简单的LED闪烁来了解对寄存器进行具体的操作,使自己能够用寄存器来进行项目的开发,其他的外设配置是一样的原理。
设计资源 培训 开发板 精华推荐
- DC745A,用于 LTC2433-1、16 位高性能 DS 模数转换器的演示板
- 2023年服创大赛A10赛题:智能家居远程控制系统
- NCL30125FW300WGEVB:用于两个开关正向拓扑的固定频率电流模式 PWM 控制器
- K375s-104键盘pcb
- OM15020: 面向ZigBee的JN5169 USB Dongle
- LT6656ACS6-3.3、3.3V 微控制器电压基准和稳压器的典型应用
- LTC4162EUFD-FAD 9V 至 35V 2 节 3.2A 充电器的典型应用,具有 PowerPath 和 2A 输入限制
- LT3741/LT3741-1 的典型应用 - 高功率、恒流、恒压、降压型控制器
- C8051F580DK,C8051F588 8051 MCU 工业应用开发系统
- 微型蓝牙运动传感器