1 /***************************************************************************
2 * Copyright (C) 2005 by Dominic Rath *
3 * Dominic.Rath@gmx.de *
5 * Copyright (C) 2008 by Spencer Oliver *
6 * spen@spen-soft.co.uk *
8 * Copyright (C) 2010 Øyvind Harboe *
9 * oyvind.harboe@zylin.com *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
16 * This program is distributed in the hope that it will be useful, *
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
19 * GNU General Public License for more details. *
21 * You should have received a copy of the GNU General Public License *
22 * along with this program; if not, write to the *
23 * Free Software Foundation, Inc., *
24 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
25 ***************************************************************************/
31 #include <target/arm.h>
32 #include <helper/binarybuffer.h>
33 #include <target/algorithm.h>
38 #define FLASH_CR0 0x00000000
39 #define FLASH_CR1 0x00000004
40 #define FLASH_DR0 0x00000008
41 #define FLASH_DR1 0x0000000C
42 #define FLASH_AR 0x00000010
43 #define FLASH_ER 0x00000014
44 #define FLASH_NVWPAR 0x0000DFB0
45 #define FLASH_NVAPR0 0x0000DFB8
46 #define FLASH_NVAPR1 0x0000DFBC
48 /* FLASH_CR0 register bits */
50 #define FLASH_WMS 0x80000000
51 #define FLASH_SUSP 0x40000000
52 #define FLASH_WPG 0x20000000
53 #define FLASH_DWPG 0x10000000
54 #define FLASH_SER 0x08000000
55 #define FLASH_SPR 0x01000000
56 #define FLASH_BER 0x04000000
57 #define FLASH_MER 0x02000000
58 #define FLASH_LOCK 0x00000010
59 #define FLASH_BSYA1 0x00000004
60 #define FLASH_BSYA0 0x00000002
62 /* FLASH_CR1 register bits */
64 #define FLASH_B1S 0x02000000
65 #define FLASH_B0S 0x01000000
66 #define FLASH_B1F1 0x00020000
67 #define FLASH_B1F0 0x00010000
68 #define FLASH_B0F7 0x00000080
69 #define FLASH_B0F6 0x00000040
70 #define FLASH_B0F5 0x00000020
71 #define FLASH_B0F4 0x00000010
72 #define FLASH_B0F3 0x00000008
73 #define FLASH_B0F2 0x00000004
74 #define FLASH_B0F1 0x00000002
75 #define FLASH_B0F0 0x00000001
77 /* FLASH_ER register bits */
79 #define FLASH_WPF 0x00000100
80 #define FLASH_RESER 0x00000080
81 #define FLASH_SEQER 0x00000040
82 #define FLASH_10ER 0x00000008
83 #define FLASH_PGER 0x00000004
84 #define FLASH_ERER 0x00000002
85 #define FLASH_ERR 0x00000001
88 struct str7x_flash_bank
90 uint32_t *sector_bits
;
93 uint32_t register_base
;
94 struct working_area
*write_algorithm
;
97 struct str7x_mem_layout
{
98 uint32_t sector_start
;
103 enum str7x_status_codes
105 STR7X_CMD_SUCCESS
= 0,
106 STR7X_INVALID_COMMAND
= 1,
107 STR7X_SRC_ADDR_ERROR
= 2,
108 STR7X_DST_ADDR_ERROR
= 3,
109 STR7X_SRC_ADDR_NOT_MAPPED
= 4,
110 STR7X_DST_ADDR_NOT_MAPPED
= 5,
111 STR7X_COUNT_ERROR
= 6,
112 STR7X_INVALID_SECTOR
= 7,
113 STR7X_SECTOR_NOT_BLANK
= 8,
114 STR7X_SECTOR_NOT_PREPARED
= 9,
115 STR7X_COMPARE_ERROR
= 10,
119 static struct str7x_mem_layout mem_layout_str7bank0
[] = {
120 {0x00000000, 0x02000, 0x01},
121 {0x00002000, 0x02000, 0x02},
122 {0x00004000, 0x02000, 0x04},
123 {0x00006000, 0x02000, 0x08},
124 {0x00008000, 0x08000, 0x10},
125 {0x00010000, 0x10000, 0x20},
126 {0x00020000, 0x10000, 0x40},
127 {0x00030000, 0x10000, 0x80}
130 static struct str7x_mem_layout mem_layout_str7bank1
[] = {
131 {0x00000000, 0x02000, 0x10000},
132 {0x00002000, 0x02000, 0x20000}
135 static int str7x_get_flash_adr(struct flash_bank
*bank
, uint32_t reg
)
137 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
138 return (str7x_info
->register_base
| reg
);
141 static int str7x_build_block_list(struct flash_bank
*bank
)
143 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
147 int b0_sectors
= 0, b1_sectors
= 0;
164 LOG_ERROR("BUG: unknown bank->size encountered");
168 num_sectors
= b0_sectors
+ b1_sectors
;
170 bank
->num_sectors
= num_sectors
;
171 bank
->sectors
= malloc(sizeof(struct flash_sector
) * num_sectors
);
172 str7x_info
->sector_bits
= malloc(sizeof(uint32_t) * num_sectors
);
176 for (i
= 0; i
< b0_sectors
; i
++)
178 bank
->sectors
[num_sectors
].offset
= mem_layout_str7bank0
[i
].sector_start
;
179 bank
->sectors
[num_sectors
].size
= mem_layout_str7bank0
[i
].sector_size
;
180 bank
->sectors
[num_sectors
].is_erased
= -1;
181 /* the reset_init handler marks all the sectors unprotected,
182 * matching hardware after reset; keep the driver in sync
184 bank
->sectors
[num_sectors
].is_protected
= 0;
185 str7x_info
->sector_bits
[num_sectors
++] = mem_layout_str7bank0
[i
].sector_bit
;
188 for (i
= 0; i
< b1_sectors
; i
++)
190 bank
->sectors
[num_sectors
].offset
= mem_layout_str7bank1
[i
].sector_start
;
191 bank
->sectors
[num_sectors
].size
= mem_layout_str7bank1
[i
].sector_size
;
192 bank
->sectors
[num_sectors
].is_erased
= -1;
193 /* the reset_init handler marks all the sectors unprotected,
194 * matching hardware after reset; keep the driver in sync
196 bank
->sectors
[num_sectors
].is_protected
= 0;
197 str7x_info
->sector_bits
[num_sectors
++] = mem_layout_str7bank1
[i
].sector_bit
;
203 /* flash bank str7x <base> <size> 0 0 <target#> <str71_variant>
205 FLASH_BANK_COMMAND_HANDLER(str7x_flash_bank_command
)
207 struct str7x_flash_bank
*str7x_info
;
211 LOG_WARNING("incomplete flash_bank str7x configuration");
212 return ERROR_FLASH_BANK_INVALID
;
215 str7x_info
= malloc(sizeof(struct str7x_flash_bank
));
216 bank
->driver_priv
= str7x_info
;
218 /* set default bits for str71x flash */
219 str7x_info
->busy_bits
= (FLASH_LOCK
| FLASH_BSYA1
| FLASH_BSYA0
);
220 str7x_info
->disable_bit
= (1 << 1);
222 if (strcmp(CMD_ARGV
[6], "STR71x") == 0)
224 str7x_info
->register_base
= 0x40100000;
226 else if (strcmp(CMD_ARGV
[6], "STR73x") == 0)
228 str7x_info
->register_base
= 0x80100000;
229 str7x_info
->busy_bits
= (FLASH_LOCK
| FLASH_BSYA0
);
231 else if (strcmp(CMD_ARGV
[6], "STR75x") == 0)
233 str7x_info
->register_base
= 0x20100000;
234 str7x_info
->disable_bit
= (1 << 0);
238 LOG_ERROR("unknown STR7x variant: '%s'", CMD_ARGV
[6]);
240 return ERROR_FLASH_BANK_INVALID
;
243 str7x_build_block_list(bank
);
245 str7x_info
->write_algorithm
= NULL
;
250 /* wait for flash to become idle or report errors.
252 FIX!!! what's the maximum timeout??? The documentation doesn't
253 state any maximum time.... by inspection it seems > 1000ms is to be
256 10000ms is long enough that it should cover anything, yet not
257 quite be equivalent to an infinite loop.
260 static int str7x_waitbusy(struct flash_bank
*bank
)
264 struct target
*target
= bank
->target
;
265 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
267 for (i
= 0 ; i
< 10000; i
++)
270 err
= target_read_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), &retval
);
274 if ((retval
& str7x_info
->busy_bits
) == 0)
279 LOG_ERROR("Timed out waiting for str7x flash");
284 static int str7x_result(struct flash_bank
*bank
)
286 struct target
*target
= bank
->target
;
290 err
= target_read_u32(target
, str7x_get_flash_adr(bank
, FLASH_ER
), &retval
);
294 if (retval
& FLASH_WPF
)
296 LOG_ERROR("str7x hw write protection set");
299 if (retval
& FLASH_RESER
)
301 LOG_ERROR("str7x suspended program erase not resumed");
304 if (retval
& FLASH_10ER
)
306 LOG_ERROR("str7x trying to set bit to 1 when it is already 0");
309 if (retval
& FLASH_PGER
)
311 LOG_ERROR("str7x program error");
314 if (retval
& FLASH_ERER
)
316 LOG_ERROR("str7x erase error");
321 if (retval
& FLASH_ERR
)
323 /* this should always be set if one of the others are set... */
324 LOG_ERROR("str7x write operation failed / bad setup");
330 LOG_ERROR("FLASH_ER register contents: 0x%" PRIx32
, retval
);
336 static int str7x_protect_check(struct flash_bank
*bank
)
338 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
339 struct target
*target
= bank
->target
;
344 if (bank
->target
->state
!= TARGET_HALTED
)
346 LOG_ERROR("Target not halted");
347 return ERROR_TARGET_NOT_HALTED
;
350 target_read_u32(target
, str7x_get_flash_adr(bank
, FLASH_NVWPAR
), &retval
);
352 for (i
= 0; i
< bank
->num_sectors
; i
++)
354 if (retval
& str7x_info
->sector_bits
[i
])
355 bank
->sectors
[i
].is_protected
= 0;
357 bank
->sectors
[i
].is_protected
= 1;
363 static int str7x_erase(struct flash_bank
*bank
, int first
, int last
)
365 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
366 struct target
*target
= bank
->target
;
370 uint32_t sectors
= 0;
373 if (bank
->target
->state
!= TARGET_HALTED
)
375 LOG_ERROR("Target not halted");
376 return ERROR_TARGET_NOT_HALTED
;
379 for (i
= first
; i
<= last
; i
++)
381 sectors
|= str7x_info
->sector_bits
[i
];
384 LOG_DEBUG("sectors: 0x%" PRIx32
"", sectors
);
386 /* clear FLASH_ER register */
387 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_ER
), 0x0);
392 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
397 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR1
), cmd
);
401 cmd
= FLASH_SER
| FLASH_WMS
;
402 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
406 err
= str7x_waitbusy(bank
);
410 err
= str7x_result(bank
);
414 for (i
= first
; i
<= last
; i
++)
415 bank
->sectors
[i
].is_erased
= 1;
420 static int str7x_protect(struct flash_bank
*bank
, int set
, int first
, int last
)
422 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
423 struct target
*target
= bank
->target
;
426 uint32_t protect_blocks
;
428 if (bank
->target
->state
!= TARGET_HALTED
)
430 LOG_ERROR("Target not halted");
431 return ERROR_TARGET_NOT_HALTED
;
434 protect_blocks
= 0xFFFFFFFF;
438 for (i
= first
; i
<= last
; i
++)
439 protect_blocks
&= ~(str7x_info
->sector_bits
[i
]);
442 /* clear FLASH_ER register */
444 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_ER
), 0x0);
449 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
453 cmd
= str7x_get_flash_adr(bank
, FLASH_NVWPAR
);
454 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_AR
), cmd
);
458 cmd
= protect_blocks
;
459 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_DR0
), cmd
);
463 cmd
= FLASH_SPR
| FLASH_WMS
;
464 err
= target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
468 err
= str7x_waitbusy(bank
);
472 err
= str7x_result(bank
);
479 static int str7x_write_block(struct flash_bank
*bank
, uint8_t *buffer
,
480 uint32_t offset
, uint32_t count
)
482 struct str7x_flash_bank
*str7x_info
= bank
->driver_priv
;
483 struct target
*target
= bank
->target
;
484 uint32_t buffer_size
= 32768;
485 struct working_area
*source
;
486 uint32_t address
= bank
->base
+ offset
;
487 struct reg_param reg_params
[6];
488 struct arm_algorithm armv4_5_info
;
489 int retval
= ERROR_OK
;
491 /* see contib/loaders/flash/str7x.s for src */
493 static const uint32_t str7x_flash_write_code
[] = {
495 0xe3a04201, /* mov r4, #0x10000000 */
496 0xe5824000, /* str r4, [r2, #0x0] */
497 0xe5821010, /* str r1, [r2, #0x10] */
498 0xe4904004, /* ldr r4, [r0], #4 */
499 0xe5824008, /* str r4, [r2, #0x8] */
500 0xe4904004, /* ldr r4, [r0], #4 */
501 0xe582400c, /* str r4, [r2, #0xc] */
502 0xe3a04209, /* mov r4, #0x90000000 */
503 0xe5824000, /* str r4, [r2, #0x0] */
505 0xe5924000, /* ldr r4, [r2, #0x0] */
506 0xe1140005, /* tst r4, r5 */
507 0x1afffffc, /* bne busy */
508 0xe5924014, /* ldr r4, [r2, #0x14] */
509 0xe31400ff, /* tst r4, #0xff */
510 0x03140c01, /* tsteq r4, #0x100 */
511 0x1a000002, /* bne exit */
512 0xe2811008, /* add r1, r1, #0x8 */
513 0xe2533001, /* subs r3, r3, #1 */
514 0x1affffec, /* bne write */
516 0xeafffffe, /* b exit */
519 /* flash write code */
520 if (target_alloc_working_area_try(target
, sizeof(str7x_flash_write_code
),
521 &str7x_info
->write_algorithm
) != ERROR_OK
)
523 return ERROR_TARGET_RESOURCE_NOT_AVAILABLE
;
526 target_write_buffer(target
, str7x_info
->write_algorithm
->address
,
527 sizeof(str7x_flash_write_code
),
528 (uint8_t*)str7x_flash_write_code
);
531 while (target_alloc_working_area_try(target
, buffer_size
, &source
) != ERROR_OK
)
534 if (buffer_size
<= 256)
536 /* if we already allocated the writing code, but failed to get a
537 * buffer, free the algorithm */
538 if (str7x_info
->write_algorithm
)
539 target_free_working_area(target
, str7x_info
->write_algorithm
);
541 LOG_WARNING("no large enough working area available, can't do block memory writes");
542 return ERROR_TARGET_RESOURCE_NOT_AVAILABLE
;
546 armv4_5_info
.common_magic
= ARM_COMMON_MAGIC
;
547 armv4_5_info
.core_mode
= ARM_MODE_SVC
;
548 armv4_5_info
.core_state
= ARM_STATE_ARM
;
550 init_reg_param(®_params
[0], "r0", 32, PARAM_OUT
);
551 init_reg_param(®_params
[1], "r1", 32, PARAM_OUT
);
552 init_reg_param(®_params
[2], "r2", 32, PARAM_OUT
);
553 init_reg_param(®_params
[3], "r3", 32, PARAM_OUT
);
554 init_reg_param(®_params
[4], "r4", 32, PARAM_IN
);
555 init_reg_param(®_params
[5], "r5", 32, PARAM_OUT
);
559 uint32_t thisrun_count
= (count
> (buffer_size
/ 8)) ? (buffer_size
/ 8) : count
;
561 target_write_buffer(target
, source
->address
, thisrun_count
* 8, buffer
);
563 buf_set_u32(reg_params
[0].value
, 0, 32, source
->address
);
564 buf_set_u32(reg_params
[1].value
, 0, 32, address
);
565 buf_set_u32(reg_params
[2].value
, 0, 32, str7x_get_flash_adr(bank
, FLASH_CR0
));
566 buf_set_u32(reg_params
[3].value
, 0, 32, thisrun_count
);
567 buf_set_u32(reg_params
[5].value
, 0, 32, str7x_info
->busy_bits
);
569 if ((retval
= target_run_algorithm(target
, 0, NULL
, 6, reg_params
,
570 str7x_info
->write_algorithm
->address
,
571 str7x_info
->write_algorithm
->address
+ (sizeof(str7x_flash_write_code
) - 4),
572 10000, &armv4_5_info
)) != ERROR_OK
)
577 if (buf_get_u32(reg_params
[4].value
, 0, 32) != 0x00)
579 retval
= str7x_result(bank
);
583 buffer
+= thisrun_count
* 8;
584 address
+= thisrun_count
* 8;
585 count
-= thisrun_count
;
588 target_free_working_area(target
, source
);
589 target_free_working_area(target
, str7x_info
->write_algorithm
);
591 destroy_reg_param(®_params
[0]);
592 destroy_reg_param(®_params
[1]);
593 destroy_reg_param(®_params
[2]);
594 destroy_reg_param(®_params
[3]);
595 destroy_reg_param(®_params
[4]);
596 destroy_reg_param(®_params
[5]);
601 static int str7x_write(struct flash_bank
*bank
, uint8_t *buffer
,
602 uint32_t offset
, uint32_t count
)
604 struct target
*target
= bank
->target
;
605 uint32_t dwords_remaining
= (count
/ 8);
606 uint32_t bytes_remaining
= (count
& 0x00000007);
607 uint32_t address
= bank
->base
+ offset
;
608 uint32_t bytes_written
= 0;
611 uint32_t check_address
= offset
;
614 if (bank
->target
->state
!= TARGET_HALTED
)
616 LOG_ERROR("Target not halted");
617 return ERROR_TARGET_NOT_HALTED
;
622 LOG_WARNING("offset 0x%" PRIx32
" breaks required 8-byte alignment", offset
);
623 return ERROR_FLASH_DST_BREAKS_ALIGNMENT
;
626 for (i
= 0; i
< bank
->num_sectors
; i
++)
628 uint32_t sec_start
= bank
->sectors
[i
].offset
;
629 uint32_t sec_end
= sec_start
+ bank
->sectors
[i
].size
;
631 /* check if destination falls within the current sector */
632 if ((check_address
>= sec_start
) && (check_address
< sec_end
))
634 /* check if destination ends in the current sector */
635 if (offset
+ count
< sec_end
)
636 check_address
= offset
+ count
;
638 check_address
= sec_end
;
642 if (check_address
!= offset
+ count
)
643 return ERROR_FLASH_DST_OUT_OF_BANK
;
645 /* clear FLASH_ER register */
646 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_ER
), 0x0);
648 /* multiple dwords (8-byte) to be programmed? */
649 if (dwords_remaining
> 0)
651 /* try using a block write */
652 if ((retval
= str7x_write_block(bank
, buffer
, offset
,
653 dwords_remaining
)) != ERROR_OK
)
655 if (retval
== ERROR_TARGET_RESOURCE_NOT_AVAILABLE
)
657 /* if block write failed (no sufficient working area),
658 * we use normal (slow) single dword accesses */
659 LOG_WARNING("couldn't use block writes, falling back to single memory accesses");
667 buffer
+= dwords_remaining
* 8;
668 address
+= dwords_remaining
* 8;
669 dwords_remaining
= 0;
673 while (dwords_remaining
> 0)
677 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
680 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_AR
), address
);
683 target_write_memory(target
, str7x_get_flash_adr(bank
, FLASH_DR0
),
684 4, 1, buffer
+ bytes_written
);
688 target_write_memory(target
, str7x_get_flash_adr(bank
, FLASH_DR1
),
689 4, 1, buffer
+ bytes_written
);
692 /* start programming cycle */
693 cmd
= FLASH_DWPG
| FLASH_WMS
;
694 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
697 err
= str7x_waitbusy(bank
);
701 err
= str7x_result(bank
);
711 uint8_t last_dword
[8] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
714 while (bytes_remaining
> 0)
716 last_dword
[i
++] = *(buffer
+ bytes_written
);
723 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
726 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_AR
), address
);
729 target_write_memory(target
, str7x_get_flash_adr(bank
, FLASH_DR0
),
734 target_write_memory(target
, str7x_get_flash_adr(bank
, FLASH_DR1
),
735 4, 1, last_dword
+ 4);
738 /* start programming cycle */
739 cmd
= FLASH_DWPG
| FLASH_WMS
;
740 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), cmd
);
743 err
= str7x_waitbusy(bank
);
747 err
= str7x_result(bank
);
755 static int str7x_probe(struct flash_bank
*bank
)
761 COMMAND_HANDLER(str7x_handle_part_id_command
)
767 static int get_str7x_info(struct flash_bank
*bank
, char *buf
, int buf_size
)
769 snprintf(buf
, buf_size
, "str7x flash driver info");
770 /* STR7x flash doesn't support sector protection interrogation.
771 * FLASH_NVWPAR acts as a write only register; its read value
772 * doesn't reflect the actual protection state of the sectors.
774 LOG_WARNING("STR7x flash lock information might not be correct "
775 "due to hardware limitations.");
779 COMMAND_HANDLER(str7x_handle_disable_jtag_command
)
781 struct target
*target
= NULL
;
782 struct str7x_flash_bank
*str7x_info
= NULL
;
785 uint16_t ProtectionLevel
= 0;
786 uint16_t ProtectionRegs
;
790 command_print(CMD_CTX
, "str7x disable_jtag <bank>");
794 struct flash_bank
*bank
;
795 int retval
= CALL_COMMAND_HANDLER(flash_command_get_bank
, 0, &bank
);
796 if (ERROR_OK
!= retval
)
799 str7x_info
= bank
->driver_priv
;
801 target
= bank
->target
;
803 if (target
->state
!= TARGET_HALTED
)
805 LOG_ERROR("Target not halted");
806 return ERROR_TARGET_NOT_HALTED
;
809 /* first we get protection status */
811 target_read_u32(target
, str7x_get_flash_adr(bank
, FLASH_NVAPR0
), ®
);
813 if (!(reg
& str7x_info
->disable_bit
))
818 target_read_u32(target
, str7x_get_flash_adr(bank
, FLASH_NVAPR1
), ®
);
819 ProtectionRegs
= ~(reg
>> 16);
821 while (((ProtectionRegs
) != 0) && (ProtectionLevel
< 16))
823 ProtectionRegs
>>= 1;
827 if (ProtectionLevel
== 0)
829 flash_cmd
= FLASH_SPR
;
830 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), flash_cmd
);
831 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_AR
), 0x4010DFB8);
832 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_DR0
), 0xFFFFFFFD);
833 flash_cmd
= FLASH_SPR
| FLASH_WMS
;
834 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), flash_cmd
);
838 flash_cmd
= FLASH_SPR
;
839 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), flash_cmd
);
840 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_AR
), 0x4010DFBC);
841 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_DR0
),
842 ~(1 << (15 + ProtectionLevel
)));
843 flash_cmd
= FLASH_SPR
| FLASH_WMS
;
844 target_write_u32(target
, str7x_get_flash_adr(bank
, FLASH_CR0
), flash_cmd
);
850 static const struct command_registration str7x_exec_command_handlers
[] = {
852 .name
= "disable_jtag",
853 .handler
= str7x_handle_disable_jtag_command
,
854 .mode
= COMMAND_EXEC
,
855 .help
= "disable jtag access",
857 COMMAND_REGISTRATION_DONE
860 static const struct command_registration str7x_command_handlers
[] = {
864 .help
= "str7x flash command group",
865 .chain
= str7x_exec_command_handlers
,
867 COMMAND_REGISTRATION_DONE
870 struct flash_driver str7x_flash
= {
872 .commands
= str7x_command_handlers
,
873 .flash_bank_command
= str7x_flash_bank_command
,
874 .erase
= str7x_erase
,
875 .protect
= str7x_protect
,
876 .write
= str7x_write
,
877 .read
= default_flash_read
,
878 .probe
= str7x_probe
,
879 .auto_probe
= str7x_probe
,
880 .erase_check
= default_flash_blank_check
,
881 .protect_check
= str7x_protect_check
,
882 .info
= get_str7x_info
,
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)