IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> STM32学习笔记3——GPIO的输入引脚 -> 正文阅读

[嵌入式]STM32学习笔记3——GPIO的输入引脚


写在前面

果然我还要学习很多东西,记录今天怎么解决上次加按键就不能正常运作的问题。


一、代码对比

1、原先代码

根据库函数版本下意识写出来的寄存器版:

void Key_GPIO_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	//PC
	RCC->APB2ENR |= (1<<4);
	GPIOC->CRL &= 0xFFFFFF0F;//PC1
	GPIOC->CRL |= 0x00000080;
	GPIOC->CRH &= 0xFF0FFFFF;//PC13
	GPIOC->CRH |= 0x00800000;
	

	//PA
	RCC->APB2ENR |= (1<<2);
	GPIOA->CRL &= 0xFFFFFFF0;//PA0
	GPIOA->CRL |= 0x00000008;
}

2、最终代码

一步步纠错修改后的程序

void Key_GPIO_Config(void)
{
	//PC
	RCC->APB2ENR |= (1<<4);
	GPIOC->CRL &= 0xFFFFFF0F;
	GPIOC->CRL |= 0x00000080;
	GPIOC->BSRR |= (1<<1);//set

	GPIOC->CRH &= 0xFF0FFFFF;
	GPIOC->CRH |= 0x00800000;
	GPIOC->BSRR |= (1<<13);//set
	

	//PA
	RCC->APB2ENR |= (1<<2);
	GPIOA->CRL &= 0xFFFFFFF0;
	GPIOA->CRL |= 0x00000008;
	GPIOC->BRR |= (1<<0);//reset
}

二、经验总结

1、怎么定位和修改代码的?

逐行注释,逐行修改,逐渐定位到问题点。

void Key_GPIO_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	//PC 第一次修改的使能时钟语句,编译运行,没有任何问题
	//RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
	RCC->APB2ENR |= (1<<4);
	
	/*   第二次修改的是PC口的引脚配置 PC两个按键不起作用
	GPIO_InitStructure.GPIO_Pin |= GPIO_Pin_1|GPIO_Pin_13; 
	GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;
	
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	*/
	GPIOC->CRL &= 0xFFFFFF0F;
	GPIOC->CRL |= 0x00000080;

	GPIOC->CRH &= 0xFF0FFFFF;
	GPIOC->CRH |= 0x00800000;
	

	//PA  第一次修改的使能时钟语句,编译运行,没有任何问题
	//RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC->APB2ENR |= (1<<2);
	
	/* 第三次修改的是PA口的引脚配置 PA口的按键正常运行,PC口按键不起作用
	GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;
	GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;
	
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	*/
	GPIOA->CRL &= 0xFFFFFFF0;
	GPIOA->CRL |= 0x00000008;
}

第一次修改的使能时钟语句,编译运行,没有任何问题。
第二次修改的是PC口的引脚配置 PC口两个按键不起作用。
第三次修改的是PA口的引脚配置 PA口的按键正常运行,PC口按键不起作用。
在检查了C口引脚没有定义错误的情况下,可以定位出我在配置输入口时有问题,转到GPIO_Init这个函数定义看我是不是少了什么。

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;
  /* Check the parameters */
  assert_param(IS_GPIO_ALL_PERIPH(GPIOx));
  assert_param(IS_GPIO_MODE(GPIO_InitStruct->GPIO_Mode));
  assert_param(IS_GPIO_PIN(GPIO_InitStruct->GPIO_Pin));  
  
/*---------------------------- GPIO Mode Configuration -----------------------*/
  currentmode = ((uint32_t)GPIO_InitStruct->GPIO_Mode) & ((uint32_t)0x0F);
  if ((((uint32_t)GPIO_InitStruct->GPIO_Mode) & ((uint32_t)0x10)) != 0x00)
  { 
    /* Check the parameters */
    assert_param(IS_GPIO_SPEED(GPIO_InitStruct->GPIO_Speed));
    /* Output mode */
    currentmode |= (uint32_t)GPIO_InitStruct->GPIO_Speed;
  }
