#include "globals.h" #include "i2c.h" #include "device-functions.h" #include "lcd.h" #include "monitor.h" #include "error_utils.h" #include "menus.h" #include <glib.h> #include <math.h> long monitor_last_forced_trig[max_channels]; void I2C_Setup_Monitor(); void I2C_Setup_Monitor() { int i; float new_os; float average_os; float integrated_error; int channel; int total; total=0; average_os = 0.0; for (channel=0; channel<(globals.Flash.ChanKey_Curr_Mon_value?globals.Flash.channels:1); ++channel) if (globals.Flash.monitor_enabled[channel]) { ++total; LCD_clear(); LCD_write(0,0,"Nulling Current Monitor ...."); /* quiescent */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xff); I2C_Write(PCF8574A+Curr_Mon_LSB+channel*4,0xff); /* remove chip-select */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xbf); /* pulse CAL line low */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0x9f); I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xbf); for(i=0; i<50; i++); /* quiescent */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xff); /* wait for monitor output to become stable */ g_usleep(1e6); do { globals.ChannelState[channel].Curr_Mon_offset=0; integrated_error=0; /* repeat until stable for 50 iterations */ int j=0; while (j<50) { g_usleep(1e4); /* pulse S/H line low for initial sample of monitor */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0x7f); for (i=0; i<100; ++i) { ; } /* quiescent */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xff); /* inactive, weak pull-up mode */ I2C_Read(PCF8574A+Curr_Mon_MSB+channel*4); new_os = 1.0 * I2C_Get_Monitor_Word(channel); average_os = (average_os*j+new_os) / (j+1.0); integrated_error += (average_os-new_os); if ( fabs(new_os-average_os) < 300.0) { ++j; } else { j=0; } } } while (fabs(integrated_error) > 25.0); globals.ChannelState[channel].Curr_Mon_offset=(int) average_os; LCD_write(0,0,"Current Monitor is now ready."); } if (total) { I2C_Check_Monitors(); } return; } void force_monitor_cal(int channel, int do_calibration); void force_monitor_cal(int channel, int do_calibration) { if (do_calibration) { /* quiescent */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xff); I2C_Write(PCF8574A+Curr_Mon_LSB+channel*4,0xff); /* remove chip-select */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xbf); /* pulse CAL line low */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0x9f); I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xbf); g_usleep(2e3); } /* pulse trigger line low and remove chip-select to force an A/D conversion */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0x3f); /* quiescent */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xff); } int I2C_Get_Monitor_Word(int channel); int I2C_Get_Monitor_Word(int channel) { int monitor_word; /* remove chip-select, so that the value doesn't change while read */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xbf); monitor_word=((I2C_Read(PCF8574A+Curr_Mon_MSB+channel*4) & 0x1f)<<8) + I2C_Read(PCF8574A+Curr_Mon_LSB+channel*4); /* restore chip-select */ I2C_Write(PCF8574A+Curr_Mon_MSB+channel*4,0xff); /* PCB 116B and earlier had 13-bit ADC, PCB 116C has 12 bit. */ if (globals.Flash.pcb116c_mon == 1) { monitor_word = (monitor_word & 0x0fff) << 1; } /* deal with sign bit */ if (monitor_word & 0x1000) { monitor_word=-( ((~monitor_word)+1) & 0x0fff); } monitor_word*=-1; monitor_word-=globals.ChannelState[channel].Curr_Mon_offset; return monitor_word; } int I2C_Check_Monitors(void); int I2C_Check_Monitors(void) { int monitor_word; float step_size; int channel; int ampl_range,point_found,UseNegData,entry,word_out; int update_display; int use_neg_data; int actual_pol; int i; int error_num; long timer_check; long seconds_since_last; update_display=0; for (channel=0; channel<(globals.Flash.ChanKey_Curr_Mon_value?globals.Flash.channels:1); ++channel) { timer_check = sec_timer(); seconds_since_last = timer_check - monitor_last_forced_trig[channel]; if (seconds_since_last > 0) { if ( (globals.ChannelState[channel].func_mode==dc_mode_on) || (globals.ChannelState[channel].output_state==output_off)) { /* force monitor measurement every second in DC mode (because no TTL trigger is present) and output-off mode */ force_monitor_cal(channel,NO); } monitor_last_forced_trig[channel]=timer_check; } if (globals.Flash.sep_posneg_mon_ratio[channel] && globals.ChannelState[channel].amplitude<0.0) { use_neg_data=1; } else { use_neg_data=0; } monitor_word=I2C_Get_Monitor_Word(channel); /* determine current ampl_range */ Set_VI_Control(pwl_ampl_values,channel,globals.ChannelState[channel].amplitude,&point_found, &l_range,&UseNegData,&entry,&word_out,&actual_pol); /* 5.0V is the full-scale voltage of the 12bit ADC */ globals.ChannelState[channel].Curr_Mon_value=((((float) monitor_word)/4095.0) * 5.0) / globals.Flash.mon_vi_ratio[channel][ampl_range][use_neg_data]; /* convert to even step size */ step_size=globals.Flash.monitor_step[channel]; globals.ChannelState[channel].Curr_Mon_value= ((float) ((long) (globals.ChannelState[channel].Curr_Mon_value/step_size)) ) * step_size; /* AVO-8D2: check for duty cycle problems, as a function of measured amplitude */ if ( globals.Flash.hard_current_limit_enabled[channel] && (globals.ChannelState[channel].output_state == output_on)) { for (i=0; i<max_channels; ++i) { TestState[i]=globals.ChannelState[i]; } TestState[channel].amplitude=globals.ChannelState[channel].Curr_Mon_value; if ((error_num=Error_check(TestState))) { Set_Output_State(channel,output_off); queue_and_broadcast_sensor_alarm(error_num); } } if (globals.Flash.soft_current_limit_enabled[channel] && (globals.Flash.fully_programmed != Being_Programmed) && (globals.ChannelState[channel].Curr_Mon_value > globals.ChannelState[channel].soft_current_limit)) { Set_Output_State(channel,output_off); queue_and_broadcast_sensor_alarm(Soft_Limit_Exceeded); } else if (globals.MenuStatus.Type_Of_Menu==Main_Menu_On && !globals.MenuStatus.Error_Screen && !globals.MenuStatus.Nonstd_Display && globals.ChannelState[channel].Curr_Mon_value!=globals.ChannelState[channel].displayed_mon_val && (seconds_since_last > 0)) /* only update display once a second */ { ++update_display; } } if (update_display) { Show_Main_Menu(); } return OK; }