| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686 |
- #include <string.h>
- #include "typedef.h"
- #include "amt630h.h"
- #include "UartPrint.h"
- #include "timer.h"
- #include "spi.h"
- #include "cp15.h"
- #include "sysinfo.h"
- #include "crc32.h"
- #include "gpio.h"
- #define SPI_CS_GPIO 32
- #define SPI_RXFIFO_FULL (1<<4)
- #define SPI_RXFIFO_NOTEMPTY (1<<3)
- #define SPI_TXFIFO_EMPTY (1<<2)
- #define SPI_TXFIFO_NOTFULL (1<<1)
- #define SPIFLASH_BUSY (1<<0)
- /* SFUD support manufacturer JEDEC ID */
- #define SFUD_MF_ID_CYPRESS 0x01
- #define SFUD_MF_ID_FUJITSU 0x04
- #define SFUD_MF_ID_EON 0x1C
- #define SFUD_MF_ID_ATMEL 0x1F
- #define SFUD_MF_ID_MICRON 0x20
- #define SFUD_MF_ID_AMIC 0x37
- #define SFUD_MF_ID_SANYO 0x62
- #define SFUD_MF_ID_INTEL 0x89
- #define SFUD_MF_ID_ESMT 0x8C
- #define SFUD_MF_ID_FUDAN 0xA1
- #define SFUD_MF_ID_HYUNDAI 0xAD
- #define SFUD_MF_ID_SST 0xBF
- #define SFUD_MF_ID_MICRONIX 0xC2
- #define SFUD_MF_ID_GIGADEVICE 0xC8
- #define SFUD_MF_ID_ISSI 0xD5
- #define SFUD_MF_ID_WINBOND 0xEF
- /* flash chip information */
- typedef struct {
- char *name; /**< flash chip name */
- uint8_t mf_id; /**< manufacturer ID */
- uint8_t type_id; /**< memory type ID */
- uint8_t capacity_id; /**< capacity ID */
- uint32_t capacity; /**< flash capacity (bytes) */
- } flash_chip;
- static const flash_chip flash_chip_table[] =
- { \
- {"W25Q40BV", SFUD_MF_ID_WINBOND, 0x40, 0x13, 512L*1024L}, \
- {"W25Q16BV", SFUD_MF_ID_WINBOND, 0x40, 0x15, 2L*1024L*1024L}, \
- {"W25Q32BV", SFUD_MF_ID_WINBOND, 0x40, 0x16, 4L*1024L*1024L}, \
- {"W25Q64CV", SFUD_MF_ID_WINBOND, 0x40, 0x17, 8L*1024L*1024L}, \
- {"W25Q64DW", SFUD_MF_ID_WINBOND, 0x60, 0x17, 8L*1024L*1024L}, \
- {"W25Q128BV", SFUD_MF_ID_WINBOND, 0x40, 0x18, 16L*1024L*1024L}, \
- {"W25Q256FV", SFUD_MF_ID_WINBOND, 0x40, 0x19, 32L*1024L*1024L}, \
- {"W25H256JV", SFUD_MF_ID_WINBOND, 0x90, 0x19, 32L*1024L*1024L}, \
- {"W25Q512JVFM", SFUD_MF_ID_WINBOND, 0x70, 0x20, 64L*1024L*1024L}, \
- {"SST25VF080B", SFUD_MF_ID_SST, 0x25, 0x8E, 1L*1024L*1024L}, \
- {"SST25VF016B", SFUD_MF_ID_SST, 0x25, 0x41, 2L*1024L*1024L}, \
- {"EN25Q32B", SFUD_MF_ID_EON, 0x30, 0x16, 4L*1024L*1024L}, \
- {"GD25Q64B", SFUD_MF_ID_GIGADEVICE, 0x40, 0x17, 8L*1024L*1024L}, \
- {"GD25Q16B", SFUD_MF_ID_GIGADEVICE, 0x40, 0x15, 2L*1024L*1024L}, \
- {"GD25Q32C", SFUD_MF_ID_GIGADEVICE, 0x40, 0x16, 4L*1024L*1024L}, \
- {"S25FL216K", SFUD_MF_ID_CYPRESS, 0x40, 0x15, 2L*1024L*1024L}, \
- {"S25FL032P", SFUD_MF_ID_CYPRESS, 0x02, 0x15, 4L*1024L*1024L}, \
- {"A25L080", SFUD_MF_ID_AMIC, 0x30, 0x14, 1L*1024L*1024L}, \
- {"F25L004", SFUD_MF_ID_ESMT, 0x20, 0x13, 512L*1024L}, \
- {"MX25L6433F", SFUD_MF_ID_MICRONIX, 0x20, 0x17, 8L*1024L*1024L}, \
- {"MX25L12845G", SFUD_MF_ID_MICRONIX, 0x20, 0x18, 16L*1024L*1024L}, \
- {"MX25L25645G", SFUD_MF_ID_MICRONIX, 0x20, 0x19, 32L*1024L*1024L},\
- {"PCT25VF016B", SFUD_MF_ID_SST, 0x25, 0x41, 2L*1024L*1024L}, \
- {"GD25B256", SFUD_MF_ID_GIGADEVICE, 0x43, 0x19, 32L*1024L*1024L},\
- {"GD25B512ME", SFUD_MF_ID_GIGADEVICE, 0x47, 0x1A, 64L*1024L*1024L},
- };
- static int addr_in_4_byte = 0;
- static void SetCSGpioEnable(int enable)
- {
- gpio_direction_output(SPI_CS_GPIO, !enable);
- }
- static void SetSpiDataMode(unsigned int bitMode)
- {
- unsigned int val = 0;
-
- (void)val;
- while((rSPI_SR & SPI_BUSY));
- rSPI_SSIENR = 0;
- val = rSPI_CTLR0;
- val &=~(0x1f<<16);
- val |=((bitMode-1)<<16);
- rSPI_CTLR0 = val;
- rSPI_SSIENR = 1;
- }
- static void SpiWaitIdle(void)
- {
- while(rSPI_SR & SPIFLASH_BUSY);
- udelay(2);
- }
- static void SpiEmptyRxFIFO(void)
- {
- INT32 data = 0;
- (void)data;
-
- while(rSPI_SR & SPI_RXFIFO_NOTEMPTY)
- data = rSPI_DR;
- }
- static UINT8 SpiReadSta(void)
- {
- UINT8 status;
- SetSpiDataMode(8);
- SetCSGpioEnable(1);
- rSPI_DR = SPI_READ_STATUS;
- rSPI_DR = 0;
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- status = rSPI_DR;
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- status = rSPI_DR;
- //PrintVariableValueHex("status:", status);
- SpiWaitIdle();
- SetCSGpioEnable(0);
- SetSpiDataMode(32);
- return status;
- }
- static void SpiDisable4ByteMode(void)
- {
- SetSpiDataMode(8);
- SetCSGpioEnable(1);
- rSPI_DR = SPI_DISABLE_4BYTE_MODE;
- while(!(rSPI_SR & SPI_TXFIFO_EMPTY));
- SpiWaitIdle();
- SetCSGpioEnable(0);
- SetSpiDataMode(32);
- }
- static void SpiEnable4ByteMode(void)
- {
- SetSpiDataMode(8);
- SetCSGpioEnable(1);
- rSPI_DR = SPI_ENABLE_4BYTE_MODE;
- while(!(rSPI_SR & SPI_TXFIFO_EMPTY));
- SpiWaitIdle();
- SetCSGpioEnable(0);
- SetSpiDataMode(32);
- }
- static void SpiReadPage(UINT32 pagenum, UINT32 *buf)
- {
- UINT32 addr;
- UINT32 val = 0;
- INT32 i, j;
- UINT8 tmpaddr[4];
- UINT8 *data = (UINT8*)buf;
- (void)val;
-
- addr = pagenum*BYTESPERPAGE;
- tmpaddr[0] = addr;
- tmpaddr[1] = addr>>8;
- tmpaddr[2] = addr>>16;
- tmpaddr[3] = addr>>24;
- SpiEmptyRxFIFO();
- SetCSGpioEnable(1);
- if (addr_in_4_byte) {
- SetSpiDataMode(8);
- rSPI_DR = SPI_4BYTEADDR_READ_DATA;
- rSPI_DR = tmpaddr[3];
- rSPI_DR = tmpaddr[2];
- rSPI_DR = tmpaddr[1];
- rSPI_DR = tmpaddr[0];
- for (i = 0; i < 5; i++) {
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- val = rSPI_DR;
- }
- } else {
- rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_READ_DATA;
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- val = rSPI_DR;
- }
- if (addr_in_4_byte) {
- for (i = 0; i < 4; i++) {
- for (j = 0; j < WORDSPERPAGE; j++) {
- while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
- rSPI_DR = 0;
- }
- for (j = 0; j < WORDSPERPAGE; j++) {
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- *data++ = rSPI_DR;
- }
- }
- } else {
- for (i = 0; i < WORDSPERPAGE; i++) {
- while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
- rSPI_DR = 0;
- }
- for(i = 0; i < WORDSPERPAGE; i++) {
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- *buf++ = rSPI_DR;
- }
- }
- SpiWaitIdle();
- SetCSGpioEnable(0);
- if (addr_in_4_byte)
- SetSpiDataMode(32);
- }
- void SpiSelectPad()
- {
- UINT32 val;
- val = rSYS_PAD_CTRL02;
- val &= ~0xfff;
- val |= (0x1<<10)|(0x1 << 8)|(0x1 << 6)|(0x1 << 4)| (0x1 << 2);
- rSYS_PAD_CTRL02 = val;
-
- val = rSYS_SSP_CLK_CFG;
- val &= ~((0x1<<31)|(0x1<<30));
- val |= (0x1<<31)|(0x1<<30);
- rSYS_SSP_CLK_CFG = val;
- //cs inactive first
- SetCSGpioEnable(0);
- }
- void Reset(void)
- {
- SetSpiDataMode(8);
- rSPI_DR = 0x66;
- while((SpiReadSta() & SPI_BUSY));
- // while(!(SpiReadSta() & SPIFLASH_WRITEENABLE));
- rSPI_DR = 0x99;
- while((SpiReadSta() & SPI_BUSY));
- // while(!(SpiReadSta() & SPIFLASH_WRITEENABLE));
-
- SetSpiDataMode(32);
- }
- #define SPI0_CS0_GPIO 32
- #define SPI0_IO0_GPIO 34
- static void dwspi_jedec252_reset(void)
- {
- int i;
- int si = 0;
- UINT32 val;
-
- val = rSYS_PAD_CTRL02;
- val &= ~((3 << 4) | 3);
- rSYS_PAD_CTRL02 = val;
- gpio_direction_output(SPI0_CS0_GPIO, 1);
- gpio_direction_output(SPI0_IO0_GPIO, 1);
- udelay(300);
- for (i = 0; i < 4; i++) {
- gpio_direction_output(SPI0_CS0_GPIO, 0);
- gpio_direction_output(SPI0_IO0_GPIO, si);
- si = !si;
- udelay(300);
- gpio_direction_output(SPI0_CS0_GPIO, 1);
- udelay(300);
- }
- }
- static void SpiReadDeviceId(UINT8 *mfid, UINT8 *devid)
- {
- UINT8 val[6];
- int i;
- SetSpiDataMode(8);
- SetCSGpioEnable(1);
- rSPI_DR = SPI_MF_DEVICE_ID;
- rSPI_DR = 0;
- rSPI_DR = 0;
- rSPI_DR = 0;
- rSPI_DR = 0;
- rSPI_DR = 0;
- for (i = 0; i < 6; i++)
- {
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- val[i] = rSPI_DR;
- }
- *mfid = val[4];
- *devid = val[5];
- SpiWaitIdle();
- SetCSGpioEnable(0);
- SetSpiDataMode(32);
- }
- static void SpiReadJedecId(UINT8 *mfid, UINT8 *memid, UINT8 *capid)
- {
- UINT8 val[4];
- int i;
- SetSpiDataMode(8);
- SetCSGpioEnable(1);
- rSPI_DR = SPI_READ_JEDEC_ID;
- rSPI_DR = 0;
- rSPI_DR = 0;
- rSPI_DR = 0;
-
- for (i = 0; i < 4; i++)
- {
- while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
- val[i] = rSPI_DR;
- }
- *mfid = val[1];
- *memid = val[2];
- *capid = val[3];
- SpiWaitIdle();
- SetCSGpioEnable(0);
- SetSpiDataMode(32);
- }
- void SpiReadId(void)
- {
- UINT8 mfid,devid,memid,capid;
- SpiReadDeviceId(&mfid,&devid);
- SpiReadJedecId(&mfid,&memid,&capid);
- PrintVariableValueHex("ManufacturerID: ", mfid);
- PrintVariableValueHex("DeviceID: ", devid);
- PrintVariableValueHex("Memory Type ID: ", memid);
- PrintVariableValueHex("Capacity ID: ", capid);
- }
- void SpiInit()
- {
- uint8_t mfid, typeid, capid;
- unsigned int val;
- int i;
- dwspi_jedec252_reset();
- SpiSelectPad();
- val = rSYS_SOFT_RST;
- val &= ~(0x1<<8);
- rSYS_SOFT_RST = val;
- udelay(10);
- val |= (0x1<<8);
- rSYS_SOFT_RST = val;
- rSPI_SSIENR = 0;
- rSPI_CTLR0 = 0;
- rSPI_CTLR0 |=(0<<21)|(0x1f<<16)|(0x0<<12)|(0x0<<8)|(0x0<<4);
- // rSPI_CTLR1 = 63;
- rSPI_BAUDR = 4;//42;//16;//2;
- rSPI_SER = 1;
- rSPI_IMR = 0;
- rSPI_SSIENR = 1;
- // Reset();
- SpiReadJedecId(&mfid, &typeid, &capid);
- for (i = 0; i < sizeof(flash_chip_table) / sizeof(flash_chip); i++) {
- if ((flash_chip_table[i].mf_id == mfid)
- && (flash_chip_table[i].type_id == typeid)
- && (flash_chip_table[i].capacity_id == capid)) {
- if (flash_chip_table[i].capacity > 0x1000000) {
- addr_in_4_byte = 1;
- SpiEnable4ByteMode();
- }
- break;
- }
- }
- if (!addr_in_4_byte)
- SpiDisable4ByteMode();
- udelay(10000);
- }
- #define SPI_READ_MAXLEN BYTESPERPAGE
- static void SpiLoadStepldr(void (*readfunc)(UINT32, UINT32 *))
- {
- unsigned int i = 0;
- UINT32 *buf = (UINT32*)STEPLDR_ENTRY;
- UINT32 offset;
- UINT32 size;
- UINT32 nPageCount;
- UINT32 nPageStart;
-
- if (ReadSysInfo()) {
- SendUartString("read sysinfo fail, try to load stepldr part a.\n");
- offset = STEPLDRA_OFFSET;
- size = STEPLDR_MAX_SIZE;
- } else {
- SysInfo *sysinfo = GetSysInfo();
- offset = sysinfo->stepldr_offset;
- size = sysinfo->stepldr_size;
- }
-
- PrintVariableValueHex("stepldr offset: ", offset);
-
- nPageCount = (size + BYTESPERPAGE - 1) / BYTESPERPAGE;
- nPageStart = offset / BYTESPERPAGE;
- for(i = nPageStart; i < nPageStart + nPageCount; i++)
- {
- readfunc(i, buf);
- buf += BYTESPERPAGE/4;
- }
-
- #ifdef MMU_ENABLE
- CP15_clean_dcache_for_dma(STEPLDR_ENTRY, STEPLDR_ENTRY + size);
- #endif
- }
- static void SpiWriteEnable(void)
- {
- SetSpiDataMode(8);
- SetCSGpioEnable(1);
- rSPI_DR = SPI_WRITE_ENABLE;
- SpiWaitIdle();
- SetCSGpioEnable(0);
- while((SpiReadSta() & SPI_BUSY));
- while(!(SpiReadSta() & SPIFLASH_WRITEENABLE));
- SetSpiDataMode(32);
- }
- static void SpiEraseSector(UINT32 sectorNum)
- {
- UINT32 addr;
- UINT8 tmpaddr[4];
- addr = BYTESPERSECTOR*sectorNum;
- tmpaddr[0] = addr;
- tmpaddr[1] = addr>>8;
- tmpaddr[2] = addr>>16;
- tmpaddr[3] = addr>>24;
- SpiWriteEnable();
- SetCSGpioEnable(1);
- if (addr_in_4_byte) {
- SetSpiDataMode(8);
- rSPI_DR = SPI_4BYTEADD_SECTOR_ERASE;
- rSPI_DR = tmpaddr[3];
- rSPI_DR = tmpaddr[2];
- rSPI_DR = tmpaddr[1];
- rSPI_DR = tmpaddr[0];
- } else {
- rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_SECTOR_ERASE;
- }
- SpiWaitIdle();
- SetCSGpioEnable(0);
- while((SpiReadSta() & SPIFLASH_WRITEENABLE));
- }
- static void SpiEraseBlock(UINT32 blockNum)
- {
- UINT32 addr;
- UINT8 tmpaddr[4];
- addr = BYTESPERBLOCK*blockNum;
- tmpaddr[0] = addr;
- tmpaddr[1] = addr>>8;
- tmpaddr[2] = addr>>16;
- tmpaddr[3] = addr>>24;
- SpiWriteEnable();
- SetCSGpioEnable(1);
- if (addr_in_4_byte) {
- SetSpiDataMode(8);
- rSPI_DR = SPI_4BYTEADD_BLOCK_ERASE;
- rSPI_DR = tmpaddr[3];
- rSPI_DR = tmpaddr[2];
- rSPI_DR = tmpaddr[1];
- rSPI_DR = tmpaddr[0];
- } else {
- rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_BLOCK_ERASE;
- }
- SpiWaitIdle();
- SetCSGpioEnable(0);
- while((SpiReadSta() & SPIFLASH_WRITEENABLE));
- }
- static void SpiWritePage(UINT32 pagenum, UINT32 *buf)
- {
- UINT32 addr;
- UINT32 val = 0;;
- INT32 i;
- UINT8 tmpaddr[4];
- UINT8 *data = (UINT8*)buf;
-
- (void)val;
- addr = pagenum*BYTESPERPAGE;
- tmpaddr[0] = addr;
- tmpaddr[1] = addr>>8;
- tmpaddr[2] = addr>>16;
- tmpaddr[3] = addr>>24;
- SpiWriteEnable();
- SetCSGpioEnable(1);
- if (addr_in_4_byte) {
- SetSpiDataMode(8);
- rSPI_DR = SPI_4BYTEADD_PAGE_PROGRAM;
- rSPI_DR = tmpaddr[3];
- rSPI_DR = tmpaddr[2];
- rSPI_DR = tmpaddr[1];
- rSPI_DR = tmpaddr[0];
- } else {
- rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_PAGE_PROGRAM;
- }
-
- if (addr_in_4_byte) {
- for (i = 0; i < BYTESPERPAGE; i++) {
- while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
- rSPI_DR = *data++;
- }
- } else {
- for (i = 0; i < WORDSPERPAGE; i++) {
- while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
- rSPI_DR = *buf++;
- }
- }
- SpiWaitIdle();
- SetCSGpioEnable(0);
- while(SpiReadSta() & SPI_BUSY);
- }
- void bootFromSPI(void)
- {
- void (*funPtr)(void);
-
- SpiLoadStepldr(SpiReadPage);
-
- funPtr = (void (*)(void))STEPLDR_ENTRY;
- funPtr();
- }
- static unsigned int pagecheck[WORDSPERPAGE];
- /* offset is at least align to SECOTR_SIZE */
- static int SpiNorBurnPage(int pagenum, unsigned int *buf)
- {
- int timeout = 3;
- unsigned int *tmp = (unsigned int *)buf;
- int i;
- retry:
- SpiWritePage(pagenum, buf);
- SpiReadPage(pagenum, pagecheck);
- for (i = 0; i < WORDSPERPAGE; i++) {
- if (tmp[i] != pagecheck[i]) {
- if (timeout-- > 0) {
- PrintVariableValueHex("write: ", tmp[i]);
- PrintVariableValueHex("read: ", pagecheck[i]);
- SendUartString("burn check data fail, retry...\r\n");
- goto retry;
- } else {
- SendUartString("burn error!\r\n");
- return -1;
- }
- }
- }
- return 0;
- }
- /* offset is at least align to SECOTR_SIZE */
- int SpiNorBurn(void *buf, unsigned int offset, unsigned int size)
- {
- int i, j;
- int secstart, secnum;
- int blkstart, blknum;
- int pagestart, pageburned;
- int remain = size;
- unsigned int *pbuf = (unsigned int *)buf;
- pagestart = offset / BYTESPERPAGE;
- pageburned = 0;
- while (remain > 0) {
- unsigned int tmp = offset & (BYTESPERBLOCK - 1);
- if (tmp) {
- tmp = (BYTESPERBLOCK - tmp) / BYTESPERSECTOR;
- secnum = (remain + BYTESPERSECTOR - 1) / BYTESPERSECTOR;
- secnum = min(tmp, secnum);
- secstart = offset / BYTESPERSECTOR;
- for (i = 0; i < secnum; i++) {
- SpiEraseSector(secstart + i);
- for (j = 0; j < PAGESPERSECTORS; j++) {
- if (SpiNorBurnPage(pagestart + j, pbuf))
- return -1;
- pbuf += WORDSPERPAGE;
- pageburned++;
- remain -= BYTESPERPAGE;
- if (remain <= 0) goto finish;
- }
- pagestart += PAGESPERSECTORS;
- }
- offset += secnum * BYTESPERSECTOR;
- } else {
- blkstart = offset / BYTESPERBLOCK;
- blknum = remain / BYTESPERBLOCK;
- for (i = 0; i < blknum; i++) {
- SpiEraseBlock(blkstart + i);
- for (j = 0; j < PAGESPERSECTORS * SECTORSPERBLOCK; j++) {
- if (SpiNorBurnPage(pagestart + j, pbuf))
- return -1;
- pbuf += WORDSPERPAGE;
- pageburned++;
- remain -= BYTESPERPAGE;
- if (remain <= 0) goto finish;
- }
- pagestart += PAGESPERSECTORS * SECTORSPERBLOCK;
- }
- offset += blknum * BYTESPERBLOCK;
- if (remain > 0) {
- secstart = offset / BYTESPERSECTOR;
- secnum = (remain + BYTESPERSECTOR - 1) / BYTESPERSECTOR;
- for (i = 0; i < secnum; i++) {
- SpiEraseSector(secstart + i);
- for (j = 0; j < PAGESPERSECTORS; j++) {
- if (SpiNorBurnPage(pagestart + j, pbuf))
- return -1;
- pbuf += WORDSPERPAGE;
- pageburned++;
- remain -= BYTESPERPAGE;
- if (remain <= 0) goto finish;
- }
- pagestart += PAGESPERSECTORS;
- }
- offset += secnum * BYTESPERSECTOR;
- }
- }
- }
- finish:
- return 0;
- }
- int SpiReadSysInfo(SysInfo *info)
- {
- UINT32 *buf = (UINT32*)IMAGE_ENTRY;
- int pagestart;
- int pagenum;
- UINT32 checksum;
- UINT32 calc_checksum;
- int i;
- pagestart = SYSINFOA_OFFSET / BYTESPERPAGE;
- pagenum = (sizeof(SysInfo) + BYTESPERPAGE - 1) / BYTESPERPAGE;
- for(i = pagestart; i < pagestart + pagenum; i++)
- {
- SpiReadPage(i, buf);
- buf += BYTESPERPAGE/4;
- }
-
- checksum = *(UINT32*)(IMAGE_ENTRY + sizeof(SysInfo) - 4);
- calc_checksum = xcrc32((unsigned char*)IMAGE_ENTRY, sizeof(SysInfo) - 4, 0xffffffff);
- if (calc_checksum == checksum) {
- memcpy(info, (void*)IMAGE_ENTRY, sizeof(SysInfo));
- return 0;
- }
- buf = (UINT32*)IMAGE_ENTRY;
- pagestart = SYSINFOB_OFFSET / BYTESPERPAGE;
- for(i = pagestart; i < pagestart + pagenum; i++)
- {
- SpiReadPage(i, buf);
- buf += BYTESPERPAGE/4;
- }
-
- checksum = *(UINT32*)(IMAGE_ENTRY + sizeof(SysInfo) - 4);
- calc_checksum = xcrc32((unsigned char*)IMAGE_ENTRY, sizeof(SysInfo) - 4, 0xffffffff);
- if (calc_checksum == checksum) {
- memcpy(info, (void*)IMAGE_ENTRY, sizeof(SysInfo));
- return 0;
- }
-
- return -1;
- }
- void SpiWriteSysInfo(SysInfo *info)
- {
- SpiNorBurn(info, SYSINFOB_OFFSET, sizeof(SysInfo));
- SpiNorBurn(info, SYSINFOA_OFFSET, sizeof(SysInfo));
- }
|