#include <avr/io.h>
#include <avr/interrupt.h>
#include <avr/sleep.h>
+#include <util/delay.h>
#include "lights.h"
static uint16_t button_start[N_BUTTONS];
static unsigned char button_pressed[N_BUTTONS];
-
-static unsigned char sleep_after_release;
+static unsigned char just_waked_up;
void init_buttons()
{
+ unsigned char i;
+
DDRB &= ~(_BV(PB4) | _BV(PB6));
PORTB |= _BV(PB4) | _BV(PB6);
- sleep_after_release = 0;
- // log_byte(PORTB);
-}
+ for (i=0; i < N_BUTTONS; i++) {
+ button_start[i] = 0;
+ button_pressed[i] = 0;
+ }
-static void inline long_press(unsigned char n)
-{
- sleep_after_release = 1;
- gpio_set(1, 1);
+ just_waked_up = 1;
+
+ // log_byte(PORTB);
}
static void do_sleep()
{
- // led_set_status(2);
- // MCUCR |= _BV(ISC00); // any edge generates IRQ
log_byte(0xb0);
log_flush();
- set_sleep_mode(SLEEP_MODE_PWR_DOWN);
- PORTA = 0;
- PORTB = 0;
- DDRA = 0;
- DDRB = 0;
- GIMSK |= _BV(INT0); // enable INT0
- sleep_enable();
- sei();
- sleep_cpu();
- sleep_disable();
- cli();
- GIMSK &= ~_BV(INT0); // disable INT0
- hw_setup();
+
+ hw_suspend();
+ gpio_before_poweroff(); // Set the status LED on again
+
+ while((PINB & _BV(PB6)) == 0)
+ ; // wait for button release
+
+ _delay_ms(100);
+
+ susp_gpio(); // disable the status LED
+
+ power_down();
+}
+
+static void inline long_press(unsigned char n)
+{
+ if (n == 0)
+ do_sleep();
}
static void inline short_press(unsigned char n)
{
- led_set_status(1);
+ if (n == 0)
+ gpio_set(2,1);
}
void timer_check_buttons()
pinb & _BV(PB6),
pinb & _BV(PB4),
};
+
for (i = 0; i < N_BUTTONS; i++) {
if (!port_states[i]) { // is pressed
+ if (i == 0 && just_waked_up)
+ continue;
if (button_pressed[i] == 0) {
// begin of button press
button_pressed[i] = 1;
button_start[i] = jiffies;
#if 0
- // log_byte(0xC0);
- // log_word(jiffies);
+ log_byte(0xC0);
+ log_word(jiffies);
#endif
} else if (button_pressed[i] == 1) {
// been already pressed
}
}
} else { // is not pressed
+ if (i == 0)
+ just_waked_up = 0;
if (button_pressed[i]) { // just depressed
uint16_t duration = jiffies - button_start[i];
#if 1
button_pressed[i] = 0;
if (duration > 6 && duration < 30)
short_press(i);
- if (sleep_after_release)
- do_sleep();
}
}
}
}
-ISR(INT0_vect)
-{
- hw_setup();
- log_byte(0xbb);
- log_flush();
-}
-
/* adc.c */
void init_adc();
+void susp_adc();
void timer_start_adcs();
/* pwm.c */
void init_pwm();
+void susp_pwm();
void pwm_off(unsigned char n);
void pwm_set(unsigned char n, uint16_t stride);
/* tmr.c */
extern volatile uint16_t jiffies;
void init_tmr();
+void susp_tmr();
/* pwmled.c */
void pwmled_init();
/* gpio.c */
void gpio_init();
+void susp_gpio();
void gpio_set(unsigned char n, unsigned char on);
+void gpio_before_poweroff();
/* ambient.c */
void ambient_init();
void init_buttons();
void timer_check_buttons();
-/* init.c */
+/* main.c */
void hw_setup();
+void hw_suspend();
+void power_down();
#endif /* !LIGHTS_H__ */
#include <util/delay.h>
#include <avr/sleep.h>
#include <avr/interrupt.h>
+#include <avr/power.h>
#include <avr/wdt.h>
#include "lights.h"
void hw_setup()
{
+ wdt_enable(WDTO_1S);
init_pwm();
init_adc();
init_tmr();
set_sleep_mode(SLEEP_MODE_IDLE);
}
-int main(void)
+void hw_suspend()
{
- log_init();
- wdt_enable(WDTO_1S);
+ susp_pwm();
+ susp_adc();
+ susp_tmr();
+ susp_gpio();
+ wdt_disable();
+}
+
+
+void power_down()
+{
+ // enable PCINT14, so that user can wake up later
+ GIMSK |= _BV(PCIE1);
+ PCMSK1 |= _BV(PCINT14);
+
+ // G'night
+ set_sleep_mode(SLEEP_MODE_PWR_DOWN);
+ sleep_enable();
+ sleep_bod_disable();
+ sei();
+ sleep_cpu();
+
+ // G'morning
+ cli();
+
+ sleep_disable();
- _delay_ms(100);
+ // Disable PCINT14
+ GIMSK &= ~_BV(PCIE1);
+ PCMSK1 &= ~_BV(PCINT14);
hw_setup();
+}
+
+
+ISR(PCINT_vect)
+{
+ GIMSK &= ~_BV(PCIE1);
+ PCMSK1 &= ~_BV(PCINT14);
+}
+
+int main(void)
+{
+ log_init();
+ power_usi_disable(); // Once for lifetime
log_set_state(3);
+ hw_setup();
+ hw_suspend();
+ power_down();
+
sei();
#if 1
while (1) {