- Improves error handling upon GDB connect
[openocd.git] / src / target / armv7m.c
1 /***************************************************************************
2 * Copyright (C) 2005 by Dominic Rath *
3 * Dominic.Rath@gmx.de *
4 * *
5 * Copyright (C) 2006 by Magnus Lundin *
6 * lundin@mlu.mine.nu *
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
26
27 #include "replacements.h"
28
29 #include "armv7m.h"
30 #include "register.h"
31 #include "target.h"
32 #include "log.h"
33 #include "jtag.h"
34 #include "arm_jtag.h"
35
36 #include <stdlib.h>
37 #include <string.h>
38
39 #if 0
40 #define _DEBUG_INSTRUCTION_EXECUTION_
41 #endif
42
43 char* armv7m_mode_strings[] =
44 {
45 "Handler", "Thread"
46 };
47
48 char* armv7m_state_strings[] =
49 {
50 "Thumb", "Debug"
51 };
52
53 char* armv7m_exception_strings[] =
54 {
55 "", "Reset", "NMI", "HardFault", "MemManage", "BusFault", "UsageFault", "RESERVED", "RESERVED", "RESERVED", "RESERVED",
56 "SVCall", "DebugMonitor", "RESERVED", "PendSV", "SysTick"
57 };
58
59 char* armv7m_core_reg_list[] =
60 {
61 /* Registers accessed through core debug */
62 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11", "r12",
63 "sp", "lr", "pc",
64 "xPSR", "msp", "psp",
65 /* Registers accessed through MSR instructions */
66 /* "apsr", "iapsr", "ipsr", "epsr", */
67 "primask", "basepri", "faultmask", "control"
68 };
69
70 char* armv7m_core_dbgreg_list[] =
71 {
72 /* Registers accessed through core debug */
73 "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7", "r8", "r9", "r10", "r11", "r12",
74 "sp", "lr", "pc",
75 "xPSR", "msp", "psp",
76 /* Registers accessed through MSR instructions */
77 /* "dbg_apsr", "iapsr", "ipsr", "epsr", */
78 "primask", "basepri", "faultmask", "dbg_control"
79 };
80
81 u8 armv7m_gdb_dummy_fp_value[] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
82
83 reg_t armv7m_gdb_dummy_fp_reg =
84 {
85 "GDB dummy floating-point register", armv7m_gdb_dummy_fp_value, 0, 1, 96, NULL, 0, NULL, 0
86 };
87
88 armv7m_core_reg_t armv7m_core_reg_list_arch_info[] =
89 {
90 /* CORE_GP are accesible using the core debug registers */
91 {0, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
92 {1, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
93 {2, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
94 {3, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
95 {4, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
96 {5, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
97 {6, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
98 {7, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
99 {8, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
100 {9, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
101 {10, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
102 {11, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
103 {12, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
104 {13, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
105 {14, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
106 {15, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL},
107
108 {16, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL}, /* xPSR */
109 {17, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL}, /* MSP */
110 {18, ARMV7M_REGISTER_CORE_GP, ARMV7M_MODE_ANY, NULL, NULL}, /* PSP */
111
112 /* CORE_SP are accesible using MSR and MRS instructions */
113 #if 0
114 // {0x00, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* APSR */
115 // {0x01, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* IAPSR */
116 // {0x05, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* IPSR */
117 // {0x06, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* EPSR */
118 #endif
119
120 {0x10, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* PRIMASK */
121 {0x11, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* BASEPRI */
122 {0x13, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL}, /* FAULTMASK */
123 {0x14, ARMV7M_REGISTER_CORE_SP, ARMV7M_MODE_ANY, NULL, NULL} /* CONTROL */
124 };
125
126 int armv7m_core_reg_arch_type = -1;
127
128 /* Keep different contexts for the process being debugged and debug algorithms */
129 enum armv7m_runcontext armv7m_get_context(target_t *target)
130 {
131 /* get pointers to arch-specific information */
132 armv7m_common_t *armv7m = target->arch_info;
133
134 if (armv7m->process_context == armv7m->core_cache)
135 return ARMV7M_PROCESS_CONTEXT;
136 if (armv7m->debug_context == armv7m->core_cache)
137 return ARMV7M_DEBUG_CONTEXT;
138
139 ERROR("Invalid runcontext");
140 exit(-1);
141 }
142
143 int armv7m_use_context(target_t *target, enum armv7m_runcontext new_ctx)
144 {
145 int i;
146 /* get pointers to arch-specific information */
147 armv7m_common_t *armv7m = target->arch_info;
148
149 if ((target->state != TARGET_HALTED) && (target->state != TARGET_RESET))
150 {
151 WARNING("target not halted, switch context ");
152 return ERROR_TARGET_NOT_HALTED;
153 }
154
155 if (new_ctx == armv7m_get_context(target))
156 return ERROR_OK;
157
158 switch (new_ctx)
159 {
160 case ARMV7M_PROCESS_CONTEXT:
161 armv7m->core_cache = armv7m->process_context;
162 break;
163 case ARMV7M_DEBUG_CONTEXT:
164 armv7m->core_cache = armv7m->debug_context;
165 break;
166 default:
167 ERROR("Invalid runcontext");
168 exit(-1);
169 }
170 /* Mark registers in new context as dirty to force reload when run */
171
172 for (i = 0; i < armv7m->core_cache->num_regs-1; i++) /* EXCLUDE CONTROL TODOLATER : CHECK THIS */
173 {
174 armv7m->core_cache->reg_list[i].dirty = 1;
175 }
176
177 return ERROR_OK;
178 }
179
180 int armv7m_restore_context(target_t *target)
181 {
182 int i;
183
184 /* get pointers to arch-specific information */
185 armv7m_common_t *armv7m = target->arch_info;
186
187 DEBUG(" ");
188
189 if (armv7m->pre_restore_context)
190 armv7m->pre_restore_context(target);
191
192 for (i = ARMV7NUMCOREREGS-1; i >= 0; i--)
193 {
194 if (armv7m->core_cache->reg_list[i].dirty)
195 {
196 armv7m->write_core_reg(target, i);
197 }
198 }
199
200 if (armv7m->post_restore_context)
201 armv7m->post_restore_context(target);
202
203 return ERROR_OK;
204 }
205
206
207 /* Core state functions */
208 char enamebuf[32];
209 char *armv7m_exception_string(int number)
210 {
211 if ((number < 0) | (number > 511))
212 return "Invalid exception";
213 if (number < 16)
214 return armv7m_exception_strings[number];
215 sprintf(enamebuf, "External Interrupt(%i)", number - 16);
216 return enamebuf;
217 }
218
219 int armv7m_get_core_reg(reg_t *reg)
220 {
221 int retval;
222 armv7m_core_reg_t *armv7m_reg = reg->arch_info;
223 target_t *target = armv7m_reg->target;
224 armv7m_common_t *armv7m_target = target->arch_info;
225
226 if (target->state != TARGET_HALTED)
227 {
228 return ERROR_TARGET_NOT_HALTED;
229 }
230
231 retval = armv7m_target->read_core_reg(target, armv7m_reg->num);
232
233 return retval;
234 }
235
236 int armv7m_set_core_reg(reg_t *reg, u8 *buf)
237 {
238 armv7m_core_reg_t *armv7m_reg = reg->arch_info;
239 target_t *target = armv7m_reg->target;
240 u32 value = buf_get_u32(buf, 0, 32);
241
242 if (target->state != TARGET_HALTED)
243 {
244 return ERROR_TARGET_NOT_HALTED;
245 }
246
247 buf_set_u32(reg->value, 0, 32, value);
248 reg->dirty = 1;
249 reg->valid = 1;
250
251 return ERROR_OK;
252 }
253
254 int armv7m_read_core_reg(struct target_s *target, int num)
255 {
256 u32 reg_value;
257 int retval;
258 armv7m_core_reg_t * armv7m_core_reg;
259
260 /* get pointers to arch-specific information */
261 armv7m_common_t *armv7m = target->arch_info;
262
263 if ((num < 0) || (num >= ARMV7NUMCOREREGS))
264 return ERROR_INVALID_ARGUMENTS;
265
266 armv7m_core_reg = armv7m->core_cache->reg_list[num].arch_info;
267 retval = armv7m->load_core_reg_u32(target, armv7m_core_reg->type, armv7m_core_reg->num, &reg_value);
268 buf_set_u32(armv7m->core_cache->reg_list[num].value, 0, 32, reg_value);
269 armv7m->core_cache->reg_list[num].valid = 1;
270 armv7m->core_cache->reg_list[num].dirty = 0;
271
272 return ERROR_OK;
273 }
274
275 int armv7m_write_core_reg(struct target_s *target, int num)
276 {
277 int retval;
278 u32 reg_value;
279 armv7m_core_reg_t *armv7m_core_reg;
280
281 /* get pointers to arch-specific information */
282 armv7m_common_t *armv7m = target->arch_info;
283
284 if ((num < 0) || (num >= ARMV7NUMCOREREGS))
285 return ERROR_INVALID_ARGUMENTS;
286
287 reg_value = buf_get_u32(armv7m->core_cache->reg_list[num].value, 0, 32);
288 armv7m_core_reg = armv7m->core_cache->reg_list[num].arch_info;
289 retval = armv7m->store_core_reg_u32(target, armv7m_core_reg->type, armv7m_core_reg->num, reg_value);
290 if (retval != ERROR_OK)
291 {
292 ERROR("JTAG failure");
293 armv7m->core_cache->reg_list[num].dirty = armv7m->core_cache->reg_list[num].valid;
294 return ERROR_JTAG_DEVICE_ERROR;
295 }
296 DEBUG("write core reg %i value 0x%x", num , reg_value);
297 armv7m->core_cache->reg_list[num].valid = 1;
298 armv7m->core_cache->reg_list[num].dirty = 0;
299
300 return ERROR_OK;
301 }
302
303 int armv7m_invalidate_core_regs(target_t *target)
304 {
305 /* get pointers to arch-specific information */
306 armv7m_common_t *armv7m = target->arch_info;
307 int i;
308
309 for (i = 0; i < armv7m->core_cache->num_regs; i++)
310 {
311 armv7m->core_cache->reg_list[i].valid = 0;
312 armv7m->core_cache->reg_list[i].dirty = 0;
313 }
314
315 return ERROR_OK;
316 }
317
318 int armv7m_get_gdb_reg_list(target_t *target, reg_t **reg_list[], int *reg_list_size)
319 {
320 /* get pointers to arch-specific information */
321 armv7m_common_t *armv7m = target->arch_info;
322 int i;
323
324 if (target->state != TARGET_HALTED)
325 {
326 ERROR("Target not halted");
327 return ERROR_TARGET_NOT_HALTED;
328 }
329
330 *reg_list_size = 26;
331 *reg_list = malloc(sizeof(reg_t*) * (*reg_list_size));
332
333 /* TODOLATER correct list of registers, names ? */
334 for (i = 0; i < *reg_list_size; i++)
335 {
336 if (i < ARMV7NUMCOREREGS)
337 (*reg_list)[i] = &armv7m->process_context->reg_list[i];
338 /* (*reg_list)[i] = &armv7m->core_cache->reg_list[i]; */
339 else
340 (*reg_list)[i] = &armv7m_gdb_dummy_fp_reg;
341 }
342 /* ARMV7M is always in thumb mode, try to make GDB understand this if it does not support this arch */
343 armv7m->process_context->reg_list[15].value[0] |= 1;
344 (*reg_list)[25] = &armv7m->process_context->reg_list[ARMV7M_xPSR];
345 return ERROR_OK;
346 }
347
348 int armv7m_run_algorithm(struct target_s *target, int num_mem_params, mem_param_t *mem_params, int num_reg_params, reg_param_t *reg_params, u32 entry_point, u32 exit_point, int timeout_ms, void *arch_info)
349 {
350 /* get pointers to arch-specific information */
351 armv7m_common_t *armv7m = target->arch_info;
352 armv7m_algorithm_t *armv7m_algorithm_info = arch_info;
353 enum armv7m_state core_state = armv7m->core_state;
354 enum armv7m_mode core_mode = armv7m->core_mode;
355 int retval = ERROR_OK;
356 u32 pc;
357 int exit_breakpoint_size = 0;
358 int i;
359
360 armv7m->core_state = core_state;
361 armv7m->core_mode = core_mode;
362
363 if (armv7m_algorithm_info->common_magic != ARMV7M_COMMON_MAGIC)
364 {
365 ERROR("current target isn't an ARMV7M target");
366 return ERROR_TARGET_INVALID;
367 }
368
369 if (target->state != TARGET_HALTED)
370 {
371 WARNING("target not halted");
372 return ERROR_TARGET_NOT_HALTED;
373 }
374
375 /* refresh core register cache */
376 /* Not needed if core register cache is always consistent with target process state */
377 armv7m_use_context(target, ARMV7M_DEBUG_CONTEXT);
378
379 for (i = 0; i < num_mem_params; i++)
380 {
381 target_write_buffer(target, mem_params[i].address, mem_params[i].size, mem_params[i].value);
382 }
383
384 for (i = 0; i < num_reg_params; i++)
385 {
386 reg_t *reg = register_get_by_name(armv7m->core_cache, reg_params[i].reg_name, 0);
387 u32 regvalue;
388
389 if (!reg)
390 {
391 ERROR("BUG: register '%s' not found", reg_params[i].reg_name);
392 exit(-1);
393 }
394
395 if (reg->size != reg_params[i].size)
396 {
397 ERROR("BUG: register '%s' size doesn't match reg_params[i].size", reg_params[i].reg_name);
398 exit(-1);
399 }
400
401 regvalue = buf_get_u32(reg_params[i].value, 0, 32);
402 armv7m_set_core_reg(reg, reg_params[i].value);
403 }
404
405 /* ARMV7M always runs in Thumb state */
406 exit_breakpoint_size = 2;
407 if ((retval = breakpoint_add(target, exit_point, exit_breakpoint_size, BKPT_SOFT)) != ERROR_OK)
408 {
409 ERROR("can't add breakpoint to finish algorithm execution");
410 return ERROR_TARGET_FAILURE;
411 }
412
413 /* This code relies on the target specific resume() and poll()->debug_entry()
414 sequence to write register values to the processor and the read them back */
415 target->type->resume(target, 0, entry_point, 1, 1);
416 target->type->poll(target);
417
418 while (target->state != TARGET_HALTED)
419 {
420 usleep(5000);
421 target->type->poll(target);
422 if ((timeout_ms -= 5) <= 0)
423 {
424 ERROR("timeout waiting for algorithm to complete, trying to halt target");
425 target->type->halt(target);
426 timeout_ms = 1000;
427 while (target->state != TARGET_HALTED)
428 {
429 usleep(10000);
430 target->type->poll(target);
431 if ((timeout_ms -= 10) <= 0)
432 {
433 ERROR("target didn't reenter debug state, exiting");
434 exit(-1);
435 }
436 }
437 armv7m->load_core_reg_u32(target, ARMV7M_REGISTER_CORE_GP, 15, &pc);
438 DEBUG("failed algoritm halted at 0x%x ", pc);
439 retval = ERROR_TARGET_TIMEOUT;
440 }
441 }
442
443 breakpoint_remove(target, exit_point);
444
445 /* Read memory values to mem_params[] */
446 for (i = 0; i < num_mem_params; i++)
447 {
448 if (mem_params[i].direction != PARAM_OUT)
449 target_read_buffer(target, mem_params[i].address, mem_params[i].size, mem_params[i].value);
450 }
451
452 /* Copy core register values to reg_params[] */
453 for (i = 0; i < num_reg_params; i++)
454 {
455 if (reg_params[i].direction != PARAM_OUT)
456 {
457 reg_t *reg = register_get_by_name(armv7m->debug_context, reg_params[i].reg_name, 0);
458
459 if (!reg)
460 {
461 ERROR("BUG: register '%s' not found", reg_params[i].reg_name);
462 exit(-1);
463 }
464
465 if (reg->size != reg_params[i].size)
466 {
467 ERROR("BUG: register '%s' size doesn't match reg_params[i].size", reg_params[i].reg_name);
468 exit(-1);
469 }
470
471 buf_set_u32(reg_params[i].value, 0, 32, buf_get_u32(reg->value, 0, 32));
472 }
473 }
474
475 return retval;
476 }
477
478 int armv7m_arch_state(struct target_s *target)
479 {
480 /* get pointers to arch-specific information */
481 armv7m_common_t *armv7m = target->arch_info;
482
483 USER("target halted in %s state due to %s, current mode: %s %s\nxPSR: 0x%8.8x pc: 0x%8.8x",
484 armv7m_state_strings[armv7m->core_state],
485 target_debug_reason_strings[target->debug_reason],
486 armv7m_mode_strings[armv7m->core_mode],
487 armv7m_exception_string(armv7m->exception_number),
488 buf_get_u32(armv7m->core_cache->reg_list[ARMV7M_xPSR].value, 0, 32),
489 buf_get_u32(armv7m->core_cache->reg_list[15].value, 0, 32));
490
491 return ERROR_OK;
492 }
493
494 reg_cache_t *armv7m_build_reg_cache(target_t *target)
495 {
496 /* get pointers to arch-specific information */
497 armv7m_common_t *armv7m = target->arch_info;
498
499 int num_regs = ARMV7NUMCOREREGS;
500 reg_cache_t **cache_p = register_get_last_cache_p(&target->reg_cache);
501 reg_cache_t *cache = malloc(sizeof(reg_cache_t));
502 reg_t *reg_list = malloc(sizeof(reg_t) * num_regs);
503 armv7m_core_reg_t *arch_info = malloc(sizeof(armv7m_core_reg_t) * num_regs);
504 int i;
505
506 if (armv7m_core_reg_arch_type == -1)
507 armv7m_core_reg_arch_type = register_reg_arch_type(armv7m_get_core_reg, armv7m_set_core_reg);
508
509 /* Build the process context cache */
510 cache->name = "arm v7m registers";
511 cache->next = NULL;
512 cache->reg_list = reg_list;
513 cache->num_regs = num_regs;
514 (*cache_p) = cache;
515 armv7m->core_cache = cache;
516 armv7m->process_context = cache;
517
518 for (i = 0; i < num_regs; i++)
519 {
520 arch_info[i] = armv7m_core_reg_list_arch_info[i];
521 arch_info[i].target = target;
522 arch_info[i].armv7m_common = armv7m;
523 reg_list[i].name = armv7m_core_reg_list[i];
524 reg_list[i].size = 32;
525 reg_list[i].value = calloc(1, 4);
526 reg_list[i].dirty = 0;
527 reg_list[i].valid = 0;
528 reg_list[i].bitfield_desc = NULL;
529 reg_list[i].num_bitfields = 0;
530 reg_list[i].arch_type = armv7m_core_reg_arch_type;
531 reg_list[i].arch_info = &arch_info[i];
532 }
533
534 /* Build the debug context cache*/
535 cache = malloc(sizeof(reg_cache_t));
536 reg_list = malloc(sizeof(reg_t) * num_regs);
537
538 cache->name = "arm v7m debug registers";
539 cache->next = NULL;
540 cache->reg_list = reg_list;
541 cache->num_regs = num_regs;
542 armv7m->debug_context = cache;
543 armv7m->process_context->next = cache;
544
545 for (i = 0; i < num_regs; i++)
546 {
547 reg_list[i].name = armv7m_core_dbgreg_list[i];
548 reg_list[i].size = 32;
549 reg_list[i].value = calloc(1, 4);
550 reg_list[i].dirty = 0;
551 reg_list[i].valid = 0;
552 reg_list[i].bitfield_desc = NULL;
553 reg_list[i].num_bitfields = 0;
554 reg_list[i].arch_type = armv7m_core_reg_arch_type;
555 reg_list[i].arch_info = &arch_info[i];
556 }
557
558 return cache;
559 }
560
561 int armv7m_init_target(struct command_context_s *cmd_ctx, struct target_s *target)
562 {
563 armv7m_build_reg_cache(target);
564
565 return ERROR_OK;
566 }
567
568 int armv7m_init_arch_info(target_t *target, armv7m_common_t *armv7m)
569 {
570 /* register arch-specific functions */
571
572 target->arch_info = armv7m;
573 armv7m->core_state = ARMV7M_STATE_THUMB;
574 armv7m->read_core_reg = armv7m_read_core_reg;
575 armv7m->write_core_reg = armv7m_write_core_reg;
576
577 return ERROR_OK;
578 }
579
580 int armv7m_register_commands(struct command_context_s *cmd_ctx)
581 {
582 return ERROR_OK;
583 }
584
585 int armv7m_checksum_memory(struct target_s *target, u32 address, u32 count, u32* checksum)
586 {
587 working_area_t *crc_algorithm;
588 armv7m_algorithm_t armv7m_info;
589 reg_param_t reg_params[2];
590 int retval;
591
592 u16 cortex_m3_crc_code[] = {
593 0x4602, /* mov r2, r0 */
594 0xF04F, 0x30FF, /* mov r0, #0xffffffff */
595 0x460B, /* mov r3, r1 */
596 0xF04F, 0x0400, /* mov r4, #0 */
597 0xE013, /* b ncomp */
598 /* nbyte: */
599 0x5D11, /* ldrb r1, [r2, r4] */
600 0xF8DF, 0x7028, /* ldr r7, CRC32XOR */
601 0xEA80, 0x6001, /* eor r0, r0, r1, asl #24 */
602
603 0xF04F, 0x0500, /* mov r5, #0 */
604 /* loop: */
605 0x2800, /* cmp r0, #0 */
606 0xEA4F, 0x0640, /* mov r6, r0, asl #1 */
607 0xF105, 0x0501, /* add r5, r5, #1 */
608 0x4630, /* mov r0, r6 */
609 0xBFB8, /* it lt */
610 0xEA86, 0x0007, /* eor r0, r6, r7 */
611 0x2D08, /* cmp r5, #8 */
612 0xD1F4, /* bne loop */
613
614 0xF104, 0x0401, /* add r4, r4, #1 */
615 /* ncomp: */
616 0x429C, /* cmp r4, r3 */
617 0xD1E9, /* bne nbyte */
618 /* end: */
619 0xE7FE, /* b end */
620 0x1DB7, 0x04C1 /* CRC32XOR: .word 0x04C11DB7 */
621 };
622
623 int i;
624
625 if (target_alloc_working_area(target, sizeof(cortex_m3_crc_code), &crc_algorithm) != ERROR_OK)
626 {
627 return ERROR_TARGET_RESOURCE_NOT_AVAILABLE;
628 }
629
630 /* convert flash writing code into a buffer in target endianness */
631 for (i = 0; i < (sizeof(cortex_m3_crc_code)/sizeof(u16)); i++)
632 target_write_u16(target, crc_algorithm->address + i*sizeof(u16), cortex_m3_crc_code[i]);
633
634 armv7m_info.common_magic = ARMV7M_COMMON_MAGIC;
635 armv7m_info.core_mode = ARMV7M_MODE_ANY;
636 armv7m_info.core_state = ARMV7M_STATE_THUMB;
637
638 init_reg_param(&reg_params[0], "r0", 32, PARAM_IN_OUT);
639 init_reg_param(&reg_params[1], "r1", 32, PARAM_OUT);
640
641 buf_set_u32(reg_params[0].value, 0, 32, address);
642 buf_set_u32(reg_params[1].value, 0, 32, count);
643
644 if ((retval = target->type->run_algorithm(target, 0, NULL, 2, reg_params,
645 crc_algorithm->address, crc_algorithm->address + (sizeof(cortex_m3_crc_code)-6), 20000, &armv7m_info)) != ERROR_OK)
646 {
647 ERROR("error executing cortex_m3 crc algorithm");
648 destroy_reg_param(&reg_params[0]);
649 destroy_reg_param(&reg_params[1]);
650 target_free_working_area(target, crc_algorithm);
651 return retval;
652 }
653
654 *checksum = buf_get_u32(reg_params[0].value, 0, 32);
655
656 destroy_reg_param(&reg_params[0]);
657 destroy_reg_param(&reg_params[1]);
658
659 target_free_working_area(target, crc_algorithm);
660
661 return ERROR_OK;
662 }
663

Linking to existing account procedure

If you already have an account and want to add another login method you MUST first sign in with your existing account and then change URL to read https://review.openocd.org/login/?link to get to this page again but this time it'll work for linking. Thank you.

SSH host keys fingerprints

1024 SHA256:YKx8b7u5ZWdcbp7/4AeXNaqElP49m6QrwfXaqQGJAOk gerrit-code-review@openocd.zylin.com (DSA)
384 SHA256:jHIbSQa4REvwCFG4cq5LBlBLxmxSqelQPem/EXIrxjk gerrit-code-review@openocd.org (ECDSA)
521 SHA256:UAOPYkU9Fjtcao0Ul/Rrlnj/OsQvt+pgdYSZ4jOYdgs gerrit-code-review@openocd.org (ECDSA)
256 SHA256:A13M5QlnozFOvTllybRZH6vm7iSt0XLxbA48yfc2yfY gerrit-code-review@openocd.org (ECDSA)
256 SHA256:spYMBqEYoAOtK7yZBrcwE8ZpYt6b68Cfh9yEVetvbXg gerrit-code-review@openocd.org (ED25519)
+--[ED25519 256]--+
|=..              |
|+o..   .         |
|*.o   . .        |
|+B . . .         |
|Bo. = o S        |
|Oo.+ + =         |
|oB=.* = . o      |
| =+=.+   + E     |
|. .=o   . o      |
+----[SHA256]-----+
2048 SHA256:0Onrb7/PHjpo6iVZ7xQX2riKN83FJ3KGU0TvI0TaFG4 gerrit-code-review@openocd.zylin.com (RSA)