You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

main.c 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680
  1. /**
  2. * Copyright (c) 2022 Brian Starkey <stark3y@gmail.com>
  3. *
  4. * Based on the Pico W tcp_server example:
  5. * Copyright (c) 2022 Raspberry Pi (Trading) Ltd.
  6. *
  7. * SPDX-License-Identifier: BSD-3-Clause
  8. */
  9. #include <string.h>
  10. #include <stdlib.h>
  11. #include "RP2040.h"
  12. #include "pico/critical_section.h"
  13. #include "pico/time.h"
  14. #include "pico/util/queue.h"
  15. #include "hardware/dma.h"
  16. #include "hardware/flash.h"
  17. #include "hardware/structs/dma.h"
  18. #include "hardware/structs/watchdog.h"
  19. #include "hardware/gpio.h"
  20. #include "hardware/resets.h"
  21. #include "hardware/uart.h"
  22. #include "hardware/watchdog.h"
  23. #include "pico/stdlib.h"
  24. #include "pico/cyw43_arch.h"
  25. #include "tcp_comm.h"
  26. #include "picowota/reboot.h"
  27. #ifdef DEBUG
  28. #include <stdio.h>
  29. #include "pico/stdio_usb.h"
  30. #define DBG_PRINTF_INIT() stdio_usb_init()
  31. #define DBG_PRINTF(...) printf(__VA_ARGS__)
  32. #else
  33. #define DBG_PRINTF_INIT() { }
  34. #define DBG_PRINTF(...) { }
  35. #endif
  36. #if PICOWOTA_WIFI_AP == 1
  37. #include "dhcpserver.h"
  38. static dhcp_server_t dhcp_server;
  39. #endif
  40. #define QUOTE(name) #name
  41. #define STR(macro) QUOTE(macro)
  42. #ifndef PICOWOTA_WIFI_SSID
  43. #warning "PICOWOTA_WIFI_SSID not defined"
  44. #else
  45. const char *wifi_ssid = STR(PICOWOTA_WIFI_SSID);
  46. #endif
  47. #ifndef PICOWOTA_WIFI_PASS
  48. #warning "PICOWOTA_WIFI_PASS not defined"
  49. #else
  50. const char *wifi_pass = STR(PICOWOTA_WIFI_PASS);
  51. #endif
  52. critical_section_t critical_section;
  53. #define EVENT_QUEUE_LENGTH 8
  54. queue_t event_queue;
  55. enum event_type {
  56. EVENT_TYPE_REBOOT = 1,
  57. EVENT_TYPE_GO,
  58. EVENT_TYPE_SERVER_DONE,
  59. };
  60. struct event {
  61. enum event_type type;
  62. union {
  63. struct {
  64. bool to_bootloader;
  65. } reboot;
  66. struct {
  67. uint32_t vtor;
  68. } go;
  69. };
  70. };
  71. #define BOOTLOADER_ENTRY_PIN 15
  72. #define TCP_PORT 4242
  73. struct image_header app_image_header;
  74. #define IMAGE_HEADER_OFFSET ((uint32_t)&app_image_header - XIP_BASE)
  75. #define WRITE_ADDR_MIN (XIP_BASE + IMAGE_HEADER_OFFSET + FLASH_SECTOR_SIZE)
  76. #define ERASE_ADDR_MIN (XIP_BASE + IMAGE_HEADER_OFFSET)
  77. #define FLASH_ADDR_MAX (XIP_BASE + PICO_FLASH_SIZE_BYTES)
  78. #define CMD_SYNC (('S' << 0) | ('Y' << 8) | ('N' << 16) | ('C' << 24))
  79. #define RSP_SYNC (('W' << 0) | ('O' << 8) | ('T' << 16) | ('A' << 24))
  80. #define CMD_INFO (('I' << 0) | ('N' << 8) | ('F' << 16) | ('O' << 24))
  81. #define CMD_READ (('R' << 0) | ('E' << 8) | ('A' << 16) | ('D' << 24))
  82. #define CMD_CSUM (('C' << 0) | ('S' << 8) | ('U' << 16) | ('M' << 24))
  83. #define CMD_CRC (('C' << 0) | ('R' << 8) | ('C' << 16) | ('C' << 24))
  84. #define CMD_ERASE (('E' << 0) | ('R' << 8) | ('A' << 16) | ('S' << 24))
  85. #define CMD_WRITE (('W' << 0) | ('R' << 8) | ('I' << 16) | ('T' << 24))
  86. #define CMD_SEAL (('S' << 0) | ('E' << 8) | ('A' << 16) | ('L' << 24))
  87. #define CMD_GO (('G' << 0) | ('O' << 8) | ('G' << 16) | ('O' << 24))
  88. #define CMD_REBOOT (('B' << 0) | ('O' << 8) | ('O' << 16) | ('T' << 24))
  89. static uint32_t handle_sync(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  90. {
  91. return RSP_SYNC;
  92. }
  93. const struct comm_command sync_cmd = {
  94. .opcode = CMD_SYNC,
  95. .nargs = 0,
  96. .resp_nargs = 0,
  97. .size = NULL,
  98. .handle = &handle_sync,
  99. };
  100. static uint32_t size_read(uint32_t *args_in, uint32_t *data_len_out, uint32_t *resp_data_len_out)
  101. {
  102. uint32_t size = args_in[1];
  103. if (size > TCP_COMM_MAX_DATA_LEN) {
  104. return TCP_COMM_RSP_ERR;
  105. }
  106. // TODO: Validate address
  107. *data_len_out = 0;
  108. *resp_data_len_out = size;
  109. return TCP_COMM_RSP_OK;
  110. }
  111. static uint32_t handle_read(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  112. {
  113. uint32_t addr = args_in[0];
  114. uint32_t size = args_in[1];
  115. memcpy(resp_data_out, (void *)addr, size);
  116. return TCP_COMM_RSP_OK;
  117. }
  118. const struct comm_command read_cmd = {
  119. // READ addr len
  120. // OKOK [data]
  121. .opcode = CMD_READ,
  122. .nargs = 2,
  123. .resp_nargs = 0,
  124. .size = &size_read,
  125. .handle = &handle_read,
  126. };
  127. static uint32_t size_csum(uint32_t *args_in, uint32_t *data_len_out, uint32_t *resp_data_len_out)
  128. {
  129. uint32_t addr = args_in[0];
  130. uint32_t size = args_in[1];
  131. if ((addr & 0x3) || (size & 0x3)) {
  132. // Must be aligned
  133. return TCP_COMM_RSP_ERR;
  134. }
  135. // TODO: Validate address
  136. *data_len_out = 0;
  137. *resp_data_len_out = 0;
  138. return TCP_COMM_RSP_OK;
  139. }
  140. static uint32_t handle_csum(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  141. {
  142. uint32_t dummy_dest;
  143. uint32_t addr = args_in[0];
  144. uint32_t size = args_in[1];
  145. int channel = dma_claim_unused_channel(true);
  146. dma_channel_config c = dma_channel_get_default_config(channel);
  147. channel_config_set_transfer_data_size(&c, DMA_SIZE_32);
  148. channel_config_set_read_increment(&c, true);
  149. channel_config_set_write_increment(&c, false);
  150. channel_config_set_sniff_enable(&c, true);
  151. dma_hw->sniff_data = 0;
  152. dma_sniffer_enable(channel, 0xf, true);
  153. dma_channel_configure(channel, &c, &dummy_dest, (void *)addr, size / 4, true);
  154. dma_channel_wait_for_finish_blocking(channel);
  155. dma_sniffer_disable();
  156. dma_channel_unclaim(channel);
  157. *resp_args_out = dma_hw->sniff_data;
  158. return TCP_COMM_RSP_OK;
  159. }
  160. struct comm_command csum_cmd = {
  161. // CSUM addr len
  162. // OKOK csum
  163. .opcode = CMD_CSUM,
  164. .nargs = 2,
  165. .resp_nargs = 1,
  166. .size = &size_csum,
  167. .handle = &handle_csum,
  168. };
  169. static uint32_t size_crc(uint32_t *args_in, uint32_t *data_len_out, uint32_t *resp_data_len_out)
  170. {
  171. uint32_t addr = args_in[0];
  172. uint32_t size = args_in[1];
  173. if ((addr & 0x3) || (size & 0x3)) {
  174. // Must be aligned
  175. return TCP_COMM_RSP_ERR;
  176. }
  177. // TODO: Validate address
  178. *data_len_out = 0;
  179. *resp_data_len_out = 0;
  180. return TCP_COMM_RSP_OK;
  181. }
  182. // ptr must be 4-byte aligned and len must be a multiple of 4
  183. static uint32_t calc_crc32(void *ptr, uint32_t len)
  184. {
  185. uint32_t dummy_dest, crc;
  186. int channel = dma_claim_unused_channel(true);
  187. dma_channel_config c = dma_channel_get_default_config(channel);
  188. channel_config_set_transfer_data_size(&c, DMA_SIZE_32);
  189. channel_config_set_read_increment(&c, true);
  190. channel_config_set_write_increment(&c, false);
  191. channel_config_set_sniff_enable(&c, true);
  192. // Seed the CRC calculation
  193. dma_hw->sniff_data = 0xffffffff;
  194. // Mode 1, then bit-reverse the result gives the same result as
  195. // golang's IEEE802.3 implementation
  196. dma_sniffer_enable(channel, 0x1, true);
  197. dma_hw->sniff_ctrl |= DMA_SNIFF_CTRL_OUT_REV_BITS;
  198. dma_channel_configure(channel, &c, &dummy_dest, ptr, len / 4, true);
  199. dma_channel_wait_for_finish_blocking(channel);
  200. // Read the result before resetting
  201. crc = dma_hw->sniff_data ^ 0xffffffff;
  202. dma_sniffer_disable();
  203. dma_channel_unclaim(channel);
  204. return crc;
  205. }
  206. static uint32_t handle_crc(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  207. {
  208. uint32_t addr = args_in[0];
  209. uint32_t size = args_in[1];
  210. resp_args_out[0] = calc_crc32((void *)addr, size);
  211. return TCP_COMM_RSP_OK;
  212. }
  213. struct comm_command crc_cmd = {
  214. // CRCC addr len
  215. // OKOK crc
  216. .opcode = CMD_CRC,
  217. .nargs = 2,
  218. .resp_nargs = 1,
  219. .size = &size_crc,
  220. .handle = &handle_crc,
  221. };
  222. static uint32_t handle_erase(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  223. {
  224. uint32_t addr = args_in[0];
  225. uint32_t size = args_in[1];
  226. if ((addr < ERASE_ADDR_MIN) || (addr + size >= FLASH_ADDR_MAX)) {
  227. // Outside flash
  228. return TCP_COMM_RSP_ERR;
  229. }
  230. if ((addr & (FLASH_SECTOR_SIZE - 1)) || (size & (FLASH_SECTOR_SIZE - 1))) {
  231. // Must be aligned
  232. return TCP_COMM_RSP_ERR;
  233. }
  234. critical_section_enter_blocking(&critical_section);
  235. flash_range_erase(addr - XIP_BASE, size);
  236. critical_section_exit(&critical_section);
  237. return TCP_COMM_RSP_OK;
  238. }
  239. struct comm_command erase_cmd = {
  240. // ERAS addr len
  241. // OKOK
  242. .opcode = CMD_ERASE,
  243. .nargs = 2,
  244. .resp_nargs = 0,
  245. .size = NULL,
  246. .handle = &handle_erase,
  247. };
  248. static uint32_t size_write(uint32_t *args_in, uint32_t *data_len_out, uint32_t *resp_data_len_out)
  249. {
  250. uint32_t addr = args_in[0];
  251. uint32_t size = args_in[1];
  252. if ((addr < WRITE_ADDR_MIN) || (addr + size >= FLASH_ADDR_MAX)) {
  253. // Outside flash
  254. return TCP_COMM_RSP_ERR;
  255. }
  256. if ((addr & (FLASH_PAGE_SIZE - 1)) || (size & (FLASH_PAGE_SIZE -1))) {
  257. // Must be aligned
  258. return TCP_COMM_RSP_ERR;
  259. }
  260. if (size > TCP_COMM_MAX_DATA_LEN) {
  261. return TCP_COMM_RSP_ERR;
  262. }
  263. // TODO: Validate address
  264. *data_len_out = size;
  265. *resp_data_len_out = 0;
  266. return TCP_COMM_RSP_OK;
  267. }
  268. static uint32_t handle_write(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  269. {
  270. uint32_t addr = args_in[0];
  271. uint32_t size = args_in[1];
  272. critical_section_enter_blocking(&critical_section);
  273. flash_range_program(addr - XIP_BASE, data_in, size);
  274. critical_section_exit(&critical_section);
  275. resp_args_out[0] = calc_crc32((void *)addr, size);
  276. return TCP_COMM_RSP_OK;
  277. }
  278. struct comm_command write_cmd = {
  279. // WRIT addr len [data]
  280. // OKOK crc
  281. .opcode = CMD_WRITE,
  282. .nargs = 2,
  283. .resp_nargs = 1,
  284. .size = &size_write,
  285. .handle = &handle_write,
  286. };
  287. struct image_header {
  288. uint32_t vtor;
  289. uint32_t size;
  290. uint32_t crc;
  291. uint8_t pad[FLASH_PAGE_SIZE - (3 * 4)];
  292. };
  293. static_assert(sizeof(struct image_header) == FLASH_PAGE_SIZE, "image_header must be FLASH_PAGE_SIZE bytes");
  294. static bool image_header_ok(struct image_header *hdr)
  295. {
  296. uint32_t *vtor = (uint32_t *)hdr->vtor;
  297. uint32_t calc = calc_crc32((void *)hdr->vtor, hdr->size);
  298. // CRC has to match
  299. if (calc != hdr->crc) {
  300. return false;
  301. }
  302. // Stack pointer needs to be in RAM
  303. if (vtor[0] < SRAM_BASE) {
  304. return false;
  305. }
  306. // Reset vector should be in the image, and thumb (bit 0 set)
  307. if ((vtor[1] < hdr->vtor) || (vtor[1] > hdr->vtor + hdr->size) || !(vtor[1] & 1)) {
  308. return false;
  309. }
  310. // Looks OK.
  311. return true;
  312. }
  313. static uint32_t handle_seal(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  314. {
  315. struct image_header hdr = {
  316. .vtor = args_in[0],
  317. .size = args_in[1],
  318. .crc = args_in[2],
  319. };
  320. if ((hdr.vtor & 0xff) || (hdr.size & 0x3)) {
  321. // Must be aligned
  322. return TCP_COMM_RSP_ERR;
  323. }
  324. if (!image_header_ok(&hdr)) {
  325. return TCP_COMM_RSP_ERR;
  326. }
  327. critical_section_enter_blocking(&critical_section);
  328. flash_range_erase(IMAGE_HEADER_OFFSET, FLASH_SECTOR_SIZE);
  329. flash_range_program(IMAGE_HEADER_OFFSET, (const uint8_t *)&hdr, sizeof(hdr));
  330. critical_section_exit(&critical_section);
  331. struct image_header *check = &app_image_header;
  332. if (memcmp(&hdr, check, sizeof(hdr))) {
  333. return TCP_COMM_RSP_ERR;
  334. }
  335. return TCP_COMM_RSP_OK;
  336. }
  337. struct comm_command seal_cmd = {
  338. // SEAL vtor len crc
  339. // OKOK
  340. .opcode = CMD_SEAL,
  341. .nargs = 3,
  342. .resp_nargs = 0,
  343. .size = NULL,
  344. .handle = &handle_seal,
  345. };
  346. static void disable_interrupts(void)
  347. {
  348. SysTick->CTRL &= ~1;
  349. NVIC->ICER[0] = 0xFFFFFFFF;
  350. NVIC->ICPR[0] = 0xFFFFFFFF;
  351. }
  352. static void reset_peripherals(void)
  353. {
  354. reset_block(~(
  355. RESETS_RESET_IO_QSPI_BITS |
  356. RESETS_RESET_PADS_QSPI_BITS |
  357. RESETS_RESET_SYSCFG_BITS |
  358. RESETS_RESET_PLL_SYS_BITS
  359. ));
  360. }
  361. static void jump_to_vtor(uint32_t vtor)
  362. {
  363. // Derived from the Leaf Labs Cortex-M3 bootloader.
  364. // Copyright (c) 2010 LeafLabs LLC.
  365. // Modified 2021 Brian Starkey <stark3y@gmail.com>
  366. // Originally under The MIT License
  367. uint32_t reset_vector = *(volatile uint32_t *)(vtor + 0x04);
  368. SCB->VTOR = (volatile uint32_t)(vtor);
  369. asm volatile("msr msp, %0"::"g"
  370. (*(volatile uint32_t *)vtor));
  371. asm volatile("bx %0"::"r" (reset_vector));
  372. }
  373. static uint32_t handle_go(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  374. {
  375. struct event ev = {
  376. .type = EVENT_TYPE_GO,
  377. .go = {
  378. .vtor = args_in[0],
  379. },
  380. };
  381. if (!queue_try_add(&event_queue, &ev)) {
  382. return TCP_COMM_RSP_ERR;
  383. }
  384. return TCP_COMM_RSP_OK;
  385. }
  386. struct comm_command go_cmd = {
  387. // GOGO vtor
  388. // NO RESPONSE
  389. .opcode = CMD_GO,
  390. .nargs = 1,
  391. .resp_nargs = 0,
  392. .size = NULL,
  393. .handle = &handle_go,
  394. };
  395. static uint32_t handle_info(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  396. {
  397. resp_args_out[0] = WRITE_ADDR_MIN;
  398. resp_args_out[1] = (XIP_BASE + PICO_FLASH_SIZE_BYTES) - WRITE_ADDR_MIN;
  399. resp_args_out[2] = FLASH_SECTOR_SIZE;
  400. resp_args_out[3] = FLASH_PAGE_SIZE;
  401. resp_args_out[4] = TCP_COMM_MAX_DATA_LEN;
  402. return TCP_COMM_RSP_OK;
  403. }
  404. const struct comm_command info_cmd = {
  405. // INFO
  406. // OKOK flash_start flash_size erase_size write_size max_data_len
  407. .opcode = CMD_INFO,
  408. .nargs = 0,
  409. .resp_nargs = 5,
  410. .size = NULL,
  411. .handle = &handle_info,
  412. };
  413. static uint32_t size_reboot(uint32_t *args_in, uint32_t *data_len_out, uint32_t *resp_data_len_out)
  414. {
  415. *data_len_out = 0;
  416. *resp_data_len_out = 0;
  417. return TCP_COMM_RSP_OK;
  418. }
  419. static uint32_t handle_reboot(uint32_t *args_in, uint8_t *data_in, uint32_t *resp_args_out, uint8_t *resp_data_out)
  420. {
  421. struct event ev = {
  422. .type = EVENT_TYPE_REBOOT,
  423. .reboot = {
  424. .to_bootloader = !!args_in[0],
  425. },
  426. };
  427. if (!queue_try_add(&event_queue, &ev)) {
  428. return TCP_COMM_RSP_ERR;
  429. }
  430. return TCP_COMM_RSP_OK;
  431. }
  432. struct comm_command reboot_cmd = {
  433. // BOOT to_bootloader
  434. // NO RESPONSE
  435. .opcode = CMD_REBOOT,
  436. .nargs = 1,
  437. .resp_nargs = 0,
  438. .size = &size_reboot,
  439. .handle = &handle_reboot,
  440. };
  441. static bool should_stay_in_bootloader()
  442. {
  443. bool wd_says_so = (watchdog_hw->scratch[5] == PICOWOTA_BOOTLOADER_ENTRY_MAGIC) &&
  444. (watchdog_hw->scratch[6] == ~PICOWOTA_BOOTLOADER_ENTRY_MAGIC);
  445. return !gpio_get(BOOTLOADER_ENTRY_PIN) || wd_says_so;
  446. }
  447. static void network_deinit()
  448. {
  449. #if PICOWOTA_WIFI_AP == 1
  450. dhcp_server_deinit(&dhcp_server);
  451. #endif
  452. cyw43_arch_deinit();
  453. }
  454. int main()
  455. {
  456. err_t err;
  457. gpio_init(BOOTLOADER_ENTRY_PIN);
  458. gpio_pull_up(BOOTLOADER_ENTRY_PIN);
  459. gpio_set_dir(BOOTLOADER_ENTRY_PIN, 0);
  460. sleep_ms(10);
  461. struct image_header *hdr = (struct image_header *)(XIP_BASE + IMAGE_HEADER_OFFSET);
  462. if (!should_stay_in_bootloader() && image_header_ok(hdr)) {
  463. uint32_t vtor = *((uint32_t *)(XIP_BASE + IMAGE_HEADER_OFFSET));
  464. disable_interrupts();
  465. reset_peripherals();
  466. jump_to_vtor(vtor);
  467. }
  468. DBG_PRINTF_INIT();
  469. queue_init(&event_queue, sizeof(struct event), EVENT_QUEUE_LENGTH);
  470. if (cyw43_arch_init()) {
  471. DBG_PRINTF("failed to initialise\n");
  472. return 1;
  473. }
  474. #if PICOWOTA_WIFI_AP == 1
  475. cyw43_arch_enable_ap_mode(wifi_ssid, wifi_pass, CYW43_AUTH_WPA2_AES_PSK);
  476. DBG_PRINTF("Enabled the WiFi AP.\n");
  477. ip4_addr_t gw, mask;
  478. IP4_ADDR(&gw, 192, 168, 4, 1);
  479. IP4_ADDR(&mask, 255, 255, 255, 0);
  480. dhcp_server_t dhcp_server;
  481. dhcp_server_init(&dhcp_server, &gw, &mask);
  482. DBG_PRINTF("Started the DHCP server.\n");
  483. #else
  484. cyw43_arch_enable_sta_mode();
  485. DBG_PRINTF("Connecting to WiFi...\n");
  486. if (cyw43_arch_wifi_connect_timeout_ms(wifi_ssid, wifi_pass, CYW43_AUTH_WPA2_AES_PSK, 30000)) {
  487. DBG_PRINTF("failed to connect.\n");
  488. return 1;
  489. } else {
  490. DBG_PRINTF("Connected.\n");
  491. }
  492. #endif
  493. critical_section_init(&critical_section);
  494. const struct comm_command *cmds[] = {
  495. &sync_cmd,
  496. &read_cmd,
  497. &csum_cmd,
  498. &crc_cmd,
  499. &erase_cmd,
  500. &write_cmd,
  501. &seal_cmd,
  502. &go_cmd,
  503. &info_cmd,
  504. &reboot_cmd,
  505. };
  506. struct tcp_comm_ctx *tcp = tcp_comm_new(cmds, sizeof(cmds) / sizeof(cmds[0]), CMD_SYNC);
  507. struct event ev = {
  508. .type = EVENT_TYPE_SERVER_DONE,
  509. };
  510. queue_add_blocking(&event_queue, &ev);
  511. for ( ; ; ) {
  512. while (queue_try_remove(&event_queue, &ev)) {
  513. switch (ev.type) {
  514. case EVENT_TYPE_SERVER_DONE:
  515. err = tcp_comm_listen(tcp, TCP_PORT);
  516. if (err != ERR_OK) {
  517. DBG_PRINTF("Failed to start server: %d\n", err);
  518. }
  519. break;
  520. case EVENT_TYPE_REBOOT:
  521. tcp_comm_server_close(tcp);
  522. network_deinit();
  523. picowota_reboot(ev.reboot.to_bootloader);
  524. /* Should never get here */
  525. break;
  526. case EVENT_TYPE_GO:
  527. tcp_comm_server_close(tcp);
  528. network_deinit();
  529. disable_interrupts();
  530. reset_peripherals();
  531. jump_to_vtor(ev.go.vtor);
  532. /* Should never get here */
  533. break;
  534. };
  535. }
  536. cyw43_arch_poll();
  537. sleep_ms(5);
  538. }
  539. network_deinit();
  540. return 0;
  541. }