STM32编程:实现LED灯闪烁(基于手写SDK的方式)

项目结构

在这里插入图片描述

stm32f10x.h 文件

//寄存器的值常常是芯片外设自动更改的,即使CPU没有执行程序,也有可能发生变化
//编译器有可能会对没有执行程序的变量进行优化

//volatile表示易变的变量,防止编译器优化,
#define     __IO    volatile
typedef unsigned int uint32_t;
typedef unsigned short uint16_t;

// GPIO 寄存器结构体定义
typedef struct
{
    __IO uint32_t CRL;       // 端口配置低寄存器,     地址偏移0X00
    __IO uint32_t CRH;       // 端口配置高寄存器,     地址偏移0X04
    __IO uint32_t IDR;       // 端口数据输入寄存器,   地址偏移0X08
    __IO uint32_t ODR;       // 端口数据输出寄存器,   地址偏移0X0C
    __IO uint32_t BSRR;      // 端口位设置/清除寄存器,地址偏移0X10
    __IO uint32_t BRR;       // 端口位清除寄存器,     地址偏移0X14
    __IO uint32_t LCKR;      // 端口配置锁定寄存器,   地址偏移0X18
} GPIO_TypeDef;


/*片上外设基地址  */
#define PERIPH_BASE           ((unsigned int)0x40000000)

/*APB2 总线基地址 */
#define APB2PERIPH_BASE       (PERIPH_BASE + 0x10000)
/* AHB总线基地址 */
#define AHBPERIPH_BASE        (PERIPH_BASE + 0x20000)

/*GPIO外设基地址*/
#define GPIOA_BASE            (APB2PERIPH_BASE + 0x0800)
#define GPIOB_BASE            (APB2PERIPH_BASE + 0x0C00)
#define GPIOC_BASE            (APB2PERIPH_BASE + 0x1000)
#define GPIOD_BASE            (APB2PERIPH_BASE + 0x1400)
#define GPIOE_BASE            (APB2PERIPH_BASE + 0x1800)
#define GPIOF_BASE            (APB2PERIPH_BASE + 0x1C00)
#define GPIOG_BASE            (APB2PERIPH_BASE + 0x2000)

/*RCC外设基地址*/
#define RCC_BASE      (AHBPERIPH_BASE + 0x1000)


// GPIO 外设声明
#define GPIOA               ((GPIO_TypeDef *) GPIOA_BASE)
#define GPIOB               ((GPIO_TypeDef *) GPIOB_BASE)
#define GPIOC               ((GPIO_TypeDef *) GPIOC_BASE)
#define GPIOD               ((GPIO_TypeDef *) GPIOD_BASE)
#define GPIOE               ((GPIO_TypeDef *) GPIOE_BASE)
#define GPIOF               ((GPIO_TypeDef *) GPIOF_BASE)
#define GPIOG               ((GPIO_TypeDef *) GPIOG_BASE)


// RCC 外设声明
#define RCC                 ((RCC_TypeDef *) RCC_BASE)

/*RCC的AHB1时钟使能寄存器地址,强制转换成指针*/
#define RCC_APB2ENR      *(unsigned int*)(RCC_BASE+0x18)
	







/*GPIO引脚号定义*/
#define GPIO_Pin_0              ((uint16_t)0x0001)  /*!< 选择Pin0 (1<<0) */
#define GPIO_Pin_1              ((uint16_t)0x0002)  /*!< 选择Pin1 (1<<1)*/
#define GPIO_Pin_2              ((uint16_t)0x0004)  /*!< 选择Pin2 (1<<2)*/
#define GPIO_Pin_3              ((uint16_t)0x0008)  /*!< 选择Pin3 (1<<3)*/
#define GPIO_Pin_4              ((uint16_t)0x0010)  /*!< 选择Pin4 */
#define GPIO_Pin_5              ((uint16_t)0x0020)  /*!< 选择Pin5 */
#define GPIO_Pin_6              ((uint16_t)0x0040)  /*!< 选择Pin6 */
#define GPIO_Pin_7              ((uint16_t)0x0080)  /*!< 选择Pin7 */
#define GPIO_Pin_8              ((uint16_t)0x0100)  /*!< 选择Pin8 */
#define GPIO_Pin_9              ((uint16_t)0x0200)  /*!< 选择Pin9 */
#define GPIO_Pin_10             ((uint16_t)0x0400)  /*!< 选择Pin10 */
#define GPIO_Pin_11             ((uint16_t)0x0800)  /*!< 选择Pin11 */
#define GPIO_Pin_12             ((uint16_t)0x1000)  /*!< 选择Pin12 */
#define GPIO_Pin_13             ((uint16_t)0x2000)  /*!< 选择Pin13 */
#define GPIO_Pin_14             ((uint16_t)0x4000)  /*!< 选择Pin14 */
#define GPIO_Pin_15             ((uint16_t)0x8000)  /*!< 选择Pin15 */
#define GPIO_Pin_All            ((uint16_t)0xFFFF)  /*!< 选择全部引脚 */



	

