#include "globals.h"
#include "lcd.h"
#include "i2c.h"
#include "flash.h"
#include "device-functions.h"
#include <stdint.h>
#include <stdio.h>
#include <stdbool.h>
#include <fcntl.h>
#include <stdlib.h>
#include <inttypes.h>
#include <unistd.h>
#include <errno.h>
#include <sys/sendfile.h>
#include <sys/stat.h>
#include <mhash.h>
#include <glib.h>

#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

// safety check toggles
#define FROZENSMALLEROK 1
#define FROZENBIGGEROK 0

// flash writing queue bits
#define MAXQUEUEDJOBS 1 // This makes the queue a little bit pointless
// but you might want to queue a few more

typedef struct {
	FlashStruct *mem;
	int addr;
	int numbytes;
} userflashjob;

static void initFlashValues(FlashStruct *mem);

static bool alive = true;
static GMutex writermutex;
static GCond writerwakeup;
static GCond writerqueueopen;
static GThread* userflashwritethread;
static GAsyncQueue* userflashwritequeue;
//

// 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
// this returns true if it is safe to continue. If there was nothing to backup
// that is considered "safe to continue"
bool persistence_copyfile(char* source, char* dest)
{
	mode_t filemode = S_IRUSR | S_IWUSR;

	int src = open(source, O_RDONLY);

	if (src < 0 && errno == ENOENT) { // If the source file doesn't exist, there isn't anything to copy
		close(src);
		return true;
	}

	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) {
			close(fd);
			errno = PERSIST_ERR_COULDNTALLOCATEBUFFER;
			return false;
		}

		// get the header
		persistencehdr hdr;
		if (read(fd, &hdr, sizeof(persistencehdr)) != sizeof(persistencehdr)) {
			errno = PERSIST_ERR_COULDNTREADHDR;
			free(payload);
			close(fd);
			return false;
		}

		// load the data
		persistence_printheader(&hdr);
		if (read(fd, payload, hdr.length) != hdr.length) {
			errno = PERSIST_ERR_COULDNTREADDATA;
			free(payload);
			close(fd);
			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;
			free(payload);
			close(fd);
			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;
		close(fd);
		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.
		close(fd);
		return false;
	}

	// write the data out to disk
	if (write(fd, ((char*) data) + offset, len) != len) {
		errno = PERSIST_ERR_COULDNTWRITE;
		close(fd);
		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)
{

	bool truncatelastbyte = false;
	int fd = open(dest, O_RDONLY);

	// get the header
	persistencehdr hdr;
	if (read(fd, &hdr, sizeof(persistencehdr)) != sizeof(persistencehdr)) {
		errno = PERSIST_ERR_COULDNTREADHDR;
		close(fd);
		return false;
	}

	persistence_printheader(&hdr);

	// check that the length of this frozen object is what we are expecting
	if (hdr.length != len) {

#if FROZENSMALLEROK
		if (hdr.length < len) {
			printf("frozen struct is %d bytes smaller than the requested size, removing end byte\n", len - hdr.length);
			truncatelastbyte = true;
			goto hdrlengthok;
		}
#endif

#if FROZENBIGGEROK
		if(hdr.length > len) {
			printf("frozen struct is bigger than the requested size, %d bytes will be truncated\n", hdr.length - len);
			goto hdrlengthok;
		}
#endif

		errno = PERSIST_ERR_HDRLENMISMATCH;
		close(fd);
		return false;
	}
hdrlengthok:

	// 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;
		close(fd);
		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.
	char* tempresult = g_malloc(hdr.length);
	if (read(fd, tempresult, hdr.length) != hdr.length) {
		errno = PERSIST_ERR_COULDNTREADDATA;
		close(fd);
		return false;
	}

	// check it's crc32 to make sure it's not corrupt
	uint32_t calculatedcrc32 = crc32(tempresult, hdr.length);
	if (calculatedcrc32 != hdr.crc32) {
		printf("Calculated CRC is 0x%08"PRIx32"\n", calculatedcrc32);
		errno = PERSIST_ERR_BADCRC32;
		close(fd);
		return false;
	}

	close(fd);
	memcpy(result, tempresult, MIN(len, truncatelastbyte ? (hdr.length - 1) : hdr.length));
	g_free(tempresult);
	return true;

}


