2015-07-28 15:49:10 +03:00

727 lines
19 KiB
C

/*
* user_proto.c
*
* Copyright 2014 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 "cdcacm.h"
#include "main.h"
#include "uart.h"
#include "hardware_ini.h"
#include "flash.h"
#include "stepper_motors.h"
#include "powerhw.h"
#include "AD7794.h"
// mode:
curmode_t mode = BYTE_MODE; // text protocol, activated on 1st meeteng of '['
// integer value given by user
static volatile int32_t User_value = 0;
// number of motor to process
static volatile uint8_t active_motor = 6;
// last active send function - to post "anonymous" replies
sendfun lastsendfun = usb_send; // make it usb_send by default (to prevent suspension)
// flag: !=0 when user value reading ends - for character terminals
enum{
UVAL_START, // user start to write integer value
UVAL_CHECKED, // user checks the value & enters + that it's right
UVAL_PRINTED, // value printed to user
UVAL_ENTERED, // value entered but not printed
UVAL_BAD // entered bad value
};
static uint8_t Uval_ready = UVAL_PRINTED;
int read_int(char *buf, int cnt);
static intfun I = NULL; // function to process entered integer
#define READINT() do{i += read_int(&buf[i+1], len-i-1);}while(0)
#define WRONG_COMMAND() do{if(mode == BYTE_MODE) command = '?';}while(0)
/**
* displays all TRD values from AD7794
* @param s -- current output
*/
void print_ad_vals(sendfun s){
int j;
if(ad7794_on){
for(j = 0; j < TRD_NO; ++j){
if(mode == LINE_MODE){
P("[ " STR_EXTADC_VALUES " ", s);
print_int(j, s);
s(' ');
}
print_int(ad7794_values[j], s);
if(mode == LINE_MODE)
P(" ]\n", s);
else s(' ');
}
if(mode == BYTE_MODE) newline(s);
}else{
if(mode == BYTE_MODE) P("no AD7794 found\n", s);
else if(mode == LINE_MODE) P("[ " STR_EXTADC_VALUES " ERR ]\n", s);
}
}
/**
* displays all TRD values from inner ADC
* @param s -- current output
*/
void print_int_ad_vals(sendfun s){
int j;
for(j = 0; j < 8; j++){
if(mode == LINE_MODE){
P("[ " STR_INTADC_VALUES " ", s);
print_int(j, s);
s(' ');
}
print_int(TRD_value(j), s);
if(mode == BYTE_MODE) s(' ');
else if(mode == LINE_MODE)
P(" ]\n", s);
}
if(OW_scan){ // print 1-wire temperatures
for(j = 0; j < OW_dev_amount; ++j){
if(OW_temperature[j] != ERR_TEMP_VAL){
if(mode == LINE_MODE)
P("[ " STR_INTADC_VALUES " ", s);
else
s('N');
print_int(j + TRD_NO, s);
s(' ');
print_int(OW_temperature[j], s);
if(mode == LINE_MODE)
P(" ]\n", s);
else s(' ');
}
}
}
}
/*
uint8_t set_shtr_delay(int32_t v, sendfun s){
uint16_t d = (uint16_t) v;
Shutter_delay = d;
P("Change delay to ", s);
print_int(d, s);
return 0;
}
*/
int adc_channel = -1;
int div_mul = 0; // 0 - multip., !0 - div.
/**
* change dividers/multipliers
* work only in BYTE_MODE
*/
uint8_t ch_divmul(int32_t v, _U_ sendfun s){
uint32_t val = (uint32_t) v;
if(adc_channel == -1) return 0;
if(div_mul){ // != 0 - divisors
ADC_divisors[adc_channel] = val;
}else{ // == 0 - mul
ADC_multipliers[adc_channel] = val;
}
adc_channel = -1;
return 1;
}
/**
* Change divisor (work only in BYTE_MODE)
* @param v - user value (sensor number)
* @param s - active sendfunction
*/
uint8_t try_ch_divmul(int32_t v, sendfun s){
if(v > ADC_CHANNELS_NUMBER || v < 0){
P("wrong channel number\n", s);
adc_channel = -1;
return 0; // error
}
adc_channel = v;
I = ch_divmul;
read_int(NULL, 0); //start reading next int
return 0;
}
/**
* prints state of end switches for motor v
*/
uint8_t endswitchstate(int32_t v, sendfun s){
int32_t i;
if(v < 0 || v > 4){ // show all end-switches
for(i = 0; i < 5; i++)
endswitchstate(i, s);
return 0;
}
i = check_ep(v);
if(mode == BYTE_MODE){
P("Endswitches for motor ", s);
s('0' + v);
P(": ", s);
}else if(mode == LINE_MODE){ // in LINE_MODE print [ E No_motor state ]
P("[ " STR_ENDSW_STATE " ",s);
s('0' + v);
s(' ');
}
if(mode != BINARY_MODE){
print_int(i, s);
if(mode == LINE_MODE) P(" ]\n", s);
else s('\n');
}
/*
uint32_t h = (GPIO_IDR(GPIOD) & 0xff) | ((GPIO_IDR(GPIOB)<<2) & 0x300) |
((GPIO_IDR(GPIOC)<<3) & 0x1C00) | ((GPIO_IDR(GPIOA)&0x100)<<5);
print_hex((uint8_t*)&h, 4, lastsendfun);
*/
return 0;
}
/**
* moves turret 'active_motor' to position v
*/
uint8_t move_turret(int32_t v, sendfun s){
const int32_t maxpos[3] = {6, 6, 8}; // maximum position number for given turret
int32_t sign = 1;
if(active_motor > 2){
if(mode == BYTE_MODE) P("Wrong turret number\n", s);
return 0;
}
int32_t m = maxpos[active_motor];
if(v > m || v < 1){
if(mode == BYTE_MODE){
P("Wrong position, shoud be not more than ", s);
print_int(m, s);
}
return 0;
}
uint8_t curpos = check_ep(active_motor);
if(curpos == v){ // we are in that position
endswitchstate(active_motor, s);
return 1;
}
move2pos[active_motor] = v;
// check the nearest direction
if(curpos){ // we are not between positions
if(((v + m - curpos) % m) > m/2) // rotation in positive direction will take more steps than in negative
sign = -1; // move CCV
}
return move_motor(active_motor, sign * TURRETS_NEXT_POS_STEPS);
}
/**
* function 'help' works only in BYTE_MODE
*/
void help(sendfun s){
#define pr(arg) do{prnt((uint8_t*)arg, s); s('\n');}while(0)
pr("0..4\t move stepper motor to N steps");
pr("Commands in char protocol:");
pr(STR_INTADC_VALUES "\tget TRD values");
pr(STR_STOP_ALL_MOTORS "\tstop all motors");
//pr("C");
pr("D\tturn AD7794 to double conversion mode");
pr(STR_ENDSW_STATE "\tshow end-switches state for given motor");
pr("F\tdump flash data");
pr(STR_SHOW_PERIOD "\tget motors' speed");
pr("H\tshow this help");
pr("I\tturn off AD7794 init flag");
pr("J\tmove slits (0) wheel to Nth position");
pr("K\tmove filters (1) wheel to Nth position");
pr("L\tmove filters (2) wheel to Nth position");
pr("M\tswitch on/off ADC monitoring");
//pr("N");
pr("O\tturn on 1-wire scan");
pr("P\tadd 1-wire sensor");
pr("Q\tturn off 1-wire scan");
pr("R\terase 1-wire IDs from memory");
pr("S\tturn AD7794 to single conversion mode");
pr(STR_PRINT_TIME "\tprint current value of time counters");
//pr("U\t(reserved)");
//pr("V");
//pr("W\t(reserved)");
pr("X\tset timer period for linear stages");
//pr("Y");
pr("Z\tShow motors' positions");
//pr("a");
//pr("b");
pr("c\tclose shutter");
pr("d\tchange value of ADC divisor No N");
//pr("e");
pr("f\tupdate flash settings");
pr("g\tchange AD7794 gain");
pr(STR_SHTR_VOLTAGE "\tshow shutter voltage");
pr(STR_EXTADC_INIT "\tinit AD7794");
//pr("j");
//pr("k");
//pr("l");
pr("m\tchange value of ADC multiplier No N");
//pr("n");
pr("o\topen shutter");
pr(STR_MOTORS_VOLTAGE "\tshow motors voltage");
//pr("q");
pr("r\treinit shutter");
pr(STR_EXTADC_VALUES "\tshow AD7794 values");
pr(STR_SHTR_STATE "\tprint shutter state");
pr("u\tcheck USB connection");
//pr("v");
//pr("w");
pr("x\tset timer period for turrets' motors");
//pr("y");
//pr("z");
pr("\nAfter entering of user value press '+' to submit or '-' to reject it");
#undef pr
}
/**
* parce command buffer buf with length len
* return 0 if buffer processed or len if there's not enough data in buffer
*/
int parce_incoming_buf(char *buf, int len, sendfun s){
uint8_t command, do_echo = 1;
//uint32_t utmp;
int i = 0, j, m;
lastsendfun = s;
if(buf[0] == '[') mode = LINE_MODE;
if(mode == BYTE_MODE){
if(Uval_ready == UVAL_START){ // we are in process of user's value reading
i += read_int(buf, len);
}
if(Uval_ready == UVAL_ENTERED){
P("confirm entered value (+/-): ", s);
print_int(User_value, s); // printout readed integer value for error control
Uval_ready = UVAL_PRINTED;
}
if(I && Uval_ready == UVAL_CHECKED){
Uval_ready = UVAL_BAD; // clear Uval_ready
I(User_value, s);
return 0;
}
}else if(mode == LINE_MODE){ // text mode: check for "]\n" presence
uint8_t bad_cmd = 1, found_end = 0;
if(buf[0] == '['){
for(j = 1; j < len; j++){
if(buf[j] == ']' && found_end == 0){
found_end = j; // store position of command's end
continue;
}
if(buf[j] != '\n' && buf[j] != '\r') continue; // search end of line
else{
if(found_end){ // open brace have a pair
bad_cmd = 0;
len = found_end+1; // save "]" in buffer for correct integer values reading
buf[len] = 0; // truncate buffer to only one command
break;
}
else{
return 0; // end of line without closing bracket
}
}
}
}else{
return 0;
}
if(bad_cmd){
return len; // not enough data in buffer
}
}
for(; i < len; i++){
command = buf[i];
if(!command) continue; // omit zero
if(command >= '0' && command <= '4'){ // stepper motors
active_motor = command - '0';
I = stepper_proc;
READINT();
}else switch (command){
case '[':
case ']':
break;
case '+': // user check number value & confirm it's right
if(mode != BYTE_MODE) return 0; // bad command - no echo
if(Uval_ready == UVAL_PRINTED) Uval_ready = UVAL_CHECKED;
else WRONG_COMMAND();
break;
case '-': // user check number value & confirm it's wrong
if(mode != BYTE_MODE) return 0; // bad command - no echo
if(Uval_ready == UVAL_PRINTED) Uval_ready = UVAL_BAD;
else WRONG_COMMAND();
break;
case CMD_INTADC_VALUES: // [A] show TRD values by inner ADC [ A 0 val0 ]\n[A 1 val1 ] ...
//adc_start_conversion_direct(ADC1);
print_int_ad_vals(s);
if(mode == BYTE_MODE) s('\n');
do_echo = 0; // don't echo command in line mode as there's output
break;
case 'b': // byte mode
mode = BYTE_MODE;
//return 0; //don't echo this command
break;
case CMD_STOP_ALL_MOTORS: // [B] stop all motors
for(m = 0; m < 5; m++)
stop_motor(m);
do_echo = 0; // echo called from sto
break;
case 'c': // [c] close shutter
do_echo = try_to_close_shutter(); // don't echo in case of state wouldn't be changed
break;
case 'd': // change ADC_divisor (only for byte mode)
if(mode != BYTE_MODE) return 0;
div_mul = 1; //divisors
I = try_ch_divmul;
READINT();
break;
case 'D': // double conversion (only for byte mode)
if(mode != BYTE_MODE) return 0;
doubleconv = 1;
break;
case CMD_ENDSW_STATE: // [E] print end-switch state for given motor
I = endswitchstate;
READINT();
break;
case 'f': // store flash data
if(mode != BYTE_MODE) return 0;
uint8_t f_flag = save_flashdata();// == 0 if all OK, or return error flag
if(f_flag){
P("error! can't store data, errno: ", s);
print_int((int32_t) f_flag, s);
s('\n');
}else
P("data stored successfully\n", s);
break;
case 'F': // dump flash data (only in byte mode)
if(mode != BYTE_MODE) return 0;
dump_flash_data(s);
break;
case 'g': // change gain (only in byte mode)
if(mode != BYTE_MODE) return 0;
I = set_ADC_gain;
READINT();
break;
case CMD_SHOW_PERIOD: // [G] get motors' speed
show_motors_period(s);
do_echo = 0;
break;
case CMD_SHTR_VOLTAGE: // [h] show sHutter voltage * 100
if(mode == LINE_MODE) P("[ " STR_SHTR_VOLTAGE " ", s);
print_int(shutter_voltage(), s);
if(mode == LINE_MODE) P(" ]", s);
newline(s);
do_echo = 0;
break;
case 'H': // show help (only in byte mode)
if(mode != BYTE_MODE) return 0;
help(s);
break;
case CMD_EXTADC_INIT: // [i] init AD7794
AD7794_init();
break;
case 'I': // turn off flag AD7794
ad7794_on = 0;
break;
case 'J': // move turret 0/1/2 to given position
case 'K':
case 'L':
active_motor = command - 'J'; // J == 0, K == 1, L == 2
I = move_turret;
READINT();
break;
case 'm': // change ADC_multiplier (only for byte mode)
if(mode != BYTE_MODE) return 0;
div_mul = 0; // multipliers
I = try_ch_divmul;
READINT();
break;
case 'M': // ADC monitoring ON
ADC_monitoring = !ADC_monitoring;
break;
case 'o': // open shutter
do_echo = try_to_open_shutter();
break;
case 'O': // 1-wire scan ON
if(!OW_scan){
OW_scan = 1;
OW_send_read_seq();
}
break;
case CMD_MOTORS_VOLTAGE: // [p] show motors voltage * 100
if(mode == LINE_MODE) P("[ " STR_MOTORS_VOLTAGE " ", s);
print_int(power_voltage(), s);
if(mode == LINE_MODE) P(" ]", s);
newline(s);
do_echo = 0;
break;
case 'P': // (only for byte mode) - add 1-wire device
if(mode != BYTE_MODE) return 0;
OW_fill_next_ID();
break;
case 'Q': // 1-wire scan OFF
OW_scan = 0;
break;
case 'R': // (only for byte mode) - reset 1-wire IDs in RAM
if(mode != BYTE_MODE) return 0;
OW_dev_amount = 0;
break;
case 'r': // reinit shutter
shutter_init();
break;
case CMD_EXTADC_VALUES: // [s] read ADC val through SPI
if(!ad7794_on){
AD7794_init();
if(mode == BYTE_MODE) P("wait: values aren't ready yet\n", s);
break;
}
if(mode == BYTE_MODE) P("AD7794 values: ", s);
print_ad_vals(s);
do_echo = 0;
break;
case 'S': // single conversion (only for byte mode)
if(mode != BYTE_MODE) return 0;
doubleconv = 0;
break;
case CMD_SHTR_STATE: // [t] print shutter state
print_shutter_state(s);
do_echo = 0;
break;
case CMD_PRINT_TIME: // print time
print_time(s);
if(mode == BYTE_MODE) newline(s);
break;
case 'u': // check USB connection (only byte mode)
if(mode != BYTE_MODE) return 0;
P("\nUSB ", s);
if(!USB_connected) P("dis", s);
P("connected\n",s);
break;
/* case 'U': // test: init USART1
UART_init(USART1);
break; */
case 'x': // set period of TIM1 (motors 1..3)
active_motor = 1;
I = set_timr;
READINT();
break;
case 'X': // set period of TIM2 (motors 4,5)
active_motor = 4;
I = set_timr;
READINT();
break;
case CMD_MOTOR_POSITION: // [Z]: show positions of all motors
get_motors_position();
do_echo = 0;
break;
/*
case '_':
P("set: ", s);
print_int(GPIO_ODR(MOTOR_EN_PORT) & MOTOR_EN_MASK, s);
P(", get: ", s);
print_int(gpio_get(MOTOR_EN_PORT, MOTOR_EN_MASK), s);
s('\n');
break;
case '(':
gpio_set(MOTOR_EN_PORT, MOTOR_EN_MASK);
break;
case ')':
gpio_clear(MOTOR_EN_PORT, MOTOR_EN_MASK);
break;
*/
case '\n': // show newline, cr, space and tab as is
case '\r':
case ' ':
case '\t':
break;
default:
WRONG_COMMAND(); // echo '?' on unknown command in byte mode
}
if(mode == BYTE_MODE) s(command); // echo readed byte in byte mode
}
if(mode == LINE_MODE){ // process command which needs for user value
if(I){
if(Uval_ready == UVAL_ENTERED)
do_echo = I(User_value, s);
else
do_echo = 0; // if entered value is wrong, don't echo command
Uval_ready = UVAL_BAD; // clear Uval_ready
I = NULL;
}
}
if(do_echo && mode == LINE_MODE)
P(buf, s);
return 0; // all data processed - 0 bytes leave in buffer
}
/*
* Send char array wrd thru USB or UART
*/
void prnt(uint8_t *wrd, sendfun s){
if(!wrd) return;
while(*wrd) s(*wrd++);
}
/*
void newline(sendfun s){
P("\n", s);
}
*/
/**
* Read from TTY integer value given by user (in DEC).
* Reading stops on first non-numeric symbol.
* To work with symbol terminals reading don't stops on buffer's end,
* it waits for first non-numeric char.
* When working on string terminals, terminate string by '\n', 0 or any other symbol
* @param buf - buffer to read from
* @param cnt - buffer length
* @return amount of readed symbols
*/
int read_int(char *buf, int cnt){
int readed = 0, i;
static int enteredDigits; // amount of entered digits
static int sign; // sign of readed value
if(Uval_ready){ // this is first run
Uval_ready = UVAL_START; // clear flag
enteredDigits = 0; // 0 digits entered
User_value = 0; // clear value
sign = 1; // clear sign
}
if(!cnt) return 0;
for(i = 0; i < cnt; i++, readed++){
uint8_t chr = buf[i];
if(chr == '-'){
if(enteredDigits == 0){ // sign should be first
sign = -1;
continue;
}else{ // '-' after number - reject entered value
Uval_ready = UVAL_BAD;
break;
}
}
if(chr < '0' || chr > '9'){
if(enteredDigits)
Uval_ready = UVAL_ENTERED;
else{
if(chr == ' ' || chr == '\t') continue; // omit leading space|tab
else Uval_ready = UVAL_BAD; // bad symbol
}
break;
}
User_value = User_value * 10 + (int32_t)(chr - '0');
enteredDigits++;
}
if(Uval_ready == UVAL_ENTERED){ // reading has met an non-numeric character
User_value *= sign;
}
return readed;
}
/**
* Print buff as hex values
* @param buf - buffer to print
* @param l - buf length
* @param s - function to send a byte
*/
void print_hex(uint8_t *buff, uint8_t l, sendfun s){
void putc(uint8_t c){
if(c < 10)
s(c + '0');
else
s(c + 'a' - 10);
}
s('0'); s('x'); // prefix 0x
while(l--){
putc(buff[l] >> 4);
putc(buff[l] & 0x0f);
}
}
/**
* Print decimal integer value
* @param N - value to print
* @param s - function to send a byte
*/
void print_int(int32_t N, sendfun s){
uint8_t buf[10], L = 0;
if(N < 0){
s('-');
N = -N;
}
if(N){
while(N){
buf[L++] = N % 10 + '0';
N /= 10;
}
while(L--) s(buf[L]);
}else s('0');
}
/*
void process_int(int32_t v, sendfun s){
newline(s);
P("You have entered a value ", s);
print_int(v, s);
newline(s);
}
*/
/**
* change AD7794 gain. Run only in byte mode!
*/
uint8_t set_ADC_gain(int32_t v, sendfun s){
if(ad7794_on){
P("Change gain to ", s);
print_int(v, s);
newline(s);
change_AD7794_gain(v);
AD7794_calibration(0);
}
return 0;
}
/**
* Process stepper
* @param v - user value
* @param s - active sendfunction
*/
uint8_t stepper_proc(int32_t v, sendfun s){
if(active_motor > 4){
if(mode == BYTE_MODE) P("wrong motor number\n", s);
return 0; // error
}
uint8_t ret = move_motor(active_motor, v);
active_motor = 6;
return ret;
}
uint8_t set_timr(int32_t v, sendfun s){
if(active_motor > 4){
if(mode == BYTE_MODE) P("wrong motor number\n", s);
return 0; // error
}
if(v < 0 || v > 0xffff){
if(mode == BYTE_MODE) P("Bad period!\n", s);
active_motor = 6;
return 0;
}
if(mode == BYTE_MODE){
P("set period: ", s);
print_int(v, s);
P("\n", s);
}
set_motor_period(active_motor, (uint16_t)v);
active_motor = 6;
return 1;
}