/* START LIBRARY DESCRIPTION *********************************************
PARSER.LIB
	Copyright (c) 1997, Avtech Electrosystems Ltd.

DESCRIPTION:
	Parser functions.

SUPPORT LIB'S:
END DESCRIPTION **********************************************************/


#include "error_utils.h"
#include "parser.h"
#include "flash.h"
#include "nicutils.h"
#include "version.h"
#include "i2c.h"
#include "lcd.h"
#include "menus.h"
#include "gpib.h"

#include <glib/gprintf.h>

extern void send_message(gchar* message);

//STATICS
static int query_int (gchar** response, int n);
static int query_min_max_float (gchar** response, char *parameter, float min_val, float max_val);
static int query_min_max_int (gchar** response, char *parameter, int min_val, int max_val);
static int query_string (gchar** response, char *in);
static int query_float (gchar** response, float value);

static int process_float_param (char *parameter, float *value, float min_val, float max_val, int zero_mode);
static int process_int_param (char *parameter, int *value, int item_count, int *valid_nums, int allow_on_off);
static int process_int_range (char *parameter, int *value, int min_val, int max_val);
static int process_on_off (char *parameter, int *value);
static int process_two_ints (char *parameter, int *value, int min, int max);

static int check_channel_ok (int channel, int enabled_channels, char chankey);
static int Parser_id_word(char *id_me, int *channel, int *with_id_code);
static int Parser_find_commands(int commands[], int command_depth);
static gchar* filter_input (gchar *raw_in);
static int Parser_get_unit(char **parameter, char **units);
static int Parser_channel (int *channel,int with_id_code,int routine_num);
static int Handle_Units(float *mult,char *units, char *base);
static int Is_Min_Command(char *text);
static int Is_Max_Command(char *text);

