/* * Copyright © 2011 Keith Packard * * 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 * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ #include #include uint8_t ao_storage_read(ao_pos_t pos, void *v_buf, uint16_t len) { uint8_t *buf = v_buf; uint16_t this_len; uint16_t this_off; ao_storage_setup(); if (pos >= ao_storage_total || pos + len > ao_storage_total) return 0; while (len) { /* Compute portion of transfer within * a single block */ this_off = (uint16_t) pos & (ao_storage_unit - 1); this_len = ao_storage_unit - this_off; if (this_len > len) this_len = len; if (!ao_storage_device_read(pos, buf, this_len)) return 0; /* See how much is left */ buf += this_len; len -= this_len; pos += this_len; } return 1; } uint8_t ao_storage_write(ao_pos_t pos, void *v_buf, uint16_t len) { uint8_t *buf = v_buf; uint16_t this_len; uint16_t this_off; ao_storage_setup(); if (pos >= ao_storage_total || pos + len > ao_storage_total) return 0; while (len) { /* Compute portion of transfer within * a single block */ this_off = (uint16_t) pos & (ao_storage_unit - 1); this_len = ao_storage_unit - this_off; if (this_len > len) this_len = len; if (!ao_storage_device_write(pos, buf, this_len)) return 0; /* See how much is left */ buf += this_len; len -= this_len; pos += this_len; } return 1; } static uint8_t storage_data[128]; static void ao_storage_dump(void) { uint32_t block; uint8_t i, j; block = ao_cmd_hex(); if (ao_cmd_status != ao_cmd_success) return; for (i = 0; ; i += 8) { if (ao_storage_read((block << 8) + i, storage_data, 8)) { ao_cmd_put16((uint16_t) i); for (j = 0; j < 8; j++) { putchar(' '); ao_cmd_put8(storage_data[j]); } putchar ('\n'); } if (i == 248) break; } } #if HAS_STORAGE_DEBUG /* not enough space for this today */ static void ao_storage_store(void) { uint16_t block; uint8_t i; uint16_t len; uint8_t b; uint32_t addr; block = ao_cmd_hex(); i = ao_cmd_hex(); addr = ((uint32_t) block << 8) | i; len = ao_cmd_hex(); if (ao_cmd_status != ao_cmd_success) return; while (len--) { b = ao_cmd_hexbyte(); if (ao_cmd_status != ao_cmd_success) return; ao_storage_write(addr, &b, 1); addr++; } } #endif static void ao_storage_zap(void) { uint32_t v = ao_cmd_hex(); if (ao_cmd_status != ao_cmd_success) return; ao_storage_erase((uint32_t) v << 8); } static void ao_storage_zapall(void) { uint32_t pos; ao_cmd_white(); if (!ao_match_word("DoIt")) return; for (pos = 0; pos < ao_storage_log_max; pos += ao_storage_block) ao_storage_erase(pos); } #if AO_STORAGE_TEST static void ao_storage_failure(uint32_t pos, char *format, ...) { va_list a; printf("TEST FAILURE AT %08x: ", pos); va_start(a, format); vprintf(format, a); va_end(a); } static uint8_t ao_storage_check_block(uint32_t pos, uint8_t value) { uint32_t offset; uint32_t byte; for (offset = 0; offset < ao_storage_block; offset += sizeof (storage_data)) { if (!ao_storage_read(pos + offset, storage_data, sizeof (storage_data))) { ao_storage_failure(pos + offset, "read failed\n"); return 0; } for (byte = 0; byte < sizeof (storage_data); byte++) if (storage_data[byte] != value) { ao_storage_failure(pos + offset + byte, "want %02x got %02x\n", value, storage_data[byte]); return 0; } } return 1; } static uint8_t ao_storage_fill_block(uint32_t pos, uint8_t value) { uint32_t offset; uint32_t byte; for (byte = 0; byte < sizeof (storage_data); byte++) storage_data[byte] = value; for (offset = 0; offset < ao_storage_block; offset += sizeof (storage_data)) { if (!ao_storage_write(pos + offset, storage_data, sizeof (storage_data))) { ao_storage_failure(pos + offset, "write failed\n"); return 0; } } return 1; } static uint8_t ao_storage_check_incr_block(uint32_t pos) { uint32_t offset; uint32_t byte; for (offset = 0; offset < ao_storage_block; offset += sizeof (storage_data)) { if (!ao_storage_read(pos + offset, storage_data, sizeof (storage_data))) { ao_storage_failure(pos + offset, "read failed\n"); return 0; } for (byte = 0; byte < sizeof (storage_data); byte++) { uint8_t value = offset + byte; if (storage_data[byte] != value) { ao_storage_failure(pos + offset + byte, "want %02x got %02x\n", value, storage_data[byte]); return 0; } } } return 1; } static uint8_t ao_storage_fill_incr_block(uint32_t pos) { uint32_t offset; uint32_t byte; for (offset = 0; offset < ao_storage_block; offset += sizeof (storage_data)) { for (byte = 0; byte < sizeof (storage_data); byte++) storage_data[byte] = offset + byte; if (!ao_storage_write(pos + offset, storage_data, sizeof (storage_data))) { ao_storage_failure(pos + offset, "write failed\n"); return 0; } } return 1; } static uint8_t ao_storage_fill_check_block(uint32_t pos, uint8_t value) { return ao_storage_fill_block(pos, value) && ao_storage_check_block(pos, value); } static uint8_t ao_storage_incr_check_block(uint32_t pos) { return ao_storage_fill_incr_block(pos) && ao_storage_check_incr_block(pos); } static uint8_t ao_storage_test_block(uint32_t pos) { ao_storage_erase(pos); printf(" erase"); flush(); if (!ao_storage_check_block(pos, 0xff)) return 0; printf(" zero"); flush(); if (!ao_storage_fill_check_block(pos, 0x00)) return 0; ao_storage_erase(pos); printf(" 0xaa"); flush(); if (!ao_storage_fill_check_block(pos, 0xaa)) return 0; ao_storage_erase(pos); printf(" 0x55"); flush(); if (!ao_storage_fill_check_block(pos, 0x55)) return 0; ao_storage_erase(pos); printf(" increment"); flush(); if (!ao_storage_incr_check_block(pos)) return 0; ao_storage_erase(pos); printf(" pass\n"); flush(); return 1; } static void ao_storage_test(void) { uint32_t pos; ao_cmd_white(); if (!ao_match_word("DoIt")) return; for (pos = 0; pos < ao_storage_log_max; pos += ao_storage_block) { printf("Testing block 0x%08x:", pos); flush(); if (!ao_storage_test_block(pos)) break; } printf("Test complete\n"); } #endif /* AO_STORAGE_TEST */ static void ao_storage_info(void) { ao_storage_setup(); printf("Storage size: %ld\n", (long) ao_storage_total); printf("Storage erase unit: %ld\n", (long) ao_storage_block); ao_storage_device_info(); } const struct ao_cmds ao_storage_cmds[] = { { ao_storage_info, "f\0Show storage" }, { ao_storage_dump, "e \0Dump flash" }, #if HAS_STORAGE_DEBUG { ao_storage_store, "w ...\0Write data to flash" }, #endif { ao_storage_zap, "z \0Erase " }, { ao_storage_zapall,"Z \0Erase all. is doit with D&I" }, #if AO_STORAGE_TEST { ao_storage_test, "V \0Validate flash (destructive). is doit with D&I" }, #endif { 0, NULL }, }; void ao_storage_init(void) { ao_storage_device_init(); ao_cmd_register(&ao_storage_cmds[0]); }