STM32 学习2 库函数控制GPIO输出

news/2024/7/24 5:28:19 标签: stm32, 学习, 嵌入式硬件

STM32 学习2 库函数控制GPIO输出

  • 一、GPIO寄存器介绍
    • 1. GPIO简介
    • 2. GPIO功能
      • (1)模式分类
      • (2)模式设置方法
        • MODE[1:0]:模式控制,用于配置端口引脚的模式:
        • CNF[1:0]:配置引脚输出速度:
        • PUPD[1:0]:上拉/下拉配置:
      • (3)**输入模式**:
      • (4)**输出模式**:
    • 3. GPIO的寄存器
      • (1)两个32位配置寄存器
      • (2)两个32位数据寄存器
      • (3)一个32位置位/复位寄存器:GPIOx_BSRR
      • (4)一个16位复位寄存器:GPIOx_BRR
      • (5)一个32位锁定寄存器:GPIOx_LCKR
    • 4. 时钟控制器
  • 二、GPIO配置
    • 1. 配置内容
    • 2. 配置结构体
      • (1)GPIO_Pin:
      • (2)Mode:
      • (3)Speed:
      • (4)OType:
      • (5)PuPd:
  • 三、GPIO使用步骤
  • 四、代码示例
    • 1. 跑马灯显示
    • 2. 显示数字

在这里插入图片描述

一、GPIO寄存器介绍

1. GPIO简介

GPIO是STM32微控制器上的一组引脚,可配置为输入或输出模式,用于与外部设备进行数字信号交换。每个GPIO引脚都有一个唯一的编号,通常称为引脚名称或引脚号码。

STM32系列微控制器通常具有多个GPIO引脚,数量取决于具体型号。

GPIO模块是STM32微控制器中非常重要的功能之一,它允许微控制器与外部设备进行数字信号交换。通过正确配置和使用GPIO,可以实现各种应用,包括传感器接口、控制输出设备等。

IO端口的基本作用

2. GPIO功能

(1)模式分类

GPIO端口的每个位可以由软件分别配置成多种模式:

  • 输入浮空
  • 输入上拉
  • 输入下拉
  • 开漏输出
  • 推挽式输出
  • 推挽式复用功能
  • 开漏复用功能

(2)模式设置方法

通过设置: GPIOx_CRL/CRH,具体有:

MODE[1:0]:模式控制,用于配置端口引脚的模式:
  • 00:输入模式
  • 01:输出模式
  • 10:复用功能模式
  • 11:模拟输入模式
CNF[1:0]:配置引脚输出速度:
  • 00:低速
  • 01:中速
  • 10:高速
  • 11:最高速
PUPD[1:0]:上拉/下拉配置:
  • 00:无上拉/下拉
  • 01:上拉
  • 10:下拉
  • 11:保留

这些值一般在使用寄存器编程需要用到,使用库函数时直接使用系统定义的宏来设置,一般定义如下:

  • GPIO_Mode_IN:输入模式,用于将GPIO引脚配置为输入模式。
  • GPIO_Mode_OUT:输出模式,用于将GPIO引脚配置为输出模式。
  • GPIO_Mode_AF:复用模式,用于将GPIO引脚配置为复用模式,通常用于连接外设。
  • GPIO_Mode_AN:模拟模式,用于将GPIO引脚配置为模拟模式,用于ADC输入等。

(3)输入模式

GPIO引脚可以用于读取外部设备的数字信号。在输入模式下,引脚可以被连接到传感器、开关、按钮等外部设备,并且微控制器可以读取这些设备的状态(高电平或低电平)。
下图是I/O端口位的输入设置:
在这里插入图片描述

(4)输出模式

GPIO引脚可以用于向外部设备发送数字信号。在输出模式下,微控制器可以控制引脚的状态,输出高电平或低电平信号,从而驱动LED、继电器、马达等外部设备。
输出IO端口位配置:
在这里插入图片描述

3. GPIO的寄存器

在STM32系列微控制器中,GPIO的配置和控制通过一系列特定的寄存器完成。这些寄存器提供了对GPIO引脚的各种配置选项和控制功能。
每个GPIO端口有:

(1)两个32位配置寄存器

控制端口引脚的模式、输出速度和上拉/下拉配置。

  • GPIOx_CRL
  • GPIOx_CRH,

(2)两个32位数据寄存器

  • GPIOx_IDR:输入数据寄存器,读取端口引脚的电平状态。
  • GPIOx_ODR:输出数据寄存器,设置端口引脚的电平状态。

(3)一个32位置位/复位寄存器:GPIOx_BSRR

