2015-02-04 18:49:15 +03:00

292 lines
8.7 KiB
C

/*
* powerhw.c - functions to work with other power loads (shutter, heaters and so on)
*
* Copyright 2015 Edward V. Emelianov <eddy@sao.ru, edward.emelianoff@gmail.com>
*
* This program 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 2 of the License, or
* (at your option) any later version.
*
* This program 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 this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
* MA 02110-1301, USA.
*/
#include "main.h"
// state of shutter - global variable to omit interface functions
shutter_state Shutter_State = SHUTTER_NOTREADY;
uint16_t Shutter_delay = SHUTTER_DELAY;
// function to be runned from timer irq
void (*shutter_timer_fn)() = NULL;
/**
* Make background pause in 'us' microsecond, after which run function fn_ready
* @param us - pause in microseconds
* @param fn_ready - function to run at end of pause
*/
void shutter_wait(int us, void(*fn_ready)()){
if(!fn_ready) return;
//DBG("wait for previous .. ");
while(shutter_timer_fn); // wait for ending of previous operation
//DBG("ok\n");
shutter_timer_fn = fn_ready;
timer_set_period(SHUTTER_TIM, us);
timer_enable_counter(SHUTTER_TIM);
}
// macro to open/close/set default state
// open shutter is 0:0 -> when MCU power is off, shutter "automatically" opens
#define shutter_open() do{gpio_clear(SHUTTER_PORT, SHUTTER_ON_PIN | SHUTTER_POLARITY_PIN);}while(0)
#define shutter_close() do{gpio_clear(SHUTTER_PORT, SHUTTER_ON_PIN); \
gpio_set(SHUTTER_PORT, SHUTTER_POLARITY_PIN);}while(0)
#define shutter_hiZ() do{gpio_set(SHUTTER_PORT, SHUTTER_ON_PIN | SHUTTER_POLARITY_PIN);}while(0)
#define shutter_off() do{gpio_set(SHUTTER_PORT, SHUTTER_ON_PIN); \
gpio_clear(SHUTTER_PORT, SHUTTER_POLARITY_PIN);}while(0)
#define shutter_error() ((gpio_get(SHUTTER_PORT, SHUTTER_FB_PIN) == 0))
/**
* Make tests for undervoltage & wire breakage
* this function calls in 2 stages:
* 1) check undervoltage & set hiZ
* 2) repeat checking undervoltage & check wire breakage
* if shutter was uninitialized we open it
*/
void shutter_test(){
static shutter_state old_State = SHUTTER_NOTREADY;
// test for undervoltage
if(shutter_voltage() < SHUTTER_UNDERVOLTAGE_THRES){
ERR("shutter undervoltage\n");
Shutter_State = SHUTTER_NOTREADY;
shutter_off();
return;
}
if(Shutter_State != SHUTTER_INITIALIZED){ // first call of this function
old_State = Shutter_State;
Shutter_State = SHUTTER_INITIALIZED;
// test for wire breakage
DBG("breakage test\n");
shutter_hiZ(); // 1,1: breakage test
shutter_wait(SHUTTER_OP_DELAY, shutter_test);
}else{ // check breakage
if(shutter_error()){ // ERR==0 -> wire breakage
ERR("shutter wire breakage\n");
Shutter_State = SHUTTER_NOTREADY;
}else{
if(old_State == SHUTTER_NOTREADY){
Shutter_State = SHUTTER_OPENING;
DBG("ready!\n");
}else{
Shutter_State = old_State;
DBG("no errors\n");
}
}
shutter_off();
}
}
/**
* after open/close pulse we should return bridge to default state
* 3 stages:
* 1) check for short-circuit & turn off power of the shutter
* 2) check for overtemperature or undervoltage
* 3) common test for undervoltage/breakage
*/
void shutter_ready(){
uint8_t test_err = 0;
//DBG("shtr ready\n");
switch (Shutter_State){
case SHUTTER_CLOSED: // repeated pulse to check errors
case SHUTTER_OPENED:
if(shutter_error()){
ERR("shutter overtemperature or undervoltage\n");
Shutter_State = SHUTTER_NOTREADY;
}
break;
case SHUTTER_PROC_CLOSING: // closing - set to closed state
case SHUTTER_PROC_OPENING: // opening - set to opened state
if(shutter_error()){
ERR("shutter short-circuit\n");
Shutter_State = SHUTTER_NOTREADY;
}else{
test_err = 1;
if(Shutter_State == SHUTTER_PROC_CLOSING)
Shutter_State = SHUTTER_CLOSED;
if(Shutter_State == SHUTTER_PROC_OPENING)
Shutter_State = SHUTTER_OPENED;
}
break;
default:
ERR("wrong shutter state\n");
print_shutter_state(lastsendfun);
}
shutter_off();
if(Shutter_State == SHUTTER_NOTREADY) return;
if(test_err){
//DBG("now test for err\n");
shutter_wait(SHUTTER_OP_DELAY, shutter_ready); // test for overtemp or undervoltage
}else{
// wait a lot of time to prevent false detections
shutter_wait(SHUTTER_DELAY, shutter_test);
}
}
/**
* Initialisation of shutter ports & test for shutter presense
* @return 1 if all OK, 0 in case of error
*/
shutter_state shutter_init(){
Shutter_State = SHUTTER_NOTREADY;
// setup timer
rcc_peripheral_enable_clock(&RCC_APB1ENR, RCC_APB1ENR_TIM5EN);// enable timer clocking
nvic_enable_irq(NVIC_SHUTTER_IRQ);
timer_reset(SHUTTER_TIM);
// timer have frequency of 1MHz, so, to make pause in Xus set period to X
// 36MHz of APB1
timer_set_mode(SHUTTER_TIM, TIM_CR1_CKD_CK_INT, TIM_CR1_CMS_EDGE, TIM_CR1_DIR_UP);
// 72MHz div 72 = 1MHz
timer_set_prescaler(SHUTTER_TIM, 71);
timer_one_shot_mode(SHUTTER_TIM); // single pulse mode
timer_disable_preload(SHUTTER_TIM); // force changing period
timer_update_on_overflow(SHUTTER_TIM); // generate UEV only on overflow
timer_enable_update_event(SHUTTER_TIM);
timer_enable_irq(SHUTTER_TIM, TIM_DIER_UIE); // update IRQ enable
//DBG("shutter timer ready\n");
// setup pins
// on/off & polarity: open drain
gpio_set_mode(SHUTTER_PORT, GPIO_MODE_OUTPUT_2_MHZ,
GPIO_CNF_OUTPUT_OPENDRAIN, SHUTTER_ON_PIN | SHUTTER_POLARITY_PIN);
// feedback: floating input
gpio_set_mode(SHUTTER_PORT, GPIO_MODE_INPUT,
GPIO_CNF_INPUT_FLOAT, SHUTTER_FB_PIN);
shutter_off();
shutter_wait(SHUTTER_OP_DELAY, shutter_test);
return SHUTTER_INITIALIZED; // we return this state in spite of the shutter isn't really initialized yet
}
/**
* Finite-state machine processing for shutter
* (wait for capasitor charge and run needed functions
*/
void process_shutter(){
if(Shutter_State == SHUTTER_NOTREADY) return;
if(Shutter_State != SHUTTER_OPENING && Shutter_State != SHUTTER_CLOSING)
return;
if(shutter_voltage() < SHUTTER_UNDERVOLTAGE_THRES){
ERR("no shutter power source\n");
Shutter_State = SHUTTER_NOTREADY;
shutter_off();
return;
}
if(shutter_error()){
ERR("some shutter error\n");
Shutter_State = SHUTTER_NOTREADY;
shutter_off();
return;
}
if(shutter_voltage() < SHUTTER_VOLTAGE_THRES) return; // capasitor isn't charged
switch (Shutter_State){
case SHUTTER_OPENING:
Shutter_State = SHUTTER_PROC_OPENING;
shutter_open();
break;
case SHUTTER_CLOSING:
Shutter_State = SHUTTER_PROC_CLOSING;
shutter_close();
break;
default:
return;
}
shutter_wait(Shutter_delay, shutter_ready);
}
void Shutter_tim_isr(){
//DBG("shutter timer\n");
void (*fn)() = shutter_timer_fn;
if(timer_get_flag(SHUTTER_TIM, TIM_SR_UIF)){
// Clear compare interrupt flag
timer_clear_flag(SHUTTER_TIM, TIM_SR_UIF);
shutter_timer_fn = NULL;
// and run needed function
if(fn){
fn();
}
}
}
/**
* printout shutter state
*/
void print_shutter_state(sendfun s){
P("shutter ", s);
switch (Shutter_State){
case SHUTTER_INITIALIZED:
P("testing", s);
break;
case SHUTTER_READY:
P("ready", s);
break;
case SHUTTER_OPENED:
P("opened", s);
break;
case SHUTTER_CLOSED:
P("closed", s);
break;
case SHUTTER_OPENING:
case SHUTTER_CLOSING:
P("charged for ", s);
if(Shutter_State == SHUTTER_OPENING) P("opening", s);
else P("closing", s);
break;
case SHUTTER_PROC_OPENING:
case SHUTTER_PROC_CLOSING:
P("in process", s);
break;
default: // not ready or error
if(shutter_error()){ // ERR==0 -> wire breakage or something else
P("error", s);
}else
P("not initialised or broken", s);
}
newline(s);
}
/**
* Try to open/close shutter depending on argument:
* state == SHUTTER_CLOSING - try to close
* state == SHUTTER_OPENING - try to open
*/
void shutter_try(shutter_state state){
switch (Shutter_State){ // right states are only OPENED, CLOSED & READY
case SHUTTER_OPENED:
if(state == SHUTTER_OPENING){ // try to open opened shutter?
P("alerady opened\n", lastsendfun);
return;
}
break;
case SHUTTER_CLOSED:
if(state == SHUTTER_CLOSING){ // try to close closed?
P("alerady closed\n", lastsendfun);
return;
}
break;
case SHUTTER_READY:
; // All OK
break;
default:
print_shutter_state(lastsendfun);
return;
}
Shutter_State = state; // all OK, now we can change current state of shutter
}