void remount_root_as_writeable (gboolean start)
{
	gchar *cmd;

	static GStaticMutex mutex = G_STATIC_MUTEX_INIT;

	if (start) {
		g_static_mutex_lock (&mutex);
		cmd = g_strdup_printf ("/usr/bin/mount -o remount,rw %s /", globals.HWDetect.remount_point);
	} else {
		cmd = g_strdup_printf ("/usr/bin/mount -o remount,ro %s /", globals.HWDetect.remount_point);
	}

	printf ("%s\n",cmd);
	system(cmd);
	g_free (cmd);

	if (!start) {
		g_static_mutex_unlock (&mutex);
	}
}


static int readUserBlock(FlashStruct *mem)
{

	// put the default values into the the struct;
	// what should happen here is that if we load
	// a smaller struct from disk it will replace
	// the top part and leave the defaults for newly
	// added values at the end
	initFlashValues(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
			if (!globals.Sys.shutdown_started) {
				globals.Sys.flash_write_in_progress = TRUE;
				printf ("start writeable: copy from backup\n");
				remount_root_as_writeable (TRUE);

				persistence_copyfile(BACKUPFILE, MAINFILE);

				globals.Sys.flash_write_in_progress = FALSE;
				printf ("end writeable: copy from backup\n");
				remount_root_as_writeable (FALSE);
			}
			return sizeof(*mem);
		}
		// deadend :(
		else {
			printf("Error unfreezing backup %d.\n", errno);
		}
	}
	return 0;
}

void writeUserBlockNow(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.				***

	static GStaticMutex mutex = G_STATIC_MUTEX_INIT;
	g_static_mutex_lock (&mutex);

	if (!globals.Flags.flash_writes_suspended) {

		// backup the main copy of the file
		if (!globals.Sys.shutdown_started) {

			globals.Sys.flash_write_in_progress = TRUE;
			printf ("start writeable: make backup\n");
			remount_root_as_writeable (TRUE);

			bool backup_ok = persistence_copyfile(MAINFILE, BACKUPFILE);

			globals.Sys.flash_write_in_progress = FALSE;
			printf ("end writeable: make backup\n");
			remount_root_as_writeable (FALSE);

			if (backup_ok && !globals.Sys.shutdown_started) {

				globals.Sys.flash_write_in_progress = TRUE;
				printf ("start writeable: write to main config file\n");
				remount_root_as_writeable (TRUE);

				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");
					}
				}
				globals.Sys.flash_write_in_progress = FALSE;
				printf ("end writeable: write to main config file\n");
				remount_root_as_writeable (FALSE);
			} else {
				printf("Could not backup current file. **Write did not happen!!!**\n");
			}
		}
	}

	g_static_mutex_unlock (&mutex);
}

void writeUserBlock(FlashStruct *mem, int addr, int numbytes)
{
	if (alive) {
		g_mutex_lock(&writermutex);
		if (g_async_queue_length(userflashwritequeue) >= MAXQUEUEDJOBS) {
			//printf("Queue closed, waiting\n");
			g_cond_wait(&writerqueueopen, &writermutex);
		}
		g_mutex_unlock(&writermutex);

		userflashjob* job = g_malloc(sizeof(userflashjob));
		job->mem = g_malloc(sizeof(FlashStruct));
		memcpy(job->mem, mem, sizeof(FlashStruct));
		job->addr = addr;
		job->numbytes = numbytes;
		g_async_queue_push(userflashwritequeue, job);
		g_cond_broadcast(&writerwakeup);
	}
}


static void initFlashValues(FlashStruct *mem)
{
	int i,j,k,m;
	float power_of_ten, power_of_two;

	mem->flash_start=1;
	mem->turn_on_dly=5;

	mem->logic_level_enabled=0;
	mem->ChanKey_logic_level=0;

	strcpy(mem->model_num,"AV-unprogrammed");
	strcpy(mem->serial_num,"00000");

	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->baud = 1200;
	mem->parity = rs232_parity_none;	// no longer used
	mem->stopbits = 1;			// no longer used
	mem->databits = 8;			// no longer used
	mem->hardhand = 1;
	mem->echo = 1;				// no longer used

	mem->on_off_used=1;

	mem->ampl_ranges_for_ch2_only=0;

	mem->output_on_delay=DEFAULT_OUTPUT_ON_DELAY;	// 0.8 seconds

	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_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->vxi_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->self_cal_ref_freq = SELF_CAL_REF_FREQ;
	mem->self_cal_min_count = SELF_CAL_MIN_COUNT;

	mem->pcb116c_mon=1;	/* more recent ADC, different reading code */
	mem->warn_even_if_output_off=0;

	mem->enable_avrq_extra_ampls=0;

	for (i=0; i<points_in_range; i++) {
		mem->vcc1_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; i<max_channels; i++) {
		power_of_ten=1.0;
		power_of_two=24.0e-9;
		for (j=0; j<timing_ranges; j++) {
			for (k=0; k<timing_polarities; k++) {
				for (m=0; m<points_in_range; m++) {
					mem->slew_pwl_time[i][j][k][m]=0.0;
					mem->slew_pwl_Vc_norm4095[i][j][k][m]=0;

					int temp_int;

					if (m==0) {
						/* These values have been determined by experiment. */
						/* Double CH2 delays for -KMPF */

						temp_int = 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]=(6e-9*power_of_ten)+20e-9;
						mem->delay_pwl_time[i][j][k][m]=((6e-9*power_of_ten)+10e-9) * (i+1.0);
						mem->burst_pwl_time[i][j][k][m]=(22e-9*power_of_ten)+25e-9;
					} else if (m==1) {
						temp_int = dac_max/2.15;

						mem->period_pwl_time[i][j][k][m]=(100e-9*power_of_ten)+50e-9;
						mem->pw_pwl_time[i][j][k][m]=(14e-9*power_of_ten)+25e-9;
						mem->delay_pwl_time[i][j][k][m]=((14e-9*power_of_ten)+30e-9) * (i+1.0);
						mem->burst_pwl_time[i][j][k][m]=(33e-9*power_of_ten)+10e-9;
					} else if (m==2) {
						temp_int = dac_max/4.6;

						mem->period_pwl_time[i][j][k][m]=(230e-9*power_of_ten)+100e-9;
						mem->pw_pwl_time[i][j][k][m]=(28e-9*power_of_ten)+50e-9;
						mem->delay_pwl_time[i][j][k][m]=((28e-9*power_of_ten)+60e-9) * (i+1.0);
						mem->burst_pwl_time[i][j][k][m]=(50e-9*power_of_ten)+10e-9;

					} else if (m==3) {
						temp_int = dac_max / 10;

						mem->period_pwl_time[i][j][k][m]=(470e-9*power_of_ten)+160e-9;
						mem->pw_pwl_time[i][j][k][m]=(70e-9*power_of_ten)+80e-9;
						mem->delay_pwl_time[i][j][k][m]=((70e-9*power_of_ten))+90e-9 * (i+1.0);
						mem->burst_pwl_time[i][j][k][m]=(110e-9*power_of_ten)+10e-9;
					} else if (m==4) {
						temp_int = dac_max / 20 ;

						mem->period_pwl_time[i][j][k][m]=(900e-9*power_of_ten)+200e-9;
						mem->pw_pwl_time[i][j][k][m]=(140e-9*power_of_ten)+80e-9;
						mem->delay_pwl_time[i][j][k][m]=((140e-9*power_of_ten)+90e-9) * (i+1.0);
						mem->burst_pwl_time[i][j][k][m]=(220e-9*power_of_ten)+10e-9;
					} else {
						temp_int = 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;
					mem->delay_pwl_Vc_norm4095[i][j][k][m]=temp_int;
					mem->period_pwl_Vc_norm4095[i][j][k][m]=temp_int;
					mem->burst_pwl_Vc_norm4095[i][j][k][m]=temp_int;
				}
			}
			power_of_ten*=10.0;
			power_of_two*=2.0;
		}


		power_of_two=20.0e-9;
		for (j=0; j<ampl_ranges; j++) {
			for (k=0; k<ampl_polarities; k++) {
				for (m=0; m<points_in_range; m++) {
					if (m==0) {
						mem->rise_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; j<timing_ranges; j++) {
			for (k=0; k<ampl_polarities; k++) {
				mem->pw_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; j<max_stored_settings; j++) {
			mem->rcl_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; i<max_channels; i++) {
		mem->routing_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->distort_fully_below_ampl[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->volt_ctrl_delay[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->pw_ab_mode_enabled[i]=1;
		mem->double_pulse_allowed[i]=1;
		mem->invert_allowed[i]=1;
		mem->ea_enabled[i]=1;
                mem->ew_enabled[i]=0;
		mem->switchable_load[i]=1;
		mem->monitor_enabled[i]=0;
		mem->use_pos_ampl_data_only[i]=0;
		mem->rise_time_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->ew_xtra_rly[i]=5;
		mem->ignore_ampl_polarity[i]=0;
		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->couple_first_N_pw_ranges_to_ampl_ranges[i]=0;

		for (j=0; j<max_attens; j++) {
			mem->attenuators[i][j] = 0.0;
		}

		mem->atten_percent_max_ampl[i] = 0.93;

		mem->force_monotonic_ext_trig_delay[i] = 0;

		mem->max_freq_for_high_ot[i] = 0.0;
		mem->high_ot[i] = 0.0;

		for (j=0; j<max_fixed_ampl_points; j++) {
			mem->fixed_ampl_points[i][j] = 0.0;
		}

		mem->ext2_enabled[i] = 0;
		mem->toggle_trig_at_boot[i] = 1;
		mem->sequential_attenuators[i] = 0;
	}

	mem->relay_delay_in_sec=0.5;
	mem->extended_relay_delay_in_sec=0.5;

	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) */


	/* originally Second_Dly_Port (for obsolete AVX-DD-A3-PS-TC) */
	mem->I2C_port_for_CH2_delay = Second_PW_Port;	// standard address for PCB 205A now

	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? */

}

static gpointer userflashwritethreadfunc(gpointer data)
{

	//printf("userflash write thread start\n");
	while (alive || g_async_queue_length(userflashwritequeue) > 0) { // make sure the last job in the queue gets written
		if (g_async_queue_length(userflashwritequeue) == 0) { // go into a sleep
			g_mutex_lock(&writermutex);
			g_cond_wait(&writerwakeup, &writermutex);
			g_mutex_unlock(&writermutex);
		} else {
			userflashjob* job = (userflashjob*) g_async_queue_pop(userflashwritequeue);
			if (job != NULL ) {
				// process job
				//printf("processing write\n");
				writeUserBlockNow(job->mem, job->addr, job->numbytes);
				g_free(job->mem);
				g_free(job);
			}
			// If someone is waiting for the queue to start accepting
			// jobs again and we are tell them about it..
			if (g_async_queue_length(userflashwritequeue) < MAXQUEUEDJOBS) {
				g_mutex_lock(&writermutex);
				g_cond_broadcast(&writerqueueopen);
				g_mutex_unlock(&writermutex);
			}
		}

	}
	//printf("userflash write thread stop\n");

	return NULL ;
}

void startFlashWriterThread()
{
	alive= true;
	userflashwritequeue = g_async_queue_new();
	userflashwritethread = g_thread_create(userflashwritethreadfunc, NULL, true, NULL);
}

void initFlash(FlashStruct *mem, gboolean reset_to_defaults, int starting_location)
{
	startFlashWriterThread();
	atexit(stopFlashWriterThread);

	int read_size = readUserBlock(mem);

	if (	(read_size == 0) ||
	                (mem->fully_programmed == Not_Programmed) ||
	                (reset_to_defaults &&
	                 (starting_location >= 0) &&
	                 (starting_location < sizeof(*mem)))	) {

		g_print_debug ("initializing flash memory\n");
		LCD_write(0,0,"Initialize Flash Memory ...");

		gchar *message = g_strdup_printf ("Initialize Flash Memory, %d - %d", starting_location, (int) sizeof(*mem));
		LCD_write(0,0,message);
		g_free (message);

		// uninitialized device!
		initFlashValues(mem);

		// save the default Flash config, for nonvolatile persistence
		writeUserBlock(mem, starting_location, sizeof(*mem) - starting_location);

		LCD_write(1,0,"Flash Init, Done!         ");
	}
}

void stopFlashWriterThread()
{
	alive = false;
	g_cond_broadcast(&writerwakeup);
	g_thread_join(userflashwritethread); // block until the write thread is totally finished.
	g_async_queue_unref(userflashwritequeue);
}

void fixFlash(FlashStruct *mem)
{

	int i, fix_initial_constants;

	fix_initial_constants = 0;

	// handle change in model number location
	if ((mem->model_num[0] != 'A') && (mem->model_num[1] != 'V')) {
		strcpy(mem->model_num, mem->model_num_old);
		++fix_initial_constants;
	}

	for (i=0; i<max_channels; i++) {

		globals.Constraints.composite_min_burst_time[i]=mem->min_burst_gap[i];
		if ((mem->min_burst_per[i] - mem->min_pw[i]) > globals.Constraints.composite_min_burst_time[i]) {
			globals.Constraints.composite_min_burst_time[i] = mem->min_burst_per[i] - mem->min_pw[i];
		}

		int j;
		float safe_val = 0.0;
		gboolean uses_fixed_ampl;

		uses_fixed_ampl = (number_of_fixed_ampl_points(i) > 0);
		safe_val = rst_ampl_value (i);

		for (j=0; j<max_stored_settings; j++) {
			if (mem->rcl_burst_time[i][j] < globals.Constraints.composite_min_burst_time[i]) {
				mem->rcl_burst_time[i][j]=globals.Constraints.composite_min_burst_time[i];
				++fix_initial_constants;
			}
			if (mem->rcl_rise_time[i][j] < mem->min_rise_time[i]) {
				mem->rcl_rise_time[i][j]=mem->min_rise_time[i];
				++fix_initial_constants;
			}
			if (mem->rcl_slew[i][j] < mem->min_slew[i]) {
				mem->rcl_slew[i][j]=mem->min_slew[i];
				++fix_initial_constants;
			}
			if (mem->rcl_soft_current_limit[i][j] < mem->min_soft_current_limit[i]) {
				mem->rcl_soft_current_limit[i][j]=mem->max_soft_current_limit[i];
				++fix_initial_constants;
			}

			if (uses_fixed_ampl && !fixed_ampl_ok(i,mem->rcl_amplitude[i][j])) {
				// AVRQ-4-B
				mem->rcl_amplitude[i][j] = safe_val;
				++fix_initial_constants;
			}

			if ((safe_val != 0.0) && (mem->rcl_amplitude[i][j] == 0.0)) {
				// AVR-D4-B, AVR-CD2-B CH2
				mem->rcl_amplitude[i][j] = safe_val;
				++fix_initial_constants;
			}

		}

		// for AVM-6-B in particular
		if ( (globals.Flash.fully_programmed==All_Programmed) &&
		     (mem->max_freq[i] >= 5e6) &&
		     (mem->fix_pw_dac_val[i] < dac_max/4)) {
			mem->fix_pw_dac_val[i] = dac_max/4;
			++fix_initial_constants;
		}
	}

	if (fix_initial_constants) {
		int eprom_loc;

		eprom_loc = (char *) &(mem->rcl_burst_time) - (char *) &(mem->flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->rcl_burst_time));

		eprom_loc = (char *) &(mem->rcl_rise_time) - (char *) &(mem->flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->rcl_rise_time));

		eprom_loc = (char *) &(mem->rcl_slew) - (char *) &(mem->flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->rcl_slew));

		eprom_loc = (char *) &(mem->rcl_soft_current_limit) - (char *) &(mem->flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->rcl_soft_current_limit));

                eprom_loc = (char *) &(mem->rcl_amplitude) - (char *) &(mem->flash_start);
                writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->rcl_amplitude));

		eprom_loc = (char *) &(mem->model_num) - (char *) &(mem->flash_start);
                writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->model_num));

		eprom_loc = (char *) &(mem->fix_pw_dac_val) - (char *) &(mem->flash_start);
		writeUserBlock(&globals.Flash, eprom_loc, sizeof(mem->fix_pw_dac_val));
	}

}