使用STM32 再实现电动车防盗

项目需求

点击遥控器 A 按键,系统进入警戒模式,一旦检测到震动(小偷偷车),则喇叭发出声响报警,
吓退小偷。
点击遥控器 B 按键,系统退出警戒模式,再怎么摇晃系统都不会报警,否则系统一直发出尖叫,
让车主尴尬。

项目框图

模块介绍

振动传感器

产生震动时,会输出低电平,绿色指示灯亮(开关信号指示灯)

继电器模块

(VCC =3.3V)当IN输入低电平时,ON 和 COM会导通,模块上的绿灯会亮

遥控收发 433M

发送端按下ABCD会在接收端被芯片调制解调,最终解析成不同的信号来识别,接收端的D0, D1, D2, D3 对应A,B,C,D,收到哪个对应的信号,相应的引脚就会给出高电平

CubeMX配置

1.初始配置

2.震动传感器的DO口接到了PA4,所以将PA4配置成中断模式,且由于DO口拉低代表有震动,所以将模式改为下降沿触发

3.先只测试振动传感器模块

打开stm32f1xx_it.c -->EXTI4_IRQHandler(void)--> HAL_GPIO_EXTI_IRQHandler(vibrate_senior_Pin); --> HAL_GPIO_EXTI_Callback( )

HAL_GPIO_EXTI_Callback()就是中断处理程序,将他在main.c中重写:

其中weak是虚拟函数,重写的话会覆盖在此函数之上的

只测试振动传感器模块的

/*重写中断服务函数,如果检测到EXTI中断请求,则进入此函数*/
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
//	一根中断线上接多个中断源,判断中断源是否来自PA4
	if(GPIO_Pin == GPIO_PIN_4)
	{
		if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_4) == GPIO_PIN_RESET)
		//	如果坚持到PA4被拉低
		{
			//点亮LED1,否则关闭LED1
			 HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_RESET);
			HAL_Delay(500);
		}
		else
		{
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_SET);
		}
	}
	
}

PS:如果直接在中断服务函数里调用 HAL_Delay ()函数,则会造成系统卡死。 

原因:程序初始化时默认把滴答定时器的中断优先级设为最低,其它中断源很容易打断它导致卡死,也就是说当Delay函数在进行中的时候,如果又发生了震动触发了外部中断,就会又立刻执行这个中断处理程序,导致喇叭一直在响无法停下。

解决办法:

在 main 函数里使用以下函数提高滴答定时器的中断优先级(提升至0);

HAL_NVIC_SetPriority(SysTick_IRQn,0,0); //必须写在" SystemClock_Config() "后面!!!

并且在CubeMX中将 EXTI4 的中断优先级设置比滴答定时器的中断优先级低,比如 2

然后可以在CubeMX重新生成代码(generate code),Keil中此时会自动弹出这个对话框问是否要重载,选择“是”此时会将刚刚Cube中做出的修改反映到Keil中,并保留之前在Keil中自己写的代码!

 加上继电器和遥控模块

遥控模块:
按下遥控器 A 按键, LED1 亮LED2灭 ;按下遥控器 B 按键, LED2 亮LED1灭
D0 -- PA5,打开中断,进入警报模式
D1 -- PA6,打开中断,退出警报模式
继电器:IN--PB6
#define state_ON 1
#define state_OFF 0

/*重写中断服务函数,如果检测到EXTI中断请求,则进入此函数*/
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)//外部中断的处理函数,一根中断线上接多个中断源,判断中断源是否来自PA4,PA5,PA6
{
	static int status = state_OFF; //此处一定要加static,不然每次调用这个函数都会被赋值一遍
	
	/*switch(GPIO_Pin)
	{
		case GPIO_PIN_4:
			if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_4) == GPIO_PIN_RESET&& (status == state_ON))//如果检测到PA4被拉低
			{
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET);//继电器打开
			}
		break;
			// 如果检测到PA5被拉高(按键A按下),设定为警报模式
		case GPIO_PIN_5:
			if(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_5) == GPIO_PIN_SET)
			{
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_RESET);//LED1亮
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_9, GPIO_PIN_SET);//LED2灭
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET);
				HAL_Delay(1000);
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET);
				status = state_ON;// 将标志位设置为ON
			}
		break;
		// 如果检测到PA6被拉高(按键B按下),设定为关闭警报模式
		case GPIO_PIN_6:
			if(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_6) == GPIO_PIN_SET)
			{
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_9, GPIO_PIN_RESET);//LED2亮
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_SET);//LED1灭
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET);
				HAL_Delay(1000);
				HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET);
				status = state_OFF;// 将标志位设置为OFF
			}
		break;
	}*/

	
	if((GPIO_Pin == GPIO_PIN_4) && (status == state_ON))	//如果检测到PA4被拉低,且模式为警戒模式,则将P6拉低,继电器打开,喇叭响;没有检测到则不响
	{
		if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_4) == GPIO_PIN_RESET)//如果检测到PA4被拉低
		{
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET);//继电器打开
		}
	}
	
	if(GPIO_Pin == GPIO_PIN_5)//D0,PA5被拉高(按键A按下),进入警戒模式,LED1亮、LED2灭
	{
		if(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_5) == GPIO_PIN_SET){// 如果检测到PA5被拉高(按键A按下),设定为警报模式
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_RESET);//LED1亮
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_9, GPIO_PIN_SET);//LED2灭
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET);
			HAL_Delay(1000);
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET);
			status = state_ON;// 将标志位设置为ON
		}

	}
	
	if(GPIO_Pin == GPIO_PIN_6)//D1,对应按键B,按下解除警戒模式,LED2亮、LED1灭
	{
		if(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_6) == GPIO_PIN_SET){// 如果检测到PA6被拉高(按键B按下),设定为关闭警报模式
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_9, GPIO_PIN_RESET);//LED2亮
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8, GPIO_PIN_SET);//LED1灭
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET);
			HAL_Delay(1000);
			HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET);
			status = state_OFF;// 将标志位设置为OFF
		}
	
}

  • 13
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值