#include "globals.h" #include "lcd.h" #include "version.h" #include #include #include #include #include #include #include #include #include #include #include #define MAINFILE "/root/flash.copy" #define BACKUPFILE "/root/flash.bup" #define PERSIST_ERR_COULDNTREADHDR 1 #define PERSIST_ERR_HDRLENMISMATCH 2 #define PERSIST_ERR_VERSIONMISMATCH 3 #define PERSIST_ERR_COULDNTREADDATA 4 #define PERSIST_ERR_BADCRC32 5 #define PERSIST_ERR_BADARGS 6 #define PERSIST_ERR_COULDNTALLOCATEBUFFER 7 #define PERSIST_ERR_COULDNTSEEK 8 #define PERSIST_ERR_COULDNTWRITE 9 #define PERSIST_ERR_COUNDNTOPENFILE 10 // crc32 routine static uint32_t crc32(uint8_t* buf, int count) { MHASH context = mhash_init(MHASH_CRC32B); mhash(context, buf, count); uint32_t hash; mhash_deinit(context, &hash); return hash; } // header to prepend to stashed objects typedef struct { uint32_t version; uint32_t length; uint32_t crc32; } persistencehdr; static void persistence_printheader(persistencehdr* hdr) { printf("Frozen struct has version %d, is %d bytes long and has the CRC32 0x%08"PRIx32"\n", hdr->version, hdr->length, hdr->crc32); } // copy a file from one place to another.. this is not portable, linux only bool persistence_copyfile(char* source, char* dest) { mode_t filemode = S_IRUSR | S_IWUSR; int src = open(source, O_RDONLY | O_CREAT); // should create the file for us if this is the first run // You would think that O_RDONLY would stop the file creation, but it seems to work int dst = open(dest, O_SYNC | O_RDWR | O_CREAT, filemode); if (src < 0 || dst < 0) { return false; } struct stat s; fstat(src, &s); ftruncate(dst, 0); sendfile(dst, src, NULL, s.st_size); close(src); close(dst); return true; } // store an object bool persistence_freeze(char* dest, void* data, unsigned int offset, unsigned int len, unsigned int total, uint32_t version) { // don't write past the end of the file.. if (offset + len > total) { errno = PERSIST_ERR_BADARGS; return false; } bool newfile = false; uint32_t crc; // open the target file with O_SYNC so that write blocks until it's on disk // fingers crossed the FS actually does what it's told.. int fd = open(dest, O_SYNC | O_RDWR); if (fd < 0) { // this is to catch if the file didn't exist and if it needed to be created mode_t filemode = S_IRUSR | S_IWUSR; fd = open(dest, O_SYNC | O_RDWR | O_CREAT, filemode); if (fd < 0) { errno = PERSIST_ERR_COUNDNTOPENFILE; return false; } newfile = true; } // if this is a new file or we're overwriting everything we can just calculate the CRC from the data passed in if (newfile || len == total) { // if this a new file we want to write everything irrespective of the offset and len passed in if (newfile) { offset = 0; len = total; } crc = crc32((uint8_t*) data, total); } // this is a modification within an existing file so we need to merge the existing data with the // new data to calculate the new crc for the file because it seems the struct getting passed in // only contains the changed data. else if (len != total) { // create a buffer for the existing data void* payload = malloc(total); if (payload == NULL) { errno = PERSIST_ERR_COULDNTALLOCATEBUFFER; return false; } // get the header persistencehdr hdr; if (read(fd, &hdr, sizeof(persistencehdr)) != sizeof(persistencehdr)) { errno = PERSIST_ERR_COULDNTREADHDR; return false; } // load the data persistence_printheader(&hdr); if (read(fd, payload, total) != total) { errno = PERSIST_ERR_COULDNTREADDATA; return false; } // check the existing data isn't already corrupt. uint32_t calculatedcrc32 = crc32((uint8_t*) payload, hdr.length); if (calculatedcrc32 != hdr.crc32) { errno = PERSIST_ERR_BADCRC32; return false; } // overlay the payload with the existing data memcpy(((char*) payload) + offset, ((char*) data) + offset, len); crc = crc32((uint8_t*) payload, total); free(payload); lseek(fd, 0, SEEK_SET); // rewind } // build the header persistencehdr hdr; hdr.version = version; hdr.length = total; hdr.crc32 = crc; persistence_printheader(&hdr); // write the data to disk ftruncate(fd, sizeof(hdr) + total); // not really needed but if we did have a file thats bigger than it // should be put a stop to that // write the header if (write(fd, &hdr, sizeof(hdr)) != sizeof(hdr)) { errno = PERSIST_ERR_COULDNTWRITE; return false; } // seek to the offset.. which could mean not seeking at all if (lseek(fd, offset, SEEK_CUR) < 0) { errno = PERSIST_ERR_COULDNTSEEK; // shouldn't ever happen really because if we're actually // seeking the file should already be the total size. return false; } // write the data out to disk if (write(fd, ((char*) data) + offset, len) != len) { errno = PERSIST_ERR_COULDNTWRITE; return false; } // pack up and go home close(fd); return true; } // try to load an object from disk bool persistence_unfreeze(char* dest, void* result, unsigned int len, uint32_t version) { int fd = open(dest, O_RDONLY); // get the header persistencehdr hdr; if (read(fd, &hdr, sizeof(persistencehdr)) != sizeof(persistencehdr)) { errno = PERSIST_ERR_COULDNTREADHDR; return false; } persistence_printheader(&hdr); // check that the length of this frozen object is what we are expecting if (hdr.length != len) { errno = PERSIST_ERR_HDRLENMISMATCH; return false; } // check that it's the same version.. the version isn't used at the moment // but if you want to change the header at some point it'll be useful if (hdr.version != version) { errno = PERSIST_ERR_VERSIONMISMATCH; return false; } // read in the data for the object.. if we couldn't read the amount of data // that the header said there was the header is either wrong or the file is truncated. if (read(fd, result, hdr.length) != hdr.length) { errno = PERSIST_ERR_COULDNTREADDATA; return false; } // check it's crc32 to make sure it's not corrupt uint32_t calculatedcrc32 = crc32(result, hdr.length); if (calculatedcrc32 != hdr.crc32) { printf("Calculated CRC is 0x%08"PRIx32"\n", calculatedcrc32); errno = PERSIST_ERR_BADCRC32; return false; } return true; } int readUserBlock(FlashStruct *mem) { // try to unfreeze the main file if (persistence_unfreeze(MAINFILE, mem, sizeof(*mem), 0)) { return sizeof(*mem); } // something went wrong else { printf("Error unfreezing %d.. trying backup\n", errno); // hopefully we can use the backup.. if (persistence_unfreeze(BACKUPFILE, mem, sizeof(*mem), 0)) { // if the backup was good overwrite the main file persistence_copyfile(BACKUPFILE, MAINFILE); return sizeof(*mem); } // deadend :( else { printf("Error unfreezing backup %d.\n", errno); } } return 0; } void writeUserBlock(FlashStruct *mem, int addr, int numbytes) { // *** There is a potential issue here.. if the mainfile is corrupt *** // *** and this gets called before readUserBlock then the *** // *** potentially workable backup will be lost .. we could check *** // *** that the main file is valid before backing it up I guess... *** // *** but I don't think this situation should arise. *** // backup the main copy of the file if (persistence_copyfile(MAINFILE, BACKUPFILE)) { if (!persistence_freeze(MAINFILE, mem, addr, numbytes, sizeof(*mem), 0)) { if (errno != PERSIST_ERR_COULDNTWRITE) { printf("Error while trying to write, %d. **Write did not happen!!!**\n", errno); } else { printf("Error while writing data to disk. **File is potentially corrupt!**\n"); } } } else { printf("Could not backup current file. **Write did not happen!!!**\n"); } } static void initFlashValues(FlashStruct *mem) { int i,j,k,m; float power_of_ten, power_of_two; float base_number; base_number=11; g_print_debug ("initializing flash memory\n"); LCD_write(0,0,"Initialize Flash Memory ..."); mem->flash_start=1; mem->turn_on_dly=2; mem->logic_level_enabled=0; mem->ChanKey_logic_level=0; strcpy(mem->model_num,"unprogrammed"); strcpy(mem->serial_num,"no S/N"); strcpy(mem->password,"default"); strcpy(mem->username,"admin"); mem->fully_programmed=Being_Programmed; mem->gpib_address=8; mem->channels=1; mem->web_session_timeout=120; /* two minutes */ mem->telnet_session_timeout=600; /* ten minutes */ mem->telnet_logon_timeout=30; /* thirty seconds */ mem->overwrite_flash_loc=0; mem->rcl_rs232=52; mem->on_off_used=1; mem->ampl_ranges_for_ch2_only=0; mem->ChanKey_frequency=0; mem->ChanKey_delay=0; mem->ChanKey_pw=0; mem->ChanKey_current_limit=0; mem->ChanKey_rise_time=0; mem->ChanKey_amplitude=0; mem->ChanKey_offset=0; mem->ChanKey_Curr_Mon_value=0; mem->ChanKey_Curr_Mon_offset=0; mem->ChanKey_zout=0; mem->ChanKey_hold_setting=0; mem->ChanKey_double_pulse=0; mem->ChanKey_ab_mode=0; mem->ChanKey_route=0; mem->ChanKey_slew=0; mem->ChanKey_func_mode=0; mem->ChanKey_polarity=0; mem->ChanKey_output_state=0; mem->ChanKey_gate_type=0; mem->ChanKey_trigger_source=0; mem->ChanKey_amp_mode=0; mem->ChanKey_gate_level=0; mem->ChanKey_load_type=0; mem->ChanKey_test_delay_mode=0; mem->ChanKey_os_mode=0; mem->ChanKey_Burst_Count=0; mem->ChanKey_Burst_Time=0; mem->network_enabled=0; mem->self_cal=0; mem->self_cal_interval=5; mem->self_cal_startups=0; mem->self_cal_pause=300; mem->self_cal_typical_time_min=6; mem->self_cal_typical_time_sec=0; mem->prf_limiter=1; mem->pcb116c_mon=1; /* more recent ADC, different reading code */ mem->warn_even_if_output_off=0; strcpy(mem->spec_func_lib,"Not used"); strcpy(mem->firmware,FW_VERSION); mem->enable_avrq_extra_ampls=0; for (i=0; ivcc1_pwl_Vc_norm4095[0][0][0][i]=0; mem->vcc1_pwl_amp[0][0][0][i]=0.0; mem->vcc2_pwl_Vc_norm4095[0][0][0][i]=0; mem->vcc2_pwl_amp[0][0][0][i]=0.0; } mem->vcc1_pwl_Vc_norm4095[0][0][0][1]=dac_max; mem->vcc1_pwl_amp[0][0][0][1]=10.0; mem->vcc2_pwl_Vc_norm4095[0][0][0][1]=dac_max; mem->vcc2_pwl_amp[0][0][0][1]=25; for (i=0; islew_pwl_time[i][j][k][m]=0.0; mem->slew_pwl_Vc_norm4095[i][j][k][m]=0; int temp_int_pw_dly, temp_int_prf; if (m==0) { /* these values have been determined by experiment */ temp_int_pw_dly=dac_max; temp_int_prf=dac_max; mem->period_pwl_time[i][j][k][m]=(47e-9*power_of_ten)+41e-9; mem->pw_pwl_time[i][j][k][m]=(base_number*0.7e-9*power_of_ten)+5e-9; mem->delay_pwl_time[i][j][k][m]=(base_number*0.7e-9*power_of_ten)+7.6e-9; mem->burst_pwl_time[i][j][k][m]=(2*base_number*1.0e-9*power_of_ten)+25e-9; } else if (m==1) { temp_int_pw_dly=dac_max/3; temp_int_prf=dac_max/2.15; mem->period_pwl_time[i][j][k][m]=(1e-7*power_of_ten)+50e-9; mem->pw_pwl_time[i][j][k][m]=(3*base_number*0.7e-9*power_of_ten)+10e-9; mem->delay_pwl_time[i][j][k][m]=(3*base_number*0.7e-9*power_of_ten)+17e-9; mem->burst_pwl_time[i][j][k][m]=(3*base_number*1.0e-9*power_of_ten)+10e-9; } else if (m==2) { temp_int_pw_dly=dac_min; temp_int_prf=dac_max/4.6; mem->period_pwl_time[i][j][k][m]=(2.3e-7*power_of_ten)+100e-9; mem->pw_pwl_time[i][j][k][m]=(base_number*0.7e-8*power_of_ten)+40e-9; mem->delay_pwl_time[i][j][k][m]=(base_number*0.7e-8*power_of_ten)+40e-9; mem->burst_pwl_time[i][j][k][m]=(base_number*1.0e-8*power_of_ten)+10e-9; } else if (m==3) { temp_int_pw_dly=0; temp_int_prf=dac_min; mem->period_pwl_time[i][j][k][m]=(4.7e-7*power_of_ten)+160e-9; mem->pw_pwl_time[i][j][k][m]=0.0; mem->delay_pwl_time[i][j][k][m]=0.0; mem->burst_pwl_time[i][j][k][m]=0.0; } else { temp_int_pw_dly=0; temp_int_prf=0; mem->pw_pwl_time[i][j][k][m]=0.0; mem->delay_pwl_time[i][j][k][m]=0.0; mem->period_pwl_time[i][j][k][m]=0.0; mem->burst_pwl_time[i][j][k][m]=0.0; } mem->pw_pwl_Vc_norm4095[i][j][k][m]=temp_int_pw_dly; mem->delay_pwl_Vc_norm4095[i][j][k][m]=temp_int_pw_dly; mem->burst_pwl_Vc_norm4095[i][j][k][m]=temp_int_pw_dly; mem->period_pwl_Vc_norm4095[i][j][k][m]=temp_int_prf; } } power_of_ten*=10.0; power_of_two*=2.0; } power_of_two=20.0e-9; for (j=0; jrise_time_pwl_Vc_norm4095[i][j][k][m]=dac_max; mem->rise_time_pwl_time[i][j][k][m]=(1e-9+power_of_two); } else if (m==1) { mem->rise_time_pwl_Vc_norm4095[i][j][k][m]=dac_max/2; mem->rise_time_pwl_time[i][j][k][m]=(1e-9+(power_of_two*1.5)); } else if (m==2) { mem->rise_time_pwl_Vc_norm4095[i][j][k][m]=dac_max/4.6; mem->rise_time_pwl_time[i][j][k][m]=(1e-9+(power_of_two*3.0)); } else { mem->rise_time_pwl_Vc_norm4095[i][j][k][m]=0; mem->rise_time_pwl_time[i][j][k][m]=0.0; } } } power_of_two*=2.0; } for (j=0; jpw_range_pol_tweaks[i][j][k] = 0.0; } } for (j=0; j<10; j++) for (k=0; k<5; k++) for (m=0; m<2; m++) { mem->ampl_pwl_Vc_norm4095[i][k][m][j]=0; mem->ampl_pwl_amp[i][k][m][j]=0.0; } mem->ampl_pwl_Vc_norm4095[i][0][0][1]=dac_max; mem->ampl_pwl_amp[i][0][0][1]=100.0; for (j=0; jrcl_frequency[i][j]=10000.0; mem->rcl_delay[i][j]=0e-9; mem->rcl_pw[i][j]=20e-9; mem->rcl_amplitude[i][j]=0.0; mem->rcl_offset[i][j]=0.0; mem->rcl_misc[i][j]=9; mem->rcl_misc2[i][j]=0; mem->rcl_burst_count[i][j]=1 && !mem->burst_func[i]; mem->rcl_burst_time[i][j]=500e-9; mem->rcl_rise_time[i][j]=50e-9; mem->rcl_soft_current_limit[i][j]=0.0; mem->rcl_route_primary[i][j]=1; mem->rcl_route_secondary[i][j]=1; mem->rcl_slew[i][j]=100e6; mem->rcl_load[i][j]=50.0; mem->rcl_vcc1[i][j]=0.0; mem->rcl_vcc2[i][j]=0.0; mem->rcl_vlogic[i][j]=0.0; } for (j=0; j<5; j++) for (k=0; k<2; k++) { mem->mon_vi_ratio[i][j][k]=0.050*(j+1); } mem->load_type_pwl_time[i][0][0][0] = 200; mem->load_type_pwl_time[i][0][0][1] = 10000; mem->load_type_pwl_Vc_norm4095[i][0][0][0] = dac_max; mem->load_type_pwl_Vc_norm4095[i][0][0][1] = dac_max / 60; for (j=2; j<10; j++) { mem->load_type_pwl_time[i][0][0][j] = 0; mem->load_type_pwl_Vc_norm4095[i][0][0][j] = 0; } mem->slew_pwl_time[i][4][0][0]=80e6; mem->slew_pwl_time[i][4][0][1]=240e6; mem->slew_pwl_time[i][3][0][0]=40e6; mem->slew_pwl_time[i][3][0][1]=120e6; mem->slew_pwl_time[i][2][0][0]=20e6; mem->slew_pwl_time[i][2][0][1]=60e6; mem->slew_pwl_time[i][1][0][0]=10e6; mem->slew_pwl_time[i][1][0][1]=30e6; mem->slew_pwl_time[i][0][0][0]=5e6; mem->slew_pwl_time[i][0][0][1]=15e6; mem->slew_pwl_Vc_norm4095[i][0][0][1]=dac_max; mem->slew_pwl_Vc_norm4095[i][1][0][1]=dac_max; mem->slew_pwl_Vc_norm4095[i][2][0][1]=dac_max; mem->slew_pwl_Vc_norm4095[i][3][0][1]=dac_max; mem->slew_pwl_Vc_norm4095[i][4][0][1]=dac_max; } /* special consideration for CH2 delay */ mem->delay_pwl_time[1][0][0][0]=-0.1e-9; for (i=0; irouting_required[i]=0; mem->routing_max_pins[i]=16; mem->min_ampl[i]=0.0; mem->max_ampl[i]=100.0; mem->min_offset[i]=0.0; mem->max_offset[i]=100.0; mem->min_vout[i]=0.0; mem->max_vout[i]=100.0; mem->min_freq[i]=1.0; mem->max_freq[i]=8e6; mem->min_pw[i]=25e-9; mem->max_pw[i]=1.0; mem->min_rise_time[i]=50e-9; mem->max_rise_time[i]=500e-9; mem->min_soft_current_limit[i]=10.0; mem->max_soft_current_limit[i]=530.0; mem->max_delay[i]=1.0; mem->min_delay[i]=0.0; mem->propagation_delay[i]=10.0e-9; mem->delay_shrink[i]=40.0e-9; mem->ampl_zero_equiv[i]=0.1; mem->max_duty_low[i]=110.0; mem->max_duty_high[i]=110.0; mem->duty_ampl[i]=30.0; mem->max_duty_mid1[i]=0.0; mem->duty_ampl_mid1[i]=0.0; mem->max_duty_mid2[i]=0.0; mem->duty_ampl_mid2[i]=0.0; mem->min_slew[i]=90e6; mem->max_slew[i]=210e6; mem->max_high_rl_duty[i]=80.0; mem->max_peak_power[i]=0.0; mem->max_avg_power[i]=0.0; mem->duty_highRL_above_v[i]=110.0; mem->duty_highRL_below_v[i]=110.0; mem->mon_pw_threshold[i]=-1.0; mem->monitor_step[i]=1.0; mem->sep_posneg_mon_ratio[i]=0; mem->volt_ctrl_pw[i]=0; mem->voltage_enabled[i]=1; mem->voltage_offset_enabled[i]=0; mem->current_enabled[i]=0; mem->current_offset_enabled[i]=0; mem->switchable_zout[i]=1; mem->dc_mode_allowed[i]=1; mem->ab_mode_allowed[i]=1; mem->double_pulse_allowed[i]=1; mem->invert_allowed[i]=1; mem->ea_enabled[i]=1; mem->switchable_load[i]=1; mem->monitor_enabled[i]=0; mem->use_pos_ampl_data_only[i]=0; mem->ampl_min_max_only[i]=0; mem->eo_enabled[i]=1; mem->ext_amplify_enabled[i]=1; mem->zout_min[i]=2; mem->zout_max[i]=50; for (j=0; j<10; j++) { for (k=0; k<5; k++) { mem->os_pwl_Vc_norm4095[i][k][0][j]=0; mem->os_pwl_amp[i][k][0][j]=0.0; } } mem->os_pwl_Vc_norm4095[i][0][0][1]=dac_max; mem->os_pwl_amp[i][0][0][1]=100.0; mem->ampl_DAC[i]=0; mem->os_DAC[i]=1; mem->polarity_xtra_rly[i]=1; mem->fixed_pw[i]=0; mem->fixed_rise_time[i]=1; mem->pcb_203a_rise_time[i]=1; mem->ext_amplify_xtra_rly[i]=4; mem->ea_xtra_rly[i]=5; mem->curr_slew[i]=0; mem->distort_X[i]=0.0; mem->distort_Y[i]=0.0; mem->distort_Z[i]=0.0; mem->distort_max_ampl[i]=0.0; mem->distort_max_os[i]=0.0; mem->ampl_os_ranges_related[i]=0; mem->ampl_coupled_to_os[i]=0; mem->pulse_width_pol_tweak[i][0]=0.0; mem->pulse_width_pol_tweak[i][1]=0.0; mem->delay_pol_tweak[i][0]=0.0; mem->delay_pol_tweak[i][1]=0.0; mem->max_burst_count[i]=1; mem->max_burst_duty[i]=50.0; mem->min_burst_per[i]=100e-9; mem->min_burst_gap[i]=100e-9; mem->max_burst_gap[i]=1.0; mem->is_func_gen[i]=0; mem->burst_func[i]=0; mem->freq_dac[i]=7; mem->is_monocycle[i]=0; mem->monocycle_dac[i]=6; mem->rise_time_dac[i]=6; mem->slew_dac[i]=6; mem->load_type_dac[i]=3; mem->output_timer[i]=0; mem->current_limit_pulse_mode[i]=220.0; mem->current_limit_dc_mode[i]=120.0; mem->current_limit_full_scale[i]=501.0; mem->current_limit_dac[i]=3; mem->hard_current_limit_enabled[i]=0; mem->soft_current_limit_enabled[i]=0; mem->invert_by_default[i]=pol_norm; mem->max_avg_ampl[i]=0.0; mem->pol_relay_high_for_pos[i]=1; mem->special_pw_range_minimum[i]=0.0; mem->pw_shift_below_this_ampl[i]=0.0; mem->pw_shift_below_ampl_by[i]=0.0; mem->ampl_min_abs_value[i]=0.0; mem->ampl_step_size[i]=0.0; mem->low_load_type[i]=50.0; mem->high_load_type[i]=10000.0; mem->fix_pw_dac_val[i]=dac_max/8; mem->max_pw_pol[i][0]=0.0; mem->max_pw_pol[i][1]=0.0; mem->vcc1_max[i]=5.1; mem->vcc2_max[i]=24.4; mem->vcc2_min[i]=3.0; mem->use_high_ampl_ranges_for_high_pw_ranges[i]=0; } mem->relay_delay_in_sec=0.5; mem->extended_relay_delay_in_sec=0.5; mem->wait_states_after_sock_init=10000; /*0123456789012345678901234567890123456789*/ strcpy(mem->aux_error_message,"PRF too high! Output disabled."); /* default PW DACs */ mem->pw_dac[0]=2; /* channel 1: ONLY used for EXTERNAL voltage-controlled PW */ /* DAC 4 is normally used for internally controlled PW */ mem->pw_dac[1]=2; /* channel 2: varies - normally 2 (for control of PCB 107C or 174) */ /* default delay DACs */ mem->delay_dac[0]=5; /* channel 1: on OP1B board - not to be changed, as a rule */ mem->delay_dac[1]=6; /* channel 2: varies (for control of PCB107C) */ mem->flash_end=99; for (i=0; i<8; i++) { mem->initial_dac_settings[i]=0L; } mem->copy_max_channels=max_channels; /* copy to flash, so it can be read by diag:eprom:int? */ /* avrq tests mem->enable_avrq_extra_ampls=1; mem->channels=2; mem->ChanKey_amplitude=1; mem->current_enabled[1]=1; mem->voltage_enabled[1]=0; mem->fully_programmed=2; mem->ampl_DAC[0]=-1; mem->ampl_pwl_amp[1][0][0][1]=0.025; mem->max_ampl[1]=0.022; mem->max_vout[1]=0.022; mem->ampl_zero_equiv[1]=0.001; */ LCD_write(1,0,"Flash Init, Done! "); } void initFlash(FlashStruct *mem, int starting_location) { int read_size = readUserBlock(mem); if ( (read_size == 0) || (mem->fully_programmed == Not_Programmed) || ((starting_location > 0) && (starting_location < sizeof(*mem))) ) { // uninitialized device! initFlashValues(mem); // save the default Flash config, for nonvolatile persistence writeUserBlock(mem, starting_location, sizeof(*mem) - starting_location); if (starting_location > 0) { readUserBlock(mem); } } }