/**
* GPIO输出速率枚举定义
*/
typedef enum
{
    GPIO_Speed_10MHz = 1,         // 10MHZ        (01)b
    GPIO_Speed_2MHz,              // 2MHZ         (10)b
    GPIO_Speed_50MHz              // 50MHZ        (11)b
} GPIOSpeed_TypeDef;

/**
* GPIO工作模式枚举定义
*/
typedef enum
{
    GPIO_Mode_AIN = 0x0,           // 模拟输入     (0000 0000)b
    GPIO_Mode_IN_FLOATING = 0x04,  // 浮空输入     (0000 0100)b
    GPIO_Mode_IPD = 0x28,          // 下拉输入     (0010 1000)b
    GPIO_Mode_IPU = 0x48,          // 上拉输入     (0100 1000)b

    GPIO_Mode_Out_OD = 0x14,       // 开漏输出     (0001 0100)b
    GPIO_Mode_Out_PP = 0x10,       // 推挽输出     (0001 0000)b
    GPIO_Mode_AF_OD = 0x1C,        // 复用开漏输出  (0001 1100)b
    GPIO_Mode_AF_PP = 0x18         // 复用推挽输出  (0001 1000)b
} GPIOMode_TypeDef;



/**
* GPIO初始化结构体类型定义
*/
typedef struct
{
    uint16_t GPIO_Pin;             /*!< 选择要配置的GPIO引脚
                                    可输入 GPIO_Pin_ 定义的宏 */

    GPIOSpeed_TypeDef GPIO_Speed;  /*!< 选择GPIO引脚的速率
                                    可输入 GPIOSpeed_TypeDef 定义的枚举值 */

    GPIOMode_TypeDef GPIO_Mode;    /*!< 选择GPIO引脚的工作模式
                                    可输入 GPIOMode_TypeDef 定义的枚举值 */
} GPIO_InitTypeDef;


void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);

void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);

void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct);

stm32f10x.c 文件



#include "stm32f10x.h"


/**
*函数功能:设置引脚为低电平
*参数说明:GPIOx:该参数为GPIO_TypeDef类型的指针,指向GPIO端口的地址
*        GPIO_Pin:选择要设置的GPIO端口引脚,可输入宏GPIO_Pin_0-15,
*                 表示GPIOx端口的0-15号引脚。
*/

void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{
    /*设置GPIOx端口BRR寄存器的第GPIO_Pin位,使其输出低电平*/
    /*因为BRR寄存器写0不影响,
    宏GPIO_Pin只是对应位为1,其它位均为0,所以可以直接赋值*/

    GPIOx->BRR = GPIO_Pin;
}



/**
*函数功能:设置引脚为高电平
*参数说明:GPIOx:该参数为GPIO_TypeDef类型的指针,指向GPIO端口的地址
*        GPIO_Pin:选择要设置的GPIO端口引脚,可输入宏GPIO_Pin_0-15,
*                 表示GPIOx端口的0-15号引脚。
*/
void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{
    /*设置GPIOx端口BSRR寄存器的第GPIO_Pin位,使其输出高电平*/
    /*因为BSRR寄存器写0不影响,
    宏GPIO_Pin只是对应位为1,其它位均为0,所以可以直接赋值*/

    GPIOx->BSRR = GPIO_Pin;
}


