summaryrefslogtreecommitdiffstats
path: root/src/drivers/nvs
diff options
context:
space:
mode:
authorMichael Brown2006-12-06 19:51:31 +0100
committerMichael Brown2006-12-06 19:51:31 +0100
commitac401f481d0573482d4a3cecec05cce6474d98a9 (patch)
tree2a0cadea3dce11803defa4ba6e9852c9a21e1eba /src/drivers/nvs
parentWork around another confused-by-RST bug (diff)
downloadipxe-ac401f481d0573482d4a3cecec05cce6474d98a9.tar.gz
ipxe-ac401f481d0573482d4a3cecec05cce6474d98a9.tar.xz
ipxe-ac401f481d0573482d4a3cecec05cce6474d98a9.zip
Verify data after writing
Diffstat (limited to 'src/drivers/nvs')
-rw-r--r--src/drivers/nvs/nvs.c36
1 files changed, 35 insertions, 1 deletions
diff --git a/src/drivers/nvs/nvs.c b/src/drivers/nvs/nvs.c
index a566cdfd..44b56f65 100644
--- a/src/drivers/nvs/nvs.c
+++ b/src/drivers/nvs/nvs.c
@@ -17,6 +17,8 @@
*/
#include <stdint.h>
+#include <string.h>
+#include <errno.h>
#include <assert.h>
#include <gpxe/nvs.h>
@@ -70,6 +72,34 @@ int nvs_read ( struct nvs_device *nvs, unsigned int address,
}
/**
+ * Verify content of non-volatile storage device
+ *
+ * @v nvs NVS device
+ * @v address Address from which to read
+ * @v data Data to compare against
+ * @v len Length of data buffer
+ * @ret rc Return status code
+ */
+static int nvs_verify ( struct nvs_device *nvs, unsigned int address,
+ const void *data, size_t len ) {
+ uint8_t read_data[len];
+ int rc;
+
+ /* Read data into temporary buffer */
+ if ( ( rc = nvs_read ( nvs, address, read_data, len ) ) != 0 )
+ return rc;
+
+ /* Compare data */
+ if ( memcmp ( data, read_data, len ) != 0 ) {
+ DBG ( "NVS %p verification failed at %#04x+%d\n",
+ nvs, address, len );
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+/**
* Write to non-volatile storage device
*
* @v nvs NVS device
@@ -99,10 +129,14 @@ int nvs_write ( struct nvs_device *nvs, unsigned int address,
if ( frag_len > len )
frag_len = len;
- /* Read this portion of the buffer from the device */
+ /* Write this portion of the buffer to the device */
if ( ( rc = nvs->write ( nvs, address, data, frag_len ) ) != 0)
return rc;
+ /* Read back and verify data */
+ if ( ( rc = nvs_verify ( nvs, address, data, frag_len ) ) != 0)
+ return rc;
+
/* Update parameters */
data += frag_len;
address += ( frag_len >> nvs->word_len_log2 );