用于设置或复位端口引脚的电平。

(4)一个16位复位寄存器:GPIOx_BRR

用于复位端口引脚的电平。

(5)一个32位锁定寄存器:GPIOx_LCKR

锁定寄存器,用于锁定GPIO端口的配置。

4. 时钟控制器

在STM32微控制器中,每个GPIO端口都需要单独打开时钟以激活其功能。时钟控制器(Clock Controller)是用来控制微控制器内部各个模块的时钟信号的,包括GPIO模块。通过打开相应GPIO端口的时钟,可以使得该端口的GPIO模块开始正常工作。

二、GPIO配置

1. 配置内容

在STM32中,使用寄存器对GPIO进行配置和控制。常见的GPIO配置包括以下几个步骤:

  1. 选择引脚模式:确定引脚是作为输入还是输出。这通常涉及设置相应的寄存器位。

  2. 配置引脚速率:根据需求选择引脚的输出速率,以适应外部设备的要求。速率通常包括低速、中速、高速等选项。

  3. 配置引脚上下拉:根据需要启用或禁用引脚的上拉或下拉电阻,以确保在未连接外部设备时引脚的稳定状态。

  4. 配置引脚中断(可选):如果需要,可以配置引脚触发中断,以便在引脚状态发生变化时触发微控制器的中断服务程序。

2. 配置结构体

GPIO配置结构体是一个用于配置GPIO引脚参数的数据结构,面是一个典型的GPIO配置结构体的说明:

typedef struct {
    uint32_t GPIO_Pin;      // GPIO引脚编号,可以是单个引脚或者多个引脚的组合,如 GPIO_Pin_0 | GPIO_Pin_1
    GPIOMode_TypeDef Mode;  // GPIO引脚的工作模式,包括输入模式、输出模式、复用模式等
    GPIOSpeed_TypeDef Speed; // GPIO引脚的输出速率,通常有低速、中速、高速可选
    GPIOOType_TypeDef OType; // GPIO引脚的输出类型,通常有推挽输出和开漏输出两种类型
    GPIOPuPd_TypeDef PuPd;   // GPIO引脚的上拉/下拉电阻状态,可以选择无上拉/下拉、上拉、下拉
} GPIO_InitTypeDef;

初始化:

GPIO_InitTypeDef GPIO_InitStruct;

(1)GPIO_Pin:

表示需要配置的GPIO引脚的编号,可以使用GPIO_Pin_X的形式表示单个引脚,也可以使用按位或操作将多个引脚组合在一起,例如 GPIO_Pin_0 | GPIO_Pin_1 表示同时配置GPIO引脚0和1。

(2)Mode:

表示GPIO引脚的工作模式,包括输入模式(GPIO_Mode_IN)、输出模式(GPIO_Mode_OUT)、复用模式(GPIO_Mode_AF)等。根据具体应用需求选择合适的工作模式。
示例:

GPIO_InitStruct.Mode = GPIO_Mode_IN; // 输入模式
GPIO_InitStruct.Mode = GPIO_Mode_OUT; // 输出模式
GPIO_InitStruct.Mode = GPIO_Mode_AF; // 复用模式

如果是输出模式,还需要选择引脚的类型,通常有推挽输出和开漏输出两种类型。

GPIO_InitStruct.Mode = GPIO_Mode_OUT; // 输出模式
GPIO_InitStruct.Mode = GPIO_Mode_Out_PP; // 推挽输出
GPIO_InitStruct.Mode = GPIO_Mode_Out_OD; // 开漏输出

(3)Speed:

表示GPIO引脚的输出速率,通常有低速(GPIO_Speed_2MHz)、中速(GPIO_Speed_50MHz)、高速(GPIO_Speed_100MHz)可选。选择合适的输出速率可以满足不同应用的需求。
示例:

GPIO_InitStruct.Speed = GPIO_Speed_2MHz; // 2MHz输出速率
GPIO_InitStruct.Speed = GPIO_Speed_50MHz; // 50MHz输出速率
GPIO_InitStruct.Speed = GPIO_Speed_100MHz; // 100MHz输出速率

(4)OType:

表示GPIO引脚的输出类型,包括推挽输出(GPIO_OType_PP)和开漏输出(GPIO_OType_OD)两种类型。推挽输出适用于驱动电平信号,而开漏输出适用于驱动开关或者双向总线。

(5)PuPd:

表示GPIO引脚的上拉/下拉电阻状态,可以选择无上拉/下拉(GPIO_PuPd_NOPULL)、上拉(GPIO_PuPd_UP)、下拉(GPIO_PuPd_DOWN)。根据实际连接情况选择合适的上拉/下拉电阻状态。
示例:

