4.3、使用寄存器版本点亮LED灯(内附代码)

发布者:Enchanted2021最新更新时间:2022-04-21 来源: eefocus关键字:寄存器版本  点亮  LED灯 手机看文章 扫描二维码
随时随地手机看文章

第一种:指针形式


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闪烁来了解对寄存器进行具体的操作,使自己能够用寄存器来进行项目的开发,其他的外设配置是一样的原理。

关键字:寄存器版本  点亮  LED灯 引用地址:4.3、使用寄存器版本点亮LED灯(内附代码)

上一篇:STM32的DMA基本原理及实现过程
下一篇:教你如何使用STM32控制无源蜂鸣器发声播放音乐

小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved