| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455 | // SPDX-License-Identifier: GPL-2.0+/* * (C) Copyright 2002 * Rich Ireland, Enterasys Networks, rireland@enterasys.com. */#include <common.h>		/* core U-Boot definitions */#include <spartan2.h>		/* Spartan-II device family *//* Define FPGA_DEBUG to get debug printf's */#ifdef	FPGA_DEBUG#define PRINTF(fmt,args...)	printf (fmt ,##args)#else#define PRINTF(fmt,args...)#endif#undef CONFIG_SYS_FPGA_CHECK_BUSY#undef CONFIG_SYS_FPGA_PROG_FEEDBACK/* Note: The assumption is that we cannot possibly run fast enough to * overrun the device (the Slave Parallel mode can free run at 50MHz). * If there is a need to operate slower, define CONFIG_FPGA_DELAY in * the board config file to slow things down. */#ifndef CONFIG_FPGA_DELAY#define CONFIG_FPGA_DELAY()#endif#ifndef CONFIG_SYS_FPGA_WAIT#define CONFIG_SYS_FPGA_WAIT CONFIG_SYS_HZ/100	/* 10 ms */#endifstatic int spartan2_sp_load(xilinx_desc *desc, const void *buf, size_t bsize);static int spartan2_sp_dump(xilinx_desc *desc, const void *buf, size_t bsize);/* static int spartan2_sp_info(xilinx_desc *desc ); */static int spartan2_ss_load(xilinx_desc *desc, const void *buf, size_t bsize);static int spartan2_ss_dump(xilinx_desc *desc, const void *buf, size_t bsize);/* static int spartan2_ss_info(xilinx_desc *desc ); *//* ------------------------------------------------------------------------- *//* Spartan-II Generic Implementation */static int spartan2_load(xilinx_desc *desc, const void *buf, size_t bsize,			 bitstream_type bstype){	int ret_val = FPGA_FAIL;	switch (desc->iface) {	case slave_serial:		PRINTF ("%s: Launching Slave Serial Load\n", __FUNCTION__);		ret_val = spartan2_ss_load(desc, buf, bsize);		break;	case slave_parallel:		PRINTF ("%s: Launching Slave Parallel Load\n", __FUNCTION__);		ret_val = spartan2_sp_load(desc, buf, bsize);		break;	default:		printf ("%s: Unsupported interface type, %d\n",				__FUNCTION__, desc->iface);	}	return ret_val;}static int spartan2_dump(xilinx_desc *desc, const void *buf, size_t bsize){	int ret_val = FPGA_FAIL;	switch (desc->iface) {	case slave_serial:		PRINTF ("%s: Launching Slave Serial Dump\n", __FUNCTION__);		ret_val = spartan2_ss_dump(desc, buf, bsize);		break;	case slave_parallel:		PRINTF ("%s: Launching Slave Parallel Dump\n", __FUNCTION__);		ret_val = spartan2_sp_dump(desc, buf, bsize);		break;	default:		printf ("%s: Unsupported interface type, %d\n",				__FUNCTION__, desc->iface);	}	return ret_val;}static int spartan2_info(xilinx_desc *desc){	return FPGA_SUCCESS;}/* ------------------------------------------------------------------------- *//* Spartan-II Slave Parallel Generic Implementation */static int spartan2_sp_load(xilinx_desc *desc, const void *buf, size_t bsize){	int ret_val = FPGA_FAIL;	/* assume the worst */	xilinx_spartan2_slave_parallel_fns *fn = desc->iface_fns;	PRINTF ("%s: start with interface functions @ 0x%p\n",			__FUNCTION__, fn);	if (fn) {		size_t bytecount = 0;		unsigned char *data = (unsigned char *) buf;		int cookie = desc->cookie;	/* make a local copy */		unsigned long ts;		/* timestamp */		PRINTF ("%s: Function Table:\n"				"ptr:\t0x%p\n"				"struct: 0x%p\n"				"pre: 0x%p\n"				"pgm:\t0x%p\n"				"init:\t0x%p\n"				"err:\t0x%p\n"				"clk:\t0x%p\n"				"cs:\t0x%p\n"				"wr:\t0x%p\n"				"read data:\t0x%p\n"				"write data:\t0x%p\n"				"busy:\t0x%p\n"				"abort:\t0x%p\n",				"post:\t0x%p\n\n",				__FUNCTION__, &fn, fn, fn->pre, fn->pgm, fn->init, fn->err,				fn->clk, fn->cs, fn->wr, fn->rdata, fn->wdata, fn->busy,				fn->abort, fn->post);		/*		 * This code is designed to emulate the "Express Style"		 * Continuous Data Loading in Slave Parallel Mode for		 * the Spartan-II Family.		 */#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		printf ("Loading FPGA Device %d...\n", cookie);#endif		/*		 * Run the pre configuration function if there is one.		 */		if (*fn->pre) {			(*fn->pre) (cookie);		}		/* Establish the initial state */		(*fn->pgm) (true, true, cookie);	/* Assert the program, commit */		/* Get ready for the burn */		CONFIG_FPGA_DELAY ();		(*fn->pgm) (false, true, cookie);	/* Deassert the program, commit */		ts = get_timer (0);		/* get current time */		/* Now wait for INIT and BUSY to go high */		do {			CONFIG_FPGA_DELAY ();			if (get_timer (ts) > CONFIG_SYS_FPGA_WAIT) {	/* check the time */				puts ("** Timeout waiting for INIT to clear.\n");				(*fn->abort) (cookie);	/* abort the burn */				return FPGA_FAIL;			}		} while ((*fn->init) (cookie) && (*fn->busy) (cookie));		(*fn->wr) (true, true, cookie); /* Assert write, commit */		(*fn->cs) (true, true, cookie); /* Assert chip select, commit */		(*fn->clk) (true, true, cookie);	/* Assert the clock pin */		/* Load the data */		while (bytecount < bsize) {			/* XXX - do we check for an Ctrl-C press in here ??? */			/* XXX - Check the error bit? */			(*fn->wdata) (data[bytecount++], true, cookie); /* write the data */			CONFIG_FPGA_DELAY ();			(*fn->clk) (false, true, cookie);	/* Deassert the clock pin */			CONFIG_FPGA_DELAY ();			(*fn->clk) (true, true, cookie);	/* Assert the clock pin */#ifdef CONFIG_SYS_FPGA_CHECK_BUSY			ts = get_timer (0);	/* get current time */			while ((*fn->busy) (cookie)) {				/* XXX - we should have a check in here somewhere to				 * make sure we aren't busy forever... */				CONFIG_FPGA_DELAY ();				(*fn->clk) (false, true, cookie);	/* Deassert the clock pin */				CONFIG_FPGA_DELAY ();				(*fn->clk) (true, true, cookie);	/* Assert the clock pin */				if (get_timer (ts) > CONFIG_SYS_FPGA_WAIT) {	/* check the time */					puts ("** Timeout waiting for BUSY to clear.\n");					(*fn->abort) (cookie);	/* abort the burn */					return FPGA_FAIL;				}			}#endif#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK			if (bytecount % (bsize / 40) == 0)				putc ('.');		/* let them know we are alive */#endif		}		CONFIG_FPGA_DELAY ();		(*fn->cs) (false, true, cookie);	/* Deassert the chip select */		(*fn->wr) (false, true, cookie);	/* Deassert the write pin */#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		putc ('\n');			/* terminate the dotted line */#endif		/* now check for done signal */		ts = get_timer (0);		/* get current time */		ret_val = FPGA_SUCCESS;		while ((*fn->done) (cookie) == FPGA_FAIL) {			CONFIG_FPGA_DELAY ();			(*fn->clk) (false, true, cookie);	/* Deassert the clock pin */			CONFIG_FPGA_DELAY ();			(*fn->clk) (true, true, cookie);	/* Assert the clock pin */			if (get_timer (ts) > CONFIG_SYS_FPGA_WAIT) {	/* check the time */				puts ("** Timeout waiting for DONE to clear.\n");				(*fn->abort) (cookie);	/* abort the burn */				ret_val = FPGA_FAIL;				break;			}		}		/*		 * Run the post configuration function if there is one.		 */		if (*fn->post)			(*fn->post) (cookie);#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		if (ret_val == FPGA_SUCCESS)			puts ("Done.\n");		else			puts ("Fail.\n");#endif	} else {		printf ("%s: NULL Interface function table!\n", __FUNCTION__);	}	return ret_val;}static int spartan2_sp_dump(xilinx_desc *desc, const void *buf, size_t bsize){	int ret_val = FPGA_FAIL;	/* assume the worst */	xilinx_spartan2_slave_parallel_fns *fn = desc->iface_fns;	if (fn) {		unsigned char *data = (unsigned char *) buf;		size_t bytecount = 0;		int cookie = desc->cookie;	/* make a local copy */		printf ("Starting Dump of FPGA Device %d...\n", cookie);		(*fn->cs) (true, true, cookie); /* Assert chip select, commit */		(*fn->clk) (true, true, cookie);	/* Assert the clock pin */		/* dump the data */		while (bytecount < bsize) {			/* XXX - do we check for an Ctrl-C press in here ??? */			(*fn->clk) (false, true, cookie);	/* Deassert the clock pin */			(*fn->clk) (true, true, cookie);	/* Assert the clock pin */			(*fn->rdata) (&(data[bytecount++]), cookie);	/* read the data */#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK			if (bytecount % (bsize / 40) == 0)				putc ('.');		/* let them know we are alive */#endif		}		(*fn->cs) (false, false, cookie);	/* Deassert the chip select */		(*fn->clk) (false, true, cookie);	/* Deassert the clock pin */		(*fn->clk) (true, true, cookie);	/* Assert the clock pin */#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		putc ('\n');			/* terminate the dotted line */#endif		puts ("Done.\n");		/* XXX - checksum the data? */	} else {		printf ("%s: NULL Interface function table!\n", __FUNCTION__);	}	return ret_val;}/* ------------------------------------------------------------------------- */static int spartan2_ss_load(xilinx_desc *desc, const void *buf, size_t bsize){	int ret_val = FPGA_FAIL;	/* assume the worst */	xilinx_spartan2_slave_serial_fns *fn = desc->iface_fns;	int i;	unsigned char val;	PRINTF ("%s: start with interface functions @ 0x%p\n",			__FUNCTION__, fn);	if (fn) {		size_t bytecount = 0;		unsigned char *data = (unsigned char *) buf;		int cookie = desc->cookie;	/* make a local copy */		unsigned long ts;		/* timestamp */		PRINTF ("%s: Function Table:\n"				"ptr:\t0x%p\n"				"struct: 0x%p\n"				"pgm:\t0x%p\n"				"init:\t0x%p\n"				"clk:\t0x%p\n"				"wr:\t0x%p\n"				"done:\t0x%p\n\n",				__FUNCTION__, &fn, fn, fn->pgm, fn->init,				fn->clk, fn->wr, fn->done);#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		printf ("Loading FPGA Device %d...\n", cookie);#endif		/*		 * Run the pre configuration function if there is one.		 */		if (*fn->pre) {			(*fn->pre) (cookie);		}		/* Establish the initial state */		(*fn->pgm) (true, true, cookie);	/* Assert the program, commit */		/* Wait for INIT state (init low)                            */		ts = get_timer (0);		/* get current time */		do {			CONFIG_FPGA_DELAY ();			if (get_timer (ts) > CONFIG_SYS_FPGA_WAIT) {	/* check the time */				puts ("** Timeout waiting for INIT to start.\n");				return FPGA_FAIL;			}		} while (!(*fn->init) (cookie));		/* Get ready for the burn */		CONFIG_FPGA_DELAY ();		(*fn->pgm) (false, true, cookie);	/* Deassert the program, commit */		ts = get_timer (0);		/* get current time */		/* Now wait for INIT to go high */		do {			CONFIG_FPGA_DELAY ();			if (get_timer (ts) > CONFIG_SYS_FPGA_WAIT) {	/* check the time */				puts ("** Timeout waiting for INIT to clear.\n");				return FPGA_FAIL;			}		} while ((*fn->init) (cookie));		/* Load the data */		while (bytecount < bsize) {			/* Xilinx detects an error if INIT goes low (active)			   while DONE is low (inactive) */			if ((*fn->done) (cookie) == 0 && (*fn->init) (cookie)) {				puts ("** CRC error during FPGA load.\n");				return (FPGA_FAIL);			}			val = data [bytecount ++];			i = 8;			do {				/* Deassert the clock */				(*fn->clk) (false, true, cookie);				CONFIG_FPGA_DELAY ();				/* Write data */				(*fn->wr) ((val & 0x80), true, cookie);				CONFIG_FPGA_DELAY ();				/* Assert the clock */				(*fn->clk) (true, true, cookie);				CONFIG_FPGA_DELAY ();				val <<= 1;				i --;			} while (i > 0);#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK			if (bytecount % (bsize / 40) == 0)				putc ('.');		/* let them know we are alive */#endif		}		CONFIG_FPGA_DELAY ();#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		putc ('\n');			/* terminate the dotted line */#endif		/* now check for done signal */		ts = get_timer (0);		/* get current time */		ret_val = FPGA_SUCCESS;		(*fn->wr) (true, true, cookie);		while (! (*fn->done) (cookie)) {			CONFIG_FPGA_DELAY ();			(*fn->clk) (false, true, cookie);	/* Deassert the clock pin */			CONFIG_FPGA_DELAY ();			(*fn->clk) (true, true, cookie);	/* Assert the clock pin */			putc ('*');			if (get_timer (ts) > CONFIG_SYS_FPGA_WAIT) {	/* check the time */				puts ("** Timeout waiting for DONE to clear.\n");				ret_val = FPGA_FAIL;				break;			}		}		putc ('\n');			/* terminate the dotted line */		/*		 * Run the post configuration function if there is one.		 */		if (*fn->post)			(*fn->post) (cookie);#ifdef CONFIG_SYS_FPGA_PROG_FEEDBACK		if (ret_val == FPGA_SUCCESS)			puts ("Done.\n");		else			puts ("Fail.\n");#endif	} else {		printf ("%s: NULL Interface function table!\n", __FUNCTION__);	}	return ret_val;}static int spartan2_ss_dump(xilinx_desc *desc, const void *buf, size_t bsize){	/* Readback is only available through the Slave Parallel and         */	/* boundary-scan interfaces.                                         */	printf ("%s: Slave Serial Dumping is unavailable\n",			__FUNCTION__);	return FPGA_FAIL;}struct xilinx_fpga_op spartan2_op = {	.load = spartan2_load,	.dump = spartan2_dump,	.info = spartan2_info,};
 |