use crate::error::{CaError, CaResult};
use crate::server::record::{FieldDesc, ProcessAction, ProcessOutcome, Record};
use crate::types::{DbFieldType, EpicsValue};
pub struct CalcoutRecord {
pub val: f64,
pub calc: String,
pub oopt: i16, cached_should_output: bool, pub dopt: i16, pub ocal: String,
pub oval: f64,
pub ivoa: i16, pub ivov: f64,
pub inpa: String,
pub inpb: String,
pub inpc: String,
pub inpd: String,
pub inpe: String,
pub inpf: String,
pub inpg: String,
pub inph: String,
pub inpi: String,
pub inpj: String,
pub inpk: String,
pub inpl: String,
pub inpm: String,
pub inpn: String,
pub inpo: String,
pub inpp: String,
pub inpq: String,
pub inpr: String,
pub inps: String,
pub inpt: String,
pub inpu: String,
pub a: f64,
pub b: f64,
pub c: f64,
pub d: f64,
pub e: f64,
pub f: f64,
pub g: f64,
pub h: f64,
pub i: f64,
pub j: f64,
pub k: f64,
pub l: f64,
pub m: f64,
pub n: f64,
pub o: f64,
pub p: f64,
pub q: f64,
pub r: f64,
pub s: f64,
pub t: f64,
pub u: f64,
pub la: f64,
pub lb: f64,
pub lc: f64,
pub ld: f64,
pub le: f64,
pub lf: f64,
pub lg: f64,
pub lh: f64,
pub li: f64,
pub lj: f64,
pub lk: f64,
pub ll: f64,
pub lm: f64,
pub ln: f64,
pub lo: f64,
pub lp: f64,
pub lq: f64,
pub lr: f64,
pub ls: f64,
pub lt: f64,
pub lu: f64,
pub egu: String,
pub prec: i16,
pub hopr: f64,
pub lopr: f64,
pub adel: f64,
pub mdel: f64,
pub lalm: f64,
pub alst: f64,
pub mlst: f64,
pub pval: f64, pub odly: f64,
pub dlya: i16,
pending_output: bool,
pub calc_alarm: bool,
rpcl: Option<crate::calc::CompiledExpr>,
orpc: Option<crate::calc::CompiledExpr>,
}
impl Default for CalcoutRecord {
fn default() -> Self {
Self {
val: 0.0,
calc: String::new(),
oopt: 0,
cached_should_output: false,
dopt: 0,
ocal: String::new(),
oval: 0.0,
ivoa: 0,
ivov: 0.0,
inpa: String::new(),
inpb: String::new(),
inpc: String::new(),
inpd: String::new(),
inpe: String::new(),
inpf: String::new(),
inpg: String::new(),
inph: String::new(),
inpi: String::new(),
inpj: String::new(),
inpk: String::new(),
inpl: String::new(),
inpm: String::new(),
inpn: String::new(),
inpo: String::new(),
inpp: String::new(),
inpq: String::new(),
inpr: String::new(),
inps: String::new(),
inpt: String::new(),
inpu: String::new(),
a: 0.0,
b: 0.0,
c: 0.0,
d: 0.0,
e: 0.0,
f: 0.0,
g: 0.0,
h: 0.0,
i: 0.0,
j: 0.0,
k: 0.0,
l: 0.0,
m: 0.0,
n: 0.0,
o: 0.0,
p: 0.0,
q: 0.0,
r: 0.0,
s: 0.0,
t: 0.0,
u: 0.0,
la: 0.0,
lb: 0.0,
lc: 0.0,
ld: 0.0,
le: 0.0,
lf: 0.0,
lg: 0.0,
lh: 0.0,
li: 0.0,
lj: 0.0,
lk: 0.0,
ll: 0.0,
lm: 0.0,
ln: 0.0,
lo: 0.0,
lp: 0.0,
lq: 0.0,
lr: 0.0,
ls: 0.0,
lt: 0.0,
lu: 0.0,
egu: String::new(),
prec: 0,
hopr: 0.0,
lopr: 0.0,
adel: 0.0,
mdel: 0.0,
lalm: 0.0,
alst: 0.0,
mlst: 0.0,
pval: 0.0,
odly: 0.0,
dlya: 0,
pending_output: false,
calc_alarm: false,
rpcl: None,
orpc: None,
}
}
}
impl CalcoutRecord {
fn advance_prev(new: f64, prev: &mut f64) {
if new != *prev {
*prev = new;
}
}
fn get_vars(&self) -> [f64; 21] {
[
self.a, self.b, self.c, self.d, self.e, self.f, self.g, self.h, self.i, self.j, self.k,
self.l, self.m, self.n, self.o, self.p, self.q, self.r, self.s, self.t, self.u,
]
}
fn should_output(&self) -> bool {
match self.oopt {
0 => true, 1 => (self.pval - self.val).abs() > self.mdel, 2 => self.val == 0.0, 3 => self.val != 0.0, 4 => self.pval != 0.0 && self.val == 0.0, 5 => self.pval == 0.0 && self.val != 0.0, _ => false, }
}
}
static CALCOUT_FIELDS: &[FieldDesc] = &[
FieldDesc {
name: "VAL",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "CALC",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "EGU",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "PREC",
dbf_type: DbFieldType::Short,
read_only: false,
},
FieldDesc {
name: "HOPR",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "LOPR",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "ADEL",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "MDEL",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "LALM",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "ALST",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "MLST",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "PVAL",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "OOPT",
dbf_type: DbFieldType::Short,
read_only: false,
},
FieldDesc {
name: "ODLY",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "DLYA",
dbf_type: DbFieldType::Short,
read_only: true,
},
FieldDesc {
name: "DOPT",
dbf_type: DbFieldType::Short,
read_only: false,
},
FieldDesc {
name: "OCAL",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "OVAL",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "IVOA",
dbf_type: DbFieldType::Short,
read_only: false,
},
FieldDesc {
name: "IVOV",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "INPA",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPB",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPC",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPD",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPE",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPF",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPG",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPH",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPI",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPJ",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPK",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPL",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPM",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPN",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPO",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPP",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPQ",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPR",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPS",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPT",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "INPU",
dbf_type: DbFieldType::String,
read_only: false,
},
FieldDesc {
name: "A",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "B",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "C",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "D",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "E",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "F",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "G",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "H",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "I",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "J",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "K",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "L",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "M",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "N",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "O",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "P",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "Q",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "R",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "S",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "T",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "U",
dbf_type: DbFieldType::Double,
read_only: false,
},
FieldDesc {
name: "LA",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LB",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LC",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LD",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LE",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LF",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LG",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LH",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LI",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LJ",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LK",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LL",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LM",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LN",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LO",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LP",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LQ",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LR",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LS",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LT",
dbf_type: DbFieldType::Double,
read_only: true,
},
FieldDesc {
name: "LU",
dbf_type: DbFieldType::Double,
read_only: true,
},
];
impl Record for CalcoutRecord {
fn record_type(&self) -> &'static str {
"calcout"
}
fn apply_invalid_output_value(&mut self, ivov: EpicsValue) -> CaResult<()> {
self.put_field("OVAL", ivov)
}
fn init_record(&mut self, pass: u8) -> CaResult<()> {
if pass == 0 {
if !self.calc.is_empty() {
self.rpcl = crate::calc::compile(&self.calc).ok();
}
if !self.ocal.is_empty() {
self.orpc = crate::calc::compile(&self.ocal).ok();
}
self.pval = self.val;
self.mlst = self.val;
self.alst = self.val;
self.lalm = self.val;
}
Ok(())
}
fn process(&mut self) -> CaResult<ProcessOutcome> {
if self.dlya == 1 {
self.dlya = 0;
self.cached_should_output = self.pending_output;
self.pending_output = false;
return Ok(ProcessOutcome::complete());
}
if let Some(ref compiled) = self.rpcl {
let mut inputs = crate::calc::NumericInputs::with_vars(self.get_vars());
match crate::calc::eval(compiled, &mut inputs) {
Ok(v) => {
self.val = v;
self.calc_alarm = false;
}
Err(_) => {
self.calc_alarm = true;
}
}
}
if self.should_output() {
if self.dopt == 1 {
if let Some(ref compiled) = self.orpc {
let mut inputs = crate::calc::NumericInputs::with_vars(self.get_vars());
match crate::calc::eval(compiled, &mut inputs) {
Ok(v) => self.oval = v,
Err(_) => self.calc_alarm = true,
}
}
} else {
self.oval = self.val;
}
}
Self::advance_prev(self.a, &mut self.la);
Self::advance_prev(self.b, &mut self.lb);
Self::advance_prev(self.c, &mut self.lc);
Self::advance_prev(self.d, &mut self.ld);
Self::advance_prev(self.e, &mut self.le);
Self::advance_prev(self.f, &mut self.lf);
Self::advance_prev(self.g, &mut self.lg);
Self::advance_prev(self.h, &mut self.lh);
Self::advance_prev(self.i, &mut self.li);
Self::advance_prev(self.j, &mut self.lj);
Self::advance_prev(self.k, &mut self.lk);
Self::advance_prev(self.l, &mut self.ll);
Self::advance_prev(self.m, &mut self.lm);
Self::advance_prev(self.n, &mut self.ln);
Self::advance_prev(self.o, &mut self.lo);
Self::advance_prev(self.p, &mut self.lp);
Self::advance_prev(self.q, &mut self.lq);
Self::advance_prev(self.r, &mut self.lr);
Self::advance_prev(self.s, &mut self.ls);
Self::advance_prev(self.t, &mut self.lt);
Self::advance_prev(self.u, &mut self.lu);
let do_output = self.should_output();
self.pval = self.val;
if do_output && self.odly > 0.0 {
self.dlya = 1;
self.pending_output = true;
self.cached_should_output = false;
let delay = std::time::Duration::from_secs_f64(self.odly);
return Ok(ProcessOutcome::complete_with(vec![
ProcessAction::ReprocessAfter(delay),
]));
}
self.cached_should_output = do_output;
Ok(ProcessOutcome::complete())
}
fn get_field(&self, name: &str) -> Option<EpicsValue> {
match name {
"VAL" => Some(EpicsValue::Double(self.val)),
"CALC" => Some(EpicsValue::String(self.calc.clone())),
"EGU" => Some(EpicsValue::String(self.egu.clone())),
"PREC" => Some(EpicsValue::Short(self.prec)),
"HOPR" => Some(EpicsValue::Double(self.hopr)),
"LOPR" => Some(EpicsValue::Double(self.lopr)),
"ADEL" => Some(EpicsValue::Double(self.adel)),
"MDEL" => Some(EpicsValue::Double(self.mdel)),
"LALM" => Some(EpicsValue::Double(self.lalm)),
"ALST" => Some(EpicsValue::Double(self.alst)),
"MLST" => Some(EpicsValue::Double(self.mlst)),
"CALC_ALARM" => Some(EpicsValue::Char(if self.calc_alarm { 1 } else { 0 })),
"PVAL" => Some(EpicsValue::Double(self.pval)),
"OOPT" => Some(EpicsValue::Short(self.oopt)),
"ODLY" => Some(EpicsValue::Double(self.odly)),
"DLYA" => Some(EpicsValue::Short(self.dlya)),
"DOPT" => Some(EpicsValue::Short(self.dopt)),
"OCAL" => Some(EpicsValue::String(self.ocal.clone())),
"OVAL" => Some(EpicsValue::Double(self.oval)),
"IVOA" => Some(EpicsValue::Short(self.ivoa)),
"IVOV" => Some(EpicsValue::Double(self.ivov)),
"INPA" => Some(EpicsValue::String(self.inpa.clone())),
"INPB" => Some(EpicsValue::String(self.inpb.clone())),
"INPC" => Some(EpicsValue::String(self.inpc.clone())),
"INPD" => Some(EpicsValue::String(self.inpd.clone())),
"INPE" => Some(EpicsValue::String(self.inpe.clone())),
"INPF" => Some(EpicsValue::String(self.inpf.clone())),
"INPG" => Some(EpicsValue::String(self.inpg.clone())),
"INPH" => Some(EpicsValue::String(self.inph.clone())),
"INPI" => Some(EpicsValue::String(self.inpi.clone())),
"INPJ" => Some(EpicsValue::String(self.inpj.clone())),
"INPK" => Some(EpicsValue::String(self.inpk.clone())),
"INPL" => Some(EpicsValue::String(self.inpl.clone())),
"INPM" => Some(EpicsValue::String(self.inpm.clone())),
"INPN" => Some(EpicsValue::String(self.inpn.clone())),
"INPO" => Some(EpicsValue::String(self.inpo.clone())),
"INPP" => Some(EpicsValue::String(self.inpp.clone())),
"INPQ" => Some(EpicsValue::String(self.inpq.clone())),
"INPR" => Some(EpicsValue::String(self.inpr.clone())),
"INPS" => Some(EpicsValue::String(self.inps.clone())),
"INPT" => Some(EpicsValue::String(self.inpt.clone())),
"INPU" => Some(EpicsValue::String(self.inpu.clone())),
"A" => Some(EpicsValue::Double(self.a)),
"B" => Some(EpicsValue::Double(self.b)),
"C" => Some(EpicsValue::Double(self.c)),
"D" => Some(EpicsValue::Double(self.d)),
"E" => Some(EpicsValue::Double(self.e)),
"F" => Some(EpicsValue::Double(self.f)),
"G" => Some(EpicsValue::Double(self.g)),
"H" => Some(EpicsValue::Double(self.h)),
"I" => Some(EpicsValue::Double(self.i)),
"J" => Some(EpicsValue::Double(self.j)),
"K" => Some(EpicsValue::Double(self.k)),
"L" => Some(EpicsValue::Double(self.l)),
"M" => Some(EpicsValue::Double(self.m)),
"N" => Some(EpicsValue::Double(self.n)),
"O" => Some(EpicsValue::Double(self.o)),
"P" => Some(EpicsValue::Double(self.p)),
"Q" => Some(EpicsValue::Double(self.q)),
"R" => Some(EpicsValue::Double(self.r)),
"S" => Some(EpicsValue::Double(self.s)),
"T" => Some(EpicsValue::Double(self.t)),
"U" => Some(EpicsValue::Double(self.u)),
"LA" => Some(EpicsValue::Double(self.la)),
"LB" => Some(EpicsValue::Double(self.lb)),
"LC" => Some(EpicsValue::Double(self.lc)),
"LD" => Some(EpicsValue::Double(self.ld)),
"LE" => Some(EpicsValue::Double(self.le)),
"LF" => Some(EpicsValue::Double(self.lf)),
"LG" => Some(EpicsValue::Double(self.lg)),
"LH" => Some(EpicsValue::Double(self.lh)),
"LI" => Some(EpicsValue::Double(self.li)),
"LJ" => Some(EpicsValue::Double(self.lj)),
"LK" => Some(EpicsValue::Double(self.lk)),
"LL" => Some(EpicsValue::Double(self.ll)),
"LM" => Some(EpicsValue::Double(self.lm)),
"LN" => Some(EpicsValue::Double(self.ln)),
"LO" => Some(EpicsValue::Double(self.lo)),
"LP" => Some(EpicsValue::Double(self.lp)),
"LQ" => Some(EpicsValue::Double(self.lq)),
"LR" => Some(EpicsValue::Double(self.lr)),
"LS" => Some(EpicsValue::Double(self.ls)),
"LT" => Some(EpicsValue::Double(self.lt)),
"LU" => Some(EpicsValue::Double(self.lu)),
_ => None,
}
}
fn put_field(&mut self, name: &str, value: EpicsValue) -> CaResult<()> {
match name {
"VAL" => match value {
EpicsValue::Double(v) => {
self.val = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("VAL".into())),
},
"CALC" => match value {
EpicsValue::String(s) => {
self.rpcl = crate::calc::compile(&s).ok();
self.calc = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("CALC".into())),
},
"EGU" => match value {
EpicsValue::String(s) => {
self.egu = s;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"PREC" => match value {
EpicsValue::Short(v) => {
self.prec = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"HOPR" => match value {
EpicsValue::Double(v) => {
self.hopr = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"LOPR" => match value {
EpicsValue::Double(v) => {
self.lopr = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"ADEL" => match value {
EpicsValue::Double(v) => {
self.adel = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"MDEL" => match value {
EpicsValue::Double(v) => {
self.mdel = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"LALM" => match value {
EpicsValue::Double(v) => {
self.lalm = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"ALST" => match value {
EpicsValue::Double(v) => {
self.alst = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"MLST" => match value {
EpicsValue::Double(v) => {
self.mlst = v;
Ok(())
}
_ => Err(CaError::TypeMismatch(name.into())),
},
"OOPT" => match value {
EpicsValue::Short(v) => {
self.oopt = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("OOPT".into())),
},
"ODLY" => match value {
EpicsValue::Double(v) => {
self.odly = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("ODLY".into())),
},
"DLYA" => Err(CaError::ReadOnlyField("DLYA".into())),
"DOPT" => match value {
EpicsValue::Short(v) => {
self.dopt = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("DOPT".into())),
},
"OCAL" => match value {
EpicsValue::String(s) => {
self.orpc = crate::calc::compile(&s).ok();
self.ocal = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("OCAL".into())),
},
"OVAL" => match value {
EpicsValue::Double(v) => {
self.oval = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("OVAL".into())),
},
"IVOA" => match value {
EpicsValue::Short(v) => {
self.ivoa = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("IVOA".into())),
},
"IVOV" => match value {
EpicsValue::Double(v) => {
self.ivov = v;
Ok(())
}
_ => Err(CaError::TypeMismatch("IVOV".into())),
},
"INPA" => match value {
EpicsValue::String(s) => {
self.inpa = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPA".into())),
},
"INPB" => match value {
EpicsValue::String(s) => {
self.inpb = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPB".into())),
},
"INPC" => match value {
EpicsValue::String(s) => {
self.inpc = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPC".into())),
},
"INPD" => match value {
EpicsValue::String(s) => {
self.inpd = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPD".into())),
},
"INPE" => match value {
EpicsValue::String(s) => {
self.inpe = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPE".into())),
},
"INPF" => match value {
EpicsValue::String(s) => {
self.inpf = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPF".into())),
},
"INPG" => match value {
EpicsValue::String(s) => {
self.inpg = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPG".into())),
},
"INPH" => match value {
EpicsValue::String(s) => {
self.inph = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPH".into())),
},
"INPI" => match value {
EpicsValue::String(s) => {
self.inpi = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPI".into())),
},
"INPJ" => match value {
EpicsValue::String(s) => {
self.inpj = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPJ".into())),
},
"INPK" => match value {
EpicsValue::String(s) => {
self.inpk = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPK".into())),
},
"INPL" => match value {
EpicsValue::String(s) => {
self.inpl = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPL".into())),
},
"INPM" => match value {
EpicsValue::String(s) => {
self.inpm = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPM".into())),
},
"INPN" => match value {
EpicsValue::String(s) => {
self.inpn = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPN".into())),
},
"INPO" => match value {
EpicsValue::String(s) => {
self.inpo = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPO".into())),
},
"INPP" => match value {
EpicsValue::String(s) => {
self.inpp = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPP".into())),
},
"INPQ" => match value {
EpicsValue::String(s) => {
self.inpq = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPQ".into())),
},
"INPR" => match value {
EpicsValue::String(s) => {
self.inpr = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPR".into())),
},
"INPS" => match value {
EpicsValue::String(s) => {
self.inps = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPS".into())),
},
"INPT" => match value {
EpicsValue::String(s) => {
self.inpt = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPT".into())),
},
"INPU" => match value {
EpicsValue::String(s) => {
self.inpu = s;
Ok(())
}
_ => Err(CaError::TypeMismatch("INPU".into())),
},
"A" | "B" | "C" | "D" | "E" | "F" | "G" | "H" | "I" | "J" | "K" | "L" | "M" | "N"
| "O" | "P" | "Q" | "R" | "S" | "T" | "U" => {
let v = value
.to_f64()
.ok_or_else(|| CaError::TypeMismatch(name.into()))?;
match name {
"A" => self.a = v,
"B" => self.b = v,
"C" => self.c = v,
"D" => self.d = v,
"E" => self.e = v,
"F" => self.f = v,
"G" => self.g = v,
"H" => self.h = v,
"I" => self.i = v,
"J" => self.j = v,
"K" => self.k = v,
"L" => self.l = v,
"M" => self.m = v,
"N" => self.n = v,
"O" => self.o = v,
"P" => self.p = v,
"Q" => self.q = v,
"R" => self.r = v,
"S" => self.s = v,
"T" => self.t = v,
"U" => self.u = v,
_ => unreachable!(),
}
Ok(())
}
_ => Err(CaError::FieldNotFound(name.to_string())),
}
}
fn field_list(&self) -> &'static [FieldDesc] {
CALCOUT_FIELDS
}
fn multi_input_links(&self) -> &[(&'static str, &'static str)] {
&[
("INPA", "A"),
("INPB", "B"),
("INPC", "C"),
("INPD", "D"),
("INPE", "E"),
("INPF", "F"),
("INPG", "G"),
("INPH", "H"),
("INPI", "I"),
("INPJ", "J"),
("INPK", "K"),
("INPL", "L"),
("INPM", "M"),
("INPN", "N"),
("INPO", "O"),
("INPP", "P"),
("INPQ", "Q"),
("INPR", "R"),
("INPS", "S"),
("INPT", "T"),
("INPU", "U"),
]
}
fn should_output(&self) -> bool {
self.cached_should_output
}
fn can_device_write(&self) -> bool {
false
}
}