写在前面
果然我还要学习很多东西,记录今天怎么解决上次加按键就不能正常运作的问题。
一、代码对比
1、原先代码
根据库函数版本下意识写出来的寄存器版:
void Key_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC->APB2ENR |= (1<<4);
GPIOC->CRL &= 0xFFFFFF0F;
GPIOC->CRL |= 0x00000080;
GPIOC->CRH &= 0xFF0FFFFF;
GPIOC->CRH |= 0x00800000;
RCC->APB2ENR |= (1<<2);
GPIOA->CRL &= 0xFFFFFFF0;
GPIOA->CRL |= 0x00000008;
}
2、最终代码
一步步纠错修改后的程序
void Key_GPIO_Config(void)
{
RCC->APB2ENR |= (1<<4);
GPIOC->CRL &= 0xFFFFFF0F;
GPIOC->CRL |= 0x00000080;
GPIOC->BSRR |= (1<<1);
GPIOC->CRH &= 0xFF0FFFFF;
GPIOC->CRH |= 0x00800000;
GPIOC->BSRR |= (1<<13);
RCC->APB2ENR |= (1<<2);
GPIOA->CRL &= 0xFFFFFFF0;
GPIOA->CRL |= 0x00000008;
GPIOC->BRR |= (1<<0);
}
二、经验总结
1、怎么定位和修改代码的?
逐行注释,逐行修改,逐渐定位到问题点。
void Key_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC->APB2ENR |= (1<<4);
GPIOC->CRL &= 0xFFFFFF0F;
GPIOC->CRL |= 0x00000080;
GPIOC->CRH &= 0xFF0FFFFF;
GPIOC->CRH |= 0x00800000;
RCC->APB2ENR |= (1<<2);
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;
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));
currentmode = ((uint32_t)GPIO_InitStruct->GPIO_Mode) & ((uint32_t)0x0F);
if ((((uint32_t)GPIO_InitStruct->GPIO_Mode) & ((uint32_t)0x10)) != 0x00)
{
assert_param(IS_GPIO_SPEED(GPIO_InitStruct->GPIO_Speed));
currentmode |= (uint32_t)GPIO_InitStruct->GPIO_Speed;
}
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;
currentpin = (GPIO_InitStruct->GPIO_Pin) & pos;
if (currentpin == pos)
{
pos = pinpos << 2;
pinmask = ((uint32_t)0x0F) << pos;
tmpreg &= ~pinmask;
tmpreg |= (currentmode << pos);
if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
{
GPIOx->BRR = (((uint32_t)0x01) << pinpos);
}
else
{
if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
{
GPIOx->BSRR = (((uint32_t)0x01) << pinpos);
}
}
}
}
GPIOx->CRL = tmpreg;
}
if (GPIO_InitStruct->GPIO_Pin > 0x00FF)
{
tmpreg = GPIOx->CRH;
for (pinpos = 0x00; pinpos < 0x08; pinpos++)
{
pos = (((uint32_t)0x01) << (pinpos + 0x08));
currentpin = ((GPIO_InitStruct->GPIO_Pin) & pos);
if (currentpin == pos)
{
pos = pinpos << 2;
pinmask = ((uint32_t)0x0F) << pos;
tmpreg &= ~pinmask;
tmpreg |= (currentmode << pos);
if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
{
GPIOx->BRR = (((uint32_t)0x01) << (pinpos + 0x08));
}
if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
{
GPIOx->BSRR = (((uint32_t)0x01) << (pinpos + 0x08));
}
}
}
GPIOx->CRH = tmpreg;
}
}
我比较笨,如果是稍微聪明一点的话,很容易定位出来是上下拉的问题。我带入了PC1,需要配置成上拉输入,一步步代入,定位到这里的:
if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPD)
{
GPIOx->BRR = (((uint32_t)0x01) << pinpos);
}
else
{
if (GPIO_InitStruct->GPIO_Mode == GPIO_Mode_IPU)
{
GPIOx->BSRR = (((uint32_t)0x01) << pinpos);
}
因为PC1和PC13平时都是高电平,按下为低电平,所以需要配置为上拉模式,但是GPIO的ODR寄存器的复位值是0,根据这个代码,是需要置位ODR对应引脚位。 于是程序添加这两行代码:
GPIOC->BSRR |= (1<<1);
GPIOC->BSRR |= (1<<13);
为了严谨,PA0的下拉模式配置也添加了复位代码。
GPIOC->BRR |= (1<<0);
2、为什么上拉输入引脚要置位?
1)什么是上拉输入、下拉输入? 这个一搜就有答案。
2)为什么上拉输入引脚配置完要置位? 关于STM32单片机GPIO口上拉与下拉输入 我现在这水平还不能完全理解作者说的什么线与不线与啥的,但是根据“当设置上拉输入时,将其输出值设置为0,这样电平就被直接拉低了,按键的接地电路基本就不起作用了”这一句话。 我就能很能明白我聪明不到哪去。平时按键不按下为高电平,不拉高永远检测不出来什么时候按下。
写在后面
感觉自己写这些总结的时候还是过于啰嗦了,就像前辈说的不要全都总结,只总结重点的,不要把精力花在不重要的事情上。要提高效率才行。
|