rtos: remove broken code for handling the deprecated qP packet
[openocd/andreasf.git] / src / target / dsp5680xx.c
blob116f609d7e67ed6a533d9ccc84bc587a53906174
1 /***************************************************************************
2 * Copyright (C) 2011 by Rodrigo L. Rosa *
3 * rodrigorosa.LG@gmail.com *
4 * *
5 * Based on dsp563xx_once.h written by Mathias Kuester *
6 * mkdorg@users.sourceforge.net *
7 * *
8 * This program is free software; you can redistribute it and/or modify *
9 * it under the terms of the GNU General Public License as published by *
10 * the Free Software Foundation; either version 2 of the License, or *
11 * (at your option) any later version. *
12 * *
13 * This program is distributed in the hope that it will be useful, *
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
16 * GNU General Public License for more details. *
17 * *
18 * You should have received a copy of the GNU General Public License *
19 * along with this program; if not, write to the *
20 * Free Software Foundation, Inc., *
21 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
22 ***************************************************************************/
23 #ifdef HAVE_CONFIG_H
24 #include "config.h"
25 #endif
27 #include "target.h"
28 #include "target_type.h"
29 #include "dsp5680xx.h"
31 struct dsp5680xx_common dsp5680xx_context;
34 #define err_check(retval,err_msg) if(retval != ERROR_OK){LOG_ERROR("%s: %d %s.",__FUNCTION__,__LINE__,err_msg);return retval;}
35 #define err_check_propagate(retval) if(retval!=ERROR_OK){return retval;}
37 int dsp5680xx_execute_queue(void){
38 int retval;
39 retval = jtag_execute_queue();
40 err_check_propagate(retval);
41 return retval;
44 static int dsp5680xx_drscan(struct target * target, uint8_t * data_to_shift_into_dr, uint8_t * data_shifted_out_of_dr, int len){
45 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
47 // Inputs:
48 // - data_to_shift_into_dr: This is the data that will be shifted into the JTAG DR reg.
49 // - data_shifted_out_of_dr: The data that will be shifted out of the JTAG DR reg will stored here
50 // - len: Length of the data to be shifted to JTAG DR.
52 // Note: If data_shifted_out_of_dr == NULL, discard incoming bits.
54 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
55 int retval = ERROR_OK;
56 if (NULL == target->tap){
57 retval = ERROR_FAIL;
58 err_check(retval,"Invalid tap");
60 if (len > 32){
61 retval = ERROR_FAIL;
62 err_check(retval,"dr_len overflow, maxium is 32");
64 //TODO what values of len are valid for jtag_add_plain_dr_scan?
65 //can i send as many bits as i want?
66 //is the casting necessary?
67 jtag_add_plain_dr_scan(len,data_to_shift_into_dr,data_shifted_out_of_dr, TAP_IDLE);
68 if(dsp5680xx_context.flush){
69 retval = dsp5680xx_execute_queue();
70 err_check_propagate(retval);
72 if(data_shifted_out_of_dr!=NULL){
73 LOG_DEBUG("Data read (%d bits): 0x%04X",len,*data_shifted_out_of_dr);
74 }else
75 LOG_DEBUG("Data read was discarded.");
76 return retval;
79 static int dsp5680xx_irscan(struct target * target, uint32_t * data_to_shift_into_ir, uint32_t * data_shifted_out_of_ir, uint8_t ir_len){
80 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
81 // Inputs:
82 // - data_to_shift_into_ir: This is the data that will be shifted into the JTAG IR reg.
83 // - data_shifted_out_of_ir: The data that will be shifted out of the JTAG IR reg will stored here
84 // - len: Length of the data to be shifted to JTAG IR.
86 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
87 int retval = ERROR_OK;
88 if (NULL == target->tap){
89 retval = ERROR_FAIL;
90 err_check(retval,"Invalid tap");
92 if (ir_len != target->tap->ir_length){
93 if(target->tap->enabled){
94 retval = ERROR_FAIL;
95 err_check(retval,"Invalid irlen");
96 }else{
97 struct jtag_tap * master_tap = jtag_tap_by_string("dsp568013.chp");
98 if((master_tap == NULL) || ((master_tap->enabled) && (ir_len != DSP5680XX_JTAG_MASTER_TAP_IRLEN))){
99 retval = ERROR_FAIL;
100 err_check(retval,"Invalid irlen");
104 jtag_add_plain_ir_scan(ir_len,(uint8_t *)data_to_shift_into_ir,(uint8_t *)data_shifted_out_of_ir, TAP_IDLE);
105 if(dsp5680xx_context.flush){
106 retval = dsp5680xx_execute_queue();
107 err_check_propagate(retval);
109 return retval;
112 static int dsp5680xx_jtag_status(struct target *target, uint8_t * status){
113 uint32_t read_from_ir;
114 uint32_t instr;
115 int retval;
116 instr = JTAG_INSTR_ENABLE_ONCE;
117 retval = dsp5680xx_irscan(target,& instr, & read_from_ir,DSP5680XX_JTAG_CORE_TAP_IRLEN);
118 err_check_propagate(retval);
119 if(status!=NULL)
120 *status = (uint8_t)read_from_ir;
121 return ERROR_OK;
124 static int jtag_data_read(struct target * target, uint8_t * data_read, int num_bits){
125 uint32_t bogus_instr = 0;
126 int retval = dsp5680xx_drscan(target,(uint8_t *) & bogus_instr,data_read,num_bits);
127 LOG_DEBUG("Data read (%d bits): 0x%04X",num_bits,*data_read);//TODO remove this or move to jtagio?
128 return retval;
131 #define jtag_data_read8(target,data_read) jtag_data_read(target,data_read,8)
132 #define jtag_data_read16(target,data_read) jtag_data_read(target,data_read,16)
133 #define jtag_data_read32(target,data_read) jtag_data_read(target,data_read,32)
135 static uint32_t data_read_dummy;
136 static int jtag_data_write(struct target * target, uint32_t instr,int num_bits, uint32_t * data_read){
137 int retval;
138 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & data_read_dummy,num_bits);
139 err_check_propagate(retval);
140 if(data_read != NULL)
141 *data_read = data_read_dummy;
142 return retval;
145 #define jtag_data_write8(target,instr,data_read) jtag_data_write(target,instr,8,data_read)
146 #define jtag_data_write16(target,instr,data_read) jtag_data_write(target,instr,16,data_read)
147 #define jtag_data_write24(target,instr,data_read) jtag_data_write(target,instr,24,data_read)
148 #define jtag_data_write32(target,instr,data_read) jtag_data_write(target,instr,32,data_read)
151 * Executes EOnCE instruction.
153 * @param target
154 * @param instr Instruction to execute.
155 * @param rw
156 * @param go
157 * @param ex
158 * @param eonce_status Value read from the EOnCE status register.
160 * @return
162 static int eonce_instruction_exec_single(struct target * target, uint8_t instr, uint8_t rw, uint8_t go, uint8_t ex,uint8_t * eonce_status){
163 int retval;
164 uint32_t dr_out_tmp;
165 uint8_t instr_with_flags = instr|(rw<<7)|(go<<6)|(ex<<5);
166 retval = jtag_data_write(target,instr_with_flags,8,&dr_out_tmp);
167 err_check_propagate(retval);
168 if(eonce_status != NULL)
169 *eonce_status = (uint8_t) dr_out_tmp;
170 return retval;
173 ///wrappers for multi opcode instructions
174 #define dsp5680xx_exe_1(target,opcode1,opcode2,opcode3) dsp5680xx_exe1(target,opcode1)
175 #define dsp5680xx_exe_2(target,opcode1,opcode2,opcode3) dsp5680xx_exe2(target,opcode1,opcode2)
176 #define dsp5680xx_exe_3(target,opcode1,opcode2,opcode3) dsp5680xx_exe3(target,opcode1,opcode2,opcode3)
177 #define dsp5680xx_exe_generic(target,words,opcode1,opcode2,opcode3) dsp5680xx_exe_##words(target,opcode1,opcode2,opcode3)
179 /// Executes one word DSP instruction
180 static int dsp5680xx_exe1(struct target * target, uint16_t opcode){
181 int retval;
182 retval = eonce_instruction_exec_single(target,0x04,0,1,0,NULL);
183 err_check_propagate(retval);
184 retval = jtag_data_write16(target,opcode,NULL);
185 err_check_propagate(retval);
186 return retval;
189 /// Executes two word DSP instruction
190 static int dsp5680xx_exe2(struct target * target,uint16_t opcode1, uint16_t opcode2){
191 int retval;
192 retval = eonce_instruction_exec_single(target,0x04,0,0,0,NULL);
193 err_check_propagate(retval);
194 retval = jtag_data_write16(target,opcode1,NULL);
195 err_check_propagate(retval);
196 retval = eonce_instruction_exec_single(target,0x04,0,1,0,NULL);
197 err_check_propagate(retval);
198 retval = jtag_data_write16(target,opcode2,NULL);
199 err_check_propagate(retval);
200 return retval;
203 /// Executes three word DSP instruction
204 static int dsp5680xx_exe3(struct target * target, uint16_t opcode1,uint16_t opcode2,uint16_t opcode3){
205 int retval;
206 retval = eonce_instruction_exec_single(target,0x04,0,0,0,NULL);
207 err_check_propagate(retval);
208 retval = jtag_data_write16(target,opcode1,NULL);
209 err_check_propagate(retval);
210 retval = eonce_instruction_exec_single(target,0x04,0,0,0,NULL);
211 err_check_propagate(retval);
212 retval = jtag_data_write16(target,opcode2,NULL);
213 err_check_propagate(retval);
214 retval = eonce_instruction_exec_single(target,0x04,0,1,0,NULL);
215 err_check_propagate(retval);
216 retval = jtag_data_write16(target,opcode3,NULL);
217 err_check_propagate(retval);
218 return retval;
222 * --------------- Real-time data exchange ---------------
223 * The EOnCE Transmit (OTX) and Receive (ORX) registers are data memory mapped, each with an upper and lower 16 bit word.
224 * Transmit and receive directions are defined from the core’s perspective.
225 * The core writes to the Transmit register and reads the Receive register, and the host through JTAG writes to the Receive register and reads the Transmit register.
226 * Both registers have a combined data memory mapped OTXRXSR which provides indication when each may be accessed.
227 *ref: eonce_rev.1.0_0208081.pdf@36
230 /// writes data into upper ORx register of the target
231 static int core_tx_upper_data(struct target * target, uint16_t data, uint32_t * eonce_status_low){
232 int retval;
233 retval = eonce_instruction_exec_single(target,DSP5680XX_ONCE_ORX1,0,0,0,NULL);
234 err_check_propagate(retval);
235 retval = jtag_data_write16(target,data,eonce_status_low);
236 err_check_propagate(retval);
237 return retval;
240 /// writes data into lower ORx register of the target
241 #define core_tx_lower_data(target,data) eonce_instruction_exec_single(target,DSP5680XX_ONCE_ORX,0,0,0,NULL);\
242 jtag_data_write16(target,data)
246 * @param target
247 * @param data_read: Returns the data read from the upper OTX register via JTAG.
248 * @return: Returns an error code (see error code documentation)
250 static int core_rx_upper_data(struct target * target, uint8_t * data_read)
252 int retval;
253 retval = eonce_instruction_exec_single(target,DSP5680XX_ONCE_OTX1,1,0,0,NULL);
254 err_check_propagate(retval);
255 retval = jtag_data_read16(target,data_read);
256 err_check_propagate(retval);
257 return retval;
262 * @param target
263 * @param data_read: Returns the data read from the lower OTX register via JTAG.
264 * @return: Returns an error code (see error code documentation)
266 static int core_rx_lower_data(struct target * target,uint8_t * data_read)
268 int retval;
269 retval = eonce_instruction_exec_single(target,DSP5680XX_ONCE_OTX,1,0,0,NULL);
270 err_check_propagate(retval);
271 retval = jtag_data_read16(target,data_read);
272 err_check_propagate(retval);
273 return retval;
277 * -- -- -- -- --- -- -- -- --- -- -- -- --- -- -- -- --- -- -- -- --- --
278 * -- -- -- -- --- -- -- -Core Instructions- -- -- -- --- -- -- -- --- --
279 * -- -- -- -- --- -- -- -- --- -- -- -- --- -- -- -- --- -- -- -- --- --
282 /// move.l #value,r0
283 #define core_move_long_to_r0(target,value) dsp5680xx_exe_generic(target,3,0xe418,value&0xffff,value>>16)
285 /// move.l #value,n
286 #define core_move_long_to_n(target,value) dsp5680xx_exe_generic(target,3,0xe41e,value&0xffff,value>>16)
288 /// move x:(r0),y0
289 #define core_move_at_r0_to_y0(target) dsp5680xx_exe_generic(target,1,0xF514,0,0)
291 /// move x:(r0),y1
292 #define core_move_at_r0_to_y1(target) dsp5680xx_exe_generic(target,1,0xF714,0,0)
294 /// move.l x:(r0),y
295 #define core_move_long_at_r0_y(target) dsp5680xx_exe_generic(target,1,0xF734,0,0)
297 /// move y0,x:(r0)
298 #define core_move_y0_at_r0(target) dsp5680xx_exe_generic(target,1,0xd514,0,0)
300 /// bfclr #value,x:(r0)
301 #define eonce_bfclr_at_r0(target,value) dsp5680xx_exe_generic(target,2,0x8040,value,0)
303 /// move #value,y0
304 #define core_move_value_to_y0(target,value) dsp5680xx_exe_generic(target,2,0x8745,value,0)
306 /// move.w y0,x:(r0)+
307 #define core_move_y0_at_r0_inc(target) dsp5680xx_exe_generic(target,1,0xd500,0,0)
309 /// move.w y0,p:(r0)+
310 #define core_move_y0_at_pr0_inc(target) dsp5680xx_exe_generic(target,1,0x8560,0,0)
312 /// move.w p:(r0)+,y0
313 #define core_move_at_pr0_inc_to_y0(target) dsp5680xx_exe_generic(target,1,0x8568,0,0)
315 /// move.w p:(r0)+,y1
316 #define core_move_at_pr0_inc_to_y1(target) dsp5680xx_exe_generic(target,1,0x8768,0,0)
318 /// move.l #value,r2
319 #define core_move_long_to_r2(target,value) dsp5680xx_exe_generic(target,3,0xe41A,value&0xffff,value>>16)
321 /// move y0,x:(r2)
322 #define core_move_y0_at_r2(target) dsp5680xx_exe_generic(target,1,0xd516,0,0)
324 /// move.w #<value>,x:(r2)
325 #define core_move_value_at_r2(target,value) dsp5680xx_exe_generic(target,2,0x8642,value,0)
327 /// move.w #<value>,x:(r0)
328 #define core_move_value_at_r0(target,value) dsp5680xx_exe_generic(target,2,0x8640,value,0)
330 /// move.w #<value>,x:(R2+<disp>)
331 #define core_move_value_at_r2_disp(target,value,disp) dsp5680xx_exe_generic(target,3,0x8646,value,disp)
333 /// move.w x:(r2),Y0
334 #define core_move_at_r2_to_y0(target) dsp5680xx_exe_generic(target,1,0xF516,0,0)
336 /// move.w p:(r2)+,y0
337 #define core_move_at_pr2_inc_to_y0(target) dsp5680xx_exe_generic(target,1,0x856A,0,0)
339 /// move.l #value,r3
340 #define core_move_long_to_r1(target,value) dsp5680xx_exe_generic(target,3,0xE419,value&0xffff,value>>16)
342 /// move.l #value,r3
343 #define core_move_long_to_r3(target,value) dsp5680xx_exe_generic(target,3,0xE41B,value&0xffff,value>>16)
345 /// move.w y0,p:(r3)+
346 #define core_move_y0_at_pr3_inc(target) dsp5680xx_exe_generic(target,1,0x8563,0,0)
348 /// move.w y0,x:(r3)
349 #define core_move_y0_at_r3(target) dsp5680xx_exe_generic(target,1,0xD503,0,0)
351 /// move.l #value,r4
352 #define core_move_long_to_r4(target,value) dsp5680xx_exe_generic(target,3,0xE41C,value&0xffff,value>>16)
354 /// move pc,r4
355 #define core_move_pc_to_r4(target) dsp5680xx_exe_generic(target,1,0xE716,0,0)
357 /// move.l r4,y
358 #define core_move_r4_to_y(target) dsp5680xx_exe_generic(target,1,0xe764,0,0)
360 /// move.w p:(r0)+,y0
361 #define core_move_at_pr0_inc_to_y0(target) dsp5680xx_exe_generic(target,1,0x8568,0,0)
363 /// move.w x:(r0)+,y0
364 #define core_move_at_r0_inc_to_y0(target) dsp5680xx_exe_generic(target,1,0xf500,0,0)
366 /// move x:(r0),y0
367 #define core_move_at_r0_y0(target) dsp5680xx_exe_generic(target,1,0xF514,0,0)
369 /// nop
370 #define eonce_nop(target) dsp5680xx_exe_generic(target,1,0xe700,0,0)
372 /// move.w x:(R2+<disp>),Y0
373 #define core_move_at_r2_disp_to_y0(target,disp) dsp5680xx_exe_generic(target,2,0xF542,disp,0)
375 /// move.w y1,x:(r2)
376 #define core_move_y1_at_r2(target) dsp5680xx_exe_generic(target,1,0xd716,0,0)
378 /// move.w y1,x:(r0)
379 #define core_move_y1_at_r0(target) dsp5680xx_exe_generic(target,1,0xd714,0,0)
381 /// move.bp y0,x:(r0)+
382 #define core_move_byte_y0_at_r0(target) dsp5680xx_exe_generic(target,1,0xd5a0,0,0)
384 /// move.w y1,p:(r0)+
385 #define core_move_y1_at_pr0_inc(target) dsp5680xx_exe_generic(target,1,0x8760,0,0)
387 /// move.w y1,x:(r0)+
388 #define core_move_y1_at_r0_inc(target) dsp5680xx_exe_generic(target,1,0xD700,0,0)
390 /// move.l #value,y
391 #define core_move_long_to_y(target,value) dsp5680xx_exe_generic(target,3,0xe417,value&0xffff,value>>16)
393 static int core_move_value_to_pc(struct target * target, uint32_t value){
394 if (!(target->state == TARGET_HALTED)){
395 LOG_ERROR("Target must be halted to move PC. Target state = %d.",target->state);
396 return ERROR_TARGET_NOT_HALTED;
398 int retval;
399 retval = dsp5680xx_exe_generic(target,3,0xE71E,value&0xffff,value>>16);
400 err_check_propagate(retval);
401 return retval;
404 static int eonce_load_TX_RX_to_r0(struct target * target)
406 int retval;
407 retval = core_move_long_to_r0(target,((MC568013_EONCE_TX_RX_ADDR)+(MC568013_EONCE_OBASE_ADDR<<16)));
408 return retval;
411 static int core_load_TX_RX_high_addr_to_r0(struct target * target)
413 int retval = 0;
414 retval = core_move_long_to_r0(target,((MC568013_EONCE_TX1_RX1_HIGH_ADDR)+(MC568013_EONCE_OBASE_ADDR<<16)));
415 return retval;
418 static int dsp5680xx_read_core_reg(struct target * target, uint8_t reg_addr, uint16_t * data_read)
420 //TODO implement a general version of this which matches what openocd uses.
421 int retval;
422 uint32_t dummy_data_to_shift_into_dr;
423 retval = eonce_instruction_exec_single(target,reg_addr,1,0,0,NULL);
424 err_check_propagate(retval);
425 retval = dsp5680xx_drscan(target,(uint8_t *)& dummy_data_to_shift_into_dr,(uint8_t *) data_read, 8);
426 err_check_propagate(retval);
427 LOG_DEBUG("Reg. data: 0x%02X.",*data_read);
428 return retval;
431 static int eonce_read_status_reg(struct target * target, uint16_t * data){
432 int retval;
433 retval = dsp5680xx_read_core_reg(target,DSP5680XX_ONCE_OSR,data);
434 err_check_propagate(retval);
435 return retval;
439 * Takes the core out of debug mode.
441 * @param target
442 * @param eonce_status Data read from the EOnCE status register.
444 * @return
446 static int eonce_exit_debug_mode(struct target * target,uint8_t * eonce_status){
447 int retval;
448 retval = eonce_instruction_exec_single(target,0x1F,0,0,1,eonce_status);
449 err_check_propagate(retval);
450 return retval;
453 static int switch_tap(struct target * target, struct jtag_tap * master_tap,struct jtag_tap * core_tap){
454 int retval = ERROR_OK;
455 uint32_t instr;
456 uint32_t ir_out;//not used, just to make jtag happy.
457 if(master_tap == NULL){
458 master_tap = jtag_tap_by_string("dsp568013.chp");
459 if(master_tap == NULL){
460 retval = ERROR_FAIL;
461 err_check(retval,"Failed to get master tap.");
464 if(core_tap == NULL){
465 core_tap = jtag_tap_by_string("dsp568013.cpu");
466 if(core_tap == NULL){
467 retval = ERROR_FAIL;
468 err_check(retval,"Failed to get core tap.");
472 if(!(((int)master_tap->enabled) ^ ((int)core_tap->enabled))){
473 LOG_WARNING("Wrong tap enabled/disabled status:\nMaster tap:%d\nCore Tap:%d\nOnly one tap should be enabled at a given time.\n",(int)master_tap->enabled,(int)core_tap->enabled);
476 if(master_tap->enabled){
477 instr = 0x5;
478 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_MASTER_TAP_IRLEN);
479 err_check_propagate(retval);
480 instr = 0x2;
481 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & ir_out,4);
482 err_check_propagate(retval);
483 core_tap->enabled = true;
484 master_tap->enabled = false;
485 }else{
486 instr = 0x08;
487 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
488 err_check_propagate(retval);
489 instr = 0x1;
490 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & ir_out,4);
491 err_check_propagate(retval);
492 core_tap->enabled = false;
493 master_tap->enabled = true;
495 return retval;
499 * Puts the core into debug mode, enabling the EOnCE module.
500 * This will not always work, eonce_enter_debug_mode executes much
501 * more complicated routine, which is guaranteed to work, but requires
502 * a reset. This will complicate comm with the flash module, since
503 * after a reset clock divisors must be set again.
504 * This implementation works most of the time, and is not accesible to the
505 * user.
507 * @param target
508 * @param eonce_status Data read from the EOnCE status register.
510 * @return
512 static int eonce_enter_debug_mode_without_reset(struct target * target, uint16_t * eonce_status){
513 int retval;
514 uint32_t instr = JTAG_INSTR_DEBUG_REQUEST;
515 uint32_t ir_out;//not used, just to make jtag happy.
516 // Debug request #1
517 retval = dsp5680xx_irscan(target,& instr,& ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
518 err_check_propagate(retval);
520 // Enable EOnCE module
521 instr = JTAG_INSTR_ENABLE_ONCE;
522 //Two rounds of jtag 0x6 (enable eonce) to enable EOnCE.
523 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
524 err_check_propagate(retval);
525 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
526 err_check_propagate(retval);
527 // Verify that debug mode is enabled
528 uint16_t data_read_from_dr;
529 retval = eonce_read_status_reg(target,&data_read_from_dr);
530 err_check_propagate(retval);
531 if((data_read_from_dr&0x30) == 0x30){
532 LOG_DEBUG("EOnCE successfully entered debug mode.");
533 target->state = TARGET_HALTED;
534 retval = ERROR_OK;
535 }else{
536 retval = ERROR_TARGET_FAILURE;
537 err_check(retval,"Failed to set EOnCE module to debug mode. Try with halt");
539 if(eonce_status!=NULL)
540 *eonce_status = data_read_from_dr;
541 return ERROR_OK;
544 #define TIME_DIV_FREESCALE 0.3
546 * Puts the core into debug mode, enabling the EOnCE module.
548 * @param target
549 * @param eonce_status Data read from the EOnCE status register.
551 * @return
553 static int eonce_enter_debug_mode(struct target * target, uint16_t * eonce_status){
554 int retval = ERROR_OK;
555 uint32_t instr = JTAG_INSTR_DEBUG_REQUEST;
556 uint32_t ir_out;//not used, just to make jtag happy.
557 uint16_t instr_16;
558 uint16_t read_16;
560 // First try the easy way
561 retval = eonce_enter_debug_mode_without_reset(target,eonce_status);
562 if(retval == ERROR_OK)
563 return retval;
565 struct jtag_tap * tap_chp;
566 struct jtag_tap * tap_cpu;
567 tap_chp = jtag_tap_by_string("dsp568013.chp");
568 if(tap_chp == NULL){
569 retval = ERROR_FAIL;
570 err_check(retval,"Failed to get master tap.");
572 tap_cpu = jtag_tap_by_string("dsp568013.cpu");
573 if(tap_cpu == NULL){
574 retval = ERROR_FAIL;
575 err_check(retval,"Failed to get master tap.");
578 // Enable master tap
579 tap_chp->enabled = true;
580 tap_cpu->enabled = false;
582 instr = MASTER_TAP_CMD_IDCODE;
583 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_MASTER_TAP_IRLEN);
584 err_check_propagate(retval);
585 jtag_add_sleep(TIME_DIV_FREESCALE*100*1000);
587 // Enable EOnCE module
588 jtag_add_reset(0,1);
589 jtag_add_sleep(TIME_DIV_FREESCALE*200*1000);
590 instr = 0x0606ffff;// This was selected experimentally.
591 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & ir_out,32);
592 err_check_propagate(retval);
593 // ir_out now hold tap idcode
595 // Enable core tap
596 tap_chp->enabled = true;
597 retval = switch_tap(target,tap_chp,tap_cpu);
598 err_check_propagate(retval);
600 instr = JTAG_INSTR_ENABLE_ONCE;
601 //Two rounds of jtag 0x6 (enable eonce) to enable EOnCE.
602 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
603 err_check_propagate(retval);
604 instr = JTAG_INSTR_DEBUG_REQUEST;
605 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
606 err_check_propagate(retval);
607 instr_16 = 0x1;
608 retval = dsp5680xx_drscan(target,(uint8_t *) & instr_16,(uint8_t *) & read_16,8);
609 instr_16 = 0x20;
610 retval = dsp5680xx_drscan(target,(uint8_t *) & instr_16,(uint8_t *) & read_16,8);
611 jtag_add_sleep(TIME_DIV_FREESCALE*100*1000);
612 jtag_add_reset(0,0);
613 jtag_add_sleep(TIME_DIV_FREESCALE*300*1000);
615 instr = JTAG_INSTR_ENABLE_ONCE;
616 //Two rounds of jtag 0x6 (enable eonce) to enable EOnCE.
617 for(int i = 0; i<3; i++){
618 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
619 err_check_propagate(retval);
622 for(int i = 0; i<3; i++){
623 instr_16 = 0x86;
624 dsp5680xx_drscan(target,(uint8_t *) & instr_16,(uint8_t *) & read_16,16);
625 instr_16 = 0xff;
626 dsp5680xx_drscan(target,(uint8_t *) & instr_16,(uint8_t *) & read_16,16);
629 // Verify that debug mode is enabled
630 uint16_t data_read_from_dr;
631 retval = eonce_read_status_reg(target,&data_read_from_dr);
632 err_check_propagate(retval);
633 if((data_read_from_dr&0x30) == 0x30){
634 LOG_DEBUG("EOnCE successfully entered debug mode.");
635 target->state = TARGET_HALTED;
636 retval = ERROR_OK;
637 }else{
638 LOG_DEBUG("Failed to set EOnCE module to debug mode.");
639 retval = ERROR_TARGET_FAILURE;
641 if(eonce_status!=NULL)
642 *eonce_status = data_read_from_dr;
643 return retval;
647 * Reads the current value of the program counter and stores it.
649 * @param target
651 * @return
653 static int eonce_pc_store(struct target * target){
654 uint8_t tmp[2];
655 int retval;
656 retval = core_move_pc_to_r4(target);
657 err_check_propagate(retval);
658 retval = core_move_r4_to_y(target);
659 err_check_propagate(retval);
660 retval = eonce_load_TX_RX_to_r0(target);
661 err_check_propagate(retval);
662 retval = core_move_y0_at_r0(target);
663 err_check_propagate(retval);
664 retval = core_rx_lower_data(target,tmp);
665 err_check_propagate(retval);
666 LOG_USER("PC value: 0x%X%X\n",tmp[1],tmp[0]);
667 dsp5680xx_context.stored_pc = (tmp[0]|(tmp[1]<<8));
668 return ERROR_OK;
671 static int dsp5680xx_target_create(struct target *target, Jim_Interp * interp){
672 struct dsp5680xx_common *dsp5680xx = calloc(1, sizeof(struct dsp5680xx_common));
673 target->arch_info = dsp5680xx;
674 return ERROR_OK;
677 static int dsp5680xx_init_target(struct command_context *cmd_ctx, struct target *target){
678 dsp5680xx_context.stored_pc = 0;
679 dsp5680xx_context.flush = 1;
680 LOG_DEBUG("target initiated!");
681 //TODO core tap must be enabled before running these commands, currently this is done in the .cfg tcl script.
682 return ERROR_OK;
685 static int dsp5680xx_arch_state(struct target *target){
686 LOG_USER("%s not implemented yet.",__FUNCTION__);
687 return ERROR_OK;
690 int dsp5680xx_target_status(struct target * target, uint8_t * jtag_st, uint16_t * eonce_st){
691 return target->state;
694 static int dsp5680xx_assert_reset(struct target *target){
695 target->state = TARGET_RESET;
696 return ERROR_OK;
699 static int dsp5680xx_deassert_reset(struct target *target){
700 target->state = TARGET_RUNNING;
701 return ERROR_OK;
704 static int dsp5680xx_halt(struct target *target){
705 int retval;
706 uint16_t eonce_status = 0xbeef;
707 if(target->state == TARGET_HALTED){
708 LOG_USER("Target already halted.");
709 return ERROR_OK;
711 retval = eonce_enter_debug_mode(target,&eonce_status);
712 err_check(retval,"Failed to halt target.");
713 retval = eonce_pc_store(target);
714 err_check_propagate(retval);
715 //TODO is it useful to store the pc?
716 return retval;
719 static int dsp5680xx_poll(struct target *target){
720 int retval;
721 uint8_t jtag_status;
722 uint8_t eonce_status;
723 uint16_t read_tmp;
724 retval = dsp5680xx_jtag_status(target,&jtag_status);
725 err_check_propagate(retval);
726 if (jtag_status == JTAG_STATUS_DEBUG)
727 if (target->state != TARGET_HALTED){
728 retval = eonce_enter_debug_mode(target,&read_tmp);
729 err_check_propagate(retval);
730 eonce_status = (uint8_t) read_tmp;
731 if((eonce_status&EONCE_STAT_MASK) != DSP5680XX_ONCE_OSCR_DEBUG_M){
732 LOG_WARNING("%s: Failed to put EOnCE in debug mode. Is flash locked?...",__FUNCTION__);
733 return ERROR_TARGET_FAILURE;
734 }else{
735 target->state = TARGET_HALTED;
736 return ERROR_OK;
739 if (jtag_status == JTAG_STATUS_NORMAL){
740 if(target->state == TARGET_RESET){
741 retval = dsp5680xx_halt(target);
742 err_check_propagate(retval);
743 retval = eonce_exit_debug_mode(target,&eonce_status);
744 err_check_propagate(retval);
745 if((eonce_status&EONCE_STAT_MASK) != DSP5680XX_ONCE_OSCR_NORMAL_M){
746 LOG_WARNING("%s: JTAG running, but cannot make EOnCE run. Try resetting...",__FUNCTION__);
747 return ERROR_TARGET_FAILURE;
748 }else{
749 target->state = TARGET_RUNNING;
750 return ERROR_OK;
753 if(target->state != TARGET_RUNNING){
754 retval = eonce_read_status_reg(target,&read_tmp);
755 err_check_propagate(retval);
756 eonce_status = (uint8_t) read_tmp;
757 if((eonce_status&EONCE_STAT_MASK) != DSP5680XX_ONCE_OSCR_NORMAL_M){
758 LOG_WARNING("Inconsistent target status. Restart!");
759 return ERROR_TARGET_FAILURE;
762 target->state = TARGET_RUNNING;
763 return ERROR_OK;
765 if(jtag_status == JTAG_STATUS_DEAD){
766 LOG_ERROR("%s: Cannot communicate with JTAG. Check connection...",__FUNCTION__);
767 target->state = TARGET_UNKNOWN;
768 return ERROR_TARGET_FAILURE;
770 if (target->state == TARGET_UNKNOWN){
771 LOG_ERROR("%s: Target status invalid - communication failure",__FUNCTION__);
772 return ERROR_TARGET_FAILURE;
774 return ERROR_OK;
777 static int dsp5680xx_resume(struct target *target, int current, uint32_t address,int handle_breakpoints, int debug_execution){
778 if(target->state == TARGET_RUNNING){
779 LOG_USER("Target already running.");
780 return ERROR_OK;
782 int retval;
783 uint8_t eonce_status;
784 if(!current){
785 retval = core_move_value_to_pc(target,address);
786 err_check_propagate(retval);
789 int retry = 20;
790 while(retry-- > 1){
791 retval = eonce_exit_debug_mode(target,&eonce_status );
792 err_check_propagate(retval);
793 if(eonce_status == DSP5680XX_ONCE_OSCR_NORMAL_M)
794 break;
796 if(retry == 0){
797 retval = ERROR_TARGET_FAILURE;
798 err_check(retval,"Failed to resume...");
799 }else{
800 target->state = TARGET_RUNNING;
802 LOG_DEBUG("EOnCE status: 0x%02X.",eonce_status);
803 return ERROR_OK;
812 * The value of @address determines if it corresponds to P: (program) or X: (data) memory. If the address is over 0x200000 then it is considered X: memory, and @pmem = 0.
813 * The special case of 0xFFXXXX is not modified, since it allows to read out the memory mapped EOnCE registers.
815 * @param address
816 * @param pmem
818 * @return
820 static int dsp5680xx_convert_address(uint32_t * address, int * pmem){
821 // Distinguish data memory (x:) from program memory (p:) by the address.
822 // Addresses over S_FILE_DATA_OFFSET are considered (x:) memory.
823 if(*address >= S_FILE_DATA_OFFSET){
824 *pmem = 0;
825 if(((*address)&0xff0000)!=0xff0000)
826 *address -= S_FILE_DATA_OFFSET;
828 return ERROR_OK;
831 static int dsp5680xx_read_16_single(struct target * target, uint32_t address, uint8_t * data_read, int r_pmem){
832 int retval;
833 retval = core_move_long_to_r0(target,address);
834 err_check_propagate(retval);
835 if(r_pmem)
836 retval = core_move_at_pr0_inc_to_y0(target);
837 else
838 retval = core_move_at_r0_to_y0(target);
839 err_check_propagate(retval);
840 retval = eonce_load_TX_RX_to_r0(target);
841 err_check_propagate(retval);
842 retval = core_move_y0_at_r0(target);
843 err_check_propagate(retval);
844 // at this point the data i want is at the reg eonce can read
845 retval = core_rx_lower_data(target,data_read);
846 err_check_propagate(retval);
847 LOG_DEBUG("%s: Data read from 0x%06X: 0x%02X%02X",__FUNCTION__, address,data_read[1],data_read[0]);
848 return retval;
851 static int dsp5680xx_read_32_single(struct target * target, uint32_t address, uint8_t * data_read, int r_pmem){
852 int retval;
853 address = (address & 0xFFFFFE);
854 // Get data to an intermediate register
855 retval = core_move_long_to_r0(target,address);
856 err_check_propagate(retval);
857 if(r_pmem){
858 retval = core_move_at_pr0_inc_to_y0(target);
859 err_check_propagate(retval);
860 retval = core_move_at_pr0_inc_to_y1(target);
861 err_check_propagate(retval);
862 }else{
863 retval = core_move_at_r0_inc_to_y0(target);
864 err_check_propagate(retval);
865 retval = core_move_at_r0_to_y1(target);
866 err_check_propagate(retval);
868 // Get lower part of data to TX/RX
869 retval = eonce_load_TX_RX_to_r0(target);
870 err_check_propagate(retval);
871 retval = core_move_y0_at_r0_inc(target); // This also load TX/RX high to r0
872 err_check_propagate(retval);
873 // Get upper part of data to TX/RX
874 retval = core_move_y1_at_r0(target);
875 err_check_propagate(retval);
876 // at this point the data i want is at the reg eonce can read
877 retval = core_rx_lower_data(target,data_read);
878 err_check_propagate(retval);
879 retval = core_rx_upper_data(target,data_read+2);
880 err_check_propagate(retval);
881 return retval;
884 static int dsp5680xx_read(struct target * target, uint32_t address, unsigned size, unsigned count, uint8_t * buffer){
885 if(target->state != TARGET_HALTED){
886 LOG_USER("Target must be halted.");
887 return ERROR_FAIL;
889 int retval = ERROR_OK;
890 int pmem = 1;
892 retval = dsp5680xx_convert_address(&address, &pmem);
893 err_check_propagate(retval);
895 dsp5680xx_context.flush = 0;
896 int counter = FLUSH_COUNT_READ_WRITE;
898 for (unsigned i=0; i<count; i++){
899 if(--counter==0){
900 dsp5680xx_context.flush = 1;
901 counter = FLUSH_COUNT_READ_WRITE;
903 switch (size){
904 case 1:
905 if(!(i%2)){
906 retval = dsp5680xx_read_16_single(target, address + i/2, buffer + i, pmem);
908 break;
909 case 2:
910 retval = dsp5680xx_read_16_single(target, address + i, buffer+2*i, pmem);
911 break;
912 case 4:
913 retval = dsp5680xx_read_32_single(target, address + 2*i, buffer + 4*i, pmem);
914 break;
915 default:
916 LOG_USER("%s: Invalid read size.",__FUNCTION__);
917 break;
919 err_check_propagate(retval);
920 dsp5680xx_context.flush = 0;
923 dsp5680xx_context.flush = 1;
924 retval = dsp5680xx_execute_queue();
925 err_check_propagate(retval);
927 return retval;
930 static int dsp5680xx_write_16_single(struct target *target, uint32_t address, uint16_t data, uint8_t w_pmem){
931 int retval = 0;
932 retval = core_move_long_to_r0(target,address);
933 err_check_propagate(retval);
934 if(w_pmem){
935 retval = core_move_value_to_y0(target,data);
936 err_check_propagate(retval);
937 retval = core_move_y0_at_pr0_inc(target);
938 err_check_propagate(retval);
939 }else{
940 retval = core_move_value_at_r0(target,data);
941 err_check_propagate(retval);
943 return retval;
946 static int dsp5680xx_write_32_single(struct target *target, uint32_t address, uint32_t data, int w_pmem){
947 int retval = 0;
948 retval = core_move_long_to_r0(target,address);
949 err_check_propagate(retval);
950 retval = core_move_long_to_y(target,data);
951 err_check_propagate(retval);
952 if(w_pmem)
953 retval = core_move_y0_at_pr0_inc(target);
954 else
955 retval = core_move_y0_at_r0_inc(target);
956 err_check_propagate(retval);
957 if(w_pmem)
958 retval = core_move_y1_at_pr0_inc(target);
959 else
960 retval = core_move_y1_at_r0_inc(target);
961 err_check_propagate(retval);
962 return retval;
965 static int dsp5680xx_write_8(struct target * target, uint32_t address, uint32_t count, const uint8_t * data, int pmem){
966 if(target->state != TARGET_HALTED){
967 LOG_ERROR("%s: Target must be halted.",__FUNCTION__);
968 return ERROR_OK;
970 int retval = 0;
971 uint16_t data_16;
972 uint32_t iter;
974 int counter = FLUSH_COUNT_READ_WRITE;
975 for(iter = 0; iter<count/2; iter++){
976 if(--counter==0){
977 dsp5680xx_context.flush = 1;
978 counter = FLUSH_COUNT_READ_WRITE;
980 data_16=(data[2*iter]|(data[2*iter+1]<<8));
981 retval = dsp5680xx_write_16_single(target,address+iter,data_16, pmem);
982 if(retval != ERROR_OK){
983 LOG_ERROR("%s: Could not write to p:0x%04X",__FUNCTION__,address);
984 dsp5680xx_context.flush = 1;
985 return retval;
987 dsp5680xx_context.flush = 0;
989 dsp5680xx_context.flush = 1;
991 // Only one byte left, let's not overwrite the other byte (mem is 16bit)
992 // Need to retrieve the part we do not want to overwrite.
993 uint16_t data_old;
994 if((count==1)||(count%2)){
995 retval = dsp5680xx_read(target,address+iter,1,1,(uint8_t *)&data_old);
996 err_check_propagate(retval);
997 if(count==1)
998 data_old=(((data_old&0xff)<<8)|data[0]);// preserve upper byte
999 else
1000 data_old=(((data_old&0xff)<<8)|data[2*iter+1]);
1001 retval = dsp5680xx_write_16_single(target,address+iter,data_old, pmem);
1002 err_check_propagate(retval);
1004 return retval;
1007 static int dsp5680xx_write_16(struct target * target, uint32_t address, uint32_t count, const uint8_t * data, int pmem){
1008 int retval = ERROR_OK;
1009 if(target->state != TARGET_HALTED){
1010 retval = ERROR_TARGET_NOT_HALTED;
1011 err_check(retval,"Target must be halted.");
1013 uint32_t iter;
1014 int counter = FLUSH_COUNT_READ_WRITE;
1016 for(iter = 0; iter<count; iter++){
1017 if(--counter==0){
1018 dsp5680xx_context.flush = 1;
1019 counter = FLUSH_COUNT_READ_WRITE;
1021 retval = dsp5680xx_write_16_single(target,address+iter,data[iter], pmem);
1022 if(retval != ERROR_OK){
1023 LOG_ERROR("%s: Could not write to p:0x%04X",__FUNCTION__,address);
1024 dsp5680xx_context.flush = 1;
1025 return retval;
1027 dsp5680xx_context.flush = 0;
1029 dsp5680xx_context.flush = 1;
1030 return retval;
1033 static int dsp5680xx_write_32(struct target * target, uint32_t address, uint32_t count, const uint8_t * data, int pmem){
1034 int retval = ERROR_OK;
1035 if(target->state != TARGET_HALTED){
1036 retval = ERROR_TARGET_NOT_HALTED;
1037 err_check(retval,"Target must be halted.");
1039 uint32_t iter;
1040 int counter = FLUSH_COUNT_READ_WRITE;
1042 for(iter = 0; iter<count; iter++){
1043 if(--counter==0){
1044 dsp5680xx_context.flush = 1;
1045 counter = FLUSH_COUNT_READ_WRITE;
1047 retval = dsp5680xx_write_32_single(target,address+(iter<<1),data[iter], pmem);
1048 if(retval != ERROR_OK){
1049 LOG_ERROR("%s: Could not write to p:0x%04X",__FUNCTION__,address);
1050 dsp5680xx_context.flush = 1;
1051 return retval;
1053 dsp5680xx_context.flush = 0;
1055 dsp5680xx_context.flush = 1;
1056 return retval;
1060 * Writes @buffer to memory.
1061 * The parameter @address determines whether @buffer should be written to P: (program) memory or X: (data) memory.
1063 * @param target
1064 * @param address
1065 * @param size Bytes (1), Half words (2), Words (4).
1066 * @param count In bytes.
1067 * @param buffer
1069 * @return
1071 static int dsp5680xx_write(struct target *target, uint32_t address, uint32_t size, uint32_t count, const uint8_t * buffer){
1072 //TODO Cannot write 32bit to odd address, will write 0x12345678 as 0x5678 0x0012
1073 if(target->state != TARGET_HALTED){
1074 LOG_USER("Target must be halted.");
1075 return ERROR_OK;
1077 int retval = 0;
1078 int p_mem = 1;
1079 retval = dsp5680xx_convert_address(&address, &p_mem);
1080 err_check_propagate(retval);
1082 switch (size){
1083 case 1:
1084 retval = dsp5680xx_write_8(target, address, count, buffer, p_mem);
1085 break;
1086 case 2:
1087 retval = dsp5680xx_write_16(target, address, count, buffer, p_mem);
1088 break;
1089 case 4:
1090 retval = dsp5680xx_write_32(target, address, count, buffer, p_mem);
1091 break;
1092 default:
1093 retval = ERROR_TARGET_DATA_ABORT;
1094 err_check(retval,"Invalid data size.");
1095 break;
1097 return retval;
1100 static int dsp5680xx_bulk_write_memory(struct target * target,uint32_t address, uint32_t aligned, const uint8_t * buffer){
1101 LOG_ERROR("Not implemented yet.");
1102 return ERROR_FAIL;
1105 static int dsp5680xx_write_buffer(struct target * target, uint32_t address, uint32_t size, const uint8_t * buffer){
1106 if(target->state != TARGET_HALTED){
1107 LOG_USER("Target must be halted.");
1108 return ERROR_OK;
1110 return dsp5680xx_write(target, address, 1, size, buffer);
1114 * This function is called by verify_image, it is used to read data from memory.
1116 * @param target
1117 * @param address Word addressing.
1118 * @param size In bytes.
1119 * @param buffer
1121 * @return
1123 static int dsp5680xx_read_buffer(struct target * target, uint32_t address, uint32_t size, uint8_t * buffer){
1124 if(target->state != TARGET_HALTED){
1125 LOG_USER("Target must be halted.");
1126 return ERROR_OK;
1128 // The "/2" solves the byte/word addressing issue.
1129 return dsp5680xx_read(target,address,2,size/2,buffer);
1133 * This function is not implemented.
1134 * It returns an error in order to get OpenOCD to do read out the data and calculate the CRC, or try a binary comparison.
1136 * @param target
1137 * @param address Start address of the image.
1138 * @param size In bytes.
1139 * @param checksum
1141 * @return
1143 static int dsp5680xx_checksum_memory(struct target * target, uint32_t address, uint32_t size, uint32_t * checksum){
1144 return ERROR_FAIL;
1148 * Calculates a signature over @word_count words in the data from @buff16. The algorithm used is the same the FM uses, so the @return may be used to compare with the one generated by the FM module, and check if flashing was successful.
1149 * This algorithm is based on the perl script available from the Freescale website at FAQ 25630.
1151 * @param buff16
1152 * @param word_count
1154 * @return
1156 static int perl_crc(uint8_t * buff8,uint32_t word_count){
1157 uint16_t checksum = 0xffff;
1158 uint16_t data,fbmisr;
1159 uint32_t i;
1160 for(i=0;i<word_count;i++){
1161 data = (buff8[2*i]|(buff8[2*i+1]<<8));
1162 fbmisr = (checksum & 2)>>1 ^ (checksum & 4)>>2 ^ (checksum & 16)>>4 ^ (checksum & 0x8000)>>15;
1163 checksum = (data ^ ((checksum << 1) | fbmisr));
1165 i--;
1166 for(;!(i&0x80000000);i--){
1167 data = (buff8[2*i]|(buff8[2*i+1]<<8));
1168 fbmisr = (checksum & 2)>>1 ^ (checksum & 4)>>2 ^ (checksum & 16)>>4 ^ (checksum & 0x8000)>>15;
1169 checksum = (data ^ ((checksum << 1) | fbmisr));
1171 return checksum;
1175 * Resets the SIM. (System Integration Module).
1177 * @param target
1179 * @return
1181 int dsp5680xx_f_SIM_reset(struct target * target){
1182 int retval = ERROR_OK;
1183 uint16_t sim_cmd = SIM_CMD_RESET;
1184 uint32_t sim_addr;
1185 if(strcmp(target->tap->chip,"dsp568013")==0){
1186 sim_addr = MC568013_SIM_BASE_ADDR+S_FILE_DATA_OFFSET;
1187 retval = dsp5680xx_write(target,sim_addr,1,2,(const uint8_t *)&sim_cmd);
1188 err_check_propagate(retval);
1190 return retval;
1194 * Halts the core and resets the SIM. (System Integration Module).
1196 * @param target
1198 * @return
1200 static int dsp5680xx_soft_reset_halt(struct target *target){
1201 //TODO is this what this function is expected to do...?
1202 int retval;
1203 retval = dsp5680xx_halt(target);
1204 err_check_propagate(retval);
1205 retval = dsp5680xx_f_SIM_reset(target);
1206 err_check_propagate(retval);
1207 return retval;
1210 int dsp5680xx_f_protect_check(struct target * target, uint16_t * protected) {
1211 int retval;
1212 if (dsp5680xx_target_status(target,NULL,NULL) != TARGET_HALTED){
1213 retval = dsp5680xx_halt(target);
1214 err_check_propagate(retval);
1216 if(protected == NULL){
1217 err_check(ERROR_FAIL,"NULL pointer not valid.");
1219 retval = dsp5680xx_read_16_single(target,HFM_BASE_ADDR|HFM_PROT,(uint8_t *)protected,0);
1220 err_check_propagate(retval);
1221 return retval;
1225 * Executes a command on the FM module. Some commands use the parameters @address and @data, others ignore them.
1227 * @param target
1228 * @param command Command to execute.
1229 * @param address Command parameter.
1230 * @param data Command parameter.
1231 * @param hfm_ustat FM status register.
1232 * @param pmem Address is P: (program) memory (@pmem==1) or X: (data) memory (@pmem==0)
1234 * @return
1236 static int dsp5680xx_f_execute_command(struct target * target, uint16_t command, uint32_t address, uint32_t data, uint16_t * hfm_ustat, int pmem){
1237 int retval;
1238 retval = core_load_TX_RX_high_addr_to_r0(target);
1239 err_check_propagate(retval);
1240 retval = core_move_long_to_r2(target,HFM_BASE_ADDR);
1241 err_check_propagate(retval);
1242 uint8_t i[2];
1243 int watchdog = 100;
1245 retval = core_move_at_r2_disp_to_y0(target,HFM_USTAT); // read HMF_USTAT
1246 err_check_propagate(retval);
1247 retval = core_move_y0_at_r0(target);
1248 err_check_propagate(retval);
1249 retval = core_rx_upper_data(target,i);
1250 err_check_propagate(retval);
1251 if((watchdog--)==1){
1252 retval = ERROR_TARGET_FAILURE;
1253 err_check(retval,"FM execute command failed.");
1255 }while (!(i[0]&0x40)); // wait until current command is complete
1257 dsp5680xx_context.flush = 0;
1259 retval = core_move_value_at_r2_disp(target,0x00,HFM_CNFG); // write to HFM_CNFG (lock=0, select bank) -- flash_desc.bank&0x03,0x01 == 0x00,0x01 ???
1260 err_check_propagate(retval);
1261 retval = core_move_value_at_r2_disp(target,0x04,HFM_USTAT); // write to HMF_USTAT, clear PVIOL, ACCERR & BLANK bits
1262 err_check_propagate(retval);
1263 retval = core_move_value_at_r2_disp(target,0x10,HFM_USTAT); // clear only one bit at a time
1264 err_check_propagate(retval);
1265 retval = core_move_value_at_r2_disp(target,0x20,HFM_USTAT);
1266 err_check_propagate(retval);
1267 retval = core_move_value_at_r2_disp(target,0x00,HFM_PROT); // write to HMF_PROT, clear protection
1268 err_check_propagate(retval);
1269 retval = core_move_value_at_r2_disp(target,0x00,HFM_PROTB); // write to HMF_PROTB, clear protection
1270 err_check_propagate(retval);
1271 retval = core_move_value_to_y0(target,data);
1272 err_check_propagate(retval);
1273 retval = core_move_long_to_r3(target,address); // write to the flash block
1274 err_check_propagate(retval);
1275 if (pmem){
1276 retval = core_move_y0_at_pr3_inc(target);
1277 err_check_propagate(retval);
1278 }else{
1279 retval = core_move_y0_at_r3(target);
1280 err_check_propagate(retval);
1282 retval = core_move_value_at_r2_disp(target,command,HFM_CMD); // write command to the HFM_CMD reg
1283 err_check_propagate(retval);
1284 retval = core_move_value_at_r2_disp(target,0x80,HFM_USTAT); // start the command
1285 err_check_propagate(retval);
1287 dsp5680xx_context.flush = 1;
1288 retval = dsp5680xx_execute_queue();
1289 err_check_propagate(retval);
1291 watchdog = 100;
1293 retval = core_move_at_r2_disp_to_y0(target,HFM_USTAT); // read HMF_USTAT
1294 err_check_propagate(retval);
1295 retval = core_move_y0_at_r0(target);
1296 err_check_propagate(retval);
1297 retval = core_rx_upper_data(target,i);
1298 err_check_propagate(retval);
1299 if((watchdog--)==1){
1300 retval = ERROR_TARGET_FAILURE;
1301 err_check(retval,"FM execution did not finish.");
1303 }while (!(i[0]&0x40)); // wait until the command is complete
1304 *hfm_ustat = ((i[0]<<8)|(i[1]));
1305 if (i[0]&HFM_USTAT_MASK_PVIOL_ACCER){
1306 retval = ERROR_TARGET_FAILURE;
1307 err_check(retval,"pviol and/or accer bits set. HFM command execution error");
1309 return ERROR_OK;
1313 * Prior to the execution of any Flash module command, the Flash module Clock Divider (CLKDIV) register must be initialized. The values of this register determine the speed of the internal Flash Clock (FCLK). FCLK must be in the range of 150kHz ≤ FCLK ≤ 200kHz for proper operation of the Flash module. (Running FCLK too slowly wears out the module, while running it too fast under programs Flash leading to bit errors.)
1315 * @param target
1317 * @return
1319 static int set_fm_ck_div(struct target * target){
1320 uint8_t i[2];
1321 int retval;
1322 retval = core_move_long_to_r2(target,HFM_BASE_ADDR);
1323 err_check_propagate(retval);
1324 retval = core_load_TX_RX_high_addr_to_r0(target);
1325 err_check_propagate(retval);
1326 retval = core_move_at_r2_to_y0(target);// read HFM_CLKD
1327 err_check_propagate(retval);
1328 retval = core_move_y0_at_r0(target);
1329 err_check_propagate(retval);
1330 retval = core_rx_upper_data(target,i);
1331 err_check_propagate(retval);
1332 unsigned int hfm_at_wrong_value = 0;
1333 if ((i[0]&0x7f)!=HFM_CLK_DEFAULT) {
1334 LOG_DEBUG("HFM CLK divisor contained incorrect value (0x%02X).",i[0]&0x7f);
1335 hfm_at_wrong_value = 1;
1336 }else{
1337 LOG_DEBUG("HFM CLK divisor was already set to correct value (0x%02X).",i[0]&0x7f);
1338 return ERROR_OK;
1340 retval = core_move_value_at_r2(target,HFM_CLK_DEFAULT); // write HFM_CLKD
1341 err_check_propagate(retval);
1342 retval = core_move_at_r2_to_y0(target); // verify HFM_CLKD
1343 err_check_propagate(retval);
1344 retval = core_move_y0_at_r0(target);
1345 err_check_propagate(retval);
1346 retval = core_rx_upper_data(target,i);
1347 err_check_propagate(retval);
1348 if (i[0]!=(0x80|(HFM_CLK_DEFAULT&0x7f))) {
1349 retval = ERROR_TARGET_FAILURE;
1350 err_check(retval,"Unable to set HFM CLK divisor.");
1352 if(hfm_at_wrong_value)
1353 LOG_DEBUG("HFM CLK divisor set to 0x%02x.",i[0]&0x7f);
1354 return ERROR_OK;
1358 * Executes the FM calculate signature command. The FM will calculate over the data from @address to @address + @words -1. The result is written to a register, then read out by this function and returned in @signature. The value @signature may be compared to the the one returned by perl_crc to verify the flash was written correctly.
1360 * @param target
1361 * @param address Start of flash array where the signature should be calculated.
1362 * @param words Number of words over which the signature should be calculated.
1363 * @param signature Value calculated by the FM.
1365 * @return
1367 static int dsp5680xx_f_signature(struct target * target, uint32_t address, uint32_t words, uint16_t * signature){
1368 int retval;
1369 uint16_t hfm_ustat;
1370 if (dsp5680xx_target_status(target,NULL,NULL) != TARGET_HALTED){
1371 retval = eonce_enter_debug_mode_without_reset(target,NULL);
1372 err_check_propagate(retval);
1374 retval = dsp5680xx_f_execute_command(target,HFM_CALCULATE_DATA_SIGNATURE,address,words,&hfm_ustat,1);
1375 err_check_propagate(retval);
1376 retval = dsp5680xx_read_16_single(target, HFM_BASE_ADDR|HFM_DATA, (uint8_t *)signature, 0);
1377 return retval;
1380 int dsp5680xx_f_erase_check(struct target * target, uint8_t * erased,uint32_t sector){
1381 int retval;
1382 uint16_t hfm_ustat;
1383 if (dsp5680xx_target_status(target,NULL,NULL) != TARGET_HALTED){
1384 retval = dsp5680xx_halt(target);
1385 err_check_propagate(retval);
1387 retval = set_fm_ck_div(target);
1388 err_check_propagate(retval);
1389 // Check if chip is already erased.
1390 retval = dsp5680xx_f_execute_command(target,HFM_ERASE_VERIFY,HFM_FLASH_BASE_ADDR+sector*HFM_SECTOR_SIZE/2,0,&hfm_ustat,1); // blank check
1391 err_check_propagate(retval);
1392 if(erased!=NULL)
1393 *erased = (uint8_t)(hfm_ustat&HFM_USTAT_MASK_BLANK);
1394 return retval;
1398 * Executes the FM page erase command.
1400 * @param target
1401 * @param sector Page to erase.
1402 * @param hfm_ustat FM module status register.
1404 * @return
1406 static int erase_sector(struct target * target, int sector, uint16_t * hfm_ustat){
1407 int retval;
1408 retval = dsp5680xx_f_execute_command(target,HFM_PAGE_ERASE,HFM_FLASH_BASE_ADDR+sector*HFM_SECTOR_SIZE/2,0,hfm_ustat,1);
1409 err_check_propagate(retval);
1410 return retval;
1414 * Executes the FM mass erase command. Erases the flash array completely.
1416 * @param target
1417 * @param hfm_ustat FM module status register.
1419 * @return
1421 static int mass_erase(struct target * target, uint16_t * hfm_ustat){
1422 int retval;
1423 retval = dsp5680xx_f_execute_command(target,HFM_MASS_ERASE,0,0,hfm_ustat,1);
1424 return retval;
1427 int dsp5680xx_f_erase(struct target * target, int first, int last){
1428 int retval;
1429 if (dsp5680xx_target_status(target,NULL,NULL) != TARGET_HALTED){
1430 retval = dsp5680xx_halt(target);
1431 err_check_propagate(retval);
1433 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1434 // Reset SIM
1435 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1436 retval = dsp5680xx_f_SIM_reset(target);
1437 err_check_propagate(retval);
1438 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1439 // Set hfmdiv
1440 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1441 retval = set_fm_ck_div(target);
1442 err_check_propagate(retval);
1444 uint16_t hfm_ustat;
1445 int do_mass_erase = ((!(first|last)) || ((first==0)&&(last == (HFM_SECTOR_COUNT-1))));
1446 if(do_mass_erase){
1447 //Mass erase
1448 retval = mass_erase(target,&hfm_ustat);
1449 err_check_propagate(retval);
1450 last = HFM_SECTOR_COUNT-1;
1451 }else{
1452 for(int i = first;i<=last;i++){
1453 retval = erase_sector(target,i,&hfm_ustat);
1454 err_check_propagate(retval);
1457 return ERROR_OK;
1461 * Algorithm for programming normal p: flash
1462 * Follow state machine from "56F801x Peripheral Reference Manual"@163.
1463 * Registers to set up before calling:
1464 * r0: TX/RX high address.
1465 * r2: FM module base address.
1466 * r3: Destination address in flash.
1468 * hfm_wait: // wait for command to finish
1469 * brclr #0x40,x:(r2+0x13),hfm_wait
1470 * rx_check: // wait for input buffer full
1471 * brclr #0x01,x:(r0-2),rx_check
1472 * move.w x:(r0),y0 // read from Rx buffer
1473 * move.w y0,p:(r3)+
1474 * move.w #0x20,x:(r2+0x14) // write PGM command
1475 * move.w #0x80,x:(r2+0x13) // start the command
1476 * brclr #0x20,X:(R2+0x13),accerr_check // protection violation check
1477 * bfset #0x20,X:(R2+0x13) // clear pviol
1478 * bra hfm_wait
1479 * accerr_check:
1480 * brclr #0x10,X:(R2+0x13),hfm_wait // access error check
1481 * bfset #0x10,X:(R2+0x13) // clear accerr
1482 * bra hfm_wait // loop
1483 *0x00000073 0x8A460013407D brclr #0x40,X:(R2+0x13),*+0
1484 *0x00000076 0xE700 nop
1485 *0x00000077 0xE700 nop
1486 *0x00000078 0x8A44FFFE017B brclr #1,X:(R0-2),*-2
1487 *0x0000007B 0xE700 nop
1488 *0x0000007C 0xF514 move.w X:(R0),Y0
1489 *0x0000007D 0x8563 move.w Y0,P:(R3)+
1490 *0x0000007E 0x864600200014 move.w #0x20,X:(R2+0x14)
1491 *0x00000081 0x864600800013 move.w #0x80,X:(R2+0x13)
1492 *0x00000084 0x8A4600132004 brclr #0x20,X:(R2+0x13),*+7
1493 *0x00000087 0x824600130020 bfset #0x20,X:(R2+0x13)
1494 *0x0000008A 0xA968 bra *-23
1495 *0x0000008B 0x8A4600131065 brclr #0x10,X:(R2+0x13),*-24
1496 *0x0000008E 0x824600130010 bfset #0x10,X:(R2+0x13)
1497 *0x00000091 0xA961 bra *-30
1499 const uint16_t pgm_write_pflash[] = {0x8A46,0x0013,0x407D,0xE700,0xE700,0x8A44,0xFFFE,0x017B,0xE700,0xF514,0x8563,0x8646,0x0020,0x0014,0x8646,0x0080,0x0013,0x8A46,0x0013,0x2004,0x8246,0x0013,0x0020,0xA968,0x8A46,0x0013,0x1065,0x8246,0x0013,0x0010,0xA961};
1500 const uint32_t pgm_write_pflash_length = 31;
1502 int dsp5680xx_f_wr(struct target * target, uint8_t *buffer, uint32_t address, uint32_t count, int is_flash_lock){
1503 int retval = ERROR_OK;
1504 if (dsp5680xx_target_status(target,NULL,NULL) != TARGET_HALTED){
1505 retval = eonce_enter_debug_mode(target,NULL);
1506 err_check_propagate(retval);
1508 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1509 // Download the pgm that flashes.
1510 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1511 uint32_t my_favourite_ram_address = 0x8700; // This seems to be a safe address. This one is the one used by codewarrior in 56801x_flash.cfg
1512 if(!is_flash_lock){
1513 retval = dsp5680xx_write(target, my_favourite_ram_address, 1, pgm_write_pflash_length*2,(uint8_t *) pgm_write_pflash);
1514 err_check_propagate(retval);
1515 retval = dsp5680xx_execute_queue();
1516 err_check_propagate(retval);
1518 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1519 // Set hfmdiv
1520 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1521 retval = set_fm_ck_div(target);
1522 err_check_propagate(retval);
1523 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1524 // Setup registers needed by pgm_write_pflash
1525 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1527 dsp5680xx_context.flush = 0;
1529 retval = core_move_long_to_r3(target,address); // Destination address to r3
1530 err_check_propagate(retval);
1531 core_load_TX_RX_high_addr_to_r0(target); // TX/RX reg address to r0
1532 err_check_propagate(retval);
1533 retval = core_move_long_to_r2(target,HFM_BASE_ADDR);// FM base address to r2
1534 err_check_propagate(retval);
1535 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1536 // Run flashing program.
1537 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1538 retval = core_move_value_at_r2_disp(target,0x00,HFM_CNFG); // write to HFM_CNFG (lock=0, select bank)
1539 err_check_propagate(retval);
1540 retval = core_move_value_at_r2_disp(target,0x04,HFM_USTAT);// write to HMF_USTAT, clear PVIOL, ACCERR & BLANK bits
1541 err_check_propagate(retval);
1542 retval = core_move_value_at_r2_disp(target,0x10,HFM_USTAT);// clear only one bit at a time
1543 err_check_propagate(retval);
1544 retval = core_move_value_at_r2_disp(target,0x20,HFM_USTAT);
1545 err_check_propagate(retval);
1546 retval = core_move_value_at_r2_disp(target,0x00,HFM_PROT);// write to HMF_PROT, clear protection
1547 err_check_propagate(retval);
1548 retval = core_move_value_at_r2_disp(target,0x00,HFM_PROTB);// write to HMF_PROTB, clear protection
1549 err_check_propagate(retval);
1550 if(count%2){
1551 //TODO implement handling of odd number of words.
1552 retval = ERROR_FAIL;
1553 err_check(retval,"Cannot handle odd number of words.");
1556 dsp5680xx_context.flush = 1;
1557 retval = dsp5680xx_execute_queue();
1558 err_check_propagate(retval);
1560 uint32_t drscan_data;
1561 uint16_t tmp = (buffer[0]|(buffer[1]<<8));
1562 retval = core_tx_upper_data(target,tmp,&drscan_data);
1563 err_check_propagate(retval);
1565 retval = dsp5680xx_resume(target,0,my_favourite_ram_address,0,0);
1566 err_check_propagate(retval);
1568 int counter = FLUSH_COUNT_FLASH;
1569 dsp5680xx_context.flush = 0;
1570 uint32_t i;
1571 for(i=1; (i<count/2)&&(i<HFM_SIZE_WORDS); i++){
1572 if(--counter==0){
1573 dsp5680xx_context.flush = 1;
1574 counter = FLUSH_COUNT_FLASH;
1576 tmp = (buffer[2*i]|(buffer[2*i+1]<<8));
1577 retval = core_tx_upper_data(target,tmp,&drscan_data);
1578 if(retval!=ERROR_OK){
1579 dsp5680xx_context.flush = 1;
1580 err_check_propagate(retval);
1582 dsp5680xx_context.flush = 0;
1584 dsp5680xx_context.flush = 1;
1585 if(!is_flash_lock){
1586 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1587 // Verify flash (skip when exec lock sequence)
1588 // -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
1589 uint16_t signature;
1590 uint16_t pc_crc;
1591 retval = dsp5680xx_f_signature(target,address,i,&signature);
1592 err_check_propagate(retval);
1593 pc_crc = perl_crc(buffer,i);
1594 if(pc_crc != signature){
1595 retval = ERROR_FAIL;
1596 err_check(retval,"Flashed data failed CRC check, flash again!");
1599 return retval;
1602 // Reset state machine
1603 static int reset_jtag(void){
1604 int retval;
1605 tap_state_t states[2];
1606 const char *cp = "RESET";
1607 states[0] = tap_state_by_name(cp);
1608 retval = jtag_add_statemove(states[0]);
1609 err_check_propagate(retval);
1610 retval = jtag_execute_queue();
1611 err_check_propagate(retval);
1612 jtag_add_pathmove(0, states + 1);
1613 retval = jtag_execute_queue();
1614 return retval;
1617 int dsp5680xx_f_unlock(struct target * target){
1618 int retval = ERROR_OK;
1619 uint16_t eonce_status;
1620 uint32_t instr;
1621 uint32_t ir_out;
1622 uint16_t instr_16;
1623 uint16_t read_16;
1624 struct jtag_tap * tap_chp;
1625 struct jtag_tap * tap_cpu;
1626 tap_chp = jtag_tap_by_string("dsp568013.chp");
1627 if(tap_chp == NULL){
1628 retval = ERROR_FAIL;
1629 err_check(retval,"Failed to get master tap.");
1631 tap_cpu = jtag_tap_by_string("dsp568013.cpu");
1632 if(tap_cpu == NULL){
1633 retval = ERROR_FAIL;
1634 err_check(retval,"Failed to get master tap.");
1637 retval = eonce_enter_debug_mode(target,&eonce_status);
1638 if(retval == ERROR_OK){
1639 LOG_WARNING("Memory was not locked.");
1642 jtag_add_reset(0,1);
1643 jtag_add_sleep(TIME_DIV_FREESCALE*200*1000);
1645 retval = reset_jtag();
1646 err_check(retval,"Failed to reset JTAG state machine");
1647 jtag_add_sleep(150);
1649 // Enable core tap
1650 tap_chp->enabled = true;
1651 retval = switch_tap(target,tap_chp,tap_cpu);
1652 err_check_propagate(retval);
1654 instr = JTAG_INSTR_DEBUG_REQUEST;
1655 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
1656 err_check_propagate(retval);
1657 jtag_add_sleep(TIME_DIV_FREESCALE*100*1000);
1658 jtag_add_reset(0,0);
1659 jtag_add_sleep(TIME_DIV_FREESCALE*300*1000);
1661 // Enable master tap
1662 tap_chp->enabled = false;
1663 retval = switch_tap(target,tap_chp,tap_cpu);
1664 err_check_propagate(retval);
1666 // Execute mass erase to unlock
1667 instr = MASTER_TAP_CMD_FLASH_ERASE;
1668 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_MASTER_TAP_IRLEN);
1669 err_check_propagate(retval);
1671 instr = HFM_CLK_DEFAULT;
1672 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & ir_out,16);
1673 err_check_propagate(retval);
1675 jtag_add_sleep(TIME_DIV_FREESCALE*150*1000);
1676 jtag_add_reset(0,1);
1677 jtag_add_sleep(TIME_DIV_FREESCALE*200*1000);
1679 retval = reset_jtag();
1680 err_check(retval,"Failed to reset JTAG state machine");
1681 jtag_add_sleep(150);
1683 instr = 0x0606ffff;
1684 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & ir_out,32);
1685 err_check_propagate(retval);
1687 // enable core tap
1688 instr = 0x5;
1689 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_MASTER_TAP_IRLEN);
1690 err_check_propagate(retval);
1691 instr = 0x2;
1692 retval = dsp5680xx_drscan(target,(uint8_t *) & instr,(uint8_t *) & ir_out,4);
1693 err_check_propagate(retval);
1695 tap_cpu->enabled = true;
1696 tap_chp->enabled = false;
1698 instr = JTAG_INSTR_ENABLE_ONCE;
1699 //Two rounds of jtag 0x6 (enable eonce) to enable EOnCE.
1700 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
1701 err_check_propagate(retval);
1702 instr = JTAG_INSTR_DEBUG_REQUEST;
1703 retval = dsp5680xx_irscan(target, & instr, & ir_out,DSP5680XX_JTAG_CORE_TAP_IRLEN);
1704 err_check_propagate(retval);
1705 instr_16 = 0x1;
1706 retval = dsp5680xx_drscan(target,(uint8_t *) & instr_16,(uint8_t *) & read_16,8);
1707 instr_16 = 0x20;
1708 retval = dsp5680xx_drscan(target,(uint8_t *) & instr_16,(uint8_t *) & read_16,8);
1709 jtag_add_sleep(TIME_DIV_FREESCALE*100*1000);
1710 jtag_add_reset(0,0);
1711 jtag_add_sleep(TIME_DIV_FREESCALE*300*1000);
1712 return retval;
1715 int dsp5680xx_f_lock(struct target * target){
1716 int retval;
1717 uint16_t lock_word[] = {HFM_LOCK_FLASH};
1718 retval = dsp5680xx_f_wr(target,(uint8_t *)(lock_word),HFM_LOCK_ADDR_L,2,1);
1719 err_check_propagate(retval);
1721 jtag_add_reset(0,1);
1722 jtag_add_sleep(TIME_DIV_FREESCALE*200*1000);
1724 retval = reset_jtag();
1725 err_check(retval,"Failed to reset JTAG state machine");
1726 jtag_add_sleep(TIME_DIV_FREESCALE*100*1000);
1727 jtag_add_reset(0,0);
1728 jtag_add_sleep(TIME_DIV_FREESCALE*300*1000);
1730 return retval;
1733 static int dsp5680xx_step(struct target * target,int current, uint32_t address, int handle_breakpoints){
1734 err_check(ERROR_FAIL,"Not implemented yet.");
1737 /** Holds methods for dsp5680xx targets. */
1738 struct target_type dsp5680xx_target = {
1739 .name = "dsp5680xx",
1741 .poll = dsp5680xx_poll,
1742 .arch_state = dsp5680xx_arch_state,
1744 .target_request_data = NULL,
1746 .halt = dsp5680xx_halt,
1747 .resume = dsp5680xx_resume,
1748 .step = dsp5680xx_step,
1750 .write_buffer = dsp5680xx_write_buffer,
1751 .read_buffer = dsp5680xx_read_buffer,
1753 .assert_reset = dsp5680xx_assert_reset,
1754 .deassert_reset = dsp5680xx_deassert_reset,
1755 .soft_reset_halt = dsp5680xx_soft_reset_halt,
1757 .read_memory = dsp5680xx_read,
1758 .write_memory = dsp5680xx_write,
1759 .bulk_write_memory = dsp5680xx_bulk_write_memory,
1761 .checksum_memory = dsp5680xx_checksum_memory,
1763 .target_create = dsp5680xx_target_create,
1764 .init_target = dsp5680xx_init_target,