stm32按键中断控制led灯的简单介绍
stm32开发板怎么用按键控制流水灯
开发板上4个LED是接到STM32的4个IO口上的,要点亮LED,只要在这4个IO上输出对应的高电平
那如何让STM32在PF6~PF9这4个IO上输出高点平呢
首先要看下《STM32F10xxx参考手册CD00171190.pdf》里关于IO端口的介绍,第8章 通用和复用功能I/O(GPIO和AFIO)
百为STM3210E-EVAL开发板上的CPU是STM32F103ZET6,有GPIOA~GPIOG七组IO,每组IO有16个引脚GPIO_Pin_0~GPIO_Pin_15,如板上的PF0~PF15
其中每个IO端口有2个32位的寄存器(GPIOx_CRL和GPIOx_CRH)配置,每个引脚由其中4位进行配置,
4位字段是由一个两位的配置字段和一个两位的模式字段组成
要设置这4位,要看下GPIOx_CRL,GPIOx_CRH的寄存器定义
由上图可以看出GPIOF_CRL的31~28位是设置PF7的,GPIOF_CRL的27~24位是设置PF6的
GPIOF_CRH的3~0位是设置PF8的,GPIOF_CRH的7~4位是设置PF6的
这里我们把每个引脚设置为
CNF[1:0] = 00:通用推挽输出模式
MODE[1:0] = 11 : 输出模式,最大速度50MH
GPIOF_CRL,GPIOF_CRH的寄存器地址分别是0x40011C00,0x40011C04
这个寄存器是怎么得来的,STM32数据手册里并没有直接给出,在《一个内幕者对STM32的介绍》4.1.1 寄存器地址查阅 这里有介绍怎么通过上面的偏移地址算出。
配置好,我们就可以直接在端口输出数据寄存器GPIOF_ODR输出高电平控制LED了,GPIOF_ODR有16位,每个位对应GPF的一个引脚,PF0~PF15
下面就可以编写LED流水灯程序了,激动啊,等了好久终于等到今天
/* main.c */
#define GPIOF_CRL (*(volatile unsigned int *)0x40011C00) //端口配置低寄存器
#define GPIOF_CRH (*(volatile unsigned int *)0x40011C04) //端口配置高寄存器
#define GPIOF_ODR (*(volatile unsigned int *)0x40011C0C) //端口输出数据寄存器
#define RCC_APB2ENR (*(volatile unsigned int *)0x40021018) //
void delay(void)
{
unsigned int i;
for(i=0;i500000;i++);
}
int main()
{
int i;
RCC_APB2ENR |= 17; //打开GPIOF的时钟
GPIOF_CRL = ( GPIOF_CRL 0x00ffffff ) | 0x33000000; //配置PF6,PF7为通用推挽输出模式,最大速度50MH
GPIOF_CRH = ( GPIOF_CRH 0xffffff00 ) | 0x00000033; //配置PF8,PF9为通用推挽输出模式,最大速度50MH
while(1)
{
for(i=6;i=9;i++)
{
GPIOF_ODR = ( GPIOF_ODR 0xfffffc3f ) | ( 1i ); //在PF6~PF9引脚轮流输出高电平
delay();
}
}
}
奋斗stm32 v5开发版按键控制led的问题
这个不是一两句话可以
讲清楚
的,你可以可以使用APP函数或者
库函数
完成开发,与
51单片机
一样,只是
STM32
需要先开启时钟线、设置端口输入还是输出模式,速度模式,以及初始化是高还是低状态,等设置好这些了,它就是一个51单片机一样的功能了。
STM32中断程序,按键控制灯的亮灭,程序运行没错,烧录到单片机就不行呢?哪位大神帮帮忙啊,不胜感谢
现在MDK在仿真试试,主要查看一下那些重要的寄存器有没有配置错误或者被其它代码修改等。其实仿真没有问题,就应该怀疑你自己的板子问题。以下可以参考的代码:
/*******************普中科技 **********************************
*
* STM32中断实验
*
* 实验目的: 掌握中断的配置
* 连接方法: 用排线或杜邦线分别连 JP10--JP1 JP11--JP5
* 实验现象: 当K7按下LED灯 再按时,LED灯会熄灭
*
*******************************************************************************/
#include "stm32f10x_lib.h"
/******************************** 变量定义 ------------------------------------*/
EXTI_InitTypeDef EXTI_InitStructure;
ErrorStatus HSEStartUpStatus;
/*********************************声明函数 -----------------------------------------------*/
void RCC_Configuration(void);
void GPIO_Configuration(void);
void NVIC_Configuration(void);
/*******************************************************************************
*
* 主函数
*
*******************************************************************************/
int main(void)
{
#ifdef DEBUG
debug();
#endif
RCC_Configuration(); //系统时钟配置
NVIC_Configuration(); //NVIC配置
GPIO_Configuration(); //配置GPIO
///*将EXTI线6连接到PB6*/
GPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource6);
/* Configure Key Button EXTI Line to generate an interrupt on falling edge */
//配置按钮中断线触发方式
EXTI_InitStructure.EXTI_Line = EXTI_Line6;
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //下降沿触发
EXTI_InitStructure.EXTI_LineCmd = ENABLE; //中断线使能
EXTI_Init(EXTI_InitStructure); //初始化中断
/* Generate software interrupt: simulate a falling edge applied on Key Button EXTI line */
EXTI_GenerateSWInterrupt(EXTI_Line6); //EXTI_Line6中断允许 到此中断配置完成,可以写中断处理函数。
while (1)
{
}
}
/*******************************************************************************
*
* RCC配置
*
*******************************************************************************/
void RCC_Configuration(void)
{
//复位RCC外部设备寄存器到默认值
RCC_DeInit();
//打开外部高速晶振
RCC_HSEConfig(RCC_HSE_ON);
//等待外部高速时钟准备好
HSEStartUpStatus = RCC_WaitForHSEStartUp();
//外部高速时钟已经准别好
if(HSEStartUpStatus == SUCCESS)
{
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
FLASH_SetLatency(FLASH_Latency_2);
//配置AHB(HCLK)时钟=SYSCLK
RCC_HCLKConfig(RCC_SYSCLK_Div1);
//配置APB2(PCLK2)钟=AHB时钟
RCC_PCLK2Config(RCC_HCLK_Div1);
//配置APB1(PCLK1)钟=AHB 1/2时钟
RCC_PCLK1Config(RCC_HCLK_Div2);
//配置ADC时钟=PCLK2 1/4
RCC_ADCCLKConfig(RCC_PCLK2_Div4);
//配置PLL时钟 == 外部高速晶体时钟*9
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
//配置ADC时钟= PCLK2/4
RCC_ADCCLKConfig(RCC_PCLK2_Div4);
//使能PLL时钟
RCC_PLLCmd(ENABLE);
//等待PLL时钟就绪
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)
{
}
//配置系统时钟 = PLL时钟
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
//检查PLL时钟是否作为系统时钟
while(RCC_GetSYSCLKSource() != 0x08)
{
}
}
/* Enable Key Button GPIO Port, GPIO_LED and AFIO clock */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_GPIOB
| RCC_APB2Periph_AFIO, ENABLE);
}
/*************************************************
函数: void GPIO_Config(void)
功能: GPIO配置
**************************************************/
void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输入
GPIO_Init(GPIOB, GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //配置浮空输入
GPIO_Init(GPIOB, GPIO_InitStructure);
}
/*******************************************************************************
* Function Name : NVIC_Configuration
* Description : Configure the nested vectored interrupt controller.
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void NVIC_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
#ifdef VECT_TAB_RAM
/* Set the Vector Table base location at 0x20000000 */
NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); //分配中断向量表
#else /* VECT_TAB_FLASH */
/* Set the Vector Table base location at 0x08000000 */
NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);
#endif
/* Configure one bit for preemption priority */
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1); //设置中断优先级
/* Enable the EXTI9_5 Interrupt */
NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel; //中断通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //强占优先级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//次优先级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //通道中断使能
NVIC_Init(NVIC_InitStructure);//初始化中断
}
#ifdef DEBUG
/*******************************************************************************
* Function Name : assert_failed
* Description : Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* Input : - file: pointer to the source file name
* - line: assert_param error line source number
* Output : None
* Return : None
*******************************************************************************/
void assert_failed(u8* file, u32 line)
{
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* Infinite loop */
while (1)
{
}
}
#endif
void EXTI9_5_IRQHandler(void)
{
if(EXTI_GetITStatus(EXTI_Line6) != RESET) //检测制定的EXTI线路触发请求是否发生。
{
/* Toggle GPIO_LED pin 7*/
GPIO_WriteBit(GPIOB, GPIO_Pin_8, (BitAction)((1-GPIO_ReadOutputDataBit(GPIOB, GPIO_Pin_8))));
/* Clear the Key Button EXTI line pending bit */
EXTI_ClearITPendingBit(EXTI_Line6); //清除EXTI线路挂起位
}
}
/******************* (C) COPYRIGHT 2008 STMicroelectronics *****END OF FILE****/
请教一个stm32程序:我写了一个按键控制LED灯翻转,调试成功的程序如下:
有问题的那个程序
u8 ReadValue=GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_10);
只在上面那条指令采集了一次按键数据。
等到执行到 while(!ReadValue);//等待按键被放开 这条指令时
数据仍然是上次采集到的那个值代表按键按下,(!ReadValue)这个值永远成立,所以无法代表按键断开。所以程序一直卡死在这条指令上。如果要通过就必须在判据里重新读取按键状态,像正确的程序那样用这样的指令
while(!GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_10));//等待按键被放开
另外因为你只读取了一次按键状态,所以下面那么多一堆软件消除抖动的程序白写了。
综上,正确的程序在每个判断的时刻都要重新读取下按键状态。
求大神指教怎么用stm32按键控制LED
把PC5-PC8都设置为输入,把PB0,1,14,15设置为输出。
在程序while中检测pc5-8的输入管脚电平,来使能输出PB管脚的高低电平来点亮LED。
逻辑很简单,希望你成功。
单片机STM32求助用3个按键控制3个LED灯
给你一个最简单的思路 CPU利用率不高但是可以完成
你上面写的我给你提供一个思路 你应该可以看的懂 看不懂追问
unsigned int key;
void main(void)
{
while(1)
{
switch(key)
{
case 0: LED0=!LED0; //你自己修改
delay_ms(200);
break;
case 1: LED1=!LED1;
delay_ms(200);
break;
case 2:LED2=!LED2;
delay_ms(200);
break;
default:
break;
}
void 在这里按键获取函数(void)
{
key=获取到的值; //没有按键按下的话是进入不到这里
}
}
}
关于stm32按键中断控制led灯和的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。微信号:ymsc_2016
相关文章
发表评论
评论列表
- 这篇文章还没有收到评论,赶紧来抢沙发吧~