SpiBooter.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686
  1. #include <string.h>
  2. #include "typedef.h"
  3. #include "amt630h.h"
  4. #include "UartPrint.h"
  5. #include "timer.h"
  6. #include "spi.h"
  7. #include "cp15.h"
  8. #include "sysinfo.h"
  9. #include "crc32.h"
  10. #include "gpio.h"
  11. #define SPI_CS_GPIO 32
  12. #define SPI_RXFIFO_FULL (1<<4)
  13. #define SPI_RXFIFO_NOTEMPTY (1<<3)
  14. #define SPI_TXFIFO_EMPTY (1<<2)
  15. #define SPI_TXFIFO_NOTFULL (1<<1)
  16. #define SPIFLASH_BUSY (1<<0)
  17. /* SFUD support manufacturer JEDEC ID */
  18. #define SFUD_MF_ID_CYPRESS 0x01
  19. #define SFUD_MF_ID_FUJITSU 0x04
  20. #define SFUD_MF_ID_EON 0x1C
  21. #define SFUD_MF_ID_ATMEL 0x1F
  22. #define SFUD_MF_ID_MICRON 0x20
  23. #define SFUD_MF_ID_AMIC 0x37
  24. #define SFUD_MF_ID_SANYO 0x62
  25. #define SFUD_MF_ID_INTEL 0x89
  26. #define SFUD_MF_ID_ESMT 0x8C
  27. #define SFUD_MF_ID_FUDAN 0xA1
  28. #define SFUD_MF_ID_HYUNDAI 0xAD
  29. #define SFUD_MF_ID_SST 0xBF
  30. #define SFUD_MF_ID_MICRONIX 0xC2
  31. #define SFUD_MF_ID_GIGADEVICE 0xC8
  32. #define SFUD_MF_ID_ISSI 0xD5
  33. #define SFUD_MF_ID_WINBOND 0xEF
  34. /* flash chip information */
  35. typedef struct {
  36. char *name; /**< flash chip name */
  37. uint8_t mf_id; /**< manufacturer ID */
  38. uint8_t type_id; /**< memory type ID */
  39. uint8_t capacity_id; /**< capacity ID */
  40. uint32_t capacity; /**< flash capacity (bytes) */
  41. } flash_chip;
  42. static const flash_chip flash_chip_table[] =
  43. { \
  44. {"W25Q40BV", SFUD_MF_ID_WINBOND, 0x40, 0x13, 512L*1024L}, \
  45. {"W25Q16BV", SFUD_MF_ID_WINBOND, 0x40, 0x15, 2L*1024L*1024L}, \
  46. {"W25Q32BV", SFUD_MF_ID_WINBOND, 0x40, 0x16, 4L*1024L*1024L}, \
  47. {"W25Q64CV", SFUD_MF_ID_WINBOND, 0x40, 0x17, 8L*1024L*1024L}, \
  48. {"W25Q64DW", SFUD_MF_ID_WINBOND, 0x60, 0x17, 8L*1024L*1024L}, \
  49. {"W25Q128BV", SFUD_MF_ID_WINBOND, 0x40, 0x18, 16L*1024L*1024L}, \
  50. {"W25Q256FV", SFUD_MF_ID_WINBOND, 0x40, 0x19, 32L*1024L*1024L}, \
  51. {"W25H256JV", SFUD_MF_ID_WINBOND, 0x90, 0x19, 32L*1024L*1024L}, \
  52. {"W25Q512JVFM", SFUD_MF_ID_WINBOND, 0x70, 0x20, 64L*1024L*1024L}, \
  53. {"SST25VF080B", SFUD_MF_ID_SST, 0x25, 0x8E, 1L*1024L*1024L}, \
  54. {"SST25VF016B", SFUD_MF_ID_SST, 0x25, 0x41, 2L*1024L*1024L}, \
  55. {"EN25Q32B", SFUD_MF_ID_EON, 0x30, 0x16, 4L*1024L*1024L}, \
  56. {"GD25Q64B", SFUD_MF_ID_GIGADEVICE, 0x40, 0x17, 8L*1024L*1024L}, \
  57. {"GD25Q16B", SFUD_MF_ID_GIGADEVICE, 0x40, 0x15, 2L*1024L*1024L}, \
  58. {"GD25Q32C", SFUD_MF_ID_GIGADEVICE, 0x40, 0x16, 4L*1024L*1024L}, \
  59. {"S25FL216K", SFUD_MF_ID_CYPRESS, 0x40, 0x15, 2L*1024L*1024L}, \
  60. {"S25FL032P", SFUD_MF_ID_CYPRESS, 0x02, 0x15, 4L*1024L*1024L}, \
  61. {"A25L080", SFUD_MF_ID_AMIC, 0x30, 0x14, 1L*1024L*1024L}, \
  62. {"F25L004", SFUD_MF_ID_ESMT, 0x20, 0x13, 512L*1024L}, \
  63. {"MX25L6433F", SFUD_MF_ID_MICRONIX, 0x20, 0x17, 8L*1024L*1024L}, \
  64. {"MX25L12845G", SFUD_MF_ID_MICRONIX, 0x20, 0x18, 16L*1024L*1024L}, \
  65. {"MX25L25645G", SFUD_MF_ID_MICRONIX, 0x20, 0x19, 32L*1024L*1024L},\
  66. {"PCT25VF016B", SFUD_MF_ID_SST, 0x25, 0x41, 2L*1024L*1024L}, \
  67. {"GD25B256", SFUD_MF_ID_GIGADEVICE, 0x43, 0x19, 32L*1024L*1024L},\
  68. {"GD25B512ME", SFUD_MF_ID_GIGADEVICE, 0x47, 0x1A, 64L*1024L*1024L},
  69. };
  70. static int addr_in_4_byte = 0;
  71. static void SetCSGpioEnable(int enable)
  72. {
  73. gpio_direction_output(SPI_CS_GPIO, !enable);
  74. }
  75. static void SetSpiDataMode(unsigned int bitMode)
  76. {
  77. unsigned int val = 0;
  78. (void)val;
  79. while((rSPI_SR & SPI_BUSY));
  80. rSPI_SSIENR = 0;
  81. val = rSPI_CTLR0;
  82. val &=~(0x1f<<16);
  83. val |=((bitMode-1)<<16);
  84. rSPI_CTLR0 = val;
  85. rSPI_SSIENR = 1;
  86. }
  87. static void SpiWaitIdle(void)
  88. {
  89. while(rSPI_SR & SPIFLASH_BUSY);
  90. udelay(2);
  91. }
  92. static void SpiEmptyRxFIFO(void)
  93. {
  94. INT32 data = 0;
  95. (void)data;
  96. while(rSPI_SR & SPI_RXFIFO_NOTEMPTY)
  97. data = rSPI_DR;
  98. }
  99. static UINT8 SpiReadSta(void)
  100. {
  101. UINT8 status;
  102. SetSpiDataMode(8);
  103. SetCSGpioEnable(1);
  104. rSPI_DR = SPI_READ_STATUS;
  105. rSPI_DR = 0;
  106. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  107. status = rSPI_DR;
  108. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  109. status = rSPI_DR;
  110. //PrintVariableValueHex("status:", status);
  111. SpiWaitIdle();
  112. SetCSGpioEnable(0);
  113. SetSpiDataMode(32);
  114. return status;
  115. }
  116. static void SpiDisable4ByteMode(void)
  117. {
  118. SetSpiDataMode(8);
  119. SetCSGpioEnable(1);
  120. rSPI_DR = SPI_DISABLE_4BYTE_MODE;
  121. while(!(rSPI_SR & SPI_TXFIFO_EMPTY));
  122. SpiWaitIdle();
  123. SetCSGpioEnable(0);
  124. SetSpiDataMode(32);
  125. }
  126. static void SpiEnable4ByteMode(void)
  127. {
  128. SetSpiDataMode(8);
  129. SetCSGpioEnable(1);
  130. rSPI_DR = SPI_ENABLE_4BYTE_MODE;
  131. while(!(rSPI_SR & SPI_TXFIFO_EMPTY));
  132. SpiWaitIdle();
  133. SetCSGpioEnable(0);
  134. SetSpiDataMode(32);
  135. }
  136. static void SpiReadPage(UINT32 pagenum, UINT32 *buf)
  137. {
  138. UINT32 addr;
  139. UINT32 val = 0;
  140. INT32 i, j;
  141. UINT8 tmpaddr[4];
  142. UINT8 *data = (UINT8*)buf;
  143. (void)val;
  144. addr = pagenum*BYTESPERPAGE;
  145. tmpaddr[0] = addr;
  146. tmpaddr[1] = addr>>8;
  147. tmpaddr[2] = addr>>16;
  148. tmpaddr[3] = addr>>24;
  149. SpiEmptyRxFIFO();
  150. SetCSGpioEnable(1);
  151. if (addr_in_4_byte) {
  152. SetSpiDataMode(8);
  153. rSPI_DR = SPI_4BYTEADDR_READ_DATA;
  154. rSPI_DR = tmpaddr[3];
  155. rSPI_DR = tmpaddr[2];
  156. rSPI_DR = tmpaddr[1];
  157. rSPI_DR = tmpaddr[0];
  158. for (i = 0; i < 5; i++) {
  159. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  160. val = rSPI_DR;
  161. }
  162. } else {
  163. rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_READ_DATA;
  164. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  165. val = rSPI_DR;
  166. }
  167. if (addr_in_4_byte) {
  168. for (i = 0; i < 4; i++) {
  169. for (j = 0; j < WORDSPERPAGE; j++) {
  170. while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
  171. rSPI_DR = 0;
  172. }
  173. for (j = 0; j < WORDSPERPAGE; j++) {
  174. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  175. *data++ = rSPI_DR;
  176. }
  177. }
  178. } else {
  179. for (i = 0; i < WORDSPERPAGE; i++) {
  180. while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
  181. rSPI_DR = 0;
  182. }
  183. for(i = 0; i < WORDSPERPAGE; i++) {
  184. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  185. *buf++ = rSPI_DR;
  186. }
  187. }
  188. SpiWaitIdle();
  189. SetCSGpioEnable(0);
  190. if (addr_in_4_byte)
  191. SetSpiDataMode(32);
  192. }
  193. void SpiSelectPad()
  194. {
  195. UINT32 val;
  196. val = rSYS_PAD_CTRL02;
  197. val &= ~0xfff;
  198. val |= (0x1<<10)|(0x1 << 8)|(0x1 << 6)|(0x1 << 4)| (0x1 << 2);
  199. rSYS_PAD_CTRL02 = val;
  200. val = rSYS_SSP_CLK_CFG;
  201. val &= ~((0x1<<31)|(0x1<<30));
  202. val |= (0x1<<31)|(0x1<<30);
  203. rSYS_SSP_CLK_CFG = val;
  204. //cs inactive first
  205. SetCSGpioEnable(0);
  206. }
  207. void Reset(void)
  208. {
  209. SetSpiDataMode(8);
  210. rSPI_DR = 0x66;
  211. while((SpiReadSta() & SPI_BUSY));
  212. // while(!(SpiReadSta() & SPIFLASH_WRITEENABLE));
  213. rSPI_DR = 0x99;
  214. while((SpiReadSta() & SPI_BUSY));
  215. // while(!(SpiReadSta() & SPIFLASH_WRITEENABLE));
  216. SetSpiDataMode(32);
  217. }
  218. #define SPI0_CS0_GPIO 32
  219. #define SPI0_IO0_GPIO 34
  220. static void dwspi_jedec252_reset(void)
  221. {
  222. int i;
  223. int si = 0;
  224. UINT32 val;
  225. val = rSYS_PAD_CTRL02;
  226. val &= ~((3 << 4) | 3);
  227. rSYS_PAD_CTRL02 = val;
  228. gpio_direction_output(SPI0_CS0_GPIO, 1);
  229. gpio_direction_output(SPI0_IO0_GPIO, 1);
  230. udelay(300);
  231. for (i = 0; i < 4; i++) {
  232. gpio_direction_output(SPI0_CS0_GPIO, 0);
  233. gpio_direction_output(SPI0_IO0_GPIO, si);
  234. si = !si;
  235. udelay(300);
  236. gpio_direction_output(SPI0_CS0_GPIO, 1);
  237. udelay(300);
  238. }
  239. }
  240. static void SpiReadDeviceId(UINT8 *mfid, UINT8 *devid)
  241. {
  242. UINT8 val[6];
  243. int i;
  244. SetSpiDataMode(8);
  245. SetCSGpioEnable(1);
  246. rSPI_DR = SPI_MF_DEVICE_ID;
  247. rSPI_DR = 0;
  248. rSPI_DR = 0;
  249. rSPI_DR = 0;
  250. rSPI_DR = 0;
  251. rSPI_DR = 0;
  252. for (i = 0; i < 6; i++)
  253. {
  254. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  255. val[i] = rSPI_DR;
  256. }
  257. *mfid = val[4];
  258. *devid = val[5];
  259. SpiWaitIdle();
  260. SetCSGpioEnable(0);
  261. SetSpiDataMode(32);
  262. }
  263. static void SpiReadJedecId(UINT8 *mfid, UINT8 *memid, UINT8 *capid)
  264. {
  265. UINT8 val[4];
  266. int i;
  267. SetSpiDataMode(8);
  268. SetCSGpioEnable(1);
  269. rSPI_DR = SPI_READ_JEDEC_ID;
  270. rSPI_DR = 0;
  271. rSPI_DR = 0;
  272. rSPI_DR = 0;
  273. for (i = 0; i < 4; i++)
  274. {
  275. while(!(rSPI_SR & SPI_RXFIFO_NOTEMPTY));
  276. val[i] = rSPI_DR;
  277. }
  278. *mfid = val[1];
  279. *memid = val[2];
  280. *capid = val[3];
  281. SpiWaitIdle();
  282. SetCSGpioEnable(0);
  283. SetSpiDataMode(32);
  284. }
  285. void SpiReadId(void)
  286. {
  287. UINT8 mfid,devid,memid,capid;
  288. SpiReadDeviceId(&mfid,&devid);
  289. SpiReadJedecId(&mfid,&memid,&capid);
  290. PrintVariableValueHex("ManufacturerID: ", mfid);
  291. PrintVariableValueHex("DeviceID: ", devid);
  292. PrintVariableValueHex("Memory Type ID: ", memid);
  293. PrintVariableValueHex("Capacity ID: ", capid);
  294. }
  295. void SpiInit()
  296. {
  297. uint8_t mfid, typeid, capid;
  298. unsigned int val;
  299. int i;
  300. dwspi_jedec252_reset();
  301. SpiSelectPad();
  302. val = rSYS_SOFT_RST;
  303. val &= ~(0x1<<8);
  304. rSYS_SOFT_RST = val;
  305. udelay(10);
  306. val |= (0x1<<8);
  307. rSYS_SOFT_RST = val;
  308. rSPI_SSIENR = 0;
  309. rSPI_CTLR0 = 0;
  310. rSPI_CTLR0 |=(0<<21)|(0x1f<<16)|(0x0<<12)|(0x0<<8)|(0x0<<4);
  311. // rSPI_CTLR1 = 63;
  312. rSPI_BAUDR = 4;//42;//16;//2;
  313. rSPI_SER = 1;
  314. rSPI_IMR = 0;
  315. rSPI_SSIENR = 1;
  316. // Reset();
  317. SpiReadJedecId(&mfid, &typeid, &capid);
  318. for (i = 0; i < sizeof(flash_chip_table) / sizeof(flash_chip); i++) {
  319. if ((flash_chip_table[i].mf_id == mfid)
  320. && (flash_chip_table[i].type_id == typeid)
  321. && (flash_chip_table[i].capacity_id == capid)) {
  322. if (flash_chip_table[i].capacity > 0x1000000) {
  323. addr_in_4_byte = 1;
  324. SpiEnable4ByteMode();
  325. }
  326. break;
  327. }
  328. }
  329. if (!addr_in_4_byte)
  330. SpiDisable4ByteMode();
  331. udelay(10000);
  332. }
  333. #define SPI_READ_MAXLEN BYTESPERPAGE
  334. static void SpiLoadStepldr(void (*readfunc)(UINT32, UINT32 *))
  335. {
  336. unsigned int i = 0;
  337. UINT32 *buf = (UINT32*)STEPLDR_ENTRY;
  338. UINT32 offset;
  339. UINT32 size;
  340. UINT32 nPageCount;
  341. UINT32 nPageStart;
  342. if (ReadSysInfo()) {
  343. SendUartString("read sysinfo fail, try to load stepldr part a.\n");
  344. offset = STEPLDRA_OFFSET;
  345. size = STEPLDR_MAX_SIZE;
  346. } else {
  347. SysInfo *sysinfo = GetSysInfo();
  348. offset = sysinfo->stepldr_offset;
  349. size = sysinfo->stepldr_size;
  350. }
  351. PrintVariableValueHex("stepldr offset: ", offset);
  352. nPageCount = (size + BYTESPERPAGE - 1) / BYTESPERPAGE;
  353. nPageStart = offset / BYTESPERPAGE;
  354. for(i = nPageStart; i < nPageStart + nPageCount; i++)
  355. {
  356. readfunc(i, buf);
  357. buf += BYTESPERPAGE/4;
  358. }
  359. #ifdef MMU_ENABLE
  360. CP15_clean_dcache_for_dma(STEPLDR_ENTRY, STEPLDR_ENTRY + size);
  361. #endif
  362. }
  363. static void SpiWriteEnable(void)
  364. {
  365. SetSpiDataMode(8);
  366. SetCSGpioEnable(1);
  367. rSPI_DR = SPI_WRITE_ENABLE;
  368. SpiWaitIdle();
  369. SetCSGpioEnable(0);
  370. while((SpiReadSta() & SPI_BUSY));
  371. while(!(SpiReadSta() & SPIFLASH_WRITEENABLE));
  372. SetSpiDataMode(32);
  373. }
  374. static void SpiEraseSector(UINT32 sectorNum)
  375. {
  376. UINT32 addr;
  377. UINT8 tmpaddr[4];
  378. addr = BYTESPERSECTOR*sectorNum;
  379. tmpaddr[0] = addr;
  380. tmpaddr[1] = addr>>8;
  381. tmpaddr[2] = addr>>16;
  382. tmpaddr[3] = addr>>24;
  383. SpiWriteEnable();
  384. SetCSGpioEnable(1);
  385. if (addr_in_4_byte) {
  386. SetSpiDataMode(8);
  387. rSPI_DR = SPI_4BYTEADD_SECTOR_ERASE;
  388. rSPI_DR = tmpaddr[3];
  389. rSPI_DR = tmpaddr[2];
  390. rSPI_DR = tmpaddr[1];
  391. rSPI_DR = tmpaddr[0];
  392. } else {
  393. rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_SECTOR_ERASE;
  394. }
  395. SpiWaitIdle();
  396. SetCSGpioEnable(0);
  397. while((SpiReadSta() & SPIFLASH_WRITEENABLE));
  398. }
  399. static void SpiEraseBlock(UINT32 blockNum)
  400. {
  401. UINT32 addr;
  402. UINT8 tmpaddr[4];
  403. addr = BYTESPERBLOCK*blockNum;
  404. tmpaddr[0] = addr;
  405. tmpaddr[1] = addr>>8;
  406. tmpaddr[2] = addr>>16;
  407. tmpaddr[3] = addr>>24;
  408. SpiWriteEnable();
  409. SetCSGpioEnable(1);
  410. if (addr_in_4_byte) {
  411. SetSpiDataMode(8);
  412. rSPI_DR = SPI_4BYTEADD_BLOCK_ERASE;
  413. rSPI_DR = tmpaddr[3];
  414. rSPI_DR = tmpaddr[2];
  415. rSPI_DR = tmpaddr[1];
  416. rSPI_DR = tmpaddr[0];
  417. } else {
  418. rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_BLOCK_ERASE;
  419. }
  420. SpiWaitIdle();
  421. SetCSGpioEnable(0);
  422. while((SpiReadSta() & SPIFLASH_WRITEENABLE));
  423. }
  424. static void SpiWritePage(UINT32 pagenum, UINT32 *buf)
  425. {
  426. UINT32 addr;
  427. UINT32 val = 0;;
  428. INT32 i;
  429. UINT8 tmpaddr[4];
  430. UINT8 *data = (UINT8*)buf;
  431. (void)val;
  432. addr = pagenum*BYTESPERPAGE;
  433. tmpaddr[0] = addr;
  434. tmpaddr[1] = addr>>8;
  435. tmpaddr[2] = addr>>16;
  436. tmpaddr[3] = addr>>24;
  437. SpiWriteEnable();
  438. SetCSGpioEnable(1);
  439. if (addr_in_4_byte) {
  440. SetSpiDataMode(8);
  441. rSPI_DR = SPI_4BYTEADD_PAGE_PROGRAM;
  442. rSPI_DR = tmpaddr[3];
  443. rSPI_DR = tmpaddr[2];
  444. rSPI_DR = tmpaddr[1];
  445. rSPI_DR = tmpaddr[0];
  446. } else {
  447. rSPI_DR = (tmpaddr[0]<<24) | (tmpaddr[1]<<16) | (tmpaddr[2]<<8) | SPI_PAGE_PROGRAM;
  448. }
  449. if (addr_in_4_byte) {
  450. for (i = 0; i < BYTESPERPAGE; i++) {
  451. while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
  452. rSPI_DR = *data++;
  453. }
  454. } else {
  455. for (i = 0; i < WORDSPERPAGE; i++) {
  456. while(!(rSPI_SR & SPI_TXFIFO_NOTFULL));
  457. rSPI_DR = *buf++;
  458. }
  459. }
  460. SpiWaitIdle();
  461. SetCSGpioEnable(0);
  462. while(SpiReadSta() & SPI_BUSY);
  463. }
  464. void bootFromSPI(void)
  465. {
  466. void (*funPtr)(void);
  467. SpiLoadStepldr(SpiReadPage);
  468. funPtr = (void (*)(void))STEPLDR_ENTRY;
  469. funPtr();
  470. }
  471. static unsigned int pagecheck[WORDSPERPAGE];
  472. /* offset is at least align to SECOTR_SIZE */
  473. static int SpiNorBurnPage(int pagenum, unsigned int *buf)
  474. {
  475. int timeout = 3;
  476. unsigned int *tmp = (unsigned int *)buf;
  477. int i;
  478. retry:
  479. SpiWritePage(pagenum, buf);
  480. SpiReadPage(pagenum, pagecheck);
  481. for (i = 0; i < WORDSPERPAGE; i++) {
  482. if (tmp[i] != pagecheck[i]) {
  483. if (timeout-- > 0) {
  484. PrintVariableValueHex("write: ", tmp[i]);
  485. PrintVariableValueHex("read: ", pagecheck[i]);
  486. SendUartString("burn check data fail, retry...\r\n");
  487. goto retry;
  488. } else {
  489. SendUartString("burn error!\r\n");
  490. return -1;
  491. }
  492. }
  493. }
  494. return 0;
  495. }
  496. /* offset is at least align to SECOTR_SIZE */
  497. int SpiNorBurn(void *buf, unsigned int offset, unsigned int size)
  498. {
  499. int i, j;
  500. int secstart, secnum;
  501. int blkstart, blknum;
  502. int pagestart, pageburned;
  503. int remain = size;
  504. unsigned int *pbuf = (unsigned int *)buf;
  505. pagestart = offset / BYTESPERPAGE;
  506. pageburned = 0;
  507. while (remain > 0) {
  508. unsigned int tmp = offset & (BYTESPERBLOCK - 1);
  509. if (tmp) {
  510. tmp = (BYTESPERBLOCK - tmp) / BYTESPERSECTOR;
  511. secnum = (remain + BYTESPERSECTOR - 1) / BYTESPERSECTOR;
  512. secnum = min(tmp, secnum);
  513. secstart = offset / BYTESPERSECTOR;
  514. for (i = 0; i < secnum; i++) {
  515. SpiEraseSector(secstart + i);
  516. for (j = 0; j < PAGESPERSECTORS; j++) {
  517. if (SpiNorBurnPage(pagestart + j, pbuf))
  518. return -1;
  519. pbuf += WORDSPERPAGE;
  520. pageburned++;
  521. remain -= BYTESPERPAGE;
  522. if (remain <= 0) goto finish;
  523. }
  524. pagestart += PAGESPERSECTORS;
  525. }
  526. offset += secnum * BYTESPERSECTOR;
  527. } else {
  528. blkstart = offset / BYTESPERBLOCK;
  529. blknum = remain / BYTESPERBLOCK;
  530. for (i = 0; i < blknum; i++) {
  531. SpiEraseBlock(blkstart + i);
  532. for (j = 0; j < PAGESPERSECTORS * SECTORSPERBLOCK; j++) {
  533. if (SpiNorBurnPage(pagestart + j, pbuf))
  534. return -1;
  535. pbuf += WORDSPERPAGE;
  536. pageburned++;
  537. remain -= BYTESPERPAGE;
  538. if (remain <= 0) goto finish;
  539. }
  540. pagestart += PAGESPERSECTORS * SECTORSPERBLOCK;
  541. }
  542. offset += blknum * BYTESPERBLOCK;
  543. if (remain > 0) {
  544. secstart = offset / BYTESPERSECTOR;
  545. secnum = (remain + BYTESPERSECTOR - 1) / BYTESPERSECTOR;
  546. for (i = 0; i < secnum; i++) {
  547. SpiEraseSector(secstart + i);
  548. for (j = 0; j < PAGESPERSECTORS; j++) {
  549. if (SpiNorBurnPage(pagestart + j, pbuf))
  550. return -1;
  551. pbuf += WORDSPERPAGE;
  552. pageburned++;
  553. remain -= BYTESPERPAGE;
  554. if (remain <= 0) goto finish;
  555. }
  556. pagestart += PAGESPERSECTORS;
  557. }
  558. offset += secnum * BYTESPERSECTOR;
  559. }
  560. }
  561. }
  562. finish:
  563. return 0;
  564. }
  565. int SpiReadSysInfo(SysInfo *info)
  566. {
  567. UINT32 *buf = (UINT32*)IMAGE_ENTRY;
  568. int pagestart;
  569. int pagenum;
  570. UINT32 checksum;
  571. UINT32 calc_checksum;
  572. int i;
  573. pagestart = SYSINFOA_OFFSET / BYTESPERPAGE;
  574. pagenum = (sizeof(SysInfo) + BYTESPERPAGE - 1) / BYTESPERPAGE;
  575. for(i = pagestart; i < pagestart + pagenum; i++)
  576. {
  577. SpiReadPage(i, buf);
  578. buf += BYTESPERPAGE/4;
  579. }
  580. checksum = *(UINT32*)(IMAGE_ENTRY + sizeof(SysInfo) - 4);
  581. calc_checksum = xcrc32((unsigned char*)IMAGE_ENTRY, sizeof(SysInfo) - 4, 0xffffffff);
  582. if (calc_checksum == checksum) {
  583. memcpy(info, (void*)IMAGE_ENTRY, sizeof(SysInfo));
  584. return 0;
  585. }
  586. buf = (UINT32*)IMAGE_ENTRY;
  587. pagestart = SYSINFOB_OFFSET / BYTESPERPAGE;
  588. for(i = pagestart; i < pagestart + pagenum; i++)
  589. {
  590. SpiReadPage(i, buf);
  591. buf += BYTESPERPAGE/4;
  592. }
  593. checksum = *(UINT32*)(IMAGE_ENTRY + sizeof(SysInfo) - 4);
  594. calc_checksum = xcrc32((unsigned char*)IMAGE_ENTRY, sizeof(SysInfo) - 4, 0xffffffff);
  595. if (calc_checksum == checksum) {
  596. memcpy(info, (void*)IMAGE_ENTRY, sizeof(SysInfo));
  597. return 0;
  598. }
  599. return -1;
  600. }
  601. void SpiWriteSysInfo(SysInfo *info)
  602. {
  603. SpiNorBurn(info, SYSINFOB_OFFSET, sizeof(SysInfo));
  604. SpiNorBurn(info, SYSINFOA_OFFSET, sizeof(SysInfo));
  605. }