/**
*函数功能:初始化引脚模式
*参数说明:GPIOx,该参数为GPIO_TypeDef类型的指针,指向GPIO端口的地址
*         GPIO_InitTypeDef:GPIO_InitTypeDef结构体指针,指向初始化变量
*/
void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct)
{
    uint32_t currentmode =0x00,currentpin = 0x00,pinpos = 0x00,pos = 0x00;
    uint32_t tmpreg = 0x00, pinmask = 0x00;

    /*---------------- GPIO 模式配置 -------------------*/
    // 把输入参数GPIO_Mode的低四位暂存在currentmode
    currentmode = ((uint32_t)GPIO_InitStruct->GPIO_Mode) &
                ((uint32_t)0x0F);

    // bit4是1表示输出,bit4是0则是输入
    // 判断bit4是1还是0,即首选判断是输入还是输出模式
    if ((((uint32_t)GPIO_InitStruct->GPIO_Mode) &
            ((uint32_t)0x10)) != 0x00)
    {
        // 输出模式则要设置输出速度
        currentmode |= (uint32_t)GPIO_InitStruct->GPIO_Speed;
    }
    /*-----GPIO CRL 寄存器配置 CRL寄存器控制着低8位IO- ----*/
    // 配置端口低8位,即Pin0~Pin7
    if (((uint32_t)GPIO_InitStruct->GPIO_Pin &
            ((uint32_t)0x00FF)) != 0x00)
    {
        // 先备份CRL寄存器的值
        tmpreg = GPIOx->CRL;

        // 循环,从Pin0开始配对,找出具体的Pin
        for (pinpos = 0x00; pinpos < 0x08; pinpos++)
        {
            // pos的值为1左移pinpos位
            pos = ((uint32_t)0x01) << pinpos;

            // 令pos与输入参数GPIO_PIN作位与运算
            currentpin = (GPIO_InitStruct->GPIO_Pin) & pos;

            //若currentpin=pos,则找到使用的引脚
            if (currentpin == pos)
            {
                //pinpos的值左移两位(乘以4),因为寄存器中4个位配置一个引脚
                pos = pinpos << 2;
                //把控制这个引脚的4个寄存器位清零,其它寄存器位不变
                pinmask = ((uint32_t)0x0F) << pos;
                tmpreg &= ~pinmask;

                // 向寄存器写入将要配置的引脚的模式
                tmpreg |= (currentmode << pos);

                // 判断是否为下拉输入模式
                if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
                {
                    // 下拉输入模式,引脚默认置0,对BRR寄存器写1对引脚置0
                    GPIOx->BRR = (((uint32_t)0x01) << pinpos);
                }
                else
                {
                    // 判断是否为上拉输入模式
                    if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
                    {
                        // 上拉输入模式,引脚默认值为1,对BSRR寄存器写1对引脚置1
                        GPIOx->BSRR = (((uint32_t)0x01) << pinpos);
                    }
                }
            }
        }
        // 把前面处理后的暂存值写入到CRL寄存器之中
        GPIOx->CRL = tmpreg;
    }
    /*--------GPIO CRH 寄存器配置 CRH寄存器控制着高8位IO- -----*/
    // 配置端口高8位,即Pin8~Pin15
    if (GPIO_InitStruct->GPIO_Pin > 0x00FF)
    {
        // // 先备份CRH寄存器的值
        tmpreg = GPIOx->CRH;

        // 循环,从Pin8开始配对,找出具体的Pin
        for (pinpos = 0x00; pinpos < 0x08; pinpos++)
        {
            pos = (((uint32_t)0x01) << (pinpos + 0x08));

            // pos与输入参数GPIO_PIN作位与运算
            currentpin = ((GPIO_InitStruct->GPIO_Pin) & pos);

            //若currentpin=pos,则找到使用的引脚
            if (currentpin == pos)
            {
                //pinpos的值左移两位(乘以4),因为寄存器中4个位配置一个引脚
                pos = pinpos << 2;

                //把控制这个引脚的4个寄存器位清零,其它寄存器位不变
                pinmask = ((uint32_t)0x0F) << pos;
                tmpreg &= ~pinmask;

                // 向寄存器写入将要配置的引脚的模式
                tmpreg |= (currentmode << pos);

                // 判断是否为下拉输入模式
                if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
                {
                    // 下拉输入模式,引脚默认置0,对BRR寄存器写1可对引脚置0
                    GPIOx->BRR = (((uint32_t)0x01) << (pinpos + 0x08));
                }
                // 判断是否为上拉输入模式
                if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
                {
                    // 上拉输入模式,引脚默认值为1,对BSRR寄存器写1可对引脚置1
                    GPIOx->BSRR = (((uint32_t)0x01) << (pinpos + 0x08));
                }
            }
        }
        // 把前面处理后的暂存值写入到CRH寄存器之中
        GPIOx->CRH = tmpreg;
    }
}

main.c 文件



#ifndef STM32F10X // 防止重复引入报错
#define STM32F10X
#endif 


#include "stm32f10x.h"

// 函数为空,目的是为了骗过编译器不报错
void SystemInit(void)
{
}

