xref: /aosp_15_r20/external/libusb/examples/ezusb.c (revision 86b64dcb59b3a0b37502ecd56e119234366a6f7e)
1*86b64dcbSAndroid Build Coastguard Worker /*
2*86b64dcbSAndroid Build Coastguard Worker  * Copyright © 2001 Stephen Williams ([email protected])
3*86b64dcbSAndroid Build Coastguard Worker  * Copyright © 2001-2002 David Brownell ([email protected])
4*86b64dcbSAndroid Build Coastguard Worker  * Copyright © 2008 Roger Williams ([email protected])
5*86b64dcbSAndroid Build Coastguard Worker  * Copyright © 2012 Pete Batard ([email protected])
6*86b64dcbSAndroid Build Coastguard Worker  * Copyright © 2013 Federico Manzan ([email protected])
7*86b64dcbSAndroid Build Coastguard Worker  *
8*86b64dcbSAndroid Build Coastguard Worker  *    This source code is free software; you can redistribute it
9*86b64dcbSAndroid Build Coastguard Worker  *    and/or modify it in source code form under the terms of the GNU
10*86b64dcbSAndroid Build Coastguard Worker  *    General Public License as published by the Free Software
11*86b64dcbSAndroid Build Coastguard Worker  *    Foundation; either version 2 of the License, or (at your option)
12*86b64dcbSAndroid Build Coastguard Worker  *    any later version.
13*86b64dcbSAndroid Build Coastguard Worker  *
14*86b64dcbSAndroid Build Coastguard Worker  *    This program is distributed in the hope that it will be useful,
15*86b64dcbSAndroid Build Coastguard Worker  *    but WITHOUT ANY WARRANTY; without even the implied warranty of
16*86b64dcbSAndroid Build Coastguard Worker  *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17*86b64dcbSAndroid Build Coastguard Worker  *    GNU General Public License for more details.
18*86b64dcbSAndroid Build Coastguard Worker  *
19*86b64dcbSAndroid Build Coastguard Worker  *    You should have received a copy of the GNU General Public License
20*86b64dcbSAndroid Build Coastguard Worker  *    along with this program; if not, write to the Free Software
21*86b64dcbSAndroid Build Coastguard Worker  *    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
22*86b64dcbSAndroid Build Coastguard Worker  */
23*86b64dcbSAndroid Build Coastguard Worker 
24*86b64dcbSAndroid Build Coastguard Worker #include <config.h>
25*86b64dcbSAndroid Build Coastguard Worker 
26*86b64dcbSAndroid Build Coastguard Worker #include <stdio.h>
27*86b64dcbSAndroid Build Coastguard Worker #include <errno.h>
28*86b64dcbSAndroid Build Coastguard Worker #include <stdlib.h>
29*86b64dcbSAndroid Build Coastguard Worker #include <string.h>
30*86b64dcbSAndroid Build Coastguard Worker #include <stdint.h>
31*86b64dcbSAndroid Build Coastguard Worker 
32*86b64dcbSAndroid Build Coastguard Worker #include "libusb.h"
33*86b64dcbSAndroid Build Coastguard Worker #include "ezusb.h"
34*86b64dcbSAndroid Build Coastguard Worker 
35*86b64dcbSAndroid Build Coastguard Worker /*
36*86b64dcbSAndroid Build Coastguard Worker  * This file contains functions for uploading firmware into Cypress
37*86b64dcbSAndroid Build Coastguard Worker  * EZ-USB microcontrollers. These chips use control endpoint 0 and vendor
38*86b64dcbSAndroid Build Coastguard Worker  * specific commands to support writing into the on-chip SRAM. They also
39*86b64dcbSAndroid Build Coastguard Worker  * support writing into the CPUCS register, which is how we reset the
40*86b64dcbSAndroid Build Coastguard Worker  * processor after loading firmware (including the reset vector).
41*86b64dcbSAndroid Build Coastguard Worker  *
42*86b64dcbSAndroid Build Coastguard Worker  * These Cypress devices are 8-bit 8051 based microcontrollers with
43*86b64dcbSAndroid Build Coastguard Worker  * special support for USB I/O.  They come in several packages, and
44*86b64dcbSAndroid Build Coastguard Worker  * some can be set up with external memory when device costs allow.
45*86b64dcbSAndroid Build Coastguard Worker  * Note that the design was originally by AnchorChips, so you may find
46*86b64dcbSAndroid Build Coastguard Worker  * references to that vendor (which was later merged into Cypress).
47*86b64dcbSAndroid Build Coastguard Worker  * The Cypress FX parts are largely compatible with the Anchorhip ones.
48*86b64dcbSAndroid Build Coastguard Worker  */
49*86b64dcbSAndroid Build Coastguard Worker 
50*86b64dcbSAndroid Build Coastguard Worker int verbose = 1;
51*86b64dcbSAndroid Build Coastguard Worker 
52*86b64dcbSAndroid Build Coastguard Worker /*
53*86b64dcbSAndroid Build Coastguard Worker  * return true if [addr,addr+len] includes external RAM
54*86b64dcbSAndroid Build Coastguard Worker  * for Anchorchips EZ-USB or Cypress EZ-USB FX
55*86b64dcbSAndroid Build Coastguard Worker  */
fx_is_external(uint32_t addr,size_t len)56*86b64dcbSAndroid Build Coastguard Worker static bool fx_is_external(uint32_t addr, size_t len)
57*86b64dcbSAndroid Build Coastguard Worker {
58*86b64dcbSAndroid Build Coastguard Worker 	/* with 8KB RAM, 0x0000-0x1b3f can be written
59*86b64dcbSAndroid Build Coastguard Worker 	 * we can't tell if it's a 4KB device here
60*86b64dcbSAndroid Build Coastguard Worker 	 */
61*86b64dcbSAndroid Build Coastguard Worker 	if (addr <= 0x1b3f)
62*86b64dcbSAndroid Build Coastguard Worker 		return ((addr + len) > 0x1b40);
63*86b64dcbSAndroid Build Coastguard Worker 
64*86b64dcbSAndroid Build Coastguard Worker 	/* there may be more RAM; unclear if we can write it.
65*86b64dcbSAndroid Build Coastguard Worker 	 * some bulk buffers may be unused, 0x1b3f-0x1f3f
66*86b64dcbSAndroid Build Coastguard Worker 	 * firmware can set ISODISAB for 2KB at 0x2000-0x27ff
67*86b64dcbSAndroid Build Coastguard Worker 	 */
68*86b64dcbSAndroid Build Coastguard Worker 	return true;
69*86b64dcbSAndroid Build Coastguard Worker }
70*86b64dcbSAndroid Build Coastguard Worker 
71*86b64dcbSAndroid Build Coastguard Worker /*
72*86b64dcbSAndroid Build Coastguard Worker  * return true if [addr,addr+len] includes external RAM
73*86b64dcbSAndroid Build Coastguard Worker  * for Cypress EZ-USB FX2
74*86b64dcbSAndroid Build Coastguard Worker  */
fx2_is_external(uint32_t addr,size_t len)75*86b64dcbSAndroid Build Coastguard Worker static bool fx2_is_external(uint32_t addr, size_t len)
76*86b64dcbSAndroid Build Coastguard Worker {
77*86b64dcbSAndroid Build Coastguard Worker 	/* 1st 8KB for data/code, 0x0000-0x1fff */
78*86b64dcbSAndroid Build Coastguard Worker 	if (addr <= 0x1fff)
79*86b64dcbSAndroid Build Coastguard Worker 		return ((addr + len) > 0x2000);
80*86b64dcbSAndroid Build Coastguard Worker 
81*86b64dcbSAndroid Build Coastguard Worker 	/* and 512 for data, 0xe000-0xe1ff */
82*86b64dcbSAndroid Build Coastguard Worker 	else if (addr >= 0xe000 && addr <= 0xe1ff)
83*86b64dcbSAndroid Build Coastguard Worker 		return ((addr + len) > 0xe200);
84*86b64dcbSAndroid Build Coastguard Worker 
85*86b64dcbSAndroid Build Coastguard Worker 	/* otherwise, it's certainly external */
86*86b64dcbSAndroid Build Coastguard Worker 	else
87*86b64dcbSAndroid Build Coastguard Worker 		return true;
88*86b64dcbSAndroid Build Coastguard Worker }
89*86b64dcbSAndroid Build Coastguard Worker 
90*86b64dcbSAndroid Build Coastguard Worker /*
91*86b64dcbSAndroid Build Coastguard Worker  * return true if [addr,addr+len] includes external RAM
92*86b64dcbSAndroid Build Coastguard Worker  * for Cypress EZ-USB FX2LP
93*86b64dcbSAndroid Build Coastguard Worker  */
fx2lp_is_external(uint32_t addr,size_t len)94*86b64dcbSAndroid Build Coastguard Worker static bool fx2lp_is_external(uint32_t addr, size_t len)
95*86b64dcbSAndroid Build Coastguard Worker {
96*86b64dcbSAndroid Build Coastguard Worker 	/* 1st 16KB for data/code, 0x0000-0x3fff */
97*86b64dcbSAndroid Build Coastguard Worker 	if (addr <= 0x3fff)
98*86b64dcbSAndroid Build Coastguard Worker 		return ((addr + len) > 0x4000);
99*86b64dcbSAndroid Build Coastguard Worker 
100*86b64dcbSAndroid Build Coastguard Worker 	/* and 512 for data, 0xe000-0xe1ff */
101*86b64dcbSAndroid Build Coastguard Worker 	else if (addr >= 0xe000 && addr <= 0xe1ff)
102*86b64dcbSAndroid Build Coastguard Worker 		return ((addr + len) > 0xe200);
103*86b64dcbSAndroid Build Coastguard Worker 
104*86b64dcbSAndroid Build Coastguard Worker 	/* otherwise, it's certainly external */
105*86b64dcbSAndroid Build Coastguard Worker 	else
106*86b64dcbSAndroid Build Coastguard Worker 		return true;
107*86b64dcbSAndroid Build Coastguard Worker }
108*86b64dcbSAndroid Build Coastguard Worker 
109*86b64dcbSAndroid Build Coastguard Worker 
110*86b64dcbSAndroid Build Coastguard Worker /*****************************************************************************/
111*86b64dcbSAndroid Build Coastguard Worker 
112*86b64dcbSAndroid Build Coastguard Worker /*
113*86b64dcbSAndroid Build Coastguard Worker  * These are the requests (bRequest) that the bootstrap loader is expected
114*86b64dcbSAndroid Build Coastguard Worker  * to recognize.  The codes are reserved by Cypress, and these values match
115*86b64dcbSAndroid Build Coastguard Worker  * what EZ-USB hardware, or "Vend_Ax" firmware (2nd stage loader) uses.
116*86b64dcbSAndroid Build Coastguard Worker  * Cypress' "a3load" is nice because it supports both FX and FX2, although
117*86b64dcbSAndroid Build Coastguard Worker  * it doesn't have the EEPROM support (subset of "Vend_Ax").
118*86b64dcbSAndroid Build Coastguard Worker  */
119*86b64dcbSAndroid Build Coastguard Worker #define RW_INTERNAL     0xA0	/* hardware implements this one */
120*86b64dcbSAndroid Build Coastguard Worker #define RW_MEMORY       0xA3
121*86b64dcbSAndroid Build Coastguard Worker 
122*86b64dcbSAndroid Build Coastguard Worker /*
123*86b64dcbSAndroid Build Coastguard Worker  * Issues the specified vendor-specific write request.
124*86b64dcbSAndroid Build Coastguard Worker  */
ezusb_write(libusb_device_handle * device,const char * label,uint8_t opcode,uint32_t addr,const unsigned char * data,size_t len)125*86b64dcbSAndroid Build Coastguard Worker static int ezusb_write(libusb_device_handle *device, const char *label,
126*86b64dcbSAndroid Build Coastguard Worker 	uint8_t opcode, uint32_t addr, const unsigned char *data, size_t len)
127*86b64dcbSAndroid Build Coastguard Worker {
128*86b64dcbSAndroid Build Coastguard Worker 	int status;
129*86b64dcbSAndroid Build Coastguard Worker 
130*86b64dcbSAndroid Build Coastguard Worker 	if (verbose > 1)
131*86b64dcbSAndroid Build Coastguard Worker 		logerror("%s, addr 0x%08x len %4u (0x%04x)\n", label, addr, (unsigned)len, (unsigned)len);
132*86b64dcbSAndroid Build Coastguard Worker 	status = libusb_control_transfer(device,
133*86b64dcbSAndroid Build Coastguard Worker 		LIBUSB_ENDPOINT_OUT | LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE,
134*86b64dcbSAndroid Build Coastguard Worker 		opcode, addr & 0xFFFF, addr >> 16,
135*86b64dcbSAndroid Build Coastguard Worker 		(unsigned char*)data, (uint16_t)len, 1000);
136*86b64dcbSAndroid Build Coastguard Worker 	if (status != (signed)len) {
137*86b64dcbSAndroid Build Coastguard Worker 		if (status < 0)
138*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s: %s\n", label, libusb_error_name(status));
139*86b64dcbSAndroid Build Coastguard Worker 		else
140*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s ==> %d\n", label, status);
141*86b64dcbSAndroid Build Coastguard Worker 	}
142*86b64dcbSAndroid Build Coastguard Worker 	if (status < 0) {
143*86b64dcbSAndroid Build Coastguard Worker 		errno = EIO;
144*86b64dcbSAndroid Build Coastguard Worker 		return -1;
145*86b64dcbSAndroid Build Coastguard Worker 	}
146*86b64dcbSAndroid Build Coastguard Worker 	return 0;
147*86b64dcbSAndroid Build Coastguard Worker }
148*86b64dcbSAndroid Build Coastguard Worker 
149*86b64dcbSAndroid Build Coastguard Worker /*
150*86b64dcbSAndroid Build Coastguard Worker  * Issues the specified vendor-specific read request.
151*86b64dcbSAndroid Build Coastguard Worker  */
ezusb_read(libusb_device_handle * device,const char * label,uint8_t opcode,uint32_t addr,const unsigned char * data,size_t len)152*86b64dcbSAndroid Build Coastguard Worker static int ezusb_read(libusb_device_handle *device, const char *label,
153*86b64dcbSAndroid Build Coastguard Worker 	uint8_t opcode, uint32_t addr, const unsigned char *data, size_t len)
154*86b64dcbSAndroid Build Coastguard Worker {
155*86b64dcbSAndroid Build Coastguard Worker 	int status;
156*86b64dcbSAndroid Build Coastguard Worker 
157*86b64dcbSAndroid Build Coastguard Worker 	if (verbose > 1)
158*86b64dcbSAndroid Build Coastguard Worker 		logerror("%s, addr 0x%08x len %4u (0x%04x)\n", label, addr, (unsigned)len, (unsigned)len);
159*86b64dcbSAndroid Build Coastguard Worker 	status = libusb_control_transfer(device,
160*86b64dcbSAndroid Build Coastguard Worker 		LIBUSB_ENDPOINT_IN | LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE,
161*86b64dcbSAndroid Build Coastguard Worker 		opcode, addr & 0xFFFF, addr >> 16,
162*86b64dcbSAndroid Build Coastguard Worker 		(unsigned char*)data, (uint16_t)len, 1000);
163*86b64dcbSAndroid Build Coastguard Worker 	if (status != (signed)len) {
164*86b64dcbSAndroid Build Coastguard Worker 		if (status < 0)
165*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s: %s\n", label, libusb_error_name(status));
166*86b64dcbSAndroid Build Coastguard Worker 		else
167*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s ==> %d\n", label, status);
168*86b64dcbSAndroid Build Coastguard Worker 	}
169*86b64dcbSAndroid Build Coastguard Worker 	if (status < 0) {
170*86b64dcbSAndroid Build Coastguard Worker 		errno = EIO;
171*86b64dcbSAndroid Build Coastguard Worker 		return -1;
172*86b64dcbSAndroid Build Coastguard Worker 	}
173*86b64dcbSAndroid Build Coastguard Worker 	return 0;
174*86b64dcbSAndroid Build Coastguard Worker }
175*86b64dcbSAndroid Build Coastguard Worker 
176*86b64dcbSAndroid Build Coastguard Worker /*
177*86b64dcbSAndroid Build Coastguard Worker  * Modifies the CPUCS register to stop or reset the CPU.
178*86b64dcbSAndroid Build Coastguard Worker  * Returns false on error.
179*86b64dcbSAndroid Build Coastguard Worker  */
ezusb_cpucs(libusb_device_handle * device,uint32_t addr,bool doRun)180*86b64dcbSAndroid Build Coastguard Worker static bool ezusb_cpucs(libusb_device_handle *device, uint32_t addr, bool doRun)
181*86b64dcbSAndroid Build Coastguard Worker {
182*86b64dcbSAndroid Build Coastguard Worker 	int status;
183*86b64dcbSAndroid Build Coastguard Worker 	uint8_t data = doRun ? 0x00 : 0x01;
184*86b64dcbSAndroid Build Coastguard Worker 
185*86b64dcbSAndroid Build Coastguard Worker 	if (verbose)
186*86b64dcbSAndroid Build Coastguard Worker 		logerror("%s\n", data ? "stop CPU" : "reset CPU");
187*86b64dcbSAndroid Build Coastguard Worker 	status = libusb_control_transfer(device,
188*86b64dcbSAndroid Build Coastguard Worker 		LIBUSB_ENDPOINT_OUT | LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE,
189*86b64dcbSAndroid Build Coastguard Worker 		RW_INTERNAL, addr & 0xFFFF, addr >> 16,
190*86b64dcbSAndroid Build Coastguard Worker 		&data, 1, 1000);
191*86b64dcbSAndroid Build Coastguard Worker 	if ((status != 1) &&
192*86b64dcbSAndroid Build Coastguard Worker 		/* We may get an I/O error from libusb as the device disappears */
193*86b64dcbSAndroid Build Coastguard Worker 		((!doRun) || (status != LIBUSB_ERROR_IO)))
194*86b64dcbSAndroid Build Coastguard Worker 	{
195*86b64dcbSAndroid Build Coastguard Worker 		const char *mesg = "can't modify CPUCS";
196*86b64dcbSAndroid Build Coastguard Worker 		if (status < 0)
197*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s: %s\n", mesg, libusb_error_name(status));
198*86b64dcbSAndroid Build Coastguard Worker 		else
199*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s\n", mesg);
200*86b64dcbSAndroid Build Coastguard Worker 		return false;
201*86b64dcbSAndroid Build Coastguard Worker 	} else
202*86b64dcbSAndroid Build Coastguard Worker 		return true;
203*86b64dcbSAndroid Build Coastguard Worker }
204*86b64dcbSAndroid Build Coastguard Worker 
205*86b64dcbSAndroid Build Coastguard Worker /*
206*86b64dcbSAndroid Build Coastguard Worker  * Send an FX3 jump to address command
207*86b64dcbSAndroid Build Coastguard Worker  * Returns false on error.
208*86b64dcbSAndroid Build Coastguard Worker  */
ezusb_fx3_jump(libusb_device_handle * device,uint32_t addr)209*86b64dcbSAndroid Build Coastguard Worker static bool ezusb_fx3_jump(libusb_device_handle *device, uint32_t addr)
210*86b64dcbSAndroid Build Coastguard Worker {
211*86b64dcbSAndroid Build Coastguard Worker 	int status;
212*86b64dcbSAndroid Build Coastguard Worker 
213*86b64dcbSAndroid Build Coastguard Worker 	if (verbose)
214*86b64dcbSAndroid Build Coastguard Worker 		logerror("transfer execution to Program Entry at 0x%08x\n", addr);
215*86b64dcbSAndroid Build Coastguard Worker 	status = libusb_control_transfer(device,
216*86b64dcbSAndroid Build Coastguard Worker 		LIBUSB_ENDPOINT_OUT | LIBUSB_REQUEST_TYPE_VENDOR | LIBUSB_RECIPIENT_DEVICE,
217*86b64dcbSAndroid Build Coastguard Worker 		RW_INTERNAL, addr & 0xFFFF, addr >> 16,
218*86b64dcbSAndroid Build Coastguard Worker 		NULL, 0, 1000);
219*86b64dcbSAndroid Build Coastguard Worker 	/* We may get an I/O error from libusb as the device disappears */
220*86b64dcbSAndroid Build Coastguard Worker 	if ((status != 0) && (status != LIBUSB_ERROR_IO))
221*86b64dcbSAndroid Build Coastguard Worker 	{
222*86b64dcbSAndroid Build Coastguard Worker 		const char *mesg = "failed to send jump command";
223*86b64dcbSAndroid Build Coastguard Worker 		if (status < 0)
224*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s: %s\n", mesg, libusb_error_name(status));
225*86b64dcbSAndroid Build Coastguard Worker 		else
226*86b64dcbSAndroid Build Coastguard Worker 			logerror("%s\n", mesg);
227*86b64dcbSAndroid Build Coastguard Worker 		return false;
228*86b64dcbSAndroid Build Coastguard Worker 	} else
229*86b64dcbSAndroid Build Coastguard Worker 		return true;
230*86b64dcbSAndroid Build Coastguard Worker }
231*86b64dcbSAndroid Build Coastguard Worker 
232*86b64dcbSAndroid Build Coastguard Worker /*****************************************************************************/
233*86b64dcbSAndroid Build Coastguard Worker 
234*86b64dcbSAndroid Build Coastguard Worker /*
235*86b64dcbSAndroid Build Coastguard Worker  * Parse an Intel HEX image file and invoke the poke() function on the
236*86b64dcbSAndroid Build Coastguard Worker  * various segments to implement policies such as writing to RAM (with
237*86b64dcbSAndroid Build Coastguard Worker  * a one or two stage loader setup, depending on the firmware) or to
238*86b64dcbSAndroid Build Coastguard Worker  * EEPROM (two stages required).
239*86b64dcbSAndroid Build Coastguard Worker  *
240*86b64dcbSAndroid Build Coastguard Worker  * image       - the hex image file
241*86b64dcbSAndroid Build Coastguard Worker  * context     - for use by poke()
242*86b64dcbSAndroid Build Coastguard Worker  * is_external - if non-null, used to check which segments go into
243*86b64dcbSAndroid Build Coastguard Worker  *               external memory (writable only by software loader)
244*86b64dcbSAndroid Build Coastguard Worker  * poke        - called with each memory segment; errors indicated
245*86b64dcbSAndroid Build Coastguard Worker  *               by returning negative values.
246*86b64dcbSAndroid Build Coastguard Worker  *
247*86b64dcbSAndroid Build Coastguard Worker  * Caller is responsible for halting CPU as needed, such as when
248*86b64dcbSAndroid Build Coastguard Worker  * overwriting a second stage loader.
249*86b64dcbSAndroid Build Coastguard Worker  */
parse_ihex(FILE * image,void * context,bool (* is_external)(uint32_t addr,size_t len),int (* poke)(void * context,uint32_t addr,bool external,const unsigned char * data,size_t len))250*86b64dcbSAndroid Build Coastguard Worker static int parse_ihex(FILE *image, void *context,
251*86b64dcbSAndroid Build Coastguard Worker 	bool (*is_external)(uint32_t addr, size_t len),
252*86b64dcbSAndroid Build Coastguard Worker 	int (*poke) (void *context, uint32_t addr, bool external,
253*86b64dcbSAndroid Build Coastguard Worker 	const unsigned char *data, size_t len))
254*86b64dcbSAndroid Build Coastguard Worker {
255*86b64dcbSAndroid Build Coastguard Worker 	unsigned char data[1023];
256*86b64dcbSAndroid Build Coastguard Worker 	uint32_t data_addr = 0;
257*86b64dcbSAndroid Build Coastguard Worker 	size_t data_len = 0;
258*86b64dcbSAndroid Build Coastguard Worker 	int rc;
259*86b64dcbSAndroid Build Coastguard Worker 	int first_line = 1;
260*86b64dcbSAndroid Build Coastguard Worker 	bool external = false;
261*86b64dcbSAndroid Build Coastguard Worker 
262*86b64dcbSAndroid Build Coastguard Worker 	/* Read the input file as an IHEX file, and report the memory segments
263*86b64dcbSAndroid Build Coastguard Worker 	 * as we go.  Each line holds a max of 16 bytes, but uploading is
264*86b64dcbSAndroid Build Coastguard Worker 	 * faster (and EEPROM space smaller) if we merge those lines into larger
265*86b64dcbSAndroid Build Coastguard Worker 	 * chunks.  Most hex files keep memory segments together, which makes
266*86b64dcbSAndroid Build Coastguard Worker 	 * such merging all but free.  (But it may still be worth sorting the
267*86b64dcbSAndroid Build Coastguard Worker 	 * hex files to make up for undesirable behavior from tools.)
268*86b64dcbSAndroid Build Coastguard Worker 	 *
269*86b64dcbSAndroid Build Coastguard Worker 	 * Note that EEPROM segments max out at 1023 bytes; the upload protocol
270*86b64dcbSAndroid Build Coastguard Worker 	 * allows segments of up to 64 KBytes (more than a loader could handle).
271*86b64dcbSAndroid Build Coastguard Worker 	 */
272*86b64dcbSAndroid Build Coastguard Worker 	for (;;) {
273*86b64dcbSAndroid Build Coastguard Worker 		char buf[512], *cp;
274*86b64dcbSAndroid Build Coastguard Worker 		char tmp, type;
275*86b64dcbSAndroid Build Coastguard Worker 		size_t len;
276*86b64dcbSAndroid Build Coastguard Worker 		unsigned idx, off;
277*86b64dcbSAndroid Build Coastguard Worker 
278*86b64dcbSAndroid Build Coastguard Worker 		cp = fgets(buf, sizeof(buf), image);
279*86b64dcbSAndroid Build Coastguard Worker 		if (cp == NULL) {
280*86b64dcbSAndroid Build Coastguard Worker 			logerror("EOF without EOF record!\n");
281*86b64dcbSAndroid Build Coastguard Worker 			break;
282*86b64dcbSAndroid Build Coastguard Worker 		}
283*86b64dcbSAndroid Build Coastguard Worker 
284*86b64dcbSAndroid Build Coastguard Worker 		/* EXTENSION: "# comment-till-end-of-line", for copyrights etc */
285*86b64dcbSAndroid Build Coastguard Worker 		if (buf[0] == '#')
286*86b64dcbSAndroid Build Coastguard Worker 			continue;
287*86b64dcbSAndroid Build Coastguard Worker 
288*86b64dcbSAndroid Build Coastguard Worker 		if (buf[0] != ':') {
289*86b64dcbSAndroid Build Coastguard Worker 			logerror("not an ihex record: %s", buf);
290*86b64dcbSAndroid Build Coastguard Worker 			return -2;
291*86b64dcbSAndroid Build Coastguard Worker 		}
292*86b64dcbSAndroid Build Coastguard Worker 
293*86b64dcbSAndroid Build Coastguard Worker 		/* ignore any newline */
294*86b64dcbSAndroid Build Coastguard Worker 		cp = strchr(buf, '\n');
295*86b64dcbSAndroid Build Coastguard Worker 		if (cp)
296*86b64dcbSAndroid Build Coastguard Worker 			*cp = 0;
297*86b64dcbSAndroid Build Coastguard Worker 
298*86b64dcbSAndroid Build Coastguard Worker 		if (verbose >= 3)
299*86b64dcbSAndroid Build Coastguard Worker 			logerror("** LINE: %s\n", buf);
300*86b64dcbSAndroid Build Coastguard Worker 
301*86b64dcbSAndroid Build Coastguard Worker 		/* Read the length field (up to 16 bytes) */
302*86b64dcbSAndroid Build Coastguard Worker 		tmp = buf[3];
303*86b64dcbSAndroid Build Coastguard Worker 		buf[3] = 0;
304*86b64dcbSAndroid Build Coastguard Worker 		len = strtoul(buf+1, NULL, 16);
305*86b64dcbSAndroid Build Coastguard Worker 		buf[3] = tmp;
306*86b64dcbSAndroid Build Coastguard Worker 
307*86b64dcbSAndroid Build Coastguard Worker 		/* Read the target offset (address up to 64KB) */
308*86b64dcbSAndroid Build Coastguard Worker 		tmp = buf[7];
309*86b64dcbSAndroid Build Coastguard Worker 		buf[7] = 0;
310*86b64dcbSAndroid Build Coastguard Worker 		off = (unsigned int)strtoul(buf+3, NULL, 16);
311*86b64dcbSAndroid Build Coastguard Worker 		buf[7] = tmp;
312*86b64dcbSAndroid Build Coastguard Worker 
313*86b64dcbSAndroid Build Coastguard Worker 		/* Initialize data_addr */
314*86b64dcbSAndroid Build Coastguard Worker 		if (first_line) {
315*86b64dcbSAndroid Build Coastguard Worker 			data_addr = off;
316*86b64dcbSAndroid Build Coastguard Worker 			first_line = 0;
317*86b64dcbSAndroid Build Coastguard Worker 		}
318*86b64dcbSAndroid Build Coastguard Worker 
319*86b64dcbSAndroid Build Coastguard Worker 		/* Read the record type */
320*86b64dcbSAndroid Build Coastguard Worker 		tmp = buf[9];
321*86b64dcbSAndroid Build Coastguard Worker 		buf[9] = 0;
322*86b64dcbSAndroid Build Coastguard Worker 		type = (char)strtoul(buf+7, NULL, 16);
323*86b64dcbSAndroid Build Coastguard Worker 		buf[9] = tmp;
324*86b64dcbSAndroid Build Coastguard Worker 
325*86b64dcbSAndroid Build Coastguard Worker 		/* If this is an EOF record, then make it so. */
326*86b64dcbSAndroid Build Coastguard Worker 		if (type == 1) {
327*86b64dcbSAndroid Build Coastguard Worker 			if (verbose >= 2)
328*86b64dcbSAndroid Build Coastguard Worker 				logerror("EOF on hexfile\n");
329*86b64dcbSAndroid Build Coastguard Worker 			break;
330*86b64dcbSAndroid Build Coastguard Worker 		}
331*86b64dcbSAndroid Build Coastguard Worker 
332*86b64dcbSAndroid Build Coastguard Worker 		if (type != 0) {
333*86b64dcbSAndroid Build Coastguard Worker 			logerror("unsupported record type: %u\n", type);
334*86b64dcbSAndroid Build Coastguard Worker 			return -3;
335*86b64dcbSAndroid Build Coastguard Worker 		}
336*86b64dcbSAndroid Build Coastguard Worker 
337*86b64dcbSAndroid Build Coastguard Worker 		if ((len * 2) + 11 > strlen(buf)) {
338*86b64dcbSAndroid Build Coastguard Worker 			logerror("record too short?\n");
339*86b64dcbSAndroid Build Coastguard Worker 			return -4;
340*86b64dcbSAndroid Build Coastguard Worker 		}
341*86b64dcbSAndroid Build Coastguard Worker 
342*86b64dcbSAndroid Build Coastguard Worker 		/* FIXME check for _physically_ contiguous not just virtually
343*86b64dcbSAndroid Build Coastguard Worker 		 * e.g. on FX2 0x1f00-0x2100 includes both on-chip and external
344*86b64dcbSAndroid Build Coastguard Worker 		 * memory so it's not really contiguous */
345*86b64dcbSAndroid Build Coastguard Worker 
346*86b64dcbSAndroid Build Coastguard Worker 		/* flush the saved data if it's not contiguous,
347*86b64dcbSAndroid Build Coastguard Worker 		* or when we've buffered as much as we can.
348*86b64dcbSAndroid Build Coastguard Worker 		*/
349*86b64dcbSAndroid Build Coastguard Worker 		if (data_len != 0
350*86b64dcbSAndroid Build Coastguard Worker 			&& (off != (data_addr + data_len)
351*86b64dcbSAndroid Build Coastguard Worker 			/* || !merge */
352*86b64dcbSAndroid Build Coastguard Worker 			|| (data_len + len) > sizeof(data))) {
353*86b64dcbSAndroid Build Coastguard Worker 				if (is_external)
354*86b64dcbSAndroid Build Coastguard Worker 					external = is_external(data_addr, data_len);
355*86b64dcbSAndroid Build Coastguard Worker 				rc = poke(context, data_addr, external, data, data_len);
356*86b64dcbSAndroid Build Coastguard Worker 				if (rc < 0)
357*86b64dcbSAndroid Build Coastguard Worker 					return -1;
358*86b64dcbSAndroid Build Coastguard Worker 				data_addr = off;
359*86b64dcbSAndroid Build Coastguard Worker 				data_len = 0;
360*86b64dcbSAndroid Build Coastguard Worker 		}
361*86b64dcbSAndroid Build Coastguard Worker 
362*86b64dcbSAndroid Build Coastguard Worker 		/* append to saved data, flush later */
363*86b64dcbSAndroid Build Coastguard Worker 		for (idx = 0, cp = buf+9 ;  idx < len ;  idx += 1, cp += 2) {
364*86b64dcbSAndroid Build Coastguard Worker 			tmp = cp[2];
365*86b64dcbSAndroid Build Coastguard Worker 			cp[2] = 0;
366*86b64dcbSAndroid Build Coastguard Worker 			data[data_len + idx] = (uint8_t)strtoul(cp, NULL, 16);
367*86b64dcbSAndroid Build Coastguard Worker 			cp[2] = tmp;
368*86b64dcbSAndroid Build Coastguard Worker 		}
369*86b64dcbSAndroid Build Coastguard Worker 		data_len += len;
370*86b64dcbSAndroid Build Coastguard Worker 	}
371*86b64dcbSAndroid Build Coastguard Worker 
372*86b64dcbSAndroid Build Coastguard Worker 
373*86b64dcbSAndroid Build Coastguard Worker 	/* flush any data remaining */
374*86b64dcbSAndroid Build Coastguard Worker 	if (data_len != 0) {
375*86b64dcbSAndroid Build Coastguard Worker 		if (is_external)
376*86b64dcbSAndroid Build Coastguard Worker 			external = is_external(data_addr, data_len);
377*86b64dcbSAndroid Build Coastguard Worker 		rc = poke(context, data_addr, external, data, data_len);
378*86b64dcbSAndroid Build Coastguard Worker 		if (rc < 0)
379*86b64dcbSAndroid Build Coastguard Worker 			return -1;
380*86b64dcbSAndroid Build Coastguard Worker 	}
381*86b64dcbSAndroid Build Coastguard Worker 	return 0;
382*86b64dcbSAndroid Build Coastguard Worker }
383*86b64dcbSAndroid Build Coastguard Worker 
384*86b64dcbSAndroid Build Coastguard Worker /*
385*86b64dcbSAndroid Build Coastguard Worker  * Parse a binary image file and write it as is to the target.
386*86b64dcbSAndroid Build Coastguard Worker  * Applies to Cypress BIX images for RAM or Cypress IIC images
387*86b64dcbSAndroid Build Coastguard Worker  * for EEPROM.
388*86b64dcbSAndroid Build Coastguard Worker  *
389*86b64dcbSAndroid Build Coastguard Worker  * image       - the BIX image file
390*86b64dcbSAndroid Build Coastguard Worker  * context     - for use by poke()
391*86b64dcbSAndroid Build Coastguard Worker  * is_external - if non-null, used to check which segments go into
392*86b64dcbSAndroid Build Coastguard Worker  *               external memory (writable only by software loader)
393*86b64dcbSAndroid Build Coastguard Worker  * poke        - called with each memory segment; errors indicated
394*86b64dcbSAndroid Build Coastguard Worker  *               by returning negative values.
395*86b64dcbSAndroid Build Coastguard Worker  *
396*86b64dcbSAndroid Build Coastguard Worker  * Caller is responsible for halting CPU as needed, such as when
397*86b64dcbSAndroid Build Coastguard Worker  * overwriting a second stage loader.
398*86b64dcbSAndroid Build Coastguard Worker  */
parse_bin(FILE * image,void * context,bool (* is_external)(uint32_t addr,size_t len),int (* poke)(void * context,uint32_t addr,bool external,const unsigned char * data,size_t len))399*86b64dcbSAndroid Build Coastguard Worker static int parse_bin(FILE *image, void *context,
400*86b64dcbSAndroid Build Coastguard Worker 	bool (*is_external)(uint32_t addr, size_t len), int (*poke)(void *context,
401*86b64dcbSAndroid Build Coastguard Worker 	uint32_t addr, bool external, const unsigned char *data, size_t len))
402*86b64dcbSAndroid Build Coastguard Worker {
403*86b64dcbSAndroid Build Coastguard Worker 	unsigned char data[4096];
404*86b64dcbSAndroid Build Coastguard Worker 	uint32_t data_addr = 0;
405*86b64dcbSAndroid Build Coastguard Worker 	size_t data_len = 0;
406*86b64dcbSAndroid Build Coastguard Worker 	int rc;
407*86b64dcbSAndroid Build Coastguard Worker 	bool external = false;
408*86b64dcbSAndroid Build Coastguard Worker 
409*86b64dcbSAndroid Build Coastguard Worker 	for (;;) {
410*86b64dcbSAndroid Build Coastguard Worker 		data_len = fread(data, 1, 4096, image);
411*86b64dcbSAndroid Build Coastguard Worker 		if (data_len == 0)
412*86b64dcbSAndroid Build Coastguard Worker 			break;
413*86b64dcbSAndroid Build Coastguard Worker 		if (is_external)
414*86b64dcbSAndroid Build Coastguard Worker 			external = is_external(data_addr, data_len);
415*86b64dcbSAndroid Build Coastguard Worker 		rc = poke(context, data_addr, external, data, data_len);
416*86b64dcbSAndroid Build Coastguard Worker 		if (rc < 0)
417*86b64dcbSAndroid Build Coastguard Worker 			return -1;
418*86b64dcbSAndroid Build Coastguard Worker 		data_addr += (uint32_t)data_len;
419*86b64dcbSAndroid Build Coastguard Worker 	}
420*86b64dcbSAndroid Build Coastguard Worker 	return feof(image)?0:-1;
421*86b64dcbSAndroid Build Coastguard Worker }
422*86b64dcbSAndroid Build Coastguard Worker 
423*86b64dcbSAndroid Build Coastguard Worker /*
424*86b64dcbSAndroid Build Coastguard Worker  * Parse a Cypress IIC image file and invoke the poke() function on the
425*86b64dcbSAndroid Build Coastguard Worker  * various segments for writing to RAM
426*86b64dcbSAndroid Build Coastguard Worker  *
427*86b64dcbSAndroid Build Coastguard Worker  * image       - the IIC image file
428*86b64dcbSAndroid Build Coastguard Worker  * context     - for use by poke()
429*86b64dcbSAndroid Build Coastguard Worker  * is_external - if non-null, used to check which segments go into
430*86b64dcbSAndroid Build Coastguard Worker  *               external memory (writable only by software loader)
431*86b64dcbSAndroid Build Coastguard Worker  * poke        - called with each memory segment; errors indicated
432*86b64dcbSAndroid Build Coastguard Worker  *               by returning negative values.
433*86b64dcbSAndroid Build Coastguard Worker  *
434*86b64dcbSAndroid Build Coastguard Worker  * Caller is responsible for halting CPU as needed, such as when
435*86b64dcbSAndroid Build Coastguard Worker  * overwriting a second stage loader.
436*86b64dcbSAndroid Build Coastguard Worker  */
parse_iic(FILE * image,void * context,bool (* is_external)(uint32_t addr,size_t len),int (* poke)(void * context,uint32_t addr,bool external,const unsigned char * data,size_t len))437*86b64dcbSAndroid Build Coastguard Worker static int parse_iic(FILE *image, void *context,
438*86b64dcbSAndroid Build Coastguard Worker 	bool (*is_external)(uint32_t addr, size_t len),
439*86b64dcbSAndroid Build Coastguard Worker 	int (*poke)(void *context, uint32_t addr, bool external, const unsigned char *data, size_t len))
440*86b64dcbSAndroid Build Coastguard Worker {
441*86b64dcbSAndroid Build Coastguard Worker 	unsigned char data[4096];
442*86b64dcbSAndroid Build Coastguard Worker 	uint32_t data_addr = 0;
443*86b64dcbSAndroid Build Coastguard Worker 	size_t data_len = 0, read_len;
444*86b64dcbSAndroid Build Coastguard Worker 	uint8_t block_header[4];
445*86b64dcbSAndroid Build Coastguard Worker 	int rc;
446*86b64dcbSAndroid Build Coastguard Worker 	bool external = false;
447*86b64dcbSAndroid Build Coastguard Worker 	long file_size, initial_pos;
448*86b64dcbSAndroid Build Coastguard Worker 
449*86b64dcbSAndroid Build Coastguard Worker 	initial_pos = ftell(image);
450*86b64dcbSAndroid Build Coastguard Worker 	if (initial_pos < 0)
451*86b64dcbSAndroid Build Coastguard Worker 		return -1;
452*86b64dcbSAndroid Build Coastguard Worker 
453*86b64dcbSAndroid Build Coastguard Worker 	if (fseek(image, 0L, SEEK_END) != 0)
454*86b64dcbSAndroid Build Coastguard Worker 		return -1;
455*86b64dcbSAndroid Build Coastguard Worker 	file_size = ftell(image);
456*86b64dcbSAndroid Build Coastguard Worker 	if (fseek(image, initial_pos, SEEK_SET) != 0)
457*86b64dcbSAndroid Build Coastguard Worker 		return -1;
458*86b64dcbSAndroid Build Coastguard Worker 	for (;;) {
459*86b64dcbSAndroid Build Coastguard Worker 		/* Ignore the trailing reset IIC data (5 bytes) */
460*86b64dcbSAndroid Build Coastguard Worker 		if (ftell(image) >= (file_size - 5))
461*86b64dcbSAndroid Build Coastguard Worker 			break;
462*86b64dcbSAndroid Build Coastguard Worker 		if (fread(&block_header, 1, sizeof(block_header), image) != 4) {
463*86b64dcbSAndroid Build Coastguard Worker 			logerror("unable to read IIC block header\n");
464*86b64dcbSAndroid Build Coastguard Worker 			return -1;
465*86b64dcbSAndroid Build Coastguard Worker 		}
466*86b64dcbSAndroid Build Coastguard Worker 		data_len = (block_header[0] << 8) + block_header[1];
467*86b64dcbSAndroid Build Coastguard Worker 		data_addr = (block_header[2] << 8) + block_header[3];
468*86b64dcbSAndroid Build Coastguard Worker 		if (data_len > sizeof(data)) {
469*86b64dcbSAndroid Build Coastguard Worker 			/* If this is ever reported as an error, switch to using malloc/realloc */
470*86b64dcbSAndroid Build Coastguard Worker 			logerror("IIC data block too small - please report this error to libusb.info\n");
471*86b64dcbSAndroid Build Coastguard Worker 			return -1;
472*86b64dcbSAndroid Build Coastguard Worker 		}
473*86b64dcbSAndroid Build Coastguard Worker 		read_len = fread(data, 1, data_len, image);
474*86b64dcbSAndroid Build Coastguard Worker 		if (read_len != data_len) {
475*86b64dcbSAndroid Build Coastguard Worker 			logerror("read error\n");
476*86b64dcbSAndroid Build Coastguard Worker 			return -1;
477*86b64dcbSAndroid Build Coastguard Worker 		}
478*86b64dcbSAndroid Build Coastguard Worker 		if (is_external)
479*86b64dcbSAndroid Build Coastguard Worker 			external = is_external(data_addr, data_len);
480*86b64dcbSAndroid Build Coastguard Worker 		rc = poke(context, data_addr, external, data, data_len);
481*86b64dcbSAndroid Build Coastguard Worker 		if (rc < 0)
482*86b64dcbSAndroid Build Coastguard Worker 			return -1;
483*86b64dcbSAndroid Build Coastguard Worker 	}
484*86b64dcbSAndroid Build Coastguard Worker 	return 0;
485*86b64dcbSAndroid Build Coastguard Worker }
486*86b64dcbSAndroid Build Coastguard Worker 
487*86b64dcbSAndroid Build Coastguard Worker /* the parse call will be selected according to the image type */
488*86b64dcbSAndroid Build Coastguard Worker static int (*parse[IMG_TYPE_MAX])(FILE *image, void *context, bool (*is_external)(uint32_t addr, size_t len),
489*86b64dcbSAndroid Build Coastguard Worker            int (*poke)(void *context, uint32_t addr, bool external, const unsigned char *data, size_t len))
490*86b64dcbSAndroid Build Coastguard Worker            = { parse_ihex, parse_iic, parse_bin };
491*86b64dcbSAndroid Build Coastguard Worker 
492*86b64dcbSAndroid Build Coastguard Worker /*****************************************************************************/
493*86b64dcbSAndroid Build Coastguard Worker 
494*86b64dcbSAndroid Build Coastguard Worker /*
495*86b64dcbSAndroid Build Coastguard Worker  * For writing to RAM using a first (hardware) or second (software)
496*86b64dcbSAndroid Build Coastguard Worker  * stage loader and 0xA0 or 0xA3 vendor requests
497*86b64dcbSAndroid Build Coastguard Worker  */
498*86b64dcbSAndroid Build Coastguard Worker typedef enum {
499*86b64dcbSAndroid Build Coastguard Worker 	_undef = 0,
500*86b64dcbSAndroid Build Coastguard Worker 	internal_only,		/* hardware first-stage loader */
501*86b64dcbSAndroid Build Coastguard Worker 	skip_internal,		/* first phase, second-stage loader */
502*86b64dcbSAndroid Build Coastguard Worker 	skip_external		/* second phase, second-stage loader */
503*86b64dcbSAndroid Build Coastguard Worker } ram_mode;
504*86b64dcbSAndroid Build Coastguard Worker 
505*86b64dcbSAndroid Build Coastguard Worker struct ram_poke_context {
506*86b64dcbSAndroid Build Coastguard Worker 	libusb_device_handle *device;
507*86b64dcbSAndroid Build Coastguard Worker 	ram_mode mode;
508*86b64dcbSAndroid Build Coastguard Worker 	size_t total, count;
509*86b64dcbSAndroid Build Coastguard Worker };
510*86b64dcbSAndroid Build Coastguard Worker 
511*86b64dcbSAndroid Build Coastguard Worker #define RETRY_LIMIT 5
512*86b64dcbSAndroid Build Coastguard Worker 
ram_poke(void * context,uint32_t addr,bool external,const unsigned char * data,size_t len)513*86b64dcbSAndroid Build Coastguard Worker static int ram_poke(void *context, uint32_t addr, bool external,
514*86b64dcbSAndroid Build Coastguard Worker 	const unsigned char *data, size_t len)
515*86b64dcbSAndroid Build Coastguard Worker {
516*86b64dcbSAndroid Build Coastguard Worker 	struct ram_poke_context *ctx = (struct ram_poke_context*)context;
517*86b64dcbSAndroid Build Coastguard Worker 	int rc;
518*86b64dcbSAndroid Build Coastguard Worker 	unsigned retry = 0;
519*86b64dcbSAndroid Build Coastguard Worker 
520*86b64dcbSAndroid Build Coastguard Worker 	switch (ctx->mode) {
521*86b64dcbSAndroid Build Coastguard Worker 	case internal_only:		/* CPU should be stopped */
522*86b64dcbSAndroid Build Coastguard Worker 		if (external) {
523*86b64dcbSAndroid Build Coastguard Worker 			logerror("can't write %u bytes external memory at 0x%08x\n",
524*86b64dcbSAndroid Build Coastguard Worker 				(unsigned)len, addr);
525*86b64dcbSAndroid Build Coastguard Worker 			errno = EINVAL;
526*86b64dcbSAndroid Build Coastguard Worker 			return -1;
527*86b64dcbSAndroid Build Coastguard Worker 		}
528*86b64dcbSAndroid Build Coastguard Worker 		break;
529*86b64dcbSAndroid Build Coastguard Worker 	case skip_internal:		/* CPU must be running */
530*86b64dcbSAndroid Build Coastguard Worker 		if (!external) {
531*86b64dcbSAndroid Build Coastguard Worker 			if (verbose >= 2) {
532*86b64dcbSAndroid Build Coastguard Worker 				logerror("SKIP on-chip RAM, %u bytes at 0x%08x\n",
533*86b64dcbSAndroid Build Coastguard Worker 					(unsigned)len, addr);
534*86b64dcbSAndroid Build Coastguard Worker 			}
535*86b64dcbSAndroid Build Coastguard Worker 			return 0;
536*86b64dcbSAndroid Build Coastguard Worker 		}
537*86b64dcbSAndroid Build Coastguard Worker 		break;
538*86b64dcbSAndroid Build Coastguard Worker 	case skip_external:		/* CPU should be stopped */
539*86b64dcbSAndroid Build Coastguard Worker 		if (external) {
540*86b64dcbSAndroid Build Coastguard Worker 			if (verbose >= 2) {
541*86b64dcbSAndroid Build Coastguard Worker 				logerror("SKIP external RAM, %u bytes at 0x%08x\n",
542*86b64dcbSAndroid Build Coastguard Worker 					(unsigned)len, addr);
543*86b64dcbSAndroid Build Coastguard Worker 			}
544*86b64dcbSAndroid Build Coastguard Worker 			return 0;
545*86b64dcbSAndroid Build Coastguard Worker 		}
546*86b64dcbSAndroid Build Coastguard Worker 		break;
547*86b64dcbSAndroid Build Coastguard Worker 	case _undef:
548*86b64dcbSAndroid Build Coastguard Worker 	default:
549*86b64dcbSAndroid Build Coastguard Worker 		logerror("bug\n");
550*86b64dcbSAndroid Build Coastguard Worker 		errno = EDOM;
551*86b64dcbSAndroid Build Coastguard Worker 		return -1;
552*86b64dcbSAndroid Build Coastguard Worker 	}
553*86b64dcbSAndroid Build Coastguard Worker 
554*86b64dcbSAndroid Build Coastguard Worker 	ctx->total += len;
555*86b64dcbSAndroid Build Coastguard Worker 	ctx->count++;
556*86b64dcbSAndroid Build Coastguard Worker 
557*86b64dcbSAndroid Build Coastguard Worker 	/* Retry this till we get a real error. Control messages are not
558*86b64dcbSAndroid Build Coastguard Worker 	 * NAKed (just dropped) so time out means is a real problem.
559*86b64dcbSAndroid Build Coastguard Worker 	 */
560*86b64dcbSAndroid Build Coastguard Worker 	while ((rc = ezusb_write(ctx->device,
561*86b64dcbSAndroid Build Coastguard Worker 		external ? "write external" : "write on-chip",
562*86b64dcbSAndroid Build Coastguard Worker 		external ? RW_MEMORY : RW_INTERNAL,
563*86b64dcbSAndroid Build Coastguard Worker 		addr, data, len)) < 0
564*86b64dcbSAndroid Build Coastguard Worker 		&& retry < RETRY_LIMIT) {
565*86b64dcbSAndroid Build Coastguard Worker 		if (rc != LIBUSB_ERROR_TIMEOUT)
566*86b64dcbSAndroid Build Coastguard Worker 			break;
567*86b64dcbSAndroid Build Coastguard Worker 		retry += 1;
568*86b64dcbSAndroid Build Coastguard Worker 	}
569*86b64dcbSAndroid Build Coastguard Worker 	return rc;
570*86b64dcbSAndroid Build Coastguard Worker }
571*86b64dcbSAndroid Build Coastguard Worker 
572*86b64dcbSAndroid Build Coastguard Worker /*
573*86b64dcbSAndroid Build Coastguard Worker  * Load a Cypress Image file into target RAM.
574*86b64dcbSAndroid Build Coastguard Worker  * See http://www.cypress.com/?docID=41351 (AN76405 PDF) for more info.
575*86b64dcbSAndroid Build Coastguard Worker  */
fx3_load_ram(libusb_device_handle * device,const char * path)576*86b64dcbSAndroid Build Coastguard Worker static int fx3_load_ram(libusb_device_handle *device, const char *path)
577*86b64dcbSAndroid Build Coastguard Worker {
578*86b64dcbSAndroid Build Coastguard Worker 	uint32_t dCheckSum, dExpectedCheckSum, dAddress, i, dLen, dLength;
579*86b64dcbSAndroid Build Coastguard Worker 	uint32_t* dImageBuf;
580*86b64dcbSAndroid Build Coastguard Worker 	unsigned char *bBuf, hBuf[4], blBuf[4], rBuf[4096];
581*86b64dcbSAndroid Build Coastguard Worker 	FILE *image;
582*86b64dcbSAndroid Build Coastguard Worker 	int ret = 0;
583*86b64dcbSAndroid Build Coastguard Worker 
584*86b64dcbSAndroid Build Coastguard Worker 	image = fopen(path, "rb");
585*86b64dcbSAndroid Build Coastguard Worker 	if (image == NULL) {
586*86b64dcbSAndroid Build Coastguard Worker 		logerror("unable to open '%s' for input\n", path);
587*86b64dcbSAndroid Build Coastguard Worker 		return -2;
588*86b64dcbSAndroid Build Coastguard Worker 	} else if (verbose)
589*86b64dcbSAndroid Build Coastguard Worker 		logerror("open firmware image %s for RAM upload\n", path);
590*86b64dcbSAndroid Build Coastguard Worker 
591*86b64dcbSAndroid Build Coastguard Worker 	// Read header
592*86b64dcbSAndroid Build Coastguard Worker 	if (fread(hBuf, sizeof(char), sizeof(hBuf), image) != sizeof(hBuf)) {
593*86b64dcbSAndroid Build Coastguard Worker 		logerror("could not read image header");
594*86b64dcbSAndroid Build Coastguard Worker 		ret = -3;
595*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
596*86b64dcbSAndroid Build Coastguard Worker 	}
597*86b64dcbSAndroid Build Coastguard Worker 
598*86b64dcbSAndroid Build Coastguard Worker 	// check "CY" signature byte and format
599*86b64dcbSAndroid Build Coastguard Worker 	if ((hBuf[0] != 'C') || (hBuf[1] != 'Y')) {
600*86b64dcbSAndroid Build Coastguard Worker 		logerror("image doesn't have a CYpress signature\n");
601*86b64dcbSAndroid Build Coastguard Worker 		ret = -3;
602*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
603*86b64dcbSAndroid Build Coastguard Worker 	}
604*86b64dcbSAndroid Build Coastguard Worker 
605*86b64dcbSAndroid Build Coastguard Worker 	// Check bImageType
606*86b64dcbSAndroid Build Coastguard Worker 	switch(hBuf[3]) {
607*86b64dcbSAndroid Build Coastguard Worker 	case 0xB0:
608*86b64dcbSAndroid Build Coastguard Worker 		if (verbose)
609*86b64dcbSAndroid Build Coastguard Worker 			logerror("normal FW binary %s image with checksum\n", (hBuf[2]&0x01)?"data":"executable");
610*86b64dcbSAndroid Build Coastguard Worker 		break;
611*86b64dcbSAndroid Build Coastguard Worker 	case 0xB1:
612*86b64dcbSAndroid Build Coastguard Worker 		logerror("security binary image is not currently supported\n");
613*86b64dcbSAndroid Build Coastguard Worker 		ret = -3;
614*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
615*86b64dcbSAndroid Build Coastguard Worker 	case 0xB2:
616*86b64dcbSAndroid Build Coastguard Worker 		logerror("VID:PID image is not currently supported\n");
617*86b64dcbSAndroid Build Coastguard Worker 		ret = -3;
618*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
619*86b64dcbSAndroid Build Coastguard Worker 	default:
620*86b64dcbSAndroid Build Coastguard Worker 		logerror("invalid image type 0x%02X\n", hBuf[3]);
621*86b64dcbSAndroid Build Coastguard Worker 		ret = -3;
622*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
623*86b64dcbSAndroid Build Coastguard Worker 	}
624*86b64dcbSAndroid Build Coastguard Worker 
625*86b64dcbSAndroid Build Coastguard Worker 	// Read the bootloader version
626*86b64dcbSAndroid Build Coastguard Worker 	if (verbose) {
627*86b64dcbSAndroid Build Coastguard Worker 		if ((ezusb_read(device, "read bootloader version", RW_INTERNAL, 0xFFFF0020, blBuf, 4) < 0)) {
628*86b64dcbSAndroid Build Coastguard Worker 			logerror("Could not read bootloader version\n");
629*86b64dcbSAndroid Build Coastguard Worker 			ret = -8;
630*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
631*86b64dcbSAndroid Build Coastguard Worker 		}
632*86b64dcbSAndroid Build Coastguard Worker 		logerror("FX3 bootloader version: 0x%02X%02X%02X%02X\n", blBuf[3], blBuf[2], blBuf[1], blBuf[0]);
633*86b64dcbSAndroid Build Coastguard Worker 	}
634*86b64dcbSAndroid Build Coastguard Worker 
635*86b64dcbSAndroid Build Coastguard Worker 	dCheckSum = 0;
636*86b64dcbSAndroid Build Coastguard Worker 	if (verbose)
637*86b64dcbSAndroid Build Coastguard Worker 		logerror("writing image...\n");
638*86b64dcbSAndroid Build Coastguard Worker 	while (1) {
639*86b64dcbSAndroid Build Coastguard Worker 		if ((fread(&dLength, sizeof(uint32_t), 1, image) != 1) ||  // read dLength
640*86b64dcbSAndroid Build Coastguard Worker 			(fread(&dAddress, sizeof(uint32_t), 1, image) != 1)) { // read dAddress
641*86b64dcbSAndroid Build Coastguard Worker 			logerror("could not read image");
642*86b64dcbSAndroid Build Coastguard Worker 			ret = -3;
643*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
644*86b64dcbSAndroid Build Coastguard Worker 		}
645*86b64dcbSAndroid Build Coastguard Worker 		if (dLength == 0)
646*86b64dcbSAndroid Build Coastguard Worker 			break; // done
647*86b64dcbSAndroid Build Coastguard Worker 
648*86b64dcbSAndroid Build Coastguard Worker 		// coverity[tainted_data]
649*86b64dcbSAndroid Build Coastguard Worker 		dImageBuf = (uint32_t*)calloc(dLength, sizeof(uint32_t));
650*86b64dcbSAndroid Build Coastguard Worker 		if (dImageBuf == NULL) {
651*86b64dcbSAndroid Build Coastguard Worker 			logerror("could not allocate buffer for image chunk\n");
652*86b64dcbSAndroid Build Coastguard Worker 			ret = -4;
653*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
654*86b64dcbSAndroid Build Coastguard Worker 		}
655*86b64dcbSAndroid Build Coastguard Worker 
656*86b64dcbSAndroid Build Coastguard Worker 		// read sections
657*86b64dcbSAndroid Build Coastguard Worker 		if (fread(dImageBuf, sizeof(uint32_t), dLength, image) != dLength) {
658*86b64dcbSAndroid Build Coastguard Worker 			logerror("could not read image");
659*86b64dcbSAndroid Build Coastguard Worker 			free(dImageBuf);
660*86b64dcbSAndroid Build Coastguard Worker 			ret = -3;
661*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
662*86b64dcbSAndroid Build Coastguard Worker 		}
663*86b64dcbSAndroid Build Coastguard Worker 		for (i = 0; i < dLength; i++)
664*86b64dcbSAndroid Build Coastguard Worker 			dCheckSum += dImageBuf[i];
665*86b64dcbSAndroid Build Coastguard Worker 		dLength <<= 2; // convert to Byte length
666*86b64dcbSAndroid Build Coastguard Worker 		bBuf = (unsigned char*) dImageBuf;
667*86b64dcbSAndroid Build Coastguard Worker 
668*86b64dcbSAndroid Build Coastguard Worker 		while (dLength > 0) {
669*86b64dcbSAndroid Build Coastguard Worker 			dLen = 4096; // 4K max
670*86b64dcbSAndroid Build Coastguard Worker 			if (dLen > dLength)
671*86b64dcbSAndroid Build Coastguard Worker 				dLen = dLength;
672*86b64dcbSAndroid Build Coastguard Worker 			if ((ezusb_write(device, "write firmware", RW_INTERNAL, dAddress, bBuf, dLen) < 0) ||
673*86b64dcbSAndroid Build Coastguard Worker 				(ezusb_read(device, "read firmware", RW_INTERNAL, dAddress, rBuf, dLen) < 0)) {
674*86b64dcbSAndroid Build Coastguard Worker 				logerror("R/W error\n");
675*86b64dcbSAndroid Build Coastguard Worker 				free(dImageBuf);
676*86b64dcbSAndroid Build Coastguard Worker 				ret = -5;
677*86b64dcbSAndroid Build Coastguard Worker 				goto exit;
678*86b64dcbSAndroid Build Coastguard Worker 			}
679*86b64dcbSAndroid Build Coastguard Worker 			// Verify data: rBuf with bBuf
680*86b64dcbSAndroid Build Coastguard Worker 			for (i = 0; i < dLen; i++) {
681*86b64dcbSAndroid Build Coastguard Worker 				if (rBuf[i] != bBuf[i]) {
682*86b64dcbSAndroid Build Coastguard Worker 					logerror("verify error");
683*86b64dcbSAndroid Build Coastguard Worker 					free(dImageBuf);
684*86b64dcbSAndroid Build Coastguard Worker 					ret = -6;
685*86b64dcbSAndroid Build Coastguard Worker 					goto exit;
686*86b64dcbSAndroid Build Coastguard Worker 				}
687*86b64dcbSAndroid Build Coastguard Worker 			}
688*86b64dcbSAndroid Build Coastguard Worker 
689*86b64dcbSAndroid Build Coastguard Worker 			dLength -= dLen;
690*86b64dcbSAndroid Build Coastguard Worker 			bBuf += dLen;
691*86b64dcbSAndroid Build Coastguard Worker 			dAddress += dLen;
692*86b64dcbSAndroid Build Coastguard Worker 		}
693*86b64dcbSAndroid Build Coastguard Worker 		free(dImageBuf);
694*86b64dcbSAndroid Build Coastguard Worker 	}
695*86b64dcbSAndroid Build Coastguard Worker 
696*86b64dcbSAndroid Build Coastguard Worker 	// read pre-computed checksum data
697*86b64dcbSAndroid Build Coastguard Worker 	if ((fread(&dExpectedCheckSum, sizeof(uint32_t), 1, image) != 1) ||
698*86b64dcbSAndroid Build Coastguard Worker 		(dCheckSum != dExpectedCheckSum)) {
699*86b64dcbSAndroid Build Coastguard Worker 		logerror("checksum error\n");
700*86b64dcbSAndroid Build Coastguard Worker 		ret = -7;
701*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
702*86b64dcbSAndroid Build Coastguard Worker 	}
703*86b64dcbSAndroid Build Coastguard Worker 
704*86b64dcbSAndroid Build Coastguard Worker 	// transfer execution to Program Entry
705*86b64dcbSAndroid Build Coastguard Worker 	if (!ezusb_fx3_jump(device, dAddress)) {
706*86b64dcbSAndroid Build Coastguard Worker 		ret = -6;
707*86b64dcbSAndroid Build Coastguard Worker 	}
708*86b64dcbSAndroid Build Coastguard Worker 
709*86b64dcbSAndroid Build Coastguard Worker exit:
710*86b64dcbSAndroid Build Coastguard Worker 	fclose(image);
711*86b64dcbSAndroid Build Coastguard Worker 	return ret;
712*86b64dcbSAndroid Build Coastguard Worker }
713*86b64dcbSAndroid Build Coastguard Worker 
714*86b64dcbSAndroid Build Coastguard Worker /*
715*86b64dcbSAndroid Build Coastguard Worker  * Load a firmware file into target RAM. device is the open libusb
716*86b64dcbSAndroid Build Coastguard Worker  * device, and the path is the name of the source file. Open the file,
717*86b64dcbSAndroid Build Coastguard Worker  * parse the bytes, and write them in one or two phases.
718*86b64dcbSAndroid Build Coastguard Worker  *
719*86b64dcbSAndroid Build Coastguard Worker  * If stage == 0, this uses the first stage loader, built into EZ-USB
720*86b64dcbSAndroid Build Coastguard Worker  * hardware but limited to writing on-chip memory or CPUCS.  Everything
721*86b64dcbSAndroid Build Coastguard Worker  * is written during one stage, unless there's an error such as the image
722*86b64dcbSAndroid Build Coastguard Worker  * holding data that needs to be written to external memory.
723*86b64dcbSAndroid Build Coastguard Worker  *
724*86b64dcbSAndroid Build Coastguard Worker  * Otherwise, things are written in two stages.  First the external
725*86b64dcbSAndroid Build Coastguard Worker  * memory is written, expecting a second stage loader to have already
726*86b64dcbSAndroid Build Coastguard Worker  * been loaded.  Then file is re-parsed and on-chip memory is written.
727*86b64dcbSAndroid Build Coastguard Worker  */
ezusb_load_ram(libusb_device_handle * device,const char * path,int fx_type,int img_type,int stage)728*86b64dcbSAndroid Build Coastguard Worker int ezusb_load_ram(libusb_device_handle *device, const char *path, int fx_type, int img_type, int stage)
729*86b64dcbSAndroid Build Coastguard Worker {
730*86b64dcbSAndroid Build Coastguard Worker 	FILE *image;
731*86b64dcbSAndroid Build Coastguard Worker 	uint32_t cpucs_addr;
732*86b64dcbSAndroid Build Coastguard Worker 	bool (*is_external)(uint32_t off, size_t len);
733*86b64dcbSAndroid Build Coastguard Worker 	struct ram_poke_context ctx;
734*86b64dcbSAndroid Build Coastguard Worker 	int status;
735*86b64dcbSAndroid Build Coastguard Worker 	uint8_t iic_header[8] = { 0 };
736*86b64dcbSAndroid Build Coastguard Worker 	int ret = 0;
737*86b64dcbSAndroid Build Coastguard Worker 
738*86b64dcbSAndroid Build Coastguard Worker 	if (fx_type == FX_TYPE_FX3)
739*86b64dcbSAndroid Build Coastguard Worker 		return fx3_load_ram(device, path);
740*86b64dcbSAndroid Build Coastguard Worker 
741*86b64dcbSAndroid Build Coastguard Worker 	image = fopen(path, "rb");
742*86b64dcbSAndroid Build Coastguard Worker 	if (image == NULL) {
743*86b64dcbSAndroid Build Coastguard Worker 		logerror("%s: unable to open for input.\n", path);
744*86b64dcbSAndroid Build Coastguard Worker 		return -2;
745*86b64dcbSAndroid Build Coastguard Worker 	} else if (verbose > 1)
746*86b64dcbSAndroid Build Coastguard Worker 		logerror("open firmware image %s for RAM upload\n", path);
747*86b64dcbSAndroid Build Coastguard Worker 
748*86b64dcbSAndroid Build Coastguard Worker 	if (img_type == IMG_TYPE_IIC) {
749*86b64dcbSAndroid Build Coastguard Worker 		if ( (fread(iic_header, 1, sizeof(iic_header), image) != sizeof(iic_header))
750*86b64dcbSAndroid Build Coastguard Worker 		  || (((fx_type == FX_TYPE_FX2LP) || (fx_type == FX_TYPE_FX2)) && (iic_header[0] != 0xC2))
751*86b64dcbSAndroid Build Coastguard Worker 		  || ((fx_type == FX_TYPE_AN21) && (iic_header[0] != 0xB2))
752*86b64dcbSAndroid Build Coastguard Worker 		  || ((fx_type == FX_TYPE_FX1) && (iic_header[0] != 0xB6)) ) {
753*86b64dcbSAndroid Build Coastguard Worker 			logerror("IIC image does not contain executable code - cannot load to RAM.\n");
754*86b64dcbSAndroid Build Coastguard Worker 			ret = -1;
755*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
756*86b64dcbSAndroid Build Coastguard Worker 		}
757*86b64dcbSAndroid Build Coastguard Worker 	}
758*86b64dcbSAndroid Build Coastguard Worker 
759*86b64dcbSAndroid Build Coastguard Worker 	/* EZ-USB original/FX and FX2 devices differ, apart from the 8051 core */
760*86b64dcbSAndroid Build Coastguard Worker 	switch(fx_type) {
761*86b64dcbSAndroid Build Coastguard Worker 	case FX_TYPE_FX2LP:
762*86b64dcbSAndroid Build Coastguard Worker 		cpucs_addr = 0xe600;
763*86b64dcbSAndroid Build Coastguard Worker 		is_external = fx2lp_is_external;
764*86b64dcbSAndroid Build Coastguard Worker 		break;
765*86b64dcbSAndroid Build Coastguard Worker 	case FX_TYPE_FX2:
766*86b64dcbSAndroid Build Coastguard Worker 		cpucs_addr = 0xe600;
767*86b64dcbSAndroid Build Coastguard Worker 		is_external = fx2_is_external;
768*86b64dcbSAndroid Build Coastguard Worker 		break;
769*86b64dcbSAndroid Build Coastguard Worker 	default:
770*86b64dcbSAndroid Build Coastguard Worker 		cpucs_addr = 0x7f92;
771*86b64dcbSAndroid Build Coastguard Worker 		is_external = fx_is_external;
772*86b64dcbSAndroid Build Coastguard Worker 		break;
773*86b64dcbSAndroid Build Coastguard Worker 	}
774*86b64dcbSAndroid Build Coastguard Worker 
775*86b64dcbSAndroid Build Coastguard Worker 	/* use only first stage loader? */
776*86b64dcbSAndroid Build Coastguard Worker 	if (stage == 0) {
777*86b64dcbSAndroid Build Coastguard Worker 		ctx.mode = internal_only;
778*86b64dcbSAndroid Build Coastguard Worker 
779*86b64dcbSAndroid Build Coastguard Worker 		/* if required, halt the CPU while we overwrite its code/data */
780*86b64dcbSAndroid Build Coastguard Worker 		if (cpucs_addr && !ezusb_cpucs(device, cpucs_addr, false))
781*86b64dcbSAndroid Build Coastguard Worker 		{
782*86b64dcbSAndroid Build Coastguard Worker 			ret = -1;
783*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
784*86b64dcbSAndroid Build Coastguard Worker 		}
785*86b64dcbSAndroid Build Coastguard Worker 
786*86b64dcbSAndroid Build Coastguard Worker 		/* 2nd stage, first part? loader was already uploaded */
787*86b64dcbSAndroid Build Coastguard Worker 	} else {
788*86b64dcbSAndroid Build Coastguard Worker 		ctx.mode = skip_internal;
789*86b64dcbSAndroid Build Coastguard Worker 
790*86b64dcbSAndroid Build Coastguard Worker 		/* let CPU run; overwrite the 2nd stage loader later */
791*86b64dcbSAndroid Build Coastguard Worker 		if (verbose)
792*86b64dcbSAndroid Build Coastguard Worker 			logerror("2nd stage: write external memory\n");
793*86b64dcbSAndroid Build Coastguard Worker 	}
794*86b64dcbSAndroid Build Coastguard Worker 
795*86b64dcbSAndroid Build Coastguard Worker 	/* scan the image, first (maybe only) time */
796*86b64dcbSAndroid Build Coastguard Worker 	ctx.device = device;
797*86b64dcbSAndroid Build Coastguard Worker 	ctx.total = ctx.count = 0;
798*86b64dcbSAndroid Build Coastguard Worker 	status = parse[img_type](image, &ctx, is_external, ram_poke);
799*86b64dcbSAndroid Build Coastguard Worker 	if (status < 0) {
800*86b64dcbSAndroid Build Coastguard Worker 		logerror("unable to upload %s\n", path);
801*86b64dcbSAndroid Build Coastguard Worker 		ret = status;
802*86b64dcbSAndroid Build Coastguard Worker 		goto exit;
803*86b64dcbSAndroid Build Coastguard Worker 	}
804*86b64dcbSAndroid Build Coastguard Worker 
805*86b64dcbSAndroid Build Coastguard Worker 	/* second part of 2nd stage: rescan */
806*86b64dcbSAndroid Build Coastguard Worker 	// TODO: what should we do for non HEX images there?
807*86b64dcbSAndroid Build Coastguard Worker 	if (stage) {
808*86b64dcbSAndroid Build Coastguard Worker 		ctx.mode = skip_external;
809*86b64dcbSAndroid Build Coastguard Worker 
810*86b64dcbSAndroid Build Coastguard Worker 		/* if needed, halt the CPU while we overwrite the 1st stage loader */
811*86b64dcbSAndroid Build Coastguard Worker 		if (cpucs_addr && !ezusb_cpucs(device, cpucs_addr, false))
812*86b64dcbSAndroid Build Coastguard Worker 		{
813*86b64dcbSAndroid Build Coastguard Worker 			ret = -1;
814*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
815*86b64dcbSAndroid Build Coastguard Worker 		}
816*86b64dcbSAndroid Build Coastguard Worker 
817*86b64dcbSAndroid Build Coastguard Worker 		/* at least write the interrupt vectors (at 0x0000) for reset! */
818*86b64dcbSAndroid Build Coastguard Worker 		status = fseek(image, 0L, SEEK_SET);
819*86b64dcbSAndroid Build Coastguard Worker 		if (status < 0) {
820*86b64dcbSAndroid Build Coastguard Worker 			logerror("unable to rewind file %s\n", path);
821*86b64dcbSAndroid Build Coastguard Worker 			ret = status;
822*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
823*86b64dcbSAndroid Build Coastguard Worker 		}
824*86b64dcbSAndroid Build Coastguard Worker 		if (verbose)
825*86b64dcbSAndroid Build Coastguard Worker 			logerror("2nd stage: write on-chip memory\n");
826*86b64dcbSAndroid Build Coastguard Worker 		status = parse_ihex(image, &ctx, is_external, ram_poke);
827*86b64dcbSAndroid Build Coastguard Worker 		if (status < 0) {
828*86b64dcbSAndroid Build Coastguard Worker 			logerror("unable to completely upload %s\n", path);
829*86b64dcbSAndroid Build Coastguard Worker 			ret = status;
830*86b64dcbSAndroid Build Coastguard Worker 			goto exit;
831*86b64dcbSAndroid Build Coastguard Worker 		}
832*86b64dcbSAndroid Build Coastguard Worker 	}
833*86b64dcbSAndroid Build Coastguard Worker 
834*86b64dcbSAndroid Build Coastguard Worker 	if (verbose && (ctx.count != 0)) {
835*86b64dcbSAndroid Build Coastguard Worker 		logerror("... WROTE: %d bytes, %d segments, avg %d\n",
836*86b64dcbSAndroid Build Coastguard Worker 			(int)ctx.total, (int)ctx.count, (int)(ctx.total/ctx.count));
837*86b64dcbSAndroid Build Coastguard Worker 	}
838*86b64dcbSAndroid Build Coastguard Worker 
839*86b64dcbSAndroid Build Coastguard Worker 	/* if required, reset the CPU so it runs what we just uploaded */
840*86b64dcbSAndroid Build Coastguard Worker 	if (cpucs_addr && !ezusb_cpucs(device, cpucs_addr, true))
841*86b64dcbSAndroid Build Coastguard Worker 		ret = -1;
842*86b64dcbSAndroid Build Coastguard Worker 
843*86b64dcbSAndroid Build Coastguard Worker exit:
844*86b64dcbSAndroid Build Coastguard Worker 	fclose(image);
845*86b64dcbSAndroid Build Coastguard Worker 	return ret;
846*86b64dcbSAndroid Build Coastguard Worker }
847