/* * spreadspace avr utils * * * Copyright (C) 2013 Christian Pointner * Othmar Gsenger * * This file is part of spreadspace avr utils. * * spreadspace avr utils is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * spreadspace avr utils is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with spreadspace avr utils. If not, see . */ #include "statemachine.h" #include "stepper.h" #include "eventqueue.h" #include "limits.h" #include "ledmatrix.h" #include typedef enum state_enum {reset, closed, closing, opened, opening, timeout_after_open, timeout_after_close, error, manual_movement} state_t; state_t state = reset; void change_state(state_t new_state) { if (new_state == state) return; printf("state: %d\n\r",new_state); switch(new_state) { case reset: break; case closed: ledmatrix(red); break; case closing: ledmatrix(red_moving); start_stepper(dir_close); break; case opened: ledmatrix(green); break; case opening: ledmatrix(green_moving); start_stepper(dir_open); break; case timeout_after_open: ledmatrix(green_blink); break; case timeout_after_close: ledmatrix(red_blink); break; case error: ledmatrix(rg_blink); break; case manual_movement: ledmatrix(rg_moving); break; break; } state = new_state; } void statemachine_task_limits(void) { limits_t limits = limits_get(); if (limits == both) return change_state(error); switch(state) { case reset: switch(limits) { case open: return change_state(opened); case close: return change_state(closed); default: return change_state(closing); } case error: case closed: case opened: switch(limits) { case open: return change_state(opened); case close: return change_state(closed); default: return change_state(manual_movement); } case manual_movement: case timeout_after_open: case timeout_after_close: switch(limits) { case open: return change_state(opened); case close: return change_state(closed); default: return; } case closing: if(limits == close) return change_state(closed); break; case opening: if(limits == open) return change_state(opened); break; } } void statemachine_task_event(void) { if (state == closing || state == opening) return; //waiting for action to finish event_t event = eventqueue_pop(); switch(state) { case closing: case opening: //Not possibe, fuck gcc case reset: case error: printf("Error: Not accepting commands in state %d\n\r",state); break; // Not accepting commands case manual_movement: case timeout_after_open: case timeout_after_close: case closed: case opened: switch(event) { case none: return; case cmd_open: return change_state(opening); case cmd_close: return change_state(closing); case cmd_toggle: case btn_toggle: case card: return change_state( (state==closed || state == timeout_after_close) ? opening: closing); } } } void statemachine_task(void) { statemachine_task_limits(); statemachine_task_event(); }