void Delay(__IO uint32_t nCount)     //简单的延时函数
{
    for (; nCount != 0; nCount--);
}

// 使用固件库点亮LED
int main(void)
{
    // 定义一个GPIO_InitTypeDef类型的结构体
    GPIO_InitTypeDef GPIO_InitStructure;

    // 开启GPIO端口时钟
    RCC_APB2ENR |= (1<<3);

    // 选择要控制的GPIO引脚
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;

    // 设置引脚模式为通用推挽输出
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

    // 设置引脚速率为50MHz
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    // 调用库函数,初始化GPIO引脚
    GPIO_Init(GPIOB, &GPIO_InitStructure);

	
		// ======PB0口闪烁======
    // 使引脚输出低电平,点亮LED1
    GPIO_ResetBits(GPIOB,GPIO_Pin_0);

    while (1)
    {
        // 使引脚输出低电平,点亮LED
        GPIO_ResetBits(GPIOB,GPIO_Pin_0);

        /*延时一段时间*/
        Delay(0xFFFF);

        /*使引脚输出高电平,关闭LED1*/
        GPIO_SetBits(GPIOB,GPIO_Pin_0);

        /*延时一段时间*/
        Delay(0xFFFF);
    }
}



LED灯一头接入PB0,一头接入GND,收录之后,即可实现闪烁效果。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/693291.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2024-6-9 石群电路-27

2024-6-9&#xff0c;星期日&#xff0c;12:49&#xff0c;天气&#xff1a;晴&#xff0c;心情&#xff1a;晴。Hello&#xff0c;大家&#xff0c;我回来啦&#xff0c;昨天断更了一天&#xff0c;是为什么捏&#xff0c;是因为&#xff0c;我通过毕业答辩啦&#xff01;&…

数据挖掘分析的一点进步分享

