spiflash.c 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. #define MODULE "spiflash"
  2. #define DEBUG 1
  3. #include "common.h"
  4. #include "spiflash.h"
  5. #include "spz.h"
  6. #include "fw.h"
  7. #include "jtag.h"
  8. /*
  9. * SPI flash parameters
  10. */
  11. #define JTAG_SPIFLASH_HZ 10000000 /* Max 26 MHz due to ESP32 */
  12. static const struct jtag_config jtag_config_spiflash = {
  13. .hz = JTAG_SPIFLASH_HZ,
  14. .pin_tms = 10, /* CS# */
  15. .pin_tdi = 12, /* MOSI */
  16. .pin_tdo = 13, /* MISO */
  17. .pin_tck = 11, /* SCLK */
  18. .be = true /* Bit order within bytes */
  19. };
  20. /*
  21. * Set up a command header with an address according to the SPI
  22. * addressing mode. Returns a pointer to the first byte past the
  23. * address.
  24. */
  25. static void *spiflash_setup_addrcmd(uint32_t addr,
  26. uint8_t cmd24, uint8_t cmd32,
  27. void *cmdbuf)
  28. {
  29. enum spiflash_addr_mode mode = SPIFLASH_ADDR_DYNAMIC;
  30. uint8_t *cmd = cmdbuf;
  31. if (!mode)
  32. mode = addr < (1 << 24) ? SPIFLASH_ADDR_24BIT : SPIFLASH_ADDR_32BIT;
  33. if (mode == SPIFLASH_ADDR_24BIT) {
  34. *cmd++ = cmd24;
  35. } else {
  36. *cmd++ = cmd32;
  37. *cmd++ = addr >> 24;
  38. }
  39. *cmd++ = addr >> 16;
  40. *cmd++ = addr >> 8;
  41. *cmd++ = addr;
  42. return cmd;
  43. }
  44. # define SHOW_COMMAND() \
  45. do { \
  46. MSG("command: ", cmdbuf, cmdlen); \
  47. for (size_t i = 0; i < cmdlen; i++) \
  48. CMSG(" %02x", ((const uint8_t *)cmdbuf)[i]); \
  49. CMSG("\n"); \
  50. } while(0)
  51. static int spiflash_simple_command(uint32_t cmd)
  52. {
  53. jtag_io(8, JIO_CS, &cmd, NULL);
  54. return 0;
  55. }
  56. static int spiflash_plain_command(const void *cmdbuf, size_t cmdlen)
  57. {
  58. #if DEBUG > 1
  59. MSG("plain: cmdbuf = %p (%zu), databuf = %p (%zu)\n",
  60. cmdbuf, cmdlen);
  61. SHOW_COMMAND();
  62. #endif
  63. jtag_io(cmdlen << 3, JIO_CS, cmdbuf, NULL);
  64. return 0;
  65. }
  66. static int spiflash_output_command(const void *cmdbuf, size_t cmdlen,
  67. const void *databuf, size_t datalen)
  68. {
  69. if (!datalen)
  70. return spiflash_plain_command(cmdbuf, cmdlen);
  71. #if DEBUG > 1
  72. MSG("output: cmdbuf = %p (%zu), databuf = %p (%zu)\n",
  73. cmdbuf, cmdlen, databuf, datalen);
  74. SHOW_COMMAND();
  75. #endif
  76. jtag_io(cmdlen << 3, 0, cmdbuf, NULL);
  77. jtag_io(datalen << 3, JIO_CS, databuf, NULL);
  78. return 0;
  79. }
  80. static int spiflash_input_command(const void *cmdbuf, size_t cmdlen,
  81. void *databuf, size_t datalen)
  82. {
  83. if (!datalen)
  84. return spiflash_plain_command(cmdbuf, cmdlen);
  85. #if DEBUG > 1
  86. MSG("input: cmdbuf = %p (%zu), databuf = %p (%zu)\n",
  87. cmdbuf, cmdlen, databuf, datalen);
  88. SHOW_COMMAND();
  89. #endif
  90. jtag_io(cmdlen << 3, 0, cmdbuf, NULL);
  91. jtag_io(datalen << 3, JIO_CS, NULL, databuf);
  92. return 0;
  93. }
  94. static int spiflash_read_status(uint32_t reg)
  95. {
  96. uint32_t val = 0;
  97. jtag_io(8, 0, (const uint8_t *)&reg, NULL);
  98. jtag_io(8, JIO_CS, NULL, (uint8_t *)&val);
  99. return val;
  100. }
  101. /* This needs a timeout function */
  102. static int spiflash_wait_status(uint8_t mask, uint8_t val)
  103. {
  104. unsigned int wait_loops = 100000;
  105. #if DEBUG > 1
  106. MSG("waiting for status %02x/%02x... ", mask, val);
  107. #endif
  108. while (wait_loops--) {
  109. uint8_t sr1 = spiflash_read_status(ROM_READ_SR1);
  110. if ((sr1 & mask) == val) {
  111. CMSG("ok\n");
  112. return 0;
  113. }
  114. yield();
  115. }
  116. CMSG("timeout\n");
  117. return -1;
  118. }
  119. static int spiflash_read(uint32_t addr, void *buffer, size_t len)
  120. {
  121. uint32_t cmdbuf[2];
  122. uint8_t *cmd = (uint8_t *)cmdbuf;
  123. const uint8_t cmd24 = ROM_FAST_READ;
  124. const uint8_t cmd32 = ROM_FAST_READ_32BIT;
  125. const size_t max_read_len = -1;
  126. int rv;
  127. while (len) {
  128. size_t clen = len;
  129. if (clen > max_read_len)
  130. clen = max_read_len;
  131. cmd = spiflash_setup_addrcmd(addr, cmd24, cmd32, cmdbuf);
  132. *cmd++ = 0; /* Dummy cycles */
  133. rv = spiflash_input_command(cmdbuf, cmd - (uint8_t *)cmdbuf,
  134. buffer, clen);
  135. if (rv)
  136. return rv;
  137. addr += clen;
  138. buffer = (uint8_t *)buffer + clen;
  139. len -= clen;
  140. }
  141. return 0;
  142. }
  143. static int spiflash_write_enable(void)
  144. {
  145. uint8_t sr1;
  146. int rv;
  147. rv = spiflash_wait_status(1, 0);
  148. if (rv)
  149. return rv;
  150. spiflash_simple_command(ROM_WRITE_ENABLE);
  151. return spiflash_wait_status(3, 2);
  152. }
  153. static int spiflash_program_sector(uint32_t addr, const void *buffer)
  154. {
  155. uint32_t cmdbuf[2];
  156. uint8_t *cmd = (uint8_t *)cmdbuf;
  157. const uint8_t cmd24 = ROM_PAGE_PROGRAM;
  158. const uint8_t cmd32 = ROM_PAGE_PROGRAM_32BIT;
  159. int rv;
  160. int loops = SPIFLASH_SECTOR_SIZE / SPIFLASH_PAGE_SIZE;
  161. const char *p = buffer;
  162. while (loops--) {
  163. rv = spiflash_write_enable();
  164. if (rv)
  165. return rv;
  166. cmd = spiflash_setup_addrcmd(addr, cmd24, cmd32, cmdbuf);
  167. spiflash_output_command(cmdbuf, cmd - (uint8_t *)cmdbuf,
  168. p, SPIFLASH_PAGE_SIZE);
  169. rv = spiflash_wait_status(3, 0);
  170. if (rv)
  171. return rv;
  172. addr += SPIFLASH_PAGE_SIZE;
  173. p += SPIFLASH_PAGE_SIZE;
  174. }
  175. return 0;
  176. }
  177. static int spiflash_erase_sector(uint32_t addr)
  178. {
  179. uint32_t cmdbuf[2];
  180. uint8_t *cmd = (uint8_t *)cmdbuf;
  181. const uint8_t cmd24 = ROM_ERASE_4K;
  182. const uint8_t cmd32 = ROM_ERASE_4K_32BIT;
  183. int rv;
  184. rv = spiflash_write_enable();
  185. if (rv)
  186. return rv;
  187. cmd = spiflash_setup_addrcmd(addr, cmd24, cmd32, cmdbuf);
  188. spiflash_plain_command(cmdbuf, cmd - (uint8_t *)cmdbuf);
  189. return spiflash_wait_status(3, 0);
  190. }
  191. /*
  192. * from: current flash contents
  193. * to: desired flash contents
  194. *
  195. * These are assumed to be aligned full block buffers
  196. */
  197. enum flashmem_status {
  198. FMS_DONE, /* All done, no programming needed */
  199. FMS_PROGRAM, /* Can be programmed */
  200. FMS_ERASE, /* Needs erase before programming */
  201. FMS_NOTCHECKED /* Not checked yet */
  202. };
  203. static enum flashmem_status
  204. spiflash_memcmp(const void *from, const void *to, size_t len)
  205. {
  206. const uint32_t *pf = from;
  207. const uint32_t *pt = to;
  208. const uint32_t *pfend = (const uint32_t *)((const char *)from + len);
  209. uint32_t doprog = 0;
  210. uint32_t doerase = 0;
  211. while (pf < pfend) {
  212. uint32_t f = *pf++;
  213. uint32_t t = *pt++;
  214. doprog += !!(f ^ t); /* Need programming if any data mismatch */
  215. doerase += !!(~f & t); /* Need erasing if any 0 -> 1 */
  216. }
  217. return doerase ? FMS_ERASE : doprog ? FMS_PROGRAM : FMS_DONE;
  218. }
  219. static int spiflash_write_sector(spz_stream *spz, unsigned int addr)
  220. {
  221. enum flashmem_status status = FMS_NOTCHECKED;
  222. MSG("flash sector at 0x%06x: ", addr);
  223. while (1) {
  224. enum flashmem_status oldstatus = status;
  225. status = spiflash_memcmp(spz->vbuf, spz->dbuf, SPIFLASH_SECTOR_SIZE);
  226. if (status >= oldstatus) {
  227. CMSG("X [%u>%u]", oldstatus, status);
  228. break;
  229. } else if (status == FMS_DONE) {
  230. CMSG("V");
  231. break;
  232. } else if (status == FMS_ERASE) {
  233. CMSG("E");
  234. if (spiflash_erase_sector(addr))
  235. break;
  236. } else if (status == FMS_PROGRAM) {
  237. CMSG("P");
  238. if (spiflash_program_sector(addr, spz->dbuf))
  239. break;
  240. }
  241. memset(spz->vbuf, 0xdd, SPIFLASH_SECTOR_SIZE);
  242. spiflash_read(addr, spz->vbuf, SPIFLASH_SECTOR_SIZE);
  243. }
  244. int rv;
  245. if (status == FMS_DONE) {
  246. CMSG(" OK\n");
  247. rv = 0;
  248. } else {
  249. CMSG(" FAILED\n");
  250. rv = (status == FMS_PROGRAM)
  251. ? FWUPDATE_ERR_PROGRAM_FAILED : FWUPDATE_ERR_ERASE_FAILED;
  252. }
  253. if (!spz->err)
  254. spz->err = rv;
  255. return rv;
  256. }
  257. static int spiflash_read_jedec_id(void)
  258. {
  259. const uint32_t cmd = ROM_JEDEC_ID;
  260. uint32_t jid = 0;
  261. spiflash_input_command((uint8_t *)&cmd, 1, (uint8_t *)&jid, 3);
  262. MSG("JEDEC ID: vendor %02x type %02x capacity %02x\n",
  263. (uint8_t)jid, (uint8_t)(jid >> 8), (uint8_t)(jid >> 16));
  264. return 0;
  265. }
  266. static void spiflash_show_status(void)
  267. {
  268. MSG("status regs: %02x %02x %02x\n",
  269. spiflash_read_status(ROM_READ_SR1),
  270. spiflash_read_status(ROM_READ_SR2),
  271. spiflash_read_status(ROM_READ_SR3));
  272. }
  273. #define PIN_FPGA_READY 9
  274. #define PIN_FPGA_BOARD_ID 1
  275. int spiflash_write_spz(spz_stream *spz)
  276. {
  277. unsigned int data_left = spz->header.len;
  278. unsigned int addr = spz->header.addr;
  279. esp_err_t rv;
  280. if (!data_left || spz->err)
  281. return spz->err;
  282. pinMode(PIN_FPGA_READY, INPUT);
  283. pinMode(PIN_FPGA_BOARD_ID, INPUT);
  284. MSG("waiting for FPGA bypass to be ready..");
  285. while (digitalRead(PIN_FPGA_READY) != LOW) {
  286. CMSG(".");
  287. yield();
  288. }
  289. CMSG("\n");
  290. MSG("FPGA bypass ready, board version v%c.\n",
  291. digitalRead(PIN_FPGA_BOARD_ID) ? '1' : '2');
  292. jtag_enable(&jtag_config_spiflash);
  293. spiflash_read_jedec_id();
  294. spiflash_show_status();
  295. while (data_left && !spz->err) {
  296. unsigned int pre_padding = addr & (SPIFLASH_SECTOR_SIZE-1);
  297. unsigned int post_padding;
  298. unsigned int bytes;
  299. bytes = SPIFLASH_SECTOR_SIZE - pre_padding;
  300. post_padding = 0;
  301. if (bytes > data_left) {
  302. post_padding = bytes - data_left;
  303. bytes = data_left;
  304. }
  305. MSG("sector at %06x bytes %u\n", addr, bytes);
  306. addr -= pre_padding;
  307. /* Read the current content of this block into vbuf */
  308. memset(spz->vbuf, 0xee, SPIFLASH_SECTOR_SIZE);
  309. rv = spiflash_read(addr, spz->vbuf, SPIFLASH_SECTOR_SIZE);
  310. if (rv)
  311. goto err;
  312. /* Copy any invariant chunk */
  313. if (pre_padding)
  314. memcpy(spz->dbuf, spz->vbuf, pre_padding);
  315. if (post_padding)
  316. memcpy(spz->dbuf+SPIFLASH_SECTOR_SIZE-post_padding,
  317. spz->vbuf+SPIFLASH_SECTOR_SIZE-post_padding,
  318. post_padding);
  319. rv = spz_read_data(spz, spz->dbuf+pre_padding, bytes);
  320. if (rv != (int)bytes) {
  321. MSG("needed %u bytes got %d\n", rv);
  322. rv = Z_DATA_ERROR;
  323. goto err;
  324. }
  325. MSG("writing new data...\n");
  326. rv = spiflash_write_sector(spz, addr);
  327. if (rv) {
  328. spz->err = rv;
  329. goto err;
  330. }
  331. addr += pre_padding + bytes;
  332. data_left -= bytes;
  333. }
  334. rv = 0;
  335. err:
  336. if (!spz->err)
  337. spz->err = rv;
  338. jtag_disable(NULL);
  339. return spz->err;
  340. }