/*---------------------------- GPIO CRL Configuration ------------------------*/
  /* Configure the eight low port pins */
  if (((uint32_t)GPIO_InitStruct->GPIO_Pin & ((uint32_t)0x00FF)) != 0x00)
  {
    tmpreg = GPIOx->CRL;
    for (pinpos = 0x00; pinpos < 0x08; pinpos++)
    {
      pos = ((uint32_t)0x01) << pinpos;
      /* Get the port pins position */
      currentpin = (GPIO_InitStruct->GPIO_Pin) & pos;
      if (currentpin == pos)
      {
        pos = pinpos << 2;
        /* Clear the corresponding low control register bits */
        pinmask = ((uint32_t)0x0F) << pos;
        tmpreg &= ~pinmask;
        /* Write the mode configuration in the corresponding bits */
        tmpreg |= (currentmode << pos);
        /* Reset the corresponding ODR bit */
        if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
        {
          GPIOx->BRR = (((uint32_t)0x01) << pinpos);
        }
        else
        {
          /* Set the corresponding ODR bit */
          if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
          {
            GPIOx->BSRR = (((uint32_t)0x01) << pinpos);
          }
        }
      }
    }
    GPIOx->CRL = tmpreg;
  }
/*---------------------------- GPIO CRH Configuration ------------------------*/
  /* Configure the eight high port pins */
  if (GPIO_InitStruct->GPIO_Pin > 0x00FF)
  {
    tmpreg = GPIOx->CRH;
    for (pinpos = 0x00; pinpos < 0x08; pinpos++)
    {
      pos = (((uint32_t)0x01) << (pinpos + 0x08));
      /* Get the port pins position */
      currentpin = ((GPIO_InitStruct->GPIO_Pin) & pos);
      if (currentpin == pos)
      {
        pos = pinpos << 2;
        /* Clear the corresponding high control register bits */
        pinmask = ((uint32_t)0x0F) << pos;
        tmpreg &= ~pinmask;
        /* Write the mode configuration in the corresponding bits */
        tmpreg |= (currentmode << pos);
        /* Reset the corresponding ODR bit */
        if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
        {
          GPIOx->BRR = (((uint32_t)0x01) << (pinpos + 0x08));
        }
        /* Set the corresponding ODR bit */
        if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
        {
          GPIOx->BSRR = (((uint32_t)0x01) << (pinpos + 0x08));
        }
      }
    }
    GPIOx->CRH = tmpreg;
  }
}

我比较笨,如果是稍微聪明一点的话,很容易定位出来是上下拉的问题。我带入了PC1,需要配置成上拉输入,一步步代入,定位到这里的:

	/* Reset the corresponding ODR bit */
    if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
    {
       GPIOx->BRR = (((uint32_t)0x01) << pinpos);
    }
    else
    {
     /* Set the corresponding ODR bit */
     if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
     {
        GPIOx->BSRR = (((uint32_t)0x01) << pinpos);
     }

因为PC1和PC13平时都是高电平,按下为低电平,所以需要配置为上拉模式,但是GPIO的ODR寄存器的复位值是0,根据这个代码,是需要置位ODR对应引脚位。
于是程序添加这两行代码:

	//PC
	GPIOC->BSRR |= (1<<1);//set
	GPIOC->BSRR |= (1<<13);//set

为了严谨,PA0的下拉模式配置也添加了复位代码。

	GPIOC->BRR |= (1<<0);//reset

2、为什么上拉输入引脚要置位?

1)什么是上拉输入、下拉输入?
这个一搜就有答案。

2)为什么上拉输入引脚配置完要置位?
关于STM32单片机GPIO口上拉与下拉输入
我现在这水平还不能完全理解作者说的什么线与不线与啥的,但是根据“当设置上拉输入时,将其输出值设置为0,这样电平就被直接拉低了,按键的接地电路基本就不起作用了”这一句话。
我就能很能明白我聪明不到哪去。平时按键不按下为高电平,不拉高永远检测不出来什么时候按下。

写在后面

感觉自己写这些总结的时候还是过于啰嗦了,就像前辈说的不要全都总结,只总结重点的,不要把精力花在不重要的事情上。要提高效率才行。

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-08-06 09:59:50  更:2021-08-06 10:00:04 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 18:47:50-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码