Hello flashrom developers,
currently flashrom programs GPIO pins it needs for a board enable
generally for output and chooses the GPIO function if that pin is
multiplexed (as long as selecting output and multiplexing is implemented
for that type of GPIO). Another policy would be to *require* that the
GPIO pin is *already* set to output and GPIO and abort otherwise. I
consider the second approach much safer than the first one, but I
suspect there are boards where flashrom finds the pins in the …
[View More]wrong
state and reprogramming them is really needed.
Still I think that disallowing GPIOs that are not already set to output
would be a good idea (and have gpio=force as programmer option) to
protect against accidental board matches disabling core functionality.
The concrete background of this post is the Asus P2B-N board. It has
write enable on GPO18 (lower to enable write). Currently
intel_piix4_gpo_set rejects this GPO pin, because it is a "dual function
pin which is most likely in use already". In fact, it is multiplexed
with PCI_STP#, an output signal to stop the PCI bus clock. It would be
fatal on a board that really stops the PCI clock on lowering the pin to
set this pin to GPO and lower it. Just lowering it without touching the
multiplex bit would be safe, because it won't do anything if the
PCI_STP# functionality of that pin is in use - but this would be
inconsistent with what flashrom is doing now.
Any other oppinions on this topic?
Regards,
Michael Karcher
[View Less]
I wanted to get some discussion about a progress bar going. I'm not
really happy with the current patch, and it will screw up in various
places if its design is not improved, but hey... it works. Sort of.
The biggest problem is deciding which operations should print a progress
bar.
Write? Sure.
Erase? Sure. But how do you handle a one-sector erase in the middle of
some write operation? You can't print two progress bars at the same time.
Read? Makes sense as well. But how do you handle the …
[View More]reads used for
verification of erase/write? The two-bars problem strikes again.
We could add a bar_type parameter to each progressbar call, and make
sure nested calls don't cause confusion. That would solve the issues above.
Another problem: How often do we call the progress bar update? Once per
byte? Per sector? Per arbitrary unit? Will the division and
multiplication performed on every progress bar update affect
performance, especially on architectures where such operations are
sloooooooooow?
This patch also starts progress bars at all the wrong places. Still, it
should give you an impression of what's needed.
Test with
flashrom -p dummy -c MX25L1005 -f -r foo.rom -V
flashrom -p dummy -c Am29LV081B -f -r foo.rom -V
or any read or erase action on real hardware.
Signed-off-by: Carl-Daniel Hailfinger <c-d.hailfinger.devel.2006(a)gmx.net>
Index: flashrom-progressbar/flash.h
===================================================================
--- flashrom-progressbar/flash.h (Revision 1072)
+++ flashrom-progressbar/flash.h (Arbeitskopie)
@@ -588,6 +588,10 @@
int check_erased_range(struct flashchip *flash, int start, int len);
int verify_range(struct flashchip *flash, uint8_t *cmpbuf, int start, int len, char *message);
int need_erase(uint8_t *have, uint8_t *want, int len, enum write_granularity gran);
+void progressbar_start(struct flashchip *flash);
+void progressbar_update(struct flashchip *flash, int bytepos);
+void progressbar_done(struct flashchip *flash);
+void progressbar_abort(struct flashchip *flash);
char *strcat_realloc(char *dest, const char *src);
void print_version(void);
void print_banner(void);
Index: flashrom-progressbar/spi25.c
===================================================================
--- flashrom-progressbar/spi25.c (Revision 1072)
+++ flashrom-progressbar/spi25.c (Arbeitskopie)
@@ -892,6 +892,7 @@
* (start + len - 1) / page_size. Since we want to include that last
* page as well, the loop condition uses <=.
*/
+ progressbar_start(flash);
for (i = start / page_size; i <= (start + len - 1) / page_size; i++) {
/* Byte position of the first byte in the range in this page. */
/* starthere is an offset to the base address of the chip. */
@@ -900,6 +901,7 @@
lenhere = min(start + len, (i + 1) * page_size) - starthere;
for (j = 0; j < lenhere; j += chunksize) {
toread = min(chunksize, lenhere - j);
+ progressbar_update(flash, starthere + j);
rc = spi_nbyte_read(starthere + j, buf + starthere - start + j, toread);
if (rc)
break;
@@ -908,6 +910,10 @@
break;
}
+ if (!rc)
+ progressbar_done(flash);
+ else
+ progressbar_abort(flash);
return rc;
}
Index: flashrom-progressbar/flashrom.c
===================================================================
--- flashrom-progressbar/flashrom.c (Revision 1072)
+++ flashrom-progressbar/flashrom.c (Arbeitskopie)
@@ -544,8 +544,12 @@
int read_memmapped(struct flashchip *flash, uint8_t *buf, int start, int len)
{
+ progressbar_start(flash);
+ /* chip_readn does not take a struct flashchip argument, and this means
+ * we can't print progress.
+ */
chip_readn(buf, flash->virtual_memory + start, len);
-
+ progressbar_done(flash);
return 0;
}
@@ -1193,7 +1197,7 @@
continue;
}
found = 1;
- msg_cdbg("trying... ");
+ progressbar_start(flash);
for (i = 0; i < NUM_ERASEREGIONS; i++) {
/* count==0 for all automatically initialized array
* members so the loop below won't be executed for them.
@@ -1201,8 +1205,7 @@
for (j = 0; j < eraser.eraseblocks[i].count; j++) {
start = done + eraser.eraseblocks[i].size * j;
len = eraser.eraseblocks[i].size;
- msg_cdbg("0x%06x-0x%06x, ", start,
- start + len - 1);
+ progressbar_update(flash, start);
ret = eraser.block_erase(flash, start, len);
if (ret)
break;
@@ -1212,10 +1215,12 @@
done += eraser.eraseblocks[i].count *
eraser.eraseblocks[i].size;
}
- msg_cdbg("\n");
/* If everything is OK, don't try another erase function. */
- if (!ret)
+ if (!ret) {
+ progressbar_done(flash);
break;
+ } else
+ progressbar_abort(flash);
}
if (!found) {
msg_cerr("ERROR: flashrom has no erase function for this flash chip.\n");
@@ -1230,6 +1235,43 @@
return ret;
}
+static int progressbar_pos = 0;
+
+void progressbar_start(struct flashchip *flash)
+{
+ msg_cdbg("|");
+ progressbar_pos = 0;
+}
+
+/**
+ * Print a progress bar with 64 dots. This allows fitting the progress bar in
+ * an 80 column screen next to some other output.
+ */
+void progressbar_update(struct flashchip *flash, int bytepos)
+{
+ int newpos;
+ /* This will be simpler once total_size is in bytes.
+ * bytepos + 1 to get the full progress bar on the highest address
+ * which is flash->total_size * 1024 - 1.
+ */
+ newpos = (bytepos + 1) * 64 / (flash->total_size * 1024);
+ for (; progressbar_pos < newpos; progressbar_pos++)
+ msg_cdbg(".");
+}
+
+void progressbar_done(struct flashchip *flash)
+{
+ progressbar_update(flash, flash->total_size * 1024 - 1);
+ msg_cdbg("|");
+}
+
+void progressbar_abort(struct flashchip *flash)
+{
+ for (; progressbar_pos < 64; progressbar_pos++)
+ msg_cdbg("!");
+ msg_cdbg("|");
+}
+
void emergency_help_message(void)
{
msg_gerr("Your flash chip is in an unknown state.\n"
--
http://www.hailfinger.org/
[View Less]
Every SPI programmer driver had its own completely different chip write
implementation, and all of them were insufficiently commented.
Create spi_write_chunked as a copy of spi_read_chunked and convert all
SPI programmers to use it.
No functional changes except:
- Bus Pirate uses 12 Byte writes instead of 8 Byte writes
- SB600 uses 5 Byte writes instead of 1 Byte writes
Should work. Not for 0.9.1, but since I had that thing on my disk
anyway, I figured I'd post it to the list in case someone …
[View More]wants to
experiment with partial writes.
Tests appreciated.
Signed-off-by: Carl-Daniel Hailfinger <c-d.hailfinger.devel.2006(a)gmx.net>
Index: flashrom-partial_write_spi/spi25.c
===================================================================
--- flashrom-partial_write_spi/spi25.c (Revision 982)
+++ flashrom-partial_write_spi/spi25.c (Arbeitskopie)
@@ -1,7 +1,7 @@
/*
* This file is part of the flashrom project.
*
- * Copyright (C) 2007, 2008, 2009 Carl-Daniel Hailfinger
+ * Copyright (C) 2007, 2008, 2009, 2010 Carl-Daniel Hailfinger
* Copyright (C) 2008 coresystems GmbH
*
* This program is free software; you can redistribute it and/or modify
@@ -874,7 +874,7 @@
}
/*
- * Read a complete flash chip.
+ * Read a part of the flash chip.
* Each page is read separately in chunks with a maximum size of chunksize.
*/
int spi_read_chunked(struct flashchip *flash, uint8_t *buf, int start, int len, int chunksize)
@@ -913,6 +913,52 @@
}
/*
+ * Write a part of the flash chip.
+ * Each page is written separately in chunks with a maximum size of chunksize.
+ */
+int spi_write_chunked(struct flashchip *flash, uint8_t *buf, int start, int len, int chunksize)
+{
+ int rc = 0;
+ int i, j, starthere, lenhere;
+ /* FIXME: page_size is the wrong variable. We need max_writechunk_size
+ * in struct flashchip to do this properly. All chips using
+ * spi_chip_write_256 have page_size set to max_writechunk_size, so
+ * we're OK for now.
+ */
+ int page_size = flash->page_size;
+ int towrite;
+
+ /* Warning: This loop has a very unusual condition and body.
+ * The loop needs to go through each page with at least one affected
+ * byte. The lowest page number is (start / page_size) since that
+ * division rounds down. The highest page number we want is the page
+ * where the last byte of the range lives. That last byte has the
+ * address (start + len - 1), thus the highest page number is
+ * (start + len - 1) / page_size. Since we want to include that last
+ * page as well, the loop condition uses <=.
+ */
+ for (i = start / page_size; i <= (start + len - 1) / page_size; i++) {
+ /* Byte position of the first byte in the range in this page. */
+ /* starthere is an offset to the base address of the chip. */
+ starthere = max(start, i * page_size);
+ /* Length of bytes in the range in this page. */
+ lenhere = min(start + len, (i + 1) * page_size) - starthere;
+ for (j = 0; j < lenhere; j += chunksize) {
+ towrite = min(chunksize, lenhere - j);
+ rc = spi_nbyte_program(starthere + j, buf + starthere - start + j, towrite);
+ if (rc)
+ break;
+ while (spi_read_status_register() & JEDEC_RDSR_BIT_WIP)
+ programmer_delay(10);
+ }
+ if (rc)
+ break;
+ }
+
+ return rc;
+}
+
+/*
* Program chip using byte programming. (SLOW!)
* This is for chips which can only handle one byte writes
* and for chips where memory mapped programming is impossible
Index: flashrom-partial_write_spi/buspirate_spi.c
===================================================================
--- flashrom-partial_write_spi/buspirate_spi.c (Revision 982)
+++ flashrom-partial_write_spi/buspirate_spi.c (Arbeitskopie)
@@ -1,7 +1,7 @@
/*
* This file is part of the flashrom project.
*
- * Copyright (C) 2009 Carl-Daniel Hailfinger
+ * Copyright (C) 2009, 2010 Carl-Daniel Hailfinger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@@ -319,7 +319,6 @@
int buspirate_spi_write_256(struct flashchip *flash, uint8_t *buf)
{
int total_size = 1024 * flash->total_size;
- int i;
spi_disable_blockprotect();
/* Erase first. */
@@ -330,25 +329,5 @@
}
msg_pinfo("done.\n");
- /* FIXME: We could do 12 byte writes, but then we'd have to make sure
- * not to cross a 256 byte page boundary. This problem only applies to
- * writes, reads can cross page boundaries just fine.
- */
- for (i = 0; i < total_size; i += 8) {
- int l, r;
- if (i + 8 <= total_size)
- l = 8;
- else
- l = total_size - i;
-
- if ((r = spi_nbyte_program(i, &buf[i], l))) {
- msg_perr("%s: write fail %d\n", __func__, r);
- return 1;
- }
-
- while (spi_read_status_register() & JEDEC_RDSR_BIT_WIP)
- /* loop */;
- }
-
- return 0;
+ return spi_write_chunked(flash, buf, 0, total_size, 12);
}
Index: flashrom-partial_write_spi/bitbang_spi.c
===================================================================
--- flashrom-partial_write_spi/bitbang_spi.c (Revision 982)
+++ flashrom-partial_write_spi/bitbang_spi.c (Arbeitskopie)
@@ -1,7 +1,7 @@
/*
* This file is part of the flashrom project.
*
- * Copyright (C) 2009 Carl-Daniel Hailfinger
+ * Copyright (C) 2009, 2010 Carl-Daniel Hailfinger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@@ -142,24 +142,7 @@
int bitbang_spi_write_256(struct flashchip *flash, uint8_t *buf)
{
int total_size = 1024 * flash->total_size;
- int i;
msg_pdbg("total_size is %d\n", total_size);
- for (i = 0; i < total_size; i += 256) {
- int l, r;
- if (i + 256 <= total_size)
- l = 256;
- else
- l = total_size - i;
-
- if ((r = spi_nbyte_program(i, &buf[i], l))) {
- msg_perr("%s: write fail %d\n", __func__, r);
- return 1;
- }
-
- while (spi_read_status_register() & JEDEC_RDSR_BIT_WIP)
- /* loop */;
- }
-
- return 0;
+ return spi_write_chunked(flash, buf, 0, total_size, 256);
}
Index: flashrom-partial_write_spi/ft2232_spi.c
===================================================================
--- flashrom-partial_write_spi/ft2232_spi.c (Revision 982)
+++ flashrom-partial_write_spi/ft2232_spi.c (Arbeitskopie)
@@ -2,7 +2,7 @@
* This file is part of the flashrom project.
*
* Copyright (C) 2009 Paul Fox <pgf(a)laptop.org>
- * Copyright (C) 2009 Carl-Daniel Hailfinger
+ * Copyright (C) 2009, 2010 Carl-Daniel Hailfinger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@@ -288,7 +288,6 @@
int ft2232_spi_write_256(struct flashchip *flash, uint8_t *buf)
{
int total_size = 1024 * flash->total_size;
- int i;
spi_disable_blockprotect();
/* Erase first. */
@@ -299,23 +298,7 @@
}
msg_pinfo("done.\n");
msg_pdbg("total_size is %d\n", total_size);
- for (i = 0; i < total_size; i += 256) {
- int l, r;
- if (i + 256 <= total_size)
- l = 256;
- else
- l = total_size - i;
-
- if ((r = spi_nbyte_program(i, &buf[i], l))) {
- msg_perr("%s: write fail %d\n", __func__, r);
- return 1;
- }
-
- while (spi_read_status_register() & JEDEC_RDSR_BIT_WIP)
- /* loop */;
- }
-
- return 0;
+ return spi_write_chunked(flash, buf, 0, total_size, 256);
}
#endif
Index: flashrom-partial_write_spi/sb600spi.c
===================================================================
--- flashrom-partial_write_spi/sb600spi.c (Revision 982)
+++ flashrom-partial_write_spi/sb600spi.c (Arbeitskopie)
@@ -4,6 +4,7 @@
* Copyright (C) 2008 Wang Qingpei <Qingpei.Wang(a)amd.com>
* Copyright (C) 2008 Joe Bao <Zheng.Bao(a)amd.com>
* Copyright (C) 2008 Advanced Micro Devices, Inc.
+ * Copyright (C) 2009, 2010 Carl-Daniel Hailfinger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@@ -49,7 +50,6 @@
/* FIXME: SB600 can write 5 bytes per transaction. */
int sb600_spi_write_1(struct flashchip *flash, uint8_t *buf)
{
- int i;
int total_size = flash->total_size * 1024;
int result = 0;
@@ -63,19 +63,7 @@
msg_pinfo("done.\n");
msg_pinfo("Programming flash");
- for (i = 0; i < total_size; i++, buf++) {
- result = spi_nbyte_program(i, buf, 1);
- if (result) {
- msg_perr("Write error!\n");
- return result;
- }
-
- /* wait program complete. */
- if (i % 0x8000 == 0)
- msg_pspew(".");
- while (spi_read_status_register() & JEDEC_RDSR_BIT_WIP)
- ;
- }
+ result = spi_write_chunked(flash, buf, 0, total_size, 5);
msg_pinfo(" done.\n");
return result;
}
Index: flashrom-partial_write_spi/ichspi.c
===================================================================
--- flashrom-partial_write_spi/ichspi.c (Revision 982)
+++ flashrom-partial_write_spi/ichspi.c (Arbeitskopie)
@@ -5,7 +5,7 @@
* Copyright (C) 2008 Claus Gindhart <claus.gindhart(a)kontron.com>
* Copyright (C) 2008 Dominik Geyer <dominik.geyer(a)kontron.com>
* Copyright (C) 2008 coresystems GmbH <info(a)coresystems.de>
- * Copyright (C) 2009 Carl-Daniel Hailfinger
+ * Copyright (C) 2009, 2010 Carl-Daniel Hailfinger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
@@ -160,8 +160,6 @@
static int program_opcodes(OPCODES * op);
static int run_opcode(OPCODE op, uint32_t offset,
uint8_t datalength, uint8_t * data);
-static int ich_spi_write_page(struct flashchip *flash, uint8_t * bytes,
- int offset, int maxdata);
/* for pairing opcodes with their required preop */
struct preop_opcode_pair {
@@ -645,28 +643,6 @@
return -1;
}
-static int ich_spi_write_page(struct flashchip *flash, uint8_t * bytes,
- int offset, int maxdata)
-{
- int page_size = flash->page_size;
- uint32_t remaining = page_size;
- int towrite;
-
- msg_comm_debug("ich_spi_write_page: offset=%d, number=%d, buf=%p\n",
- offset, page_size, bytes);
-
- for (; remaining > 0; remaining -= towrite) {
- towrite = min(remaining, maxdata);
- if (spi_nbyte_program(offset + (page_size - remaining),
- &bytes[page_size - remaining], towrite)) {
- printf_debug("Error writing");
- return 1;
- }
- }
-
- return 0;
-}
-
int ich_spi_read(struct flashchip *flash, uint8_t * buf, int start, int len)
{
int maxdata = 64;
@@ -679,12 +655,14 @@
int ich_spi_write_256(struct flashchip *flash, uint8_t * buf)
{
- int i, j, rc = 0;
+ int i, ret = 0;
int total_size = flash->total_size * 1024;
- int page_size = flash->page_size;
int erase_size = 64 * 1024;
int maxdata = 64;
+ if (spi_controller == SPI_CONTROLLER_VIA)
+ maxdata = 16;
+
spi_disable_blockprotect();
/* Erase first */
printf("Erasing flash before programming... ");
@@ -696,19 +674,15 @@
printf("Programming page: \n");
for (i = 0; i < total_size / erase_size; i++) {
- if (spi_controller == SPI_CONTROLLER_VIA)
- maxdata = 16;
-
- for (j = 0; j < erase_size / page_size; j++) {
- ich_spi_write_page(flash,
- (void *)(buf + (i * erase_size) + (j * page_size)),
- (i * erase_size) + (j * page_size), maxdata);
- }
+ ret = spi_write_chunked(flash, buf + (i * erase_size),
+ i * erase_size, erase_size, maxdata);
+ if (ret)
+ break;
}
printf("\n");
- return rc;
+ return ret;
}
int ich_spi_send_command(unsigned int writecnt, unsigned int readcnt,
Index: flashrom-partial_write_spi/chipdrivers.h
===================================================================
--- flashrom-partial_write_spi/chipdrivers.h (Revision 982)
+++ flashrom-partial_write_spi/chipdrivers.h (Arbeitskopie)
@@ -51,6 +51,7 @@
int spi_nbyte_program(int addr, uint8_t *bytes, int len);
int spi_nbyte_read(int addr, uint8_t *bytes, int len);
int spi_read_chunked(struct flashchip *flash, uint8_t *buf, int start, int len, int chunksize);
+int spi_write_chunked(struct flashchip *flash, uint8_t *buf, int start, int len, int chunksize);
int spi_aai_write(struct flashchip *flash, uint8_t *buf);
/* 82802ab.c */
--
http://www.hailfinger.org/
[View Less]
Hi,
I've tested a Winbond W25Q80BV (W25Q80BVDAIG/25Q80BVAIG) socketed on a Asus
M4A87TD/USB3. Read, erase, write and verify worked fine. Attached is
flashrom -V output.
Jonathan Kollasch
Check if disabling the block protection on generic SPI chips works and
complain otherwise.
This patch will also appear at the top of
http://patchwork.coreboot.org/project/flashrom/list/
arunkumarm, please download latest flashrom (at least r1072) from
subversion:
svn co svn://coreboot.org/flashrom/trunk flashrom
and then apply this patch and try to write again in verbose mode.
Signed-off-by: Carl-Daniel Hailfinger <c-d.hailfinger.devel.2006(a)gmx.net>
Index: flashrom-…
[View More]spi25_disable_blockprotect_check/spi25.c
===================================================================
--- flashrom-spi25_disable_blockprotect_check/spi25.c (Revision 1072)
+++ flashrom-spi25_disable_blockprotect_check/spi25.c (Arbeitskopie)
@@ -855,6 +855,11 @@
msg_cerr("spi_write_status_register failed\n");
return result;
}
+ status = spi_read_status_register();
+ if ((status & 0x3c) != 0) {
+ msg_cerr("Block protection could not be disabled!\n");
+ /* Should we error out here? */
+ }
}
return 0;
}
--
http://www.hailfinger.org/
[View Less]