static int Go_syst_err_11(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_syst_errcnt66(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_Str_eprom_47(gchar** response, int channel, char *loc_string,char *store_string,int command_type);
static int Go_int_eprom_48(gchar** response, int channel, char *loc_string,char *store_string,int command_type);
static int Go_Float_eprom51(gchar** response, int channel, char *loc_string,char *store_string,int command_type);
static int Go_char_eprom_70(gchar** response, int channel, char *loc_string,char *store_string,int command_type);
static int Go_eprom_siz_86(gchar** response, int channel, int command_type);
static int Go_sys_net_91(gchar** response, int channel, char *parameter,char *units,int command_type);

static int Go_ampl_26(gchar** response, int channel, char *parameter,char *units,int command_type,char *mode);
static int Go_pw_36(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_duty_37(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_offset_29(gchar** response, int channel, char *parameter,char *units,int command_type,char *mode);
static int Go_idn_5(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_freq_32_33(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_period_35(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_delay_39(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_dbl_pulse_40(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_func_34(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_polarity_42(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_hold_38(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_output_55(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_trig_source46(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_gate_type_56(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_gate_level_67(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_delay_test58(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_wai_10(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_opc_5(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_ese_2(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_esr_3(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_cls_1(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_sre_7(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_stb_8(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_tst_9(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_syst_ver_12(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_event_13(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_oper_enable15(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_ques_enable18(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_preset_19(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_zout_20(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_prot_trip_21(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_gpib_addr_59(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_ser_baud_60(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_ser_rts_64(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_load_68(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_meas_ampl_69(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_rst_6(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_calib_amp_72(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_calib_mon_73(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_mon_step_74(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_outputtype_75(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_calib_os_76(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_calib_pw_77(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_routeclose_78(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Parse_chan_list(int channel,char *parameter,int *primary_selected, int *secondary_selected,gchar** response);
static int Go_dly_shift_82(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_amp_pnt_83(gchar** response, int channel, char *parameter,char *units,int command_type,int cal_type);
static int Go_puls_count_88(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_puls_sep_89(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_sys_pwd_92(gchar** response, int channel, char *parameter,int command_type);
static int Go_eprom_sus_93(gchar** response, int channel, char *parameter,int command_type);
static int Go_rise_time_94(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_rcl_53(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_sav_54(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_soft_current_limit_96(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_curr_slew_98(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_cal_100(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_cal_interval_101(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_eprom_reset_102(gchar** response, int channel, char *parameter,char *units,int command_type);
static int Go_atten_103(gchar** response, int channel, char *parameter,char *units,int command_type);

static int Parser_id_word(char *id_me, int *channel, int *with_id_code)
{
	int id_code;
	int set_new_channel;

	set_new_channel=0;

	/* detect channel suffixes in the range of 0-99 */
	if (isdigit((int) id_me[strlen(id_me)-1])) {
		set_new_channel=1;
		*channel=atoi(id_me+strlen(id_me)-1);
		if (isdigit((int) id_me[strlen(id_me)-2])) {
			*channel=atoi(id_me+strlen(id_me)-2);
			id_me[strlen(id_me)-2]=0;
		} else {
			id_me[strlen(id_me)-1]=0;
		}
	}


	if (!strcmp(id_me,"") ) {
		id_code = 0;
	} else if (!strcmp(id_me,"*cls") ) {
		id_code = 1;
	} else if (!strcmp(id_me,"*ese") ) {
		id_code = 2;
	} else if (!strcmp(id_me,"*esr") ) {
		id_code = 4;
	} else if (!strcmp(id_me,"*idn") ) {
		id_code = 5;
	} else if (!strcmp(id_me,"*opc") ) {
		id_code = 6;
	} else if (!strcmp(id_me,"*rst") ) {
		id_code = 8;
	} else if (!strcmp(id_me,"*sre") ) {
		id_code = 9;
	} else if (!strcmp(id_me,"*stb") ) {
		id_code = 10;
	} else if (!strcmp(id_me,"*tst") ) {
		id_code = 11;
	} else if (!strcmp(id_me,"*wai") ) {
		id_code = 12;
	} else if (!strcmp(id_me,"system") || !strcmp(id_me,"syst")) {
		id_code = 13;
	} else if (!strcmp(id_me,"error") || !strcmp(id_me,"err"))	{
		id_code = 14;
	} else if (!strcmp(id_me,"version") || !strcmp(id_me,"vers")) {
		id_code = 15;
	} else if (!strcmp(id_me,"status") || !strcmp(id_me,"stat") || !strcmp(id_me,"state")) {
		id_code = 16;
	} else if (!strcmp(id_me,"operation") || !strcmp(id_me,"oper")) {
		id_code = 17;
	} else if (!strcmp(id_me,"event") || !strcmp(id_me,"even")) {
		id_code = 18;
	} else if (!strcmp(id_me,"condition") || !strcmp(id_me,"cond")) {
		id_code = 19;
	} else if (!strcmp(id_me,"enable") || !strcmp(id_me,"enab")) {
		id_code = 20;
	} else if (!strcmp(id_me,"questionable") || !strcmp(id_me,"ques")) {
		id_code = 21;
	} else if (!strcmp(id_me,"preset") || !strcmp(id_me,"pres")) {
		id_code = 22;
	} else if (!strcmp(id_me,"diagnostic") || !strcmp(id_me,"diag")) {
		id_code = 23;
	} else if (!strcmp(id_me,"output") || !strcmp(id_me,"outp")) {
		id_code = 24;
	} else if (!strcmp(id_me,"impedance") || !strcmp(id_me,"imp"))	{
		id_code = 25;
	} else if (!strcmp(id_me,"protection") || !strcmp(id_me,"prot")) {
		id_code = 26;
	} else if (!strcmp(id_me,"tripped") || !strcmp(id_me,"trip")) {
		id_code = 27;
	} else if (!strcmp(id_me,"source") || !strcmp(id_me,"sour")) {
		id_code = 28;
	} else if (!strcmp(id_me,"current") || !strcmp(id_me,"curr")) {
		id_code = 29;
	} else if (!strcmp(id_me,"level") || !strcmp(id_me,"lev"))	{
		id_code = 30;
	} else if (!strcmp(id_me,"immediate") || !strcmp(id_me,"imm"))	{
		id_code = 31;
	} else if (!strcmp(id_me,"amplitude") || !strcmp(id_me,"ampl")) {
		id_code = 32;
	} else if (!strcmp(id_me,"offset") || !strcmp(id_me,"offs")) {
		id_code = 33;
	} else if (!strcmp(id_me,"high") ) {
		id_code = 34;
	} else if (!strcmp(id_me,"low") ) {
		id_code = 35;
	} else if (!strcmp(id_me,"voltage") || !strcmp(id_me,"volt")) {
		id_code = 36;
	} else if (!strcmp(id_me,"frequency") || !strcmp(id_me,"freq")) {
		id_code = 37;
	} else if (!strcmp(id_me,"function") || !strcmp(id_me,"func")) {
		id_code = 38;
	} else if (!strcmp(id_me,"shape") || !strcmp(id_me,"shap")) {
		id_code = 39;
	} else if (!strcmp(id_me,"pulse") || !strcmp(id_me,"puls")) {
		id_code = 40;
	} else if (!strcmp(id_me,"period") || !strcmp(id_me,"per"))	{
		id_code = 41;
	} else if (!strcmp(id_me,"width") || !strcmp(id_me,"widt")) {
		id_code = 42;
	} else if (!strcmp(id_me,"dcycle") || !strcmp(id_me,"dcyc")) {
		id_code = 43;
	} else if (!strcmp(id_me,"hold") ) {
		id_code = 44;
	} else if (!strcmp(id_me,"delay") || !strcmp(id_me,"del"))	{
		id_code = 45;
	} else if (!strcmp(id_me,"double") || !strcmp(id_me,"doub")) {
		id_code = 46;
	} else if (!strcmp(id_me,"polarity") || !strcmp(id_me,"pol"))	{
		id_code = 47;
	} else if (!strcmp(id_me,"beeper") || !strcmp(id_me,"beep")) {
		id_code = 48;
	} else if (!strcmp(id_me,"cw") )	{
		id_code = 49;
	} else if (!strcmp(id_me,"fixed") || !strcmp(id_me,"fix"))	{
		id_code = 50;
	} else if (!strcmp(id_me,"abort") || !strcmp(id_me,"abor")) {
		id_code = 51;
	} else if (!strcmp(id_me,"initiate") || !strcmp(id_me,"init")) {
		id_code = 52;
	} else if (!strcmp(id_me,"continuous")) {
		id_code = 53;
	} else if (!strcmp(id_me,"trigger") || !strcmp(id_me,"trig")) {
		id_code = 54;
	} else if (!strcmp(id_me,"integer") || !strcmp(id_me,"int"))	{
		id_code = 55;
	} else if (!strcmp(id_me,"float") )	{
		id_code = 56;
	} else if (!strcmp(id_me,"eprom") ) {
		id_code = 57;
	} else if (!strcmp(id_me,"string") ) {
		id_code = 58;
	} else if (!strcmp(id_me,"bit") ) {
		id_code = 59;
	} else if (!strcmp(id_me,"set") ) {
		id_code = 60;
	} else if (!strcmp(id_me,"clear") )	{
		id_code = 61;
	} else if (!strcmp(id_me,"shiftreg") )	{
		id_code = 62;
	} else if (!strcmp(id_me,"*rcl") ) {
		id_code = 63;
	} else if (!strcmp(id_me,"*sav") ) {
		id_code = 64;
	} else if (!strcmp(id_me,"gate") ) {
		id_code = 65;
	} else if (!strcmp(id_me,"local") || !strcmp(id_me,"logout") || !strcmp(id_me,"exit") || !strcmp(id_me,"quit") )	{
		id_code = 66;
	} else if (!strcmp(id_me,"test") ) {
		id_code = 67;
	} else if (!strcmp(id_me,"communicate") || !strcmp(id_me,"comm")) {
		id_code = 68;
	} else if (!strcmp(id_me,"address") || !strcmp(id_me,"addr")) {
		id_code = 69;
	} else if (!strcmp(id_me,"gpib") ) {
		id_code = 70;
	} else if (!strcmp(id_me,"serial") || !strcmp(id_me,"ser")) {
		id_code = 71;
	} else if (!strcmp(id_me,"receive") || !strcmp(id_me,"rec")) {
		id_code = 72;
	} else if (!strcmp(id_me,"baud") ) {
		id_code = 73;
	} else if (!strcmp(id_me,"parity") || !strcmp(id_me,"par")) {
		id_code = 74;
	} else if (!strcmp(id_me,"type") ) {
		id_code = 75;
	} else if (!strcmp(id_me,"bits") ) {
		id_code = 76;
	} else if (!strcmp(id_me,"sbits") || !strcmp(id_me,"sbit")) {
		id_code = 77;
	} else if (!strcmp(id_me,"echo") ) {
		id_code = 78;
	} else if (!strcmp(id_me,"rts") ) {
		id_code = 79;
	} else if (!strcmp(id_me,"control") || !strcmp(id_me,"cont"))	{
		id_code = 80;
	} else if (!strcmp(id_me,"next") ) {
		id_code = 81;
	} else if (!strcmp(id_me,"count") || !strcmp(id_me,"coun"))	{
		id_code = 82;
	} else if (!strcmp(id_me,"display") || !strcmp(id_me,"disp"))	{
		id_code = 83;
	} else if (!strcmp(id_me,"brightness") || !strcmp(id_me,"brig"))	{
		id_code = 84;
	} else if (!strcmp(id_me,"load") ) {
		id_code = 85;
	} else if (!strcmp(id_me,"measure") || !strcmp(id_me,"meas"))	{
		id_code = 86;
	} else if (!strcmp(id_me,"char") ) {
		id_code = 87;
	} else if (!strcmp(id_me,"calibration") || !strcmp(id_me,"cal"))	{
		id_code = 88;
	} else if (!strcmp(id_me,"monitor") || !strcmp(id_me,"mon"))	{
		id_code = 89;
	} else if (!strcmp(id_me,"step") ) {
		id_code = 90;
	} else if (!strcmp(id_me,"route") || !strcmp(id_me,"rout")) {
		id_code = 91;
	} else if (!strcmp(id_me,"close") || !strcmp(id_me,"clos")) {
		id_code = 92;
	} else if (!strcmp(id_me,"point") ) {
		id_code = 93;
	} else if (!strcmp(id_me,"shift") ) {
		id_code = 94;
	} else if (!strcmp(id_me,"scale") ) {
		id_code = 95;
	} else if (!strcmp(id_me,"null") ) {
		id_code = 96;
	} else if (!strcmp(id_me,"size") ) {
		id_code = 97;
	} else if (!strcmp(id_me,"separation") || !strcmp(id_me,"sep"))	{
		id_code = 98;
	} else if (!strcmp(id_me,"network") || !strcmp(id_me,"net"))	{
		id_code = 99;
	} else if (!strcmp(id_me,"password") || !strcmp(id_me,"pass"))	{
		id_code = 100;
	} else if (!strcmp(id_me,"new") ) {
		id_code = 101;
	} else if (!strcmp(id_me,"suspend") ) {
		id_code = 102;
	} else if (!strcmp(id_me,"transition") || !strcmp(id_me,"tran"))	{
		id_code = 103;
	} else if (!strcmp(id_me,"leading") || !strcmp(id_me,"lead"))	{
		id_code = 104;
	} else if (!strcmp(id_me,"limit") || !strcmp(id_me,"lim"))	{
		id_code = 105;
	} else if (!strcmp(id_me,"slew") )	{
		id_code = 106;
	} else if (!strcmp(id_me,"all") ) {
		id_code = 107;
	} else if (!strcmp(id_me,"reset") || !strcmp(id_me,"res"))      {
		id_code = 108;
	} else if (!strcmp(id_me,"attenuator") || !strcmp(id_me,"att"))      {
		id_code = 109;
	} else {
		id_code = 9999;
	}

	if (set_new_channel) {
		*with_id_code=id_code;
	}

	return id_code;

}

static int Parser_find_commands(int commands[], int command_depth)
{
#define optional  16384 /* set bit high if optional */
#define opt_mask 16383  /* default bit mask */
#define max_tokens_in_sentence 7
#define max_no_of_sentences 110

	/* Valid command -sequences-. The -individual- commands are defined in id_word. */

	static const int sentences[max_no_of_sentences][max_tokens_in_sentence] =	{
		{0},	/* null - error */
		{1},	/* cls - sentence 1 */
		{2},	/* ese - sentence 2 */
		{4},	/* esr - sentence 3 */
		{5},	/* idn - sentence 4 */
		{6},	/* opc - sentence 5 */
		{8},	/* rst - sentence 6 */
		{9},	/* sre - sentence 7 */
		{10},	/* stb - sentence 8 */
		{11},	/* tst - sentence 9 */
		{12},	/* wai - sentence 10 */
		{13,14,81|optional},	/* syst:err:next - 11 */
		{13,15},		/* syst:vers - 12 */
		{16,17,18|optional},	/* stat:oper:event - 13 */
		{16,17,19},		/* stat:oper:cond - 14 */
		{16,17,20},		/* stat:oper:enable - 15 */
		{16,21,18|optional},	/* stat:ques:event - 16 */
		{16,21,19},		/* stat:ques:cond - 17 */
		{16,21,20},		/* stat:ques:enable - 18 */
		{16,22},		/* stat:preset - 19 */
		{24,25},		/* output:impedance - 20 */
		{24,26,27},		/* output:prot:tripped - 21 */
		{28|optional,29,30|optional,31|optional,32|optional},	/* sour:curr:lev:imm:ampl - 22 */
		{28|optional,29,30|optional,31|optional,33},		/* sour:curr:lev:imm:offset - 23 - not used */
		{28|optional,29,30|optional,31|optional,34},		/* sour:curr:lev:imm:high - 24 - not used */
		{28|optional,29,30|optional,31|optional,35},		/* sour:curr:lev:imm:low - 25 */
		{28|optional,36,30|optional,31|optional,32|optional},	/* sour:volt:lev:imm:ampl - 26 */
		{28|optional,36,30|optional,31|optional,33},		/* sour:volt:lev:imm:offset - 27 - not used */
		{28|optional,36,30|optional,31|optional,34},	/* sour:volt:lev:imm:high - 28 - not used */
		{28|optional,36,30|optional,31|optional,35},	/* sour:volt:lev:imm:low - 29 */
		{28|optional,29,26,27},		/* sour:curr:prot:tripped? - 30 */
		{28|optional,36,26,27},		/* sour:volt:prot:tripped? - 31 */
		{28|optional,37,49|optional},	/* sour:freq:cw - 32 */
		{28|optional,37,50|optional},	/* sour:freq:fixed - 33 */
		{28|optional,38,39|optional},	/* sour:func:shape - 34 */
		{28|optional,40,41},		/* sour:puls:per - 35 */
		{28|optional,40,42},		/* sour:puls:width - 36 */
		{28|optional,40,43},		/* sour:puls:dcyc - 37 */
		{28|optional,40,44},		/* sour:puls:hold - 38 */
		{28|optional,40,45},		/* sour:puls:delay - 39 */
		{28|optional,40,46,16|optional},	/* sour:puls:doub:state - 40 */
		{28|optional,40,46,45},			/* sour:puls:doub:delay - 41 */
		{28|optional,40,47},			/* sour:puls:pol - 42 */
		{13,48,31|optional},	/* syst:beep:imm - 43 - not used */
		{52,31|optional,53},	/* init:imm:cont - 44 - not used */
		{51},			/* abort - 45 - not used */
		{54,31|optional,28},	/* trig:imm:sour - 46 */
		{23,57,58},		/* diag:eprom:string - 47 */
		{23,57,55},		/* diag:eprom:int - 48 */
		{23,57,59,60},		/* diag:eprom:bit:set - 49 - not used */
		{23,57,59,61},		/* diag:eprom:bit:clear - 50 - not used */
		{23,57,56},		/* diag:eprom:float - 51 */
		{23,62},		/* diag:shiftreg - 52 */
		{63},			/* *rcl - 53 */
		{64},			/* *sav - 54 */
		{24,16|optional},	/* output:state - 55 */
		{28|optional,40,65,75},	/* sour:puls:gate:type - 56 */
		{66},			/* local (RS232 mode only) - 57 */
		{23,67,45},		/* diag:test:delay - 58 */
		{13,68,70,69},			/* syst:comm:gpib:addr - 59 */
		{13,68,71,72|optional,73},	/* syst:comm:ser:rec:baud - 60 */
		{13,68,71,72|optional,74,75|optional},	/* syst:comm:ser:rec:parity - 61 - not used */
		{13,68,71,72|optional,76},	/* syst:comm:ser:rec:bits - 62 - not used */
		{13,68,71,72|optional,77},	/* syst:comm:ser:rec:sbits - 63 - not used */
		{13,68,71,80,79},		/* syst:comm:ser:control:rts - 64 */
		{13,68,71,72|optional,78},	/* syst:comm:ser:rec:echo - 65 - not used */
		{13,14,82},	  		/* syst:err:count - 66 */
		{28|optional,40,65,30},		/* sour:puls:gate:level - 67 */
		{24,85},			/* output:load - 68 */
		{86,32},			/* meas:ampl? - 69 */
		{23,57,87},			/* diag:eprom:char - 70 */
		{23,88},			/* diag:calib - 71 */
		{23,32,88,95|optional}, 	/* diag:ampl:calib:scale - 72 */
		{23,89,88,95|optional}, 	/* diag:mon:calib:scale - 73 */
		{23,89,90},			/* diag:mon:step - 74 */
		{24,75},			/* output:type - 75 */
		{23,33,88,95|optional}, 	/* diag:offset:calib:scale - 76 */
		{23,40,42,88,94},		/* diag:pulse:width:calib:shift - 77 */
		{91,92},			/* route:close - 78 */
		{23,40,42,88,93},		/* diag:pulse:width:calib:point - 79 */
		{23,40,45,88,93},		/* diag:pulse:delay:calib:point - 80 */
		{23,40,41,88,93},		/* diag:pulse:period:calib:point - 81 */
		{23,40,45,88,94},		/* diag:pulse:delay:calib:shift - 82 */
		{23,32,88,93},			/* diag:ampl:calib:point - 83 */
		{23,33,88,93},			/* diag:offset:calib:point - 84 */
		{23,33,96,93},			/* diag:offset:null:point - 85 */
		{23,57,97},			/* diag:eprom:size - 86 */
		{83,84},			/* display:brightness - 87 */
		{28|optional,40,82},		/* sour:puls:count - 88 */
		{28|optional,40,98},		/* sour:puls:separation - 89 */
		{23,40,98,88,93},		/* diag:pulse:separation:calib:point - 90 */
		{13,68,99},			/* system:comm:network - 91 */
		{13,100,101},			/* system:password:new - 92 */
		{23,57,102},				/* diag:eprom:suspend - 93 */
		{28|optional,40,103,104|optional},	/* sour:puls:transition:leading - 94 */
		{23,40,103,88,93},			/* diag:pulse:transition:calib:point - 95 */
		{28|optional,29,105,32|optional},	/* sour:curr:limit:ampl - 96 */
		{23,24,85,88,93},			/* diag:output:load:calib:point - 97 */
		{28|optional,29,106},  		/* sour:curr:slew - 98 */
		{23,106,88,93},			/* diag:slew:calib:point - 99 */
		{88,107|optional},  		/* calibration:all - 100 */
		{88,37},			/* calibration:frequency - 101 */
		{23,57,108},			/* diag:eprom:reset - 102 */
		{23,109,16|optional},       	/* diag:attenuator:state - 103 */
	};


	int token_num;		 /* which token in the command[] list is being considered */
	int sentence_num;	 /* which sentence is being considered as a match */
	int sentence_found;  /* matching sentence found yet? */
	int sent_pos;		  /* position in current sentence */
	int invalid_match;	/* flag if definitely the wrong match */
	int local_match_pos; /* position (or pos+1, depends) of most-recently-matched token in sentence */
	int local_match;	  /* match for current token found yet? */
	int prev_match;		/* were there previous local matches? */
	int i;					/* a counter */

	sentence_num = 1;
	sent_pos = 0;

	/* check each "sentence" (corresponding to a different routine or function)
		until a match is found, or until their are no possible sentences left */
	for (sentence_found = 0; (!sentence_found && sentence_num < max_no_of_sentences); ++sentence_num) {
		local_match_pos=0;
		invalid_match=0;
		prev_match=0;
		/* start with the first input token, until all have been matched, or one
		is unmatchable */
		for (token_num = 0; (token_num < command_depth) && !invalid_match; ++token_num) {
			/* start after most recent matching token */
			sent_pos=local_match_pos;

			/* compare current token to the sentence components until a matching component
				is found, or until there are none left */
			for (local_match=0; (!local_match) && (sent_pos<max_tokens_in_sentence); ++sent_pos) {

				/* compare here */
				if (commands[token_num]==((sentences[sentence_num][sent_pos]) & opt_mask)) {
					/* flag a local token match (not a whole sentence match) */
					local_match=1;
				}
			}
			if (local_match) {
				/* if any of the components in the sentence were skipped between the last two
					matches, make sure that they were optional ones */
				if (!prev_match) {
					local_match_pos=-1;
				}
				/* if there was no local match previously, then the initial value
					of local_match_pos is misleading - there was no match at 0. */
				for (i=(local_match_pos+1); i<(sent_pos-1); ++i) {
					/* if they weren't optional, the match is not valid */
					if (!(sentences[sentence_num][i] & optional)) {
						invalid_match=1;
					}
				}
				/* counteract the last "++i" due to the for loop */
				local_match_pos = sent_pos - 1;
				if (local_match && !invalid_match) {
					prev_match=1;
				}
			}

			/* if no match for this one token existed in this sentence, mark this sentence as invalid */
			if (!local_match) {
				invalid_match = 1;
			}
		}
		/* if no invalid token match was found for any of the tokens in this sentence,
			then it's an overall sentence match! */
		if (!invalid_match) {
			sentence_found=1;
		}
	}
	--sentence_num;

	if (!sentence_found) {
		sentence_num = 0;
	}
	/* 0 is the no-match value */

	return sentence_num;
}


static int Parser_get_unit(char **parameter, char **units)
{
	/* this function takes a parameter like "1e+6 ms" and breaks it into parameter="1e+6" and
		units="ms" */

	if (*parameter == NULL) {
		*parameter = g_strdup ("");
	}

	g_assert (*units == NULL);

	gchar *end_ptr;
	g_strtod (*parameter, &end_ptr);

	if (end_ptr == *parameter) {
		*units = g_strdup ("");
	} else {
		*units = g_strdup (end_ptr);
		end_ptr[0] = 0;	// truncates parameter
	}

	g_strstrip (*units);
	g_strstrip (*parameter);

	int units_present = (*units[0] != 0);
	return units_present;
}


static gchar* regex_replace (gchar* in_string, gchar* regex_string, gchar* replace_with)
{
	return conditional_regex_replace (TRUE, in_string, regex_string, replace_with);
}


static gchar* filter_input (gchar *raw_in)
{
	g_strstrip (raw_in);
	gchar *step1 = g_strdup (raw_in);

	// replace multiple whitespace with single space
	gchar *step2 = regex_replace (step1, "\\s+", " ");
	g_free (step1);

	// remove spaces before and after semicolons in compound messages
	gchar *step3 = regex_replace (step2, "\\s*;\\s*", ";");
	g_free (step2);

	// remove semicolon at end
	gchar *step4 = regex_replace (step3, ";$", "");
	g_free (step3);

	// remove leading spaces, colons, semicolons
	gchar *step5 = regex_replace (step4, "^[\\s:;]*", "");
	g_free (step4);

	// last step may leave hanging whitespace at end
	g_strstrip (step5);

	// add single space to terminate
	gchar *filt = g_strdup_printf ("%s ", step5);
	g_free (step5);

	return filt;
}


void Parser_main (char *raw_in, int interactive_terminal, void(*cbfunc)(gpointer, gchar *), gpointer user_data)
{
	static GStaticMutex mutex = G_STATIC_MUTEX_INIT;
	g_static_mutex_lock (&mutex);

	int in_pos;				 /* this identifies the single character of in being processed */
	int command_depth;		/* how many command words have been parsed */
	int old_command_depth;	/* save this if in case of a compound message */
	int commands[max_commands_in_input];			/* list of identified command tokens */
	int old_commands[max_commands_in_input];	  /* list of identified command tokens from */
	/* last non-common, non-colon-started command*/
	int space_found;		  /* if true, a space was found. parameter may follow */
	int parameter_found;	 /* if true, there is a parameter at the end of the command string */
	int semicolon_found;								 /* to break up lines */
	int units_found;		  /* text found after parameter must be units */
	int i;													/* counter */
	int routine_num;		  	/* routine to execute, based on command */
	int dont_use_this_header;	/* the old head is saved in case of compound message, unless this is set */
	int compound_message;		/* this indicates that at least one semicolon has been found, so */
	/* the message is a compound one */

	int is_query;
	int command_type;			/* combination of is_query, parameter_found, and units_found */

	int channel;
	int with_id_code;

	gchar *in = filter_input (raw_in);

	in_pos = 0;							 /* start at the beginning of the input string */
	semicolon_found = 0;
	compound_message = 0;
	command_depth=0;
	old_command_depth=0;
	dont_use_this_header=NO;

	/* examine each letter in the string until EOS */
	while (in[in_pos] != 0) {

		gchar *units = NULL;
		gchar *response = NULL;
		gchar *error_response = NULL;
		gchar *parameter = NULL;

		channel=-1;
		with_id_code=0;
		space_found = 0;
		parameter_found = 0;							 /* no numeric parameters yet */
		units_found = 0;
		is_query = 0;

		GString *current_word = g_string_new ("");

		if (semicolon_found && !dont_use_this_header) {
			old_command_depth=command_depth-2;
			for (i=0; i<old_command_depth; ++i) {
				old_commands[i]=commands[i];
			}
		}

		command_depth=old_command_depth;
		for (i=0; i<command_depth; ++i) {
			commands[i]=old_commands[i];
		}

		semicolon_found=0;
		dont_use_this_header=NO;

		/* break it up if semicolons are present */
		while (!semicolon_found && in[in_pos] != 0) {

			in[in_pos]=tolower(in[in_pos]);
			/* everything to lowercase */

			/* words are separated by white space or colons or semicolons */
			if ( isspace(in[in_pos]) || in[in_pos] == ':' || in[in_pos] == ';')

			{

				if (in[in_pos] == ';') {
					++semicolon_found;
				}
				++compound_message;

				/* if it is a separator, ignore it if it is duplicated */
				if ( !isspace(in[in_pos-1])  && (in[in_pos-1] != ':') && (in[in_pos-1] != ';')) {
					/* valid separator, so terminate current command word string */

					if (space_found) {
						/* Just end things if it is a semicolon */
						if (in[in_pos]!=';') {
							g_string_append_c (current_word, in[in_pos]);
						}

						++parameter_found;

						// the parameter may have multiple words,
						// so this branch may be called repeatedly
						g_free (parameter);
						parameter = g_strdup (current_word->str);

					} else if (!space_found) {
						/* end of a regular command word - use it */

						commands[command_depth]=Parser_id_word(	current_word->str,
						                                        &channel,
						                                        &with_id_code);

						/* get ready for the next token */
						g_string_free (current_word, 1);
						current_word = g_string_new ("");

						if (	  commands[command_depth]<13
						                || commands[command_depth]==63
						                || commands[command_depth]==64) {
							dont_use_this_header=YES;
							commands[0]=commands[command_depth];
							command_depth=0;
						}

						++command_depth;		  /* indicate that a new command word is ready */

						/* indicate if the separator was a space - next bit parameter then */
						if (in[in_pos]==' ') {
							++space_found;
						}
					}
				} /* non-repeated separator */
				else if (compound_message && in[in_pos-1]==';' && in[in_pos]==':') {
					dont_use_this_header=YES;
					command_depth=0;
				}

			} /* a separator */
			else if (in[in_pos]!='?') {
				/* if not white space or separator, add character to current command word */
				g_string_append_c (current_word, in[in_pos]);
			} /* not a separator */
			else {
				++is_query;
				/* question mark found */
			}


			++in_pos;							/* move pointer to next letter */

		} /* not a semi-colon */

		commands[command_depth]=0;
		++command_depth;
		/* add end of tokens marker (0) */

		int error_num=OK;

		units_found = Parser_get_unit(&parameter,&units);

		if (!globals.Sys.startup_complete) {
			error_num=Startup_Not_Finished;
		}

		if (!error_num) {
			command_type=(is_query<<2) | (parameter_found?2:0) | units_found;

			if (commands[0]!=0) {
				routine_num = Parser_find_commands(commands,command_depth);

				/* check for valid channel number, based on position in command */
				if ( (error_num=Parser_channel(&channel,with_id_code,routine_num)) )
					/* not a valid command due to improper channel suffix */
				{
					routine_num=9999;
				}
			} else {
				routine_num=9999;
				/* ignore empty commands lists, generated by a white space string */
			}

			switch (routine_num) {
			case 0:
				error_num=Unrecognized;
				break;

			case 1:
				error_num=Go_cls_1(&response,channel,parameter,units,command_type);
				break;
			case 2:
				error_num=Go_ese_2(&response,channel,parameter,units,command_type);
				break;
			case 3:
				error_num=Go_esr_3(&response,channel,parameter,units,command_type);
				break;
			case 4:
				error_num=Go_idn_5(&response,channel,parameter,units,command_type);
				break;
			case 5:
				error_num=Go_opc_5(&response,channel,parameter,units,command_type);
				break;
			case 6:
				error_num=Go_rst_6(&response,channel,parameter,units,command_type);
				break;
			case 7:
				error_num=Go_sre_7(&response,channel,parameter,units,command_type);
				break;
			case 8:
				error_num=Go_stb_8(&response,channel,parameter,units,command_type);
				break;
			case 9:
				error_num=Go_tst_9(&response,channel,parameter,units,command_type);
				break;
			case 10:
				error_num=Go_wai_10(&response,channel,parameter,units,command_type);
				break;
			case 11:
				error_num=Go_syst_err_11(&response,channel,parameter,units,command_type);
				break;
			case 12:
				error_num=Go_syst_ver_12(&response,channel,parameter,units,command_type);
				break;
			case 13:
			case 14:
			case 16:
			case 17:
				error_num=Go_event_13(&response,channel,parameter,units,command_type);
				break;
			case 15:
				error_num=Go_oper_enable15(&response,channel,parameter,units,command_type);
				break;
			case 18:
				error_num=Go_ques_enable18(&response,channel,parameter,units,command_type);
				break;
			case 19:
				error_num=Go_preset_19(&response,channel,parameter,units,command_type);
				break;
			case 20:
				error_num=Go_zout_20(&response,channel,parameter,units,command_type);
				break;
			case 21:
			case 30:
			case 31:
				error_num=Go_prot_trip_21(&response,channel,parameter,units,command_type);
				break;
			case 22:
				error_num=Go_ampl_26(&response,channel,parameter,units,command_type,"a");
				break;
			case 25:
				error_num=Go_offset_29(&response,channel,parameter,units,command_type,"a");
				break;
			case 26:
				error_num=Go_ampl_26(&response,channel,parameter,units,command_type,"v");
				break;
			case 29:
				error_num=Go_offset_29(&response,channel,parameter,units,command_type,"v");
				break;
			case 32:
			case 33:
				error_num=Go_freq_32_33(&response,channel,parameter,units,command_type);
				break;
			case 34:
				error_num=Go_func_34(&response,channel,parameter,units,command_type);
				break;
			case 35:
				error_num=Go_period_35(&response,channel,parameter,units,command_type);
				break;
			case 36:
				error_num=Go_pw_36(&response,channel,parameter,units,command_type);
				break;
			case 37:
				error_num=Go_duty_37(&response,channel,parameter,units,command_type);
				break;
			case 38:
				error_num=Go_hold_38(&response,channel,parameter,units,command_type);
				break;
			case 39:
			case 41:
				error_num=Go_delay_39(&response,channel,parameter,units,command_type);
				break;
			case 40:
				error_num=Go_dbl_pulse_40(&response,channel,parameter,units,command_type);
				break;
			case 42:
				error_num=Go_polarity_42(&response,channel,parameter,units,command_type);
				break;
			case 46:
				error_num=Go_trig_source46(&response,channel,parameter,units,command_type);
				break;
			case 47:
				error_num=Go_Str_eprom_47(&response,channel,parameter,units,command_type);
				break;
			case 48:
				error_num=Go_int_eprom_48(&response,channel,parameter,units,command_type);
				break;
			case 51:
				error_num=Go_Float_eprom51(&response,channel,parameter,units,command_type);
				break;
			case 53:
				error_num=Go_rcl_53(&response,channel,parameter,units,command_type);
				break;
			case 54:
				error_num=Go_sav_54(&response,channel,parameter,units,command_type);
				break;
			case 55:
				error_num=Go_output_55(&response,channel,parameter,units,command_type);
				break;
			case 56:
				error_num=Go_gate_type_56(&response,channel,parameter,units,command_type);
				break;
			case 58:
				error_num=Go_delay_test58(&response,channel,parameter,units,command_type);
				break;
			case 59:
				error_num=Go_gpib_addr_59(&response,channel,parameter,units,command_type);
				break;
			case 60:
				error_num=Go_ser_baud_60(&response,channel,parameter,units,command_type);
				break;
			case 64:
				error_num=Go_ser_rts_64(&response,channel,parameter,units,command_type);
				break;
			case 66:
				error_num=Go_syst_errcnt66(&response,channel,parameter,units,command_type);
				break;
			case 67:
				error_num=Go_gate_level_67(&response,channel,parameter,units,command_type);
				break;
			case 68:
				error_num=Go_load_68(&response,channel,parameter,units,command_type);
				break;
			case 69:
				error_num=Go_meas_ampl_69(&response,channel,parameter,units,command_type);
				break;
			case 70:
				error_num=Go_char_eprom_70(&response,channel,parameter,units,command_type);
				break;
			case 72:
				error_num=Go_calib_amp_72(&response,channel,parameter,units,command_type);
				break;
			case 73:
				error_num=Go_calib_mon_73(&response,channel,parameter,units,command_type);
				break;
			case 74:
				error_num=Go_mon_step_74(&response,channel,parameter,units,command_type);
				break;
			case 75:
				error_num=Go_outputtype_75(&response,channel,parameter,units,command_type);
				break;
			case 76:
				error_num=Go_calib_os_76(&response,channel,parameter,units,command_type);
				break;
			case 77:
				error_num=Go_calib_pw_77(&response,channel,parameter,units,command_type);
				break;
			case 78:
				error_num=Go_routeclose_78(&response,channel,parameter,units,command_type);
				break;
			case 79:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_pw_values);
				break;
			case 80:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_delay_values);
				break;
			case 81:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_period_values);
				break;
			case 82:
				error_num=Go_dly_shift_82(&response,channel,parameter,units,command_type);
				break;
			case 83:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_ampl_values);
				break;
			case 84:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_os_values);
				break;
			case 86:
				error_num=Go_eprom_siz_86(&response,channel,command_type);
				break;
			case 88:
				error_num=Go_puls_count_88(&response,channel,parameter,units,command_type);
				break;
			case 89:
				error_num=Go_puls_sep_89(&response,channel,parameter,units,command_type);
				break;
			case 90:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_burst_values);
				break;
			case 91:
				error_num=Go_sys_net_91(&response,channel,parameter,units,command_type);
				break;
			case 92:
				error_num=Go_sys_pwd_92(&response,channel,parameter,command_type);
				break;
			case 93:
				error_num=Go_eprom_sus_93(&response,channel,parameter,command_type);
				break;
			case 94:
				error_num=Go_rise_time_94(&response,channel,parameter,units,command_type);
				break;
			case 95:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_rise_time_values);
				break;
			case 96:
				error_num=Go_soft_current_limit_96(&response,channel,parameter,units,command_type);
				break;
			case 98:
				error_num=Go_curr_slew_98(&response,channel,parameter,units,command_type);
				break;
			case 99:
				error_num=Go_amp_pnt_83(&response,channel,parameter,units,command_type,pwl_slew_values);
				break;
			case 100:
				error_num=Go_cal_100(&response,channel,parameter,units,command_type);
				break;
			case 101:
				error_num=Go_cal_interval_101(&response,channel,parameter,units,command_type);
				break;
			case 102:
				error_num=Go_eprom_reset_102(&response,channel,parameter,units,command_type);
				break;
			case 103:
				error_num=Go_atten_103(&response,channel,parameter,units,command_type);
				break;
			case 9999:
				// was only whitespace, ignore
				break;

			default:
				error_num=Unrecognized;
				break;

			}
		}

		if (error_num) {
			queue_error_from_parser(&error_response, error_num);
			if (interactive_terminal) {
				(*cbfunc)(user_data, error_response);
			}
		} else if (response != NULL) {
			(*cbfunc)(user_data, response);
		}

		g_free (units);
		g_free (response);
		g_free (error_response);
		g_free (parameter);
		g_string_free (current_word, 1);

		if (!is_query) {
			Main_update_shift_registers();    /* update values in pulse generator circuit */
		}

	} /* not EOS */

	/* update display if it wasn't a query, and if the control menu isn't on (this gives the user
	a chance to press "Go To Local" to override control */
	if (!is_query && !(globals.MenuStatus.Selected_Submenu==Submenu1_rem_loc && globals.MenuStatus.Type_Of_Menu==Submenu_On)) {
		Show_Main_Menu();
	}

	/* re-run error_check to update min/max values based on actual settings, not proposed settings */
	Error_check(globals.ChannelState);

	// trigger a prompt
	if (interactive_terminal) {
		(*cbfunc)(user_data, "");
	}

	g_free (in);

	g_static_mutex_unlock (&mutex);
}


static int Parser_channel (int *channel,int with_id_code,int routine_num)
{
	/* return if no channel suffixes appeared */
	if (*channel==-1) {
		*channel=0;
		return OK;
	}


	/* compare to maximum number of channels allowed */
	if (*channel>1 && *channel>globals.Flash.channels) {
		return InvalidChannel;
	}


	/* SCPI uses 1-N counting, change to 0 - (N-1) */
	if (*channel > 0) {
		--(*channel);
	} else {
		return InvalidChannel;
	}

	return OK;
}


static int Go_freq_32_33(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_freq;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_frequency)) {
		return status;
	}

	new_freq=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_freq,units,"hz")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_freq, globals.Constraints.err_min_freq[channel], globals.Constraints.err_max_freq[channel], NORMAL_ZERO)) {
			return status;
		}
		return Set_frequency(0,0,0,channel,new_freq);
		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].frequency);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_freq[channel], globals.Constraints.err_max_freq[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_Str_eprom_47(gchar** response, int channel, char *loc_string,char *store_string,int command_type)
{
	/* diag:eprom:string - 47 */
	int eprom_loc;

	if (!String_is_it_pos_int(loc_string)) {
		return SyntaxError;
	}

	eprom_loc = atoi(loc_string);
	/* convert location string to a number */

	if ((eprom_loc < 0) || (eprom_loc>sizeof(globals.Flash))) {
		return OutOfRange;
	}

	switch (command_type) {
	case command_param_units:
		if ( (strlen(store_string) < sizeof(globals.Flash.model_num)) && (eprom_loc >= 0)) {
			int i;
			for (i=0; i<strlen(store_string); ++i) {
				/* convert ~ to spaces */
				if (store_string[i]=='~') {
					store_string[i]=' ';
				}
				*(char *)(&globals.Flash.flash_start + eprom_loc+i)=g_ascii_toupper(store_string[i]);
			}
			*(char *)(&globals.Flash.flash_start + eprom_loc + i)=(char) 0;  /* end of string */
			writeUserBlock(&globals.Flash, eprom_loc, strlen(store_string)+1);
			return OK;
		} else {
			return OutOfRange;
		}
		break;

	case query_param:
		return query_string (response, (char *)(&globals.Flash.flash_start + eprom_loc));
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_int_eprom_48(gchar** response, int channel, char *loc_string,char *store_string,int command_type)
{
	/* diag:eprom:int - 48 */
	int eprom_loc;
	short the_number;

	if (!String_is_it_pos_int(loc_string)) {
		return SyntaxError;
	}

	eprom_loc = atoi(loc_string);
	/* convert location string to a number */

	if ((eprom_loc < 0) || (eprom_loc>sizeof(globals.Flash))) {
		return OutOfRange;
	}

	switch (command_type) {
	case command_param_units:
		if (!String_is_it_pos_neg_int(store_string)) {
			return SyntaxError;
		}
		the_number=(short) atoi(store_string);
		*(short *)(&globals.Flash.flash_start + eprom_loc) = the_number;
		writeUserBlock(&globals.Flash, eprom_loc, sizeof (the_number));
		return OK;
		break;

	case query_param:
		return query_int (response, (int) *(short *)(&globals.Flash.flash_start + eprom_loc));
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_Float_eprom51(gchar** response, int channel, char *loc_string,char *store_string,int command_type)
{
	/* diag:eprom:float - 51 */
	int eprom_loc;
	float the_number;

	if (!String_is_it_pos_int(loc_string)) {
		return SyntaxError;
	}

	eprom_loc = atoi(loc_string);

	/* convert location string to a number */

	if ((eprom_loc < 0) || (eprom_loc>sizeof(globals.Flash))) {
		return OutOfRange;
	}

	switch (command_type) {
	case command_param_units:
		if (!String_is_it_numeric(store_string)) {
			return SyntaxError;
		}
		the_number=atof(store_string);
		*(float *)(&globals.Flash.flash_start + eprom_loc)=the_number;
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(the_number));
		return OK;
		break;

	case query_param:{
		// The basic issue here is that this pointer isn't aligned. Which
		// should actually be a problem as ARMv7 has unaligned access support
		// and it's hardcoded to be on.. the issue is that not all instructions
		// can do unaliged accesses and the previous code generated one of those
		// instructions. The kernel traps the access and kills the process.
		// You should see something like this in dmesg:
		// [494223.471575] Alignment trap: not handling instruction edd37a00 at [<00011d12>]
		// [494223.479380] Unhandled fault: alignment exception (0x001) at 0x00050d66
		//
		// I use memcpy here to copy the variable out into a temp
		// variable on the stack which is aligned.
		// There are probably other bits of code that will trigger this because of
		// the use of offsets into the struct.. It might be an idea to write some macros
		// for pushing and pulling bits from the struct.
		float temp;
		memcpy(&temp, (float *)(&globals.Flash.flash_start + eprom_loc),sizeof(float));
		return query_float(response, temp);
	}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_char_eprom_70(gchar** response, int channel, char *loc_string,char *store_string,int command_type)
{
	/* diag:eprom:char - 70 */
	int eprom_loc;
	char the_number;

	if (!String_is_it_pos_int(loc_string)) {
		return SyntaxError;
	}

	eprom_loc = atoi(loc_string);
	/* convert location string to a number */

	if ((eprom_loc < 0) || (eprom_loc>sizeof(globals.Flash))) {
		return OutOfRange;
	}

	switch (command_type) {
	case command_param_units:
		if (!String_is_it_pos_int(store_string)) {
			return SyntaxError;
		}
		the_number=(char) atoi(store_string);
		*(char *)(&globals.Flash.flash_start + eprom_loc)=the_number;
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(the_number));

		return OK;
		break;

	case query_param:
		return query_int (response, (int) *(char *)(&globals.Flash.flash_start + eprom_loc));
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_eprom_siz_86(gchar** response, int channel, int command_type)
{
	switch (command_type) {
	case query_simple:
		return query_int (response, sizeof(globals.Flash));
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_sys_net_91(gchar **response, int channel, char *parameter,char *units,int command_type)
{
	nicinfo info;

	switch (command_type) {
	case query_simple:
		if (nicutils_infofordefaultroute(&info)) {
			*response = g_strdup_printf ("MAC address: %s, IP address: %s", info.mac, info.ip);
			return OK;
		} else {
			return NetworkNotFound;
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Handle_Units(float *mult,char *units, char *in_base)
{
	int len_base, len_all, pos;
	int errornum = OK;

	*mult=1.0;
	if (!strcmp(units,in_base)) {
		return OK;
	}

	len_all = strlen (units);
	if (len_all == 0) {
		return OK;
	}

	gchar *prefix = g_strdup (units);
	gchar *base = g_strdup (in_base);

	len_base = strlen (base);
	pos = len_all - len_base;

	if (!strcmp(base,"%") && (len_all >= 3) && !strcmp(prefix + len_all - 3,"pct")) {
		g_free (base);
		base = g_strdup ("pct");
		prefix[len_all - 3] = 0;
	} else if (!strcmp(base,"pct") && !strcmp(prefix + len_all - 1,"%")) {
		g_free (base);
		base = g_strdup ("%");
		prefix[len_all - 1] = 0;
	} else if (strcmp(prefix+pos,base)) {
		errornum = UnknownUnits;
	} else {
		prefix[pos]=0;
	}

	if (strlen(prefix) && !errornum) {

		// special exceptions
		if (!strcmp(prefix,"m") && !strcmp(base,"ohm")) {
			*mult=1.0e6;
		} else if (!strcmp(prefix,"m") && !strcmp(base,"hz")) {
			*mult=1.0e6;
		}
		// normal rules
		else if (!strcmp(prefix,"ex")) {
			*mult=1.0e18;
		} else if (!strcmp(prefix,"pe")) {
			*mult=1.0e15;
		} else if (!strcmp(prefix,"t")) {
			*mult=1.0e12;
		} else if (!strcmp(prefix,"g")) {
			*mult=1.0e9;
		} else if (!strcmp(prefix,"ma")) {
			*mult=1.0e6;
		} else if (!strcmp(prefix,"k")) {
			*mult=1.0e3;
		} else if (!strcmp(prefix,"m")) {
			*mult=1.0e-3;
		} else if (!strcmp(prefix,"u")) {
			*mult=1.0e-6;
		} else if (!strcmp(prefix,"n")) {
			*mult=1.0e-9;
		} else if (!strcmp(prefix,"p")) {
			*mult=1.0e-12;
		} else if (!strcmp(prefix,"f")) {
			*mult=1.0e-15;
		} else if (!strcmp(prefix,"a")) {
			*mult=1.0e-18;
		} else {
			errornum = UnknownUnits;
		}
	}

	g_free (base);
	g_free (prefix);

	return errornum;
}


static int Is_Min_Command(gchar *text)
{
	return (!strcmp(text,"min") || !strcmp(text,"minimum"));
}


static int Is_Max_Command(char *text)
{
	return (!strcmp(text,"max") || !strcmp(text,"maximum"));
}

static int query_int (gchar** response, int n)
{
	*response = g_strdup_printf ("%d", n);
	return OK;
}

static int query_min_max_int (gchar** response, char *parameter, int min_val, int max_val)
{
	int report_val;

	if (Is_Min_Command(parameter)) {
		report_val=min_val;
	} else if (Is_Max_Command(parameter)) {
		report_val=max_val;
	} else {
		return SyntaxError;
	}

	*response = g_strdup_printf ("%d", report_val);
	return OK;
}

static int query_min_max_float (gchar** response, char *parameter, float min_val, float max_val)
{
	float report_val;

	if (Is_Min_Command(parameter)) {
		report_val=min_val;
	} else if (Is_Max_Command(parameter)) {
		report_val=max_val;
	} else {
		return SyntaxError;
	}

	Float_To_Text(remote_digits_after_decimal,report_val,response);
	return OK;
}

static int query_float (gchar** response, float value)
{
	Float_To_Text(remote_digits_after_decimal,value,response);
	return OK;
}

static int query_string (gchar** response, char *in)
{
	*response = g_strdup (in);
	return OK;
}

static int process_float_param (char *parameter, float *value, float min_val, float max_val, int zero_mode)
{
	int status;
	status = OK;

	if (String_is_it_numeric(parameter)) {
		*value *= atof(parameter);

		if (zero_mode == ALLOW_NEG_ZERO) {
			/* allow a value of "-0" */
			if (fabs(*value) < smallest_allowed_number)
				if (parameter[0] == '-') {
					*value = -0.1*smallest_allowed_number;
				}
		}
	}

	else if (!strcmp(parameter,"+") && (zero_mode == ALLOW_NEG_ZERO)) {
		*value = smallest_allowed_number;
	} else if (!strcmp(parameter,"-") && (zero_mode == ALLOW_NEG_ZERO)) {
		*value = -smallest_allowed_number;
	} else if (Is_Min_Command(parameter)) {
		*value = min_val;
	} else if (Is_Max_Command(parameter)) {
		*value = max_val;
	} else {
		status=SyntaxError;
	}

	return status;
}

#define USE_ON_OFF 0
#define NO_ON_OFF 1
static int process_int_param (char *parameter, int *value, int item_count, int *valid_nums, int allow_on_off)
{
	int status, min_val, max_val, is_valid;

	status = OK;
	is_valid = 0;

	min_val = valid_nums[0];
	max_val = valid_nums[item_count - 1];

	if (String_is_it_pos_int(parameter)) {
		*value = atoi(parameter);
		int i;
		for (i = 0; (i < item_count) && (is_valid == 0); i++) {
			if (*value == valid_nums[i]) {
				is_valid = 1;
			}
		}
		if (is_valid == 0) {
			status = IllegalParameter;
		}
	} else if (!strcmp(parameter,"off") && (allow_on_off == USE_ON_OFF)) {
		*value = 0;
	} else if (!strcmp(parameter,"on") && (allow_on_off == USE_ON_OFF)) {
		*value = 1;
	} else if (Is_Min_Command(parameter) && (allow_on_off == NO_ON_OFF)) {
		*value = min_val;
	} else if (Is_Max_Command(parameter) && (allow_on_off == NO_ON_OFF)) {
		*value = max_val;
	} else {
		status=SyntaxError;
	}

	return status;
}

static int process_int_range (char *parameter, int *value, int min_val, int max_val)
{
	int status;
	status = OK;

	if (String_is_it_pos_int(parameter)) {
		*value = atoi(parameter);
		if ((*value<min_val) || (*value>max_val)) {
			status = OutOfRange;
		}
	} else if (Is_Min_Command(parameter)) {
		*value = min_val;
	} else if (Is_Max_Command(parameter)) {
		*value = max_val;
	} else {
		status=SyntaxError;
	}

	return status;
}

static int process_on_off (char *parameter, int *value)
{
	int valid_choices[2] = {0, 1};
	return process_int_param (parameter, value, 2, valid_choices, USE_ON_OFF);
}

static int process_two_ints (char *parameter, int *value, int min, int max)
{
	int valid_choices[2] = {min, max};
	return process_int_param (parameter, value, 2, valid_choices, NO_ON_OFF);
}

static int check_channel_ok (int channel, int enabled_channels, char chankey)
{
	/* how many channels overall */
	if (channel >= enabled_channels) {
		return InvalidChannel;
	}

	/* abandon if high channel selected by user but not enabled by firmware for that parameter*/
	if (channel && !chankey) {
		return InvalidChannel;
	}

	return OK;
}

static int Go_syst_err_11(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		if (globals.Errors.number_of_errors==0) {
			get_error_text(response, OK);
		} else {
			get_error_text(response,globals.Errors.error_queue[1]);
			Error_Remove_From_Queue();
		}
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}

static int Go_syst_errcnt66(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		return query_int (response, globals.Errors.number_of_errors);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_ampl_26(gchar** response, int channel, char *parameter,char *units,int command_type,char *mode)
{
	float new_ampl;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_amplitude)) {
		return status;
	}
	if (!strcmp(mode,"v")) {
		if (!globals.Flash.voltage_enabled[channel]) {
			return Unrecognized;
		}
	}
	if (!strcmp(mode,"a") && !globals.Flash.current_enabled[channel]) {
		return Unrecognized;
	}

	new_ampl=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_ampl,units,mode)) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (!strcmp(parameter,"ext") || !strcmp(parameter,"external")) {
			return Set_EA(channel,amp_mode_ea);
		} else if (!strcmp(parameter,"amp") || !strcmp(parameter,"amplify")) {
			return Set_EA(channel,amp_mode_amplify);
		} else if (status = process_float_param (parameter, &new_ampl, globals.Constraints.err_min_ampl[channel], globals.Constraints.err_max_ampl[channel], ALLOW_NEG_ZERO)) {
			return status;
		}

		Set_EA(channel,amp_mode_normal);
		return Set_Amplitude(0,0,0,0,0,0,0,channel,new_ampl,0);
		break;

	case query_simple:
		if (globals.ChannelState[channel].amp_mode==amp_mode_ea) {
			return query_string(response, "EXT");
		} else if (globals.ChannelState[channel].amp_mode==amp_mode_amplify) {
			return query_string(response, "AMP");
		} else {
			return query_float(response, globals.ChannelState[channel].amplitude);
		}
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_ampl[channel], globals.Constraints.err_max_ampl[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_pw_36(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_pw;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_pw)) {
		return status;
	}
	if (globals.Flash.fixed_pw[channel]) {
		return Unrecognized;
	}

	new_pw=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_pw,units,"s")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (!strcmp(parameter,"in")) {
			return Set_Pwmode(channel,pw_in_out);
		} else if (!strcmp(parameter,"ext")) {
                        return Set_Pwmode(channel,pw_ew_ext);
		} else if (status = process_float_param (parameter, &new_pw, globals.Constraints.err_min_pw[channel], globals.Constraints.err_max_pw[channel], NORMAL_ZERO)) {
			return status;
		}

		if (status = Set_Pwmode(channel,pw_normal)) {
			return status;
		}
		return Set_Pw(0,0,0,channel,new_pw,0);
		break;

	case query_simple:
		switch (globals.ChannelState[channel].pw_ctrl_mode) {
		case pw_in_out:
			return query_string(response, "IN");
			break;
                case pw_ew_ext:
                        return query_string(response, "EXT");
                        break;
		default:
		case pw_normal:
			return query_float(response, globals.ChannelState[channel].pw);
			break;
		}

		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_pw[channel], globals.Constraints.err_max_pw[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_duty_37(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_duty;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_pw)) {
		return status;
	}
	if (globals.ChannelState[channel].trigger_source!=source_internal) {
		return DutyTriggerError;
	}

	new_duty=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_duty,units,"%")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_duty, globals.Constraints.err_min_pw[channel]*globals.ChannelState[channel].frequency*100, globals.Constraints.err_max_pw[channel]*globals.ChannelState[channel].frequency*100, NORMAL_ZERO)) {
			return status;
		}
		return Set_Pw(0,0,0,channel,new_duty/(100*globals.ChannelState[channel].frequency),0);
		break;

	case query_simple:
		return query_float(response, 100*globals.ChannelState[channel].pw*globals.ChannelState[channel].frequency);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_pw[channel]*globals.ChannelState[channel].frequency*100, globals.Constraints.err_max_pw[channel]*globals.ChannelState[channel].frequency*100);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}



static int Go_offset_29(gchar** response, int channel, char *parameter,char *units,int command_type,char *mode)
{
	float new_offset;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_offset)) {
		return status;
	}
	if (!strcmp(mode,"v") && !globals.Flash.voltage_offset_enabled[channel]) {
		return Unrecognized;
	}
	if (!strcmp(mode,"a") && !globals.Flash.current_offset_enabled[channel]) {
		return Unrecognized;
	}

	new_offset=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_offset,units,mode)) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (!strcmp(parameter,"ext") || !strcmp(parameter,"external")) {
			return Set_EO(channel,os_mode_eo);
		} else if (status = process_float_param (parameter, &new_offset, globals.Constraints.err_min_offset[channel], globals.Constraints.err_max_offset[channel], NORMAL_ZERO)) {
			return status;
		}

		Set_EO(channel,os_mode_normal);
		return Set_Offset(0,0,0,0,channel,new_offset);
		break;

	case query_simple:
		if (globals.ChannelState[channel].os_mode) {
			return query_string(response, "EXT");
		} else {
			return query_float(response, globals.ChannelState[channel].offset);
		}
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_offset[channel], globals.Constraints.err_max_offset[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_idn_5(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	/* *idn? - 5 */

	switch (command_type) {
	case query_simple:
		idn_string(response);
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_period_35(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_period;
	float min_period;
	float max_period;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_frequency)) {
		return status;
	}

	new_period=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_period,units,"s")) {
			return status;
		}
		/* no break */

	case command_withparam:
		min_period = 1.0/globals.Constraints.err_max_freq[channel];

		max_period = globals.Constraints.err_min_freq[channel];
		if (max_period==0.0) {
			return OutOfRange;
		} else {
			max_period = 1.0/max_period;
		}

		if (status = process_float_param (parameter, &new_period, min_period, max_period, NORMAL_ZERO)) {
			return status;
		}
		return Set_frequency(0,0,0,channel,1.0/new_period);
		break;

	case query_simple:
		return query_float(response, 1/globals.ChannelState[channel].frequency);
		break;

	case query_param:
		return query_min_max_float (response, parameter, 1.0/globals.Constraints.err_max_freq[channel], 1.0/globals.Constraints.err_min_freq[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_delay_39(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_delay;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_delay)) {
		return status;
	}

	if (!(globals.Flash.max_delay[channel]>0.0)) {
		return Unrecognized;
	}

	new_delay=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_delay,units,"s")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_delay, globals.Constraints.err_min_delay[channel], globals.Constraints.err_max_delay[channel], NORMAL_ZERO)) {
			return status;
		}
		return Set_Delay(0,0,0,channel,new_delay);
		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].delay);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_delay[channel], globals.Constraints.err_max_delay[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_dbl_pulse_40(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int on_off, status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_double_pulse)) {
		return status;
	}
	if (!(globals.Flash.double_pulse_allowed[channel])) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (status=process_on_off (parameter, &on_off)) {
			return status;
		}
		return Set_Double(channel,on_off);
		break;

	case query_simple:
		return query_int (response, globals.ChannelState[channel].double_pulse);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_func_34(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_func_mode)) {
		return status;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"dc")) {
			return Set_Func(channel,dc_mode_on);
		} else if (!strcmp(parameter,"pulse") || !strcmp(parameter,"puls")) {
			return Set_Func(channel,pulse_mode_on);
		} else if (!strcmp(parameter,"sinusoid") || !strcmp(parameter,"sin") || !strcmp(parameter,"sine")) {
			return Set_Func(channel,sin_mode_on);
		} else if (!strcmp(parameter,"triangle") || !strcmp(parameter,"tri")) {
			return Set_Func(channel,tri_mode_on);
		} else if (!strcmp(parameter,"square") || !strcmp(parameter,"squ")) {
			return Set_Func(channel,squ_mode_on);
		} else if (!strcmp(parameter,"amplify") || !strcmp(parameter,"amp")) {
			return Set_Func(channel,amp_mode_on);
		} else {
			return SyntaxError;
		}
		break;

	case query_simple:
		if (globals.ChannelState[channel].func_mode==dc_mode_on) {
			return query_string(response, "DC");
		} else if (globals.ChannelState[channel].func_mode==pulse_mode_on) {
			return query_string(response, "PULSE");
		} else if (globals.ChannelState[channel].func_mode==sin_mode_on) {
			return query_string(response, "SIN");
		} else if (globals.ChannelState[channel].func_mode==tri_mode_on) {
			return query_string(response, "TRI");
		} else if (globals.ChannelState[channel].func_mode==squ_mode_on) {
			return query_string(response, "SQU");
		} else if (globals.ChannelState[channel].func_mode==amp_mode_on) {
			return query_string(response, "AMP");
		}
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_polarity_42(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_inverted)) {
		return status;
	}
	if (!(globals.Flash.invert_allowed[channel])) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"norm") || !strcmp(parameter,"normal")) {
			return Set_Inverted(channel,NO);
		} else if (!strcmp(parameter,"comp") || !strcmp(parameter,"complement")
		                || !strcmp(parameter,"inv") || !strcmp(parameter,"inverted") )

		{
			return Set_Inverted(channel,YES);
		} else {
			return SyntaxError;
		}
		break;

	case query_simple:
		if (globals.ChannelState[channel].inverted==NO) {
			return query_string(response, "NORM");
		} else {
			return query_string(response, "COMP");
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_hold_38(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_hold_setting)) {
		return status;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"widt") || !strcmp(parameter,"width")) {
			Set_Hold(channel,hold_width);
			return OK;
		} else if (!strcmp(parameter,"dcyc") || !strcmp(parameter,"dcycle")) {
			Set_Hold(channel,hold_duty);
			return OK;
		} else {
			return SyntaxError;
		}
		break;

	case query_simple:
		if (globals.ChannelState[channel].hold_setting==hold_width) {
			return query_string(response, "WIDT");
		} else {
			return query_string(response, "DCYC");
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_output_55(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int on_off, status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_output_state)) {
		return status;
	}
	if(!globals.Flash.on_off_used) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (status=process_on_off (parameter, &on_off)) {
			return status;
		}
		return Set_Output_State(channel,on_off);
		break;

	case query_simple:
		return query_int (response, globals.ChannelState[channel].output_state);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_trig_source46(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_trigger_source)) {
		return status;
	}

	if (globals.Flash.is_func_gen[channel]) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"int") || !strcmp(parameter,"internal")) {
			return Set_Trig_Source(channel,source_internal);
		} else if (!strcmp(parameter,"ext") || !strcmp(parameter,"external")) {
			return Set_Trig_Source(channel,source_external);
                } else if (!strcmp(parameter,"externaldouble") || !strcmp(parameter,"extdbl")) {
                        return Set_Trig_Source(channel,source_external2);
		} else if (!strcmp(parameter,"man") || !strcmp(parameter,"manual")) {
			return Set_Trig_Source(channel,source_manual);
		} else if (!strcmp(parameter,"hold")) {
			return Set_Trig_Source(channel,source_hold);
		} else if (!strcmp(parameter,"imm") || !strcmp(parameter,"immediate")) {
			return Set_Trig_Source(channel,source_immediate);
		} else {
			return SyntaxError;
		}
		break;

	case query_simple:
		if (globals.ChannelState[channel].trigger_source==source_internal) {
			return query_string(response, "INT");
		} else if (globals.ChannelState[channel].trigger_source==source_external) {
			return query_string(response, "EXT");
                } else if (globals.ChannelState[channel].trigger_source==source_external2) {
                        return query_string(response, "EXTDBL");
		} else if (globals.ChannelState[channel].trigger_source==source_manual) {
			return query_string(response, "MAN");
		} else if (globals.ChannelState[channel].trigger_source==source_hold) {
			return query_string(response, "HOLD");
		} else if (globals.ChannelState[channel].trigger_source==source_immediate) {
			return query_string(response, "IMM");
		}
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_gate_type_56(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_gate_type)) {
		return status;
	}
	if (globals.Flash.is_func_gen[channel]) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"sync")) {
			return Set_Gate_Sync(channel,gate_sync);
		} else if (!strcmp(parameter,"async")) {
			/* async mode doesn't work properly in dual-channel units */
			if ( (globals.Flash.channels>1) || globals.Flash.sync_only) {
				return AsyncModeDisabled;
			}

			return Set_Gate_Sync(channel,gate_async);
		} else {
			return SyntaxError;
		}
		break;

	case query_simple:
		if (globals.ChannelState[channel].gate_type==gate_sync) {
			return query_string(response, "SYNC");
		} else {
			return query_string(response, "ASYNC");
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_gate_level_67(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_gate_level)) {
		return status;
	}
	if (globals.Flash.is_func_gen[channel]) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"hi") || !strcmp(parameter,"high")) {
			return Set_Gate_Level(channel,gate_high);
		} else if (!strcmp(parameter,"low") || !strcmp(parameter,"lo")) {
			return Set_Gate_Level(channel,gate_low);
		} else {
			return SyntaxError;
		}
		break;

	case query_simple:
		if (globals.ChannelState[channel].gate_level==gate_high) {
			return query_string(response, "HI");
		} else {
			return query_string(response, "LO");
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_delay_test58(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int on_off, status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_delay)) {
		return status;
	}

	switch (command_type) {
	case command_withparam:
		if (status=process_on_off (parameter, &on_off)) {
			return status;
		}
		globals.ChannelState[channel].test_delay_mode=on_off;
		Set_Mux(channel);
		return OK;
		break;

	default:
		return IllegalParameter;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_wai_10(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	/* the *WAI command has no effect since all OP1B commands are non-overlapping */

	switch (command_type) {
	case command_simple:
		return OK;
		break;

	default:

		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_opc_5(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	/* the OPC commands automatically signal "operation complete" */
	/* since all OP1B commands are non-overlapping */

	switch (command_type) {
	case command_simple:
		GPIB_set_ESR(0x01,1);
		return OK;
		break;

	case query_simple:
		return query_int (response, 1);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_ese_2(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 255)) {
			return status;
		}
		GPIB_set_ESE(0xff,0);
		GPIB_set_ESE(value,1);
		return OK;
		break;

	case query_simple:
		return query_int (response, GPIB_get_ESE());
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_esr_3(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int orig_esr;

	switch (command_type) {
	case query_simple:
		orig_esr =  GPIB_get_ESR();
		GPIB_clear_events();
		return query_int (response, orig_esr);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_cls_1(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case command_simple:
		GPIB_clear_events();
		globals.Errors.number_of_errors=0;
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_sre_7(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 255)) {
			return status;
		}
		GPIB_set_SRE(0xbf,0);
		GPIB_set_SRE(value,1);
		return OK;
		break;

	case query_simple:
		return query_int (response, GPIB_get_SRE());
		break;

	default:
		return SyntaxError;
		break;
	}
}


static int Go_stb_8(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		return query_int (response, GPIB_and_VXI_get_STB());
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_tst_9(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		return query_int (response, 0);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_syst_ver_12(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		return query_string(response, SCPI_version);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_event_13(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		return query_int (response, 0);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_oper_enable15(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 255)) {
			return status;
		}
		globals.Registers.oper_enable_register=value;
		return OK;
		break;

	case query_simple:
		return query_int (response, globals.Registers.oper_enable_register);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_ques_enable18(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 255)) {
			return status;
		}
		globals.Registers.ques_enable_register=value;
		return OK;
		break;

	case query_simple:
		return query_int (response, globals.Registers.ques_enable_register);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_preset_19(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case command_simple:
		globals.Registers.oper_enable_register=0;
		globals.Registers.ques_enable_register=0;
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_zout_20(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_zout)) {
		return status;
	}
	if (!globals.Flash.switchable_zout[channel] || globals.Flash.volt_ctrl_pw[channel]) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_param_units:
		// non-std unit handling, because zout is an integer, not a float
		if (strcmp(units,"ohm")) {
			return UnknownUnits;
		}
		/* no break */

	case command_withparam:
		if (status=process_two_ints (parameter, &value, globals.Flash.zout_min[channel], globals.Flash.zout_max[channel])) {
			return status;
		}
		return Set_zout(channel,value,1);
		break;

	case query_simple:
		return query_int (response, globals.ChannelState[channel].zout);
		break;

	case query_param:
		return query_min_max_int(response, parameter, globals.Flash.zout_min[channel], globals.Flash.zout_max[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_prot_trip_21(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case query_simple:
		if (I2C_Read(PCF8574A+Button_Press_Port) & Overload_Input) {
			return query_int (response, 0);    /* overload not tripped */
		} else {
			return query_int (response, 1);    /* overload tripped */
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_gpib_addr_59(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 30)) {
			return status;
		}
		GPIB_change_address(value);
		return OK;
		break;

	case query_simple:
		return query_int (response, globals.Flash.gpib_address);
		break;

	case query_param:
		return query_min_max_int (response, parameter, 0, 30);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_ser_baud_60(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int new_baud, status;
	int valid_choices[] = {1200, 2400, 4800, 9600, 19200, 38400, 57600, 115200};

	switch (command_type) {
	case command_withparam:
		if (status = process_int_param (parameter, &new_baud, 8, valid_choices, NO_ON_OFF)) {
			return status;
		}
		return IO_Setup_RS232(new_baud, globals.Flash.hardhand);
		break;

	case query_simple:
		return query_int (response, globals.Flash.baud);
		break;

	case query_param:
		return query_min_max_int (response, parameter, 1200, 115200);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_ser_rts_64(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	char new_hardhand;

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"rfr") || !strcmp(parameter,"ibf")
		                || !strcmp(parameter,"ibfull")) {
			// these are valid hardware handshake modes
			new_hardhand = 1;
		} else if (!strcmp(parameter,"on")) {
			new_hardhand = 0;
		} else {
			return IllegalParameter;
		}

		return IO_Setup_RS232(globals.Flash.baud, new_hardhand);
		break;

	case query_simple:
		if (globals.Flash.hardhand) {
			return query_string(response, "IBF");
		} else {
			return query_string(response, "ON");
		}
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_load_68(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_load_type;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_load_type)) {
		return status;
	}
	if (!globals.Flash.switchable_load[channel]) {
		return Unrecognized;
	}

	new_load_type=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_load_type,units,"ohm")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_load_type, globals.Constraints.err_min_load_type[channel], globals.Constraints.err_max_load_type[channel], NORMAL_ZERO)) {
			return status;
		}
		return Set_Load(channel,new_load_type);
		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].load_type);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_load_type[channel], globals.Constraints.err_max_load_type[channel]);
		break;

	default:

		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_meas_ampl_69(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_amplitude)) {
		return status;
	}
	if (!globals.Flash.monitor_enabled[channel]) {
		return Unrecognized;
	}

	switch (command_type) {
	case query_simple:
		return query_float(response, globals.ChannelState[channel].Curr_Mon_value);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_rst_6(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	switch (command_type) {
	case command_simple:
		Main_Rst();
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_calib_amp_72(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float cal_val;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_amplitude)) {
		return status;
	}
	if (!globals.Flash.voltage_enabled[channel] && !globals.Flash.current_enabled[channel]) {
		return Unrecognized;
	}

	cal_val=1.0;

	switch (command_type) {
	case command_param_units:
		if (globals.Flash.current_enabled[channel]) {
			if (status=Handle_Units(&cal_val,units,"a")) {
				return status;
			}
		} else if (globals.Flash.voltage_enabled[channel]) {
			if (status=Handle_Units(&cal_val,units,"v")) {
				return status;
			}
			/* no break */
		}

	case command_withparam:
		if (!String_is_it_numeric(parameter)) {
			return SyntaxError;
		}

		return Set_Amp_Calib(channel,atof(parameter)*cal_val);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_calib_mon_73(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float mon_val;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_amplitude)) {
		return status;
	}
	if (!globals.Flash.monitor_enabled[channel]) {
		return Unrecognized;
	}

	mon_val=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&mon_val,units,"a")) {
			return status;
		}

	case command_withparam:
		if (!String_is_it_numeric(parameter)) {
			return SyntaxError;
		}

		return Set_Mon_Calib(channel,atof(parameter)*mon_val);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_mon_step_74(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float mon_val;
	float span , step_min, step_max;
	int status;
	int eprom_loc;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_amplitude)) {
		return status;
	}
	if (!globals.Flash.monitor_enabled[channel]) {
		return Unrecognized;
	}

	mon_val=1.0;

	span = fabs(globals.Flash.max_ampl[channel]-globals.Flash.min_ampl[channel]);
	if (globals.Flash.voltage_enabled[channel] || globals.Flash.current_enabled[channel]) {
		step_min = span / 5000.0;
		step_max = span / 5.0;
	} else {
		step_min = 0.0001;
		step_max = 10.0;
	}

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&mon_val,units,"a")) {
			return status;
		}

	case command_withparam:
		if (status = process_float_param (parameter, &mon_val, step_min, step_max, NORMAL_ZERO)) {
			return status;
		}
		mon_val=fabs(mon_val);

		if ((mon_val < step_min) || (mon_val > step_max)) {
			return OutOfRange;
		}

		globals.Flash.monitor_step[channel]=mon_val;
		eprom_loc = (char *) &(globals.Flash.monitor_step) - (char *) &(globals.Flash.flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(globals.Flash.monitor_step));
		break;

	case query_simple:
		return query_float(response, globals.Flash.monitor_step[channel]);
		break;

	case query_param:
		return query_min_max_float (response, parameter, step_min, step_max);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_outputtype_75(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int new_logic_level, status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_logic_level)) {
		return status;
	}
	if (!globals.Flash.logic_level_enabled) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"ttl")) {
			new_logic_level=logic_ttl;
		} else if (!strcmp(parameter,"ecl")) {
			new_logic_level=logic_ecl;
		} else {
			return SyntaxError;
		}

		return Set_Logic_Level(channel,new_logic_level);
		break;

	case query_simple:
		if (globals.ChannelState[channel].logic_level==logic_ttl) {
			return query_string(response, "TTL");
		} else {
			return query_string(response, "ECL");
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}



static int Go_calib_os_76(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float mon_val;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_offset)) {
		return status;
	}
	if ( !globals.Flash.current_offset_enabled[channel] && !globals.Flash.voltage_offset_enabled[channel]) {
		return Unrecognized;
	}

	mon_val=1.0;

	switch (command_type) {
	case command_param_units:
		if (globals.Flash.current_offset_enabled[channel]) {
			if (status=Handle_Units(&mon_val,units,"a")) {
				return status;
			}
		} else {
			if (status=Handle_Units(&mon_val,units,"v")) {
				return status;
			}
		}

	case command_withparam:
		if (!String_is_it_numeric(parameter)) {
			return SyntaxError;
		}

		return Set_OS_Calib(channel,atof(parameter)*mon_val);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_calib_pw_77(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float pw_adjust;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_pw)) {
		return status;
	}

	pw_adjust=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&pw_adjust,units,"s")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &pw_adjust, -1.0e-6, 1.0e-6, NORMAL_ZERO)) {
			return status;
		}
		return Set_PW_shift(channel,pw_adjust);
		break;

	case query_simple:
		return query_float(response, Get_PW_shift(channel));
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_routeclose_78(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int error_num;
	int temp1, temp2, status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_route)) {
		return status;
	}
	if (!globals.Flash.routing_required[channel]) {
		return Unrecognized;
	}

	temp1 = globals.ChannelState[channel].route_primary;
	temp2 = globals.ChannelState[channel].route_secondary;

	switch (command_type) {
	case command_withparam:
		if (error_num=Parse_chan_list(channel,
		                              parameter,
		                              &temp1,
		                              &temp2,
		                              NULL)) {
			return error_num;
		} else {
			Set_Route(channel,ROUTE_PRIMARY,temp1);
			Set_Route(channel,ROUTE_SECONDARY,temp2);
			return OK;
		}
		break;

	case query_param:
		if (error_num=Parse_chan_list(channel,
		                              parameter,
		                              &temp1,
		                              &temp2,
		                              response)) {
			return error_num;
		} else {
			return OK;
		}
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Parse_chan_list(int channel,char *parameter,int *primary_selected, int *secondary_selected, gchar** response)
{
	int i;
	int n;
	int pointer;
	int has_module_name;
	int *value_pointer;

#define PARSE_MAX_STRING 20


	if ( !(parameter[0]=='(' && parameter[1]=='@')) {
		return SyntaxError;
	}

	if (strchr (parameter, ':')) {
		return Route_Range_Error;
	}

	if (strchr (parameter, '!')) {
		return Route_Range_Error;
	}


	pointer=1;

	do {
		int error_num = OK;

		pointer++;
		has_module_name = FALSE;

		/* get module name, if any */

		GString *name = g_string_new ("");

		for (i=0; isalpha(parameter[pointer]) && (i<PARSE_MAX_STRING); i++) {
			name = g_string_append_c (name, parameter[pointer]);
			pointer++;
		}

		value_pointer = primary_selected;
		if (i>0) {
			has_module_name = TRUE;

			if (!strcmp (name->str, "anod")) {
				value_pointer = primary_selected;
			} else if (!strcmp (name->str, "cath")) {
				if (globals.Flash.routing_required[channel]<2) {
					error_num = IllegalParameter;
				}
				value_pointer = secondary_selected;
			} else {
				error_num = IllegalParameter;
			}

			if (!error_num && (parameter[pointer] == '(' )) {
				pointer++;
			} else {
				error_num =SyntaxError;
			}

		}

		g_string_free (name, TRUE);

		if (error_num) {
			return error_num;
		}

		GString *digits = g_string_new ("");

		/* get digits */
		for (i=0; isdigit(parameter[pointer]) && (i<PARSE_MAX_STRING); i++) {
			digits = g_string_append_c (digits, parameter[pointer]);
			pointer++;
		}

		if (i==0) {
			error_num = SyntaxError;
		}

		n = atoi(digits->str);
		g_string_free (digits, TRUE);

		if (error_num) {
			return error_num;
		}

		if ((n < 1) || (n > globals.Flash.routing_max_pins[channel])) {
			return OutOfRange;
		}

		if (has_module_name) {
			if (parameter[pointer] == ')' ) {
				pointer++;
			} else {
				return SyntaxError;
			}
		}

		/* query or command? */
		if (response) {
			if (*value_pointer == n) {
				*response = g_strdup ("1 ");
			} else {
				*response = g_strdup ("0 ");
			}
		} else {
			if (value_pointer) {
				*value_pointer=n;
			}
		}


	} while (parameter[pointer]==',');

	if ( parameter[pointer]==')') {
		pointer++;
		for(; parameter[pointer]; pointer++)
			if (!isspace(parameter[pointer])) {
				return SyntaxError;
			}
	} else {
		return SyntaxError;
	}

	return OK;
}


static int Go_dly_shift_82(gchar** response, int channel, char *parameter,char *units,int command_type)
{
#define num_of_dly_shift_points 2

	int value, status;
	float cal_point;
	float min_one_shot_delay;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_delay)) {
		return status;
	}

	switch (command_type) {
	case command_param_units:
		/* if param=point number, and units=measurement, then update calibration */
		if (String_is_it_pos_int(parameter) && String_is_it_numeric(units)) {
			value=atoi(parameter);
			if (value>num_of_dly_shift_points || value<1) {
				return IllegalParameter;
			}
			cal_point=atof(units);
			return Set_Dly_Shr_Cal(channel,value,cal_point);
		} else if (String_is_it_pos_int(parameter) && !strcmp(units,"go")) {
			value=atoi(parameter);
			if (value>num_of_dly_shift_points || value<1) {
				return IllegalParameter;
			}
			return Set_Dly_Shr_Nom(channel,value);
		} else {
			return SyntaxError;
		}

		return OK;
		break;

	case query_param:
		return query_min_max_int(response, parameter, 1, num_of_dly_shift_points);
		break;

	case query_simple:

		min_one_shot_delay = globals.Flash.delay_pwl[channel][0][0][0];
		
		*response = g_strdup_printf (
			"Fixed delay: %f ns, PG-SYNC propagation delay: %f ns, narrowest variable delay: %f ns, all delays valid: %s, delay monotonic in external trigger mode: %s", 
			globals.Flash.delay_shrink[channel]*1e9, 
			globals.Flash.propagation_delay[channel]*1e9, 
			globals.Flash.delay_pwl[channel][0][0][0]*1e9,
			(globals.Flash.delay_shrink[channel] >= min_one_shot_delay) ? "TRUE" : "FALSE",
			(( globals.Flash.delay_shrink[channel] - globals.Flash.propagation_delay[channel] ) >= min_one_shot_delay) ? "TRUE" : "FALSE");
		return OK;
		break; 

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_amp_pnt_83(gchar** response, int channel, char *parameter,char *units,int command_type,int cal_type)
{
	int value,status;
	float cal_point;

	switch (cal_type) {
	case pwl_burst_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_Burst_Time)) {
			return status;
		}
		if (globals.Flash.max_burst_count[channel]<2) {
			return Unrecognized;
		}
		break;
	case pwl_ampl_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_amplitude)) {
			return status;
		}
		if (!globals.Flash.voltage_enabled[channel] && !globals.Flash.current_enabled[channel]) {
			return Unrecognized;
		}
		break;
	case pwl_rise_time_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_rise_time)) {
			return status;
		}
		if (globals.Flash.fixed_rise_time[channel]) {
			return Unrecognized;
		}
		break;
	case pwl_slew_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_slew)) {
			return status;
		}
		if (!globals.Flash.curr_slew[channel]) {
			return Unrecognized;
		}
		break;
	case pwl_os_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_offset)) {
			return status;
		}
		if (!globals.Flash.voltage_offset_enabled[channel] && !globals.Flash.current_offset_enabled[channel]) {
			return Unrecognized;
		}
		break;
	case pwl_pw_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_pw)) {
			return status;
		}
		break;
	case pwl_delay_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_delay)) {
			return status;
		}
		break;
	case pwl_period_values:
		if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_frequency)) {
			return status;
		}
		break;
	}

	switch (command_type) {
	case command_param_units:
		/* if param=point number, and units=measurement, then update calibration */
		if (String_is_it_pos_int(parameter) && String_is_it_numeric(units)) {
			value=atoi(parameter);
			if (value>Get_VI_Num_Pnts(cal_type,channel) || value<1) {
				return IllegalParameter;
			}
			cal_point=atof(units);
			return Set_VI_Cal_Pnt(cal_type,channel,value,cal_point);
		} else if (String_is_it_pos_int(parameter) && !strcmp(units,"go")) {
			value=atoi(parameter);
			if (value>Get_VI_Num_Pnts(cal_type,channel) || value<1) {
				return IllegalParameter;
			}
			if (status=Set_Cal_Nom(channel,value,cal_type,NULL)) {
				return status;
			}
		} else if (String_is_it_pos_int(parameter) && !strcmp(units,"delete")) {
			value=atoi(parameter);
			if (value>Get_VI_Num_Pnts(cal_type,channel) || value<1) {
				return IllegalParameter;
			}
			if (status=Set_VI_Del_Cal(cal_type,channel,value)) {
				return status;
			}
		} else if (String_is_it_numeric(parameter) && !strcmp(units,"add")) {
			cal_point=atof(parameter);
			if (status=Set_VI_Add_Cal(cal_type,channel,cal_point)) {
				return status;
			}
		} else {
			return SyntaxError;
		}

		return OK;
		break;

	case query_param:
		return query_min_max_int(response, parameter, 1, Get_VI_Num_Pnts(cal_type,channel));
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


int Handle_Curr_Slew_Unit(float *new_val,char *units);
int Handle_Curr_Slew_Unit(float *new_val,char *units)
{
	char *slash;
	float tmp1, tmp2;

	*new_val = 1.0;

	slash = strstr (units,"/");
	if (slash == NULL) {
		return UnknownUnits;
	}

	slash[0]=0;

	if (Handle_Units (&tmp1, units, "a") != OK) {
		return UnknownUnits;
	}

	if (Handle_Units (&tmp2, slash+1, "s") != OK) {
		return UnknownUnits;
	}

	*new_val = tmp1 / tmp2;

	return OK;
}


static int Go_puls_count_88(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;
	int min_val;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_Burst_Count)) {
		return status;
	}

	/* if the maximum burst count is 0 or 1, disable this command */
	if (globals.Flash.max_burst_count[channel]<2) {
		return Unrecognized;
	}

	min_val = 1;                         	// -BR
	if (globals.Flash.burst_func[channel]) {
		min_val = 0;    // -PANB
	}

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, min_val, globals.Constraints.err_max_burst_count[channel])) {
			return status;
		}
		return Set_Burst_Count(channel,value,globals.ChannelState[channel].burst_time);
		break;

	case query_simple:
		return query_int (response, globals.ChannelState[channel].burst_count);
		break;

	case query_param:
		return query_min_max_int(response, parameter, min_val, globals.Constraints.err_max_burst_count[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_puls_sep_89(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_pw;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_Burst_Time)) {
		return status;
	}

	/* if the maximum burst count is 0 or 1, disable this command */
	if ((globals.Flash.max_burst_count[channel]<2) || (globals.Flash.burst_func[channel])) {
		return Unrecognized;
	}

	new_pw=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_pw,units,"s")) {
			return status;
		}
		/* no break */

	case command_withparam:
		globals.ChannelState[channel].pw_ctrl_mode=pw_normal;
		if (status = process_float_param (parameter, &new_pw, globals.Constraints.err_min_burst_time[channel], globals.Constraints.err_max_burst_time[channel], NORMAL_ZERO)) {
			return status;
		}
		return Set_Burst_Time(0,0,0,channel,new_pw);
		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].burst_time);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_burst_time[channel], globals.Constraints.err_max_burst_time[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_sys_pwd_92(gchar** response, int channel, char *parameter,int command_type)
{
	gchar *old_password = NULL;
	gchar *new_password = NULL;
	char *new_loc;
	int error_num;

	switch (command_type) {
	case command_withparam:
		/* new password follows comma */

		new_loc = strchr(parameter,',');
		if (new_loc == NULL) {
			return password_change_error;
		}

		new_password = g_strdup_printf ("%s", new_loc + 1);
		new_loc[0]=0;
		old_password = g_strdup (parameter);

		error_num = OK;
		if ((strlen(new_password)< 6) || (strlen(new_password)>32)) {
			error_num = password_change_error;
		}

		if (!error_num) {
			error_num = change_password (old_password, new_password);
		}

		g_free (new_password);
		g_free (old_password);
		return error_num;

	default:
		return SyntaxError;
		break;
	}
}


static int Go_eprom_sus_93(gchar** response, int channel, char *parameter,int command_type)
{
	if (channel) {
		return InvalidChannel;
	}

	switch (command_type) {
	case command_withparam:
		if (!strcmp(parameter,"1")) {
			globals.Flags.flash_writes_suspended=1;
		} else if (!strcmp(parameter,"0")) {
			globals.Flags.flash_writes_suspended=0;
			writeUserBlock(&globals.Flash,0,sizeof(globals.Flash));
		} else {
			return SyntaxError;
		}

		Menu_Clear_Buttons();
		return OK;
		break;

	case query_simple:
		return query_int (response, globals.Flags.flash_writes_suspended);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_eprom_reset_102(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int status, value;

	if (channel) {
		return InvalidChannel;
	}

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, sizeof(globals.Flash)-1)) {
			return status;
		}
		initFlash(&globals.Flash, TRUE, value);
		Main_Rst();
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_rise_time_94(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_rise_time;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_rise_time)) {
		return status;
	}
	if (globals.Flash.fixed_rise_time[channel]) {
		return Unrecognized;
	}

	new_rise_time=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_rise_time,units,"s")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_rise_time, globals.Constraints.err_min_rise_time[channel], globals.Constraints.err_max_rise_time[channel], NORMAL_ZERO)) {
			return status;
		}
		return Set_rise_time(0,0,0,channel,new_rise_time);
		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].rise_time);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_rise_time[channel], globals.Constraints.err_max_rise_time[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_rcl_53(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 3)) {
			return status;
		}
		Set_Rcl(value);
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_sav_54(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int value, status;

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &value, 0, 3)) {
			return status;
		}
		Set_Sav(value);
		return OK;
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_soft_current_limit_96(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_soft_current_limit;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_current_limit)) {
		return status;
	}
	if (!globals.Flash.soft_current_limit_enabled[channel]) {
		return Unrecognized;
	}

	new_soft_current_limit=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Units(&new_soft_current_limit,units,"a")) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_soft_current_limit, globals.Constraints.err_min_soft_current_limit[channel], globals.Constraints.err_max_soft_current_limit[channel], NORMAL_ZERO)) {
			return status;
		}

		return Set_current_limit(0,channel,new_soft_current_limit);

		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].soft_current_limit);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_soft_current_limit[channel], globals.Constraints.err_max_soft_current_limit[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_curr_slew_98(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	float new_slew;
	int status;

	if (status = check_channel_ok (channel, globals.Flash.channels, globals.Flash.ChanKey_slew)) {
		return status;
	}
	if (!globals.Flash.curr_slew[channel]) {
		return Unrecognized;
	}

	new_slew=1.0;

	switch (command_type) {
	case command_param_units:
		if (status=Handle_Curr_Slew_Unit(&new_slew,units)) {
			return status;
		}
		/* no break */

	case command_withparam:
		if (status = process_float_param (parameter, &new_slew, globals.Constraints.err_min_slew[channel], globals.Constraints.err_max_slew[channel], NORMAL_ZERO)) {
			return status;
		}
		return Set_slew(0,0,0,channel,new_slew);
		break;

	case query_simple:
		return query_float(response, globals.ChannelState[channel].slew);
		break;

	case query_param:
		return query_min_max_float (response, parameter, globals.Constraints.err_min_slew[channel], globals.Constraints.err_max_slew[channel]);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_cal_100(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int no_report, status;
	CalStruct caldata;

	if (!globals.Flash.self_cal) {
		return Unrecognized;
	}

	no_report = 0;

	switch (command_type) {
	case command_simple:
		no_report = 1;
		// continue below

	case query_simple:
		status = do_full_self_cal(&caldata);
		if (!no_report) {
			query_string(response, caldata.response->str);
			g_string_free (caldata.response, TRUE);
		}
		return status;
		break;

	default:
		return SyntaxError;
		break;
	}
}


static int Go_cal_interval_101(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int eprom_loc, status, the_number;

	if (!globals.Flash.self_cal) {
		return Unrecognized;
	}

	switch (command_type) {
	case command_withparam:
		if (status = process_int_range (parameter, &the_number, 0, 365)) {
			return status;
		}

		globals.Flash.self_cal_interval = the_number;
		eprom_loc = (char *) &(globals.Flash.self_cal_interval) - (char *) &(globals.Flash.flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(globals.Flash.self_cal_interval));
		return OK;
		break;

	case query_simple:
		return query_int (response, globals.Flash.self_cal_interval);
		break;

	case query_param:
		return query_min_max_int(response, parameter, 0, 365);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}


static int Go_atten_103(gchar** response, int channel, char *parameter,char *units,int command_type)
{
	int on_off, status;

	switch (command_type) {
	case command_withparam:
		if (status=process_on_off (parameter, &on_off)) {
			return status;
		}
		globals.Flags.attenuators_enabled = on_off;

		return OK;
		break;

	case query_simple:
		return query_int (response, globals.Flags.attenuators_enabled);
		break;

	default:
		return SyntaxError;
		break;
	}

	return ThisShouldntHappen;
}