GPIO_InitStruct.Pull = GPIO_PuPd_NOPULL; // 无上拉/下拉
GPIO_InitStruct.Pull = GPIO_PuPd_UP; // 上拉
GPIO_InitStruct.Pull = GPIO_PuPd_DOWN; // 下拉

配置完成后,可以使用GPIO_Init()函数将配置应用到GPIO引脚上:

GPIO_Init(GPIOA, &GPIO_InitStruct); // 将配置应用到 GPIOA 端口上

三、GPIO使用步骤

以下是使用STM32的GPIO的一般步骤:

  1. 打开GPIO端口时钟;
  2. 初始化GPIO模块:在程序开始时,需要初始化GPIO模块,配置所需的引脚为输入或输出,并设置其他相关参数;
  3. 读取输入引脚状态:如果引脚配置为输入模式,可以使用相应的函数或指令来读取引脚的状态,判断外部设备的状态;
  4. 设置输出引脚状态:如果引脚配置为输出模式,可以使用相应的函数或指令来设置引脚的状态,向外部设备发送所需的信号;
  5. 处理中断(可选):如果配置了中断,需要编写相应的中断服务程序来处理引脚状态变化时的事件。

四、代码示例

1. 跑马灯显示

实验使用普中开发板,LED电路如下:
在这里插入图片描述
LED1的阴极连接在芯片的26号引脚PC0(GPIO端口C的第0管脚)上,当PC0输出低电平时,发光二极管正向导通,D1被点亮。因此,对PC0进行操作即可控制LED1。

#include "stm32f10x_lib.h" //包含所有的STM32F10x库的头文件
#include <stdio.h>

// GPIO配置函数
void GPIO_Configuration(void)
{
    GPIO_InitTypeDef GPIO_InitStructure; //定义GPIO初始化结构体

    // 开启GPIOC的时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

    // 设置GPIOC的模式为推挽输出
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All; //选择所有的pin
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置输出速度为50MHz
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设置为推挽输出模式
    GPIO_Init(GPIOC, &GPIO_InitStructure); //初始化GPIOC
}

// 延时函数
void delay(u32 i)
{
    while (i--) //当i不为0时,持续减1,实现延时
        ;
}

// 打开指定位置的LED
void on(int position)
{
    // GPIOC->BSRR = (1 << (16 + position)); //通过设置BSRR寄存器,关闭指定位置的LED
	GPIO_ResetBits(GPIOC, GPIO_Pin_0 << position); //
}

// 关闭指定位置的LED
void off(int position)
{
    // GPIOC->BSRR = (1 << (position)); //通过设置BSRR寄存器,打开指定位置的LED
	GPIO_SetBits(GPIOC, GPIO_Pin_0 << position);
}

// 主函数
int main(void)
{
    GPIO_Configuration(); //调用GPIO配置函数

    int j;

    //GPIOC->BRR = GPIO_Pin_0; //关闭0号位的LED
		GPIO_SetBits(GPIOC, GPIO_Pin_All);
    while (1) //无限循环
    {
        for (j = 0; j < 8; j++) //遍历0到7号位
        {
            on(j); //打开j号位的LED
            delay(0xfffff); //延时
            off(j); //关闭j号位的LED
            delay(0xfffff); //延时
        }
    }
}

可以设置断点,打开GPIOA的查看窗口查看运行状态 :
在这里插入图片描述

2. 显示数字

#include "stm32f10x_lib.h" //包含所有的STM32F10x库的头文件
#include <stdio.h>

int arr[10][7] = {
	{0,1,2,3,4,5},
	{1,2},
	{0,1,6,4,3},
	{0,1,6,2,3},
	{5,6,1,2},
	{0,5,6,2,3},
	{0,5,6,4,3,2},
	{0,1,2},
	{0,1,2,3,4,5,6},
	{0,1,2,3,5,6}
};

// GPIO配置函数
void GPIO_Configuration(void)
{
    GPIO_InitTypeDef GPIO_InitStructure; //定义GPIO初始化结构体

    // 开启GPIOC的时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

    // 设置GPIOC的模式为推挽输出
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All; //选择所有的pin
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置输出速度为50MHz
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设置为推挽输出模式
    GPIO_Init(GPIOC, &GPIO_InitStructure); //初始化GPIOC
}

// 延时函数
void delay(u32 i)
{
    while (i--) //当i不为0时,持续减1,实现延时
        ;
}

// 打开指定位置的LED
void on(int position)
{
	GPIO_ResetBits(GPIOC, GPIO_Pin_0 << position); //
}

