c arduino 按钮中断(isr),使多个 LED 点亮和熄灭
c arduino button interrupts (isr), make multiple leds go on and off
我一直在尝试通过按下相应的按钮让我的 Arduino 上的 LED 亮起和熄灭。
我正在使用中断来实现它,按钮按下确实被注册了,但由于某种原因它没有改变全局变量的值(int button_pressed1,...);
应该发生的是,当我按下按钮 1 时,Led 1 应该点亮和熄灭,与按钮 2 和按钮 3 相同。
我真的很感谢你看一看,中断对我来说很新,所以它可能是一个小疏忽。 <3
*我省略了按钮 2 和 3 的代码。如果我可以让 LED 在按钮 1 上打开,我就能让它们为其他按钮打开。
#include <util/delay.h>
#include <avr/io.h>
#include <avr/interrupt.h>
#include "usart.h"
#define LED_DDR DDRB
#define LED_PORT PORTB
#define BUTTON_DDR DDRC
#define BUTTON_PORT PORTC
#define BUTTON_PIN PINC
int button_pressed1 = 0; //globale variabele to turn on functions
ISR(PCINT1_vect)
{
if (bit_is_clear(BUTTON_PIN, PC1))
{
_delay_us(500); //debounce
if (bit_is_clear(BUTTON_PIN, PC1))
{
button_pressed1 = 1;
printf("button 1 pressed\n");
}
}
}
int main()
{
LED_DDR |= _BV(PB2) | _BV(PB3) | _BV(PB4); //registrer pins output(bit = 1)
LED_PORT |= _BV(PB2) | _BV(PB3) | _BV(PB4);
BUTTON_DDR &= ~_BV(PC1) & ~_BV(PC2) & ~_BV(PC3); //registrer inputs(bit = 0)
BUTTON_PORT |= _BV(PC1) | _BV(PC2) | _BV(PC3); // pull up ( bit =1 )
PCICR |= _BV(PCIE1); //type pins doorgeven
PCMSK1 |= _BV(PC1) | _BV(PC2) | _BV(PC3); //pin/button doorgeven aan change mask
initUSART();
sei();
while (1)
{ //infinte loop
if (button_pressed1 == 1)
{
LED_PORT &= ~_BV(PB2); //turn led on
_delay_ms(500);
LED_PORT |= _BV(PB2); //turn led off
_delay_ms(500);
}
}
return 0;
}
几个基本问题:
- 与 ISR 共享的所有变量都应声明
volatile
并防止竞争条件。有关详细信息,请参阅 this。
- 你不应该在 ISR 中有 busy-delays。相反,您应该将定时器中断设置为在特定时间段内再次触发。通常,很难专门针对按钮使用 GPIO 中断,最好从循环定时器中断进行轮询。您可以使用中断,但它相当复杂,详情here。
- 500us 的时间可能不足以 de-bounce。机械信号反弹相对较慢。等待 ~10 毫秒左右更为常见。您可以很容易地用示波器测量反弹特性,方法是向按钮添加电源,然后按下它并捕获边缘。它看起来像正弦曲线,您可以轻松测量它会持续多长时间。
- 主循环中的 500 毫秒延迟将被用户体验为“滞后”。他们可能会开始怀疑按钮坏了。您可能希望至少跳过关闭延迟。
我一直在尝试通过按下相应的按钮让我的 Arduino 上的 LED 亮起和熄灭。
我正在使用中断来实现它,按钮按下确实被注册了,但由于某种原因它没有改变全局变量的值(int button_pressed1,...);
应该发生的是,当我按下按钮 1 时,Led 1 应该点亮和熄灭,与按钮 2 和按钮 3 相同。
我真的很感谢你看一看,中断对我来说很新,所以它可能是一个小疏忽。 <3
*我省略了按钮 2 和 3 的代码。如果我可以让 LED 在按钮 1 上打开,我就能让它们为其他按钮打开。
#include <util/delay.h>
#include <avr/io.h>
#include <avr/interrupt.h>
#include "usart.h"
#define LED_DDR DDRB
#define LED_PORT PORTB
#define BUTTON_DDR DDRC
#define BUTTON_PORT PORTC
#define BUTTON_PIN PINC
int button_pressed1 = 0; //globale variabele to turn on functions
ISR(PCINT1_vect)
{
if (bit_is_clear(BUTTON_PIN, PC1))
{
_delay_us(500); //debounce
if (bit_is_clear(BUTTON_PIN, PC1))
{
button_pressed1 = 1;
printf("button 1 pressed\n");
}
}
}
int main()
{
LED_DDR |= _BV(PB2) | _BV(PB3) | _BV(PB4); //registrer pins output(bit = 1)
LED_PORT |= _BV(PB2) | _BV(PB3) | _BV(PB4);
BUTTON_DDR &= ~_BV(PC1) & ~_BV(PC2) & ~_BV(PC3); //registrer inputs(bit = 0)
BUTTON_PORT |= _BV(PC1) | _BV(PC2) | _BV(PC3); // pull up ( bit =1 )
PCICR |= _BV(PCIE1); //type pins doorgeven
PCMSK1 |= _BV(PC1) | _BV(PC2) | _BV(PC3); //pin/button doorgeven aan change mask
initUSART();
sei();
while (1)
{ //infinte loop
if (button_pressed1 == 1)
{
LED_PORT &= ~_BV(PB2); //turn led on
_delay_ms(500);
LED_PORT |= _BV(PB2); //turn led off
_delay_ms(500);
}
}
return 0;
}
几个基本问题:
- 与 ISR 共享的所有变量都应声明
volatile
并防止竞争条件。有关详细信息,请参阅 this。 - 你不应该在 ISR 中有 busy-delays。相反,您应该将定时器中断设置为在特定时间段内再次触发。通常,很难专门针对按钮使用 GPIO 中断,最好从循环定时器中断进行轮询。您可以使用中断,但它相当复杂,详情here。
- 500us 的时间可能不足以 de-bounce。机械信号反弹相对较慢。等待 ~10 毫秒左右更为常见。您可以很容易地用示波器测量反弹特性,方法是向按钮添加电源,然后按下它并捕获边缘。它看起来像正弦曲线,您可以轻松测量它会持续多长时间。
- 主循环中的 500 毫秒延迟将被用户体验为“滞后”。他们可能会开始怀疑按钮坏了。您可能希望至少跳过关闭延迟。