536 lines
14 KiB
C

/*
* This file is part of the multistepper project.
* Copyright 2023 Edward V. Emelianov <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 3 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, see <http://www.gnu.org/licenses/>.
*/
#include "adc.h"
#include "buttons.h"
#include "commonproto.h"
#include "flash.h"
#include "hardware.h"
#include "hdr.h"
#include "pdnuart.h"
#include "proto.h"
#include "steppers.h"
#include "usb_dev.h"
#define NOPARCHK(par) do{if(PARBASE(par) != CANMESG_NOPAR) return ERR_BADPAR;}while(0)
#define CHECKN(val, par) do{val = PARBASE(par); \
if(val > MOTORSNO-1) return ERR_BADPAR;}while(0)
extern volatile uint32_t Tms;
// common functions for CAN and USB (or CAN only functions)
/*
static errcodes cu_nosuchfn(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}*/
errcodes cu_abspos(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
errcodes ret = ERR_OK;
if(ISSETTER(par)){
ret = setmotpos(n, *val);
}
getpos(n, val);
return ret;
}
errcodes cu_accel(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
errcodes ret = ERR_OK;
if(ISSETTER(par)){
if(*val/the_conf.microsteps[n] > ACCELMAXSTEPS || *val < 1) return ERR_BADVAL;
uint16_t acc = the_conf.accel[n];
the_conf.accel[n] = *val;
if(!update_stepper(n)){
the_conf.accel[n] = acc;
ret = ERR_CANTRUN;
}
}
*val = the_conf.accel[n];
return ret;
}
static const uint8_t extADCchnl[NUMBER_OF_EXT_ADC_CHANNELS] = {ADC_AIN0, ADC_AIN1, ADC_AIN2, ADC_AIN3};
// V*1000
errcodes cu_adc(uint8_t par, int32_t *val){
uint8_t n = PARBASE(par);
if(n > NUMBER_OF_EXT_ADC_CHANNELS - 1) return ERR_BADPAR;
*val = getADCvoltage(extADCchnl[n]);
return ERR_OK;
}
// NON-STANDARD COMMAND!!!!!!!
// errcode == keystate, value = last time!!!!
errcodes cu_button(uint8_t par, int32_t *val){
uint8_t n = PARBASE(par);
if(n > BTNSNO-1){
*val = CANMESG_NOPAR; // the only chance to understand error
return ERR_BADPAR;
}
return (uint8_t) keystate(n, (uint32_t*)val);
}
errcodes cu_diagn(uint8_t par, int32_t *val){
uint8_t n = PARBASE(par);
#if MOTORSNO > 8
#error "Change this code!"
#endif
if(n == CANMESG_NOPAR){ // all motors
n = 0;
for(int i = MOTORSNO-1; i > -1; --i){
n <<= 1;
n |= motdiagn(i);
}
*val = n;
return ERR_OK;
}
CHECKN(n, par);
*val = motdiagn(n);
return ERR_OK;
}
errcodes cu_drvtype(uint8_t par, int32_t *val){
uint8_t n; CHECKN(n, par);
motflags_t *fl = &the_conf.motflags[n];
if(ISSETTER(par)){
if(*val >= DRVTYPE_AMOUNT) return ERR_BADVAL;
fl->drvtype = *val;
}
*val = fl->drvtype;
return ERR_OK;
}
errcodes cu_emstop(uint8_t par, int32_t _U_ *val){
uint8_t n = PARBASE(par);
if(n == CANMESG_NOPAR){
for(int i = 0; i < MOTORSNO; ++i)
emstopmotor(i);
return ERR_OK;
}
CHECKN(n, par);
emstopmotor(n);
return ERR_OK;
}
errcodes cu_eraseflash(uint8_t _U_ par, int32_t _U_ *val){
NOPARCHK(par);
if(ISSETTER(par)){
if(erase_storage(*val)) return ERR_BADVAL;
}else if(erase_storage(-1)) return ERR_CANTRUN;
return ERR_OK;
}
// par - motor number
errcodes cu_esw(uint8_t par, int32_t *val){
uint8_t n; CHECKN(n, par);
*val = ESW_state(n);
return ERR_OK;
}
errcodes cu_eswreact(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
errcodes ret = ERR_OK;
if(ISSETTER(par)){
if(*val < 0 || *val > ESW_AMOUNT-1) return ERR_BADVAL;
uint8_t react = the_conf.ESW_reaction[n];
the_conf.ESW_reaction[n] = *val;
if(!update_stepper(n)){
the_conf.ESW_reaction[n] = react;
ret = ERR_CANTRUN;
}
}
*val = geteswreact(n);
return ret;
}
errcodes cu_goto(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
if(ISSETTER(par)) return motor_absmove(n, *val);
return getpos(n, val);
}
errcodes cu_gotoz(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
return motor_goto0(n);
}
TRUE_INLINE void setextpar(uint8_t val, uint8_t i){
switch(val){
case 0:
EXT_CLEAR(i);
break;
case 1:
EXT_SET(i);
break;
default:
EXT_TOGGLE(i);
}
}
errcodes cu_gpio(uint8_t par, int32_t *val){
#if EXTNO > 8
#error "change the code!!!"
#endif
uint8_t n = PARBASE(par);
#ifdef EBUG
USND("par="); printu(par);
USND(", n="); USB_putbyte('0'+n); newline();
#endif
if(n == CANMESG_NOPAR){ // all
#ifdef EBUG
USND("ALL\n");
#endif
uint8_t g = (uint8_t)*val;
if(ISSETTER(par)){
for(int i = 0; i < EXTNO; ++i){
setextpar(g & 1, i);
g >>= 1;
}
}
g = 0;
for(int i = EXTNO-1; i > -1; --i){
g <<= 1;
g |= EXT_CHK(i);
}
*val = g;
return ERR_OK;
}else if(n > EXTNO-1) return ERR_BADPAR;
// single channel setter/getter
if(ISSETTER(par))
setextpar((uint8_t)*val, n);
*val = (int32_t) EXT_CHK(n);
return ERR_OK;
}
// TODO: configure PU/PD, IN/OUT
errcodes cu_gpioconf(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}
// calculate ARR value for given speed, return nearest possible speed
static uint16_t getSPD(uint8_t n, int32_t speed){
uint32_t ARR = PCLK/(MOTORTIM_PSC+1) / the_conf.microsteps[n] / speed - 1;
if(ARR < MOTORTIM_ARRMIN) ARR = MOTORTIM_ARRMIN;
else if(ARR > 0xffff) ARR = 0xffff;
speed = PCLK/(MOTORTIM_PSC+1) / the_conf.microsteps[n] / (ARR + 1);
if(speed > 0xffff) speed = 0xffff;
return (uint16_t)speed;
}
errcodes cu_maxspeed(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
errcodes ret = ERR_OK;
if(ISSETTER(par)){
if(*val <= the_conf.minspd[n]) return ERR_BADVAL;
uint16_t maxspd = the_conf.maxspd[n];
the_conf.maxspd[n] = getSPD(n, *val);
if(!update_stepper(n)){
the_conf.maxspd[n] = maxspd;
ret = ERR_CANTRUN;
}
}
*val = the_conf.maxspd[n];
return ret;
}
errcodes cu_maxsteps(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
if(ISSETTER(par)){
if(*val < 1) return ERR_BADVAL;
the_conf.maxsteps[n] = *val;
}
*val = the_conf.maxsteps[n];
return ERR_OK;
}
errcodes cu_mcut(uint8_t par, int32_t *val){
NOPARCHK(par);
getMCUtemp();
*val = getMCUtemp();
return ERR_OK;
}
errcodes cu_mcuvdd(uint8_t par, int32_t *val){
NOPARCHK(par);
*val = getVdd();
return ERR_OK;
}
errcodes cu_microsteps(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
if(ISSETTER(par)){
#if MICROSTEPSMAX > 512
#error "Change the code anywhere!"
#endif
uint16_t m = (uint16_t)*val, old = the_conf.microsteps[n];
if(m < 1 || m > MICROSTEPSMAX) return ERR_BADVAL;
// find most significant bit
if(m != 1<<MSB(m)) return ERR_BADVAL;
if(the_conf.maxspd[n] * m > PCLK/(MOTORTIM_PSC+1)/(MOTORTIM_ARRMIN+1)) return ERR_BADVAL;
the_conf.microsteps[n] = m;
motflags_t *f = the_conf.motflags;
if(f->drvtype == DRVTYPE_UART){
if(!pdnuart_microsteps(n, m)) return ERR_CANTRUN;
}
if(!update_stepper(n)){
the_conf.microsteps[n] = old;
return ERR_CANTRUN;
}
}
*val = the_conf.microsteps[n];
return ERR_OK;
}
errcodes cu_minspeed(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
errcodes ret = ERR_OK;
if(ISSETTER(par)){
if(*val >= the_conf.maxspd[n] || *val < 0) return ERR_BADVAL;
uint16_t minspd = the_conf.minspd[n];
the_conf.minspd[n] = getSPD(n, *val);
if(!update_stepper(n)){
the_conf.minspd[n] = minspd;
ret = ERR_CANTRUN;
}
}
*val = the_conf.minspd[n];
return ret;
}
errcodes cu_motcurrent(uint8_t par, int32_t *val){
uint8_t n; CHECKN(n, par);
if(ISSETTER(par)){
if(*val < 0 || *val > 31) return ERR_BADVAL;
the_conf.motcurrent[n] = *val;
motflags_t *f = the_conf.motflags;
if(f->drvtype == DRVTYPE_UART){
if(!pdnuart_setcurrent(n, *val)) return ERR_CANTRUN;
} else return ERR_BADCMD;
}
*val = the_conf.motcurrent[n];
return ERR_OK;
}
errcodes cu_motflags(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
errcodes ret = ERR_OK;
if(ISSETTER(par)){
motflags_t flags = the_conf.motflags[n];
the_conf.motflags[n] = *((motflags_t*)val);
if(!update_stepper(n)){
the_conf.motflags[n] = flags;
ret = ERR_CANTRUN;
}
}
*(motflags_t*)val = the_conf.motflags[n];
return ret;
}
errcodes cu_motno(uint8_t _U_ par, int32_t _U_ *val){
if(*val < 0 || *val >= MOTORSNO) return ERR_BADVAL;
if(ISSETTER(par)){
if(!pdnuart_setmotno(*val)) return ERR_CANTRUN;
}
*val = pdnuart_getmotno();
return ERR_OK;
}
errcodes cu_motmul(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}
errcodes cu_motreinit(uint8_t _U_ par, int32_t _U_ *val){
NOPARCHK(par);
init_steppers();
return ERR_OK;
}
errcodes cu_pdn(uint8_t par, int32_t *val){
uint8_t n = PARBASE(par);
if(ISSETTER(par)){
if(!pdnuart_writereg(n, *val)) return ERR_CANTRUN;
}
if(!pdnuart_readreg(n, (uint32_t*)val)) return ERR_CANTRUN;
return ERR_OK;
}
errcodes cu_ping(uint8_t _U_ par, int32_t _U_ *val){
return ERR_OK;
}
errcodes cu_relpos(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
if(ISSETTER(par)) return motor_relmove(n, *val);
return getremainsteps(n, val);
}
errcodes cu_relslow(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
if(ISSETTER(par)) return motor_relslow(n, *val);
return getremainsteps(n, val);
}
static errcodes cu_reset(uint8_t par, int32_t _U_ *val){
NOPARCHK(par);
NVIC_SystemReset();
return ERR_OK;
}
errcodes cu_saveconf(uint8_t _U_ par, int32_t _U_ *val){
NOPARCHK(par);
if(store_userconf()) return ERR_CANTRUN;
return ERR_OK;
}
errcodes cu_screen(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}
errcodes cu_speedlimit(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
*val = getSPD(n, 0xffff);
return ERR_OK;
}
errcodes cu_state(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
*val = getmotstate(n);
return ERR_OK;
}
errcodes cu_stop(uint8_t _U_ par, int32_t _U_ *val){
uint8_t n; CHECKN(n, par);
stopmotor(n);
return ERR_OK;
}
errcodes cu_time(uint8_t par, int32_t *val){
NOPARCHK(par);
*val = Tms;
return ERR_OK;
}
errcodes cu_tmcbus(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}
errcodes cu_udata(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}
errcodes cu_usartstatus(uint8_t _U_ par, int32_t _U_ *val){
return ERR_BADCMD;
}
// V*10
errcodes cu_vdrive(uint8_t par, int32_t _U_ *val){
NOPARCHK(par);
*val = getADCvoltage(ADC_VDRIVE) * 11;
return ERR_OK;
}
errcodes cu_vfive(uint8_t par, int32_t *val){
NOPARCHK(par);
*val = getADCvoltage(ADC_VFIVE) * 2;
return ERR_OK;
}
const fpointer cancmdlist[CCMD_AMOUNT] = {
// different commands
[CCMD_PING] = cu_ping,
[CCMD_ADC] = cu_adc,
[CCMD_BUTTONS] = cu_button,
[CCMD_ESWSTATE] = cu_esw,
[CCMD_MCUT] = cu_mcut,
[CCMD_MCUVDD] = cu_mcuvdd,
[CCMD_RESET] = cu_reset,
[CCMD_TIMEFROMSTART] = cu_time,
[CCMD_EXT] = cu_gpio,
// configuration
[CCMD_SAVECONF] = cu_saveconf,
[CCMD_MICROSTEPS] = cu_microsteps,
[CCMD_ACCEL] = cu_accel,
[CCMD_MAXSPEED] = cu_maxspeed,
[CCMD_MINSPEED] = cu_minspeed,
[CCMD_SPEEDLIMIT] = cu_speedlimit,
[CCMD_MAXSTEPS] = cu_maxsteps,
[CCMD_MOTFLAGS] = cu_motflags,
[CCMD_ESWREACT] = cu_eswreact,
// motor's commands
[CCMD_ABSPOS] = cu_goto,
[CCMD_RELPOS] = cu_relpos,
[CCMD_RELSLOW] = cu_relslow,
[CCMD_EMERGSTOP] = cu_emstop,
[CCMD_STOP] = cu_stop,
[CCMD_REINITMOTORS] = cu_motreinit,
[CCMD_MOTORSTATE] = cu_state,
[CCMD_SETPOS] = cu_abspos,
[CCMD_GOTOZERO] = cu_gotoz,
[CCMD_MOTMUL] = cu_motmul,
[CCMD_DIAGN] = cu_diagn,
[CCMD_ERASEFLASH] = cu_eraseflash,
// [CCMD_UDATA] = cu_udata,
// [CCMD_USARTSTATUS] = cu_usartstatus,
[CCMD_VDRIVE] = cu_vdrive,
[CCMD_VFIVE] = cu_vfive
// Leave all commands upper for back-compatability with 3steppers
};
const char* cancmds[CCMD_AMOUNT] = {
[CCMD_PING] = STR_PING,
[CCMD_ADC] = STR_ADC,
[CCMD_BUTTONS] = STR_BUTTON,
[CCMD_ESWSTATE] = STR_ESW,
[CCMD_MCUT] = STR_MCUT,
[CCMD_MCUVDD] = STR_MCUVDD,
[CCMD_RESET] = STR_RESET,
[CCMD_TIMEFROMSTART] = STR_TIME,
[CCMD_EXT] = STR_GPIO,
[CCMD_SAVECONF] = STR_SAVECONF,
[CCMD_MICROSTEPS] = STR_MICROSTEPS,
[CCMD_ACCEL] = STR_ACCEL,
[CCMD_MAXSPEED] = STR_MAXSPEED,
[CCMD_MINSPEED] = STR_MINSPEED,
[CCMD_SPEEDLIMIT] = STR_SPEEDLIMIT,
[CCMD_MAXSTEPS] = STR_MAXSTEPS,
[CCMD_MOTFLAGS] = STR_MOTFLAGS,
[CCMD_ESWREACT] = STR_ESWREACT,
[CCMD_ABSPOS] = STR_GOTO,
[CCMD_RELPOS] = STR_RELPOS,
[CCMD_RELSLOW] = STR_RELSLOW,
[CCMD_EMERGSTOP] = STR_EMSTOP,
[CCMD_STOP] = STR_STOP,
[CCMD_REINITMOTORS] = STR_MOTREINIT,
[CCMD_MOTORSTATE] = STR_STATE,
[CCMD_SETPOS] = STR_ABSPOS,
[CCMD_GOTOZERO] = STR_GOTOZ,
[CCMD_MOTMUL] = STR_MOTMUL,
[CCMD_DIAGN] = STR_DIAGN,
[CCMD_ERASEFLASH] = STR_ERASEFLASH,
// [CCMD_UDATA] = STR_UDATA,
// [CCMD_USARTSTATUS] = STR_USARTSTATUS,
[CCMD_VDRIVE] = STR_VDRIVE,
[CCMD_VFIVE] = STR_VFIVE,
[CCMD_PDN] = STR_PDN,
[CCMD_MOTNO] = STR_MOTNO,
[CCMD_DRVTYPE] = STR_DRVTYPE,
[CCMD_MOTCURRENT] = STR_MOTCURRENT,
};