// 关闭指定位置的LED
void off(int position)
{
	GPIO_SetBits(GPIOC, GPIO_Pin_0 << position);
}
void onArray(int array[], int len){
	int j;
	for(j=0;j<len;j++){
		on(array[j]);
	}
}
int count(int n){
	int len = 1;
	for (int i = 1; i < 7; i++) {
			if (arr[n][i] != '\0') {
					len++;
			} else {
					break;
			}
	}	
return len;
}
void lightn(int n){

	// 计算onArray 第二个参数,即二维数组第二维长度
	int len = count(n);
	onArray(arr[n], len);
}

void allOff(){
	GPIO_SetBits(GPIOC, GPIO_Pin_All);
}
// 主函数
int main(void)
{
    GPIO_Configuration(); //调用GPIO配置函数

    int j;

	GPIO_SetBits(GPIOC, GPIO_Pin_All);
    while (1) //无限循环
    {
       
	   for(int i=0;i<10;i++){
		   lightn(i);
		   delay(0xfffff);
		   allOff();
		   delay(0xff);
	   }
    }
}

本文代码开源地址:
https://gitee.com/xundh/stm32_arm_learn


http://www.niftyadmin.cn/n/5381318.html

相关文章

260.【华为OD机试真题】信道分配(贪心算法-JavaPythonC++JS实现)

🚀点击这里可直接跳转到本专栏,可查阅顶置最新的华为OD机试宝典~ 本专栏所有题目均包含优质解题思路,高质量解题代码(Java&Python&C++&JS分别实现),详细代码讲解,助你深入学习,深度掌握! 文章目录 一. 题目-信道分配二.解题思路三.题解代码Python题解代码…

14个常见的Java课程设计/毕业设计合集(源码+文档)

从网上整理收集了14个常见的java系统设计源码&#xff0c;可以用于课程作业或者毕业设计。 1.基于java的家政预约网站系统 平台采用B/S结构&#xff0c;后端采用主流的Springboot框架进行开发&#xff0c;前端采用主流的Vue.js进行开发。 整个平台包括前台和后台两个部分。 …

【软考高级信息系统项目管理师--第十二章:项目质量管理】

&#x1f680; 作者 &#xff1a;“码上有前” &#x1f680; 文章简介 &#xff1a;软考高级–信息系统项目管理师 &#x1f680; 欢迎小伙伴们 点赞&#x1f44d;、收藏⭐、留言&#x1f4ac; 第十二章&#xff1a;项目质量管理 质量管理过程全面质量管理(TQM )质量管理计划质…

深入理解Spring Boot Starter:概念、特点、场景、原理及自定义starter

这是目录 **一、引言****二、Spring Boot Starter基本概念****三、Spring Boot Starter的主要特点****四、Spring Boot Starter的应用场景****五、Spring Boot Starter的实现原理****六、自定义spring boot starter****为什么要创建自定义Starter&#xff1f;****创建自定义Spr…

ktutil编写生成keytab文件的脚本、通过keytab文件认证用户

文章目录 1. 生成keytab文件脚本2. 通过keytab文件认证3. 查看认证的用户4. 失效认证的用户 1. 生成keytab文件脚本 生成keytab文件的脚本 vim generate_kb.sh #!/usr/bin/bash ktutil <<EOF add_entry -password -p $1 -k 1 -e arcfour-hmac $2 write_kt $3 EOF示例&am…

error: src refspec main does not match any解决办法

一、问题描述&#xff1a; 用GitHub Actions自动部署Hexo&#xff0c;到了最关键的一步&#xff1b;突然报错&#xff1a;error: src refspec main does not match any 1、错误一&#xff1a; main分支应填写为master分支&#xff1b;但是只改这里也会报其他错误 2、错误二&a…

蓝桥杯:C++模运算、快速幂

模运算 模运算是大数运算中的常用操作。如果一个数太大&#xff0c;无法直接输出&#xff0c;或者不需要直接输出&#xff0c;则可以对它取模&#xff0c;缩小数值再输出。取模可以防止溢出&#xff0c;这是常见的操作。 模是英文mod的音译&#xff0c;取模实际上是求余。 取…

腾讯云OSS文件上传功能

腾讯云COS介绍 腾讯云COS&#xff08;Cloud Object Storage&#xff09;是一种基于对象的存储服务&#xff0c;用于存储和管理海量的非结构化数据&#xff0c;如图片、音视频文件、备份数据等。它具有以下特点和优势&#xff1a; 高可靠性&#xff1a;采用分布式存储架构&…