import pandas as pd import matplotlib.pyplot as plt import numpy as npdata pd.read_csv(heros.csv,encoding"gbk") data.head() 导入数据集 进行分析 df_datadata.copy() df_data.describe()df_data.info() df_data.drop(英雄,axis1,inplaceTrue) df_data[最…

[C++初阶]string类的详解

一、string类的模拟实现 上面已经对string类进行了简单的介绍&#xff0c;大家只要能够正常使用即可。在面试中&#xff0c;面试官总喜欢让我们来模拟实现string类&#xff0c;最主要是实现string类的构造、拷贝构造、赋值运算符重载以及析构函数。大家看下以下string类的实现是…

Flink 问题之 No Watermark (Watermarks are only available if EventTime is used)

问题背景 Flink-1.17.0在集群下&#xff0c;获取Kafka集群数据&#xff0c;进行流模式实时计算&#xff0c;Watermarks提示&#xff1a;No Watermark (Watermarks are only available if EventTime is used)&#xff1b; source数据源是从kafka中读取topic数据&#xff0c;经…

【最新鸿蒙应用开发】——ArkUI两种开发范式

在进行鸿蒙应用开发&#xff0c;openHarmony提供了一种页面开发框架叫做ArkUI方舟框架来进行页面布局的开发。 ArkUI方舟UI框架 针对不同的应用场景及技术背景&#xff0c;方舟UI框架提供了两种开发范式&#xff0c;分别是基于ArkTS的声明式开发范式&#xff08;简称“声明式…

分享美好,高清无阻 - 直播极简联网解决方案

1、需求背景 随着移动互联网、UGC模式和直播平台的发展&#xff0c;网络直播的门槛日益降低&#xff0c;越来越多的人希望成为直播的主角。基于物联网的户外直播无线联网解决方案应运而生&#xff0c;满足直播者的需求。 户外直播无线联网解决方案提供了无处不在的直播体验&a…

UniAnimate:华科提出人类跳舞视频生成新框架,支持合成一分钟高清视频

节前&#xff0c;我们星球组织了一场算法岗技术&面试讨论会&#xff0c;邀请了一些互联网大厂朋友、参加社招和校招面试的同学。 针对算法岗技术趋势、大模型落地项目经验分享、新手如何入门算法岗、该如何准备、面试常考点分享等热门话题进行了深入的讨论。 合集&#x…

Java 环境配置 -- Java 语言的安装、配置、编译与运行

大家好&#xff0c;我是栗筝i&#xff0c;这篇文章是我的 “栗筝i 的 Java 技术栈” 专栏的第 002 篇文章&#xff0c;在 “栗筝i 的 Java 技术栈” 这个专栏中我会持续为大家更新 Java 技术相关全套技术栈内容。专栏的主要目标是已经有一定 Java 开发经验&#xff0c;并希望进…

未来AI大模型的发展趋势

大家好&#xff0c;我是小悟 未来AI大模型的发展趋势无疑将是多元化、高效化、普及化以及人性化。随着技术的飞速进步&#xff0c;AI大模型将在各个领域中展现出更加广泛和深入的应用&#xff0c;成为推动社会进步的重要力量。 多元化是AI大模型发展的重要方向。随着数据量的…

vue26:vue的环境搭建

vue环境安装配置 在点击上方链接前&#xff0c;注意&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 下方的红字&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&am…

在 Qt Creator 上创建 ROS 项目并新建/导入 ROS 包

0、引言 ⚠️ 在开始之前&#xff0c;您需要确保您已经为 Qt 配置好了 ROS 开发环境了。如果您还没有配置好&#xff0c;可以参考这篇文章 本文将着手探讨如何在 Qt Creator 上编辑 ROS 项目&#xff08;工作空间&#xff09;。 1、本教程使用到的相关软件或产品 Ubuntu 20.0…

【AI基础】第四步:保姆喂饭级-langchain+chatglm2-6b+m3e-base

在第三步手动安装chatglm2-6b时&#xff0c;已经可以通过web进行交互。langchain重新封装了一下AI框架&#xff0c;提供更加友好的开发功能&#xff0c;类似于AI届的spring框架。langchain的安装过程也类似于上一步说的&#xff1a;【AI基础】第三步&#xff1a;纯天然手动安装…

github异常问题总结

问题1&#xff1a; gitgitlab.gz.cvte.cn: Permission denied (publickey). fatal: Could not read from remote repository.Please make sure you have the correct access rights and the repository exists.解决方法&#xff1a; 这个错误表示 GitLab 服务器拒绝了你的 SSH…

【MySQL】(基础篇二) —— MySQL初始用

MySQL初始用 目录 MySQL初始用基本语法约定选择数据库查看数据库和表其它的SHOW 在Navicat中&#xff0c;大部分数据库管理相关的操作都可以通过图形界面完成&#xff0c;这个很简单&#xff0c;大家可以自行探索。虽然Navicat等图形化数据库管理工具为操作和管理数据库提供了非…

【小白专用24.6.8】c#异步方法 async task调用及 await运行机制

await是C#中用于等待异步操作完成的关键字。它通常用于异步方法内部&#xff0c;使得在等待异步操作期间&#xff0c;线程可以继续执行其他操作&#xff0c;从而保持程序的响应性。 在使用await时&#xff0c;需要注意以下几点&#xff1a; 1. async修饰符&#xff1a; 使用…

PgSQL技术内幕 - psql与服务端连接与交互机制

PgSQL技术内幕 - 客户端psql与服务端连接与交互机制 简单来说&#xff0c;PgSQL的psql客户端向服务端发起连接请求&#xff0c;服务端接收到请求后&#xff0c;fork出一个子进程&#xff0c;之后由该子进程和客户端进行交互&#xff0c;处理客户端的SQL等&#xff0c;并将结果返…

用HAL库改写江科大的stm32入门-输入捕获原理图示

原理与接线&#xff1a; &#xff08;输入捕获的结构&#xff09; cubeMx: PA11&#xff1a;

【复现】含能量路由器的交直流混合配电网潮流计算

目录 1 主要内容 2 理论及模型 3 程序结果 4 下载链接 1 主要内容 程序复现《含能量路由器的交直流混合配电网潮流计算》&#xff0c;主要是对算例4.1进行建模分析&#xff0c;理论和方法按照文献所述。能量路由器&#xff08;ER&#xff09;作为新兴的电力元器件&#xff…

应用matplotlib.animation.FuncAnimation绘制摆线

上次尝试了用matplotlib.animation.ArtistAnimation绘制摆线&#xff0c;实际上也可以用matplotlib.animation.FuncAnimation实现同样的功能。 导入相关文件 引用的库包括numpy&#xff0c;matplotlib&#xff0c;代码如下&#xff1a; import numpy as np import matplotli…

字符串形成树形

字符串形成树形 有的时候我们形成树形不是以ID的关系进行匹配的而是以字符串进行形成 数据 CREATE TABLE `contract_main_org_info` (`id` bigint(20) NOT NULL COMMENT 组织单位id,`parent_id` int(11) NULL DEFAULT NULL COMMENT 父组织单位id,`org_name` varchar(255) CHA…