blob: dd13fcee1e1d5a7d8117ec95daf9193ca04b56ff [file] [log] [blame]
Harald Welted8a003d2017-02-27 20:31:09 +01001#include "board.h"
2#include "utils.h"
3#include "usb/device/dfu/dfu.h"
4#include "usb/common/dfu/usb_dfu.h"
5#include "manifest.h"
6
Harald Welte32852bc2017-02-28 00:21:45 +01007#define ALTIF_RAM 0
8#define ALTIF_FLASH 1
9
Harald Welted8a003d2017-02-27 20:31:09 +010010unsigned int g_unique_id[4];
11
12/*----------------------------------------------------------------------------
13 * Callbacks
14 *----------------------------------------------------------------------------*/
15
Harald Welte32852bc2017-02-28 00:21:45 +010016#define RAM_ADDR(offset) (IRAM_ADDR + BOARD_DFU_RAM_SIZE + offset)
17#define FLASH_ADDR(offset) (IFLASH_ADDR + BOARD_DFU_BOOT_SIZE + offset)
18
19#define IFLASH_END ((uint8_t *)IFLASH_ADDR + IFLASH_SIZE)
20#define IRAM_END ((uint8_t *)IRAM_ADDR + IRAM_SIZE)
21
22/* incoming call-back: Host has transfered 'len' bytes (stored at
23 * 'data'), which we shall write to 'offset' into the partition
24 * associated with 'altif'. Guaranted to be les than
25 * BOARD_DFU_PAGE_SIZE */
26int USBDFU_handle_dnload(uint8_t altif, unsigned int offset,
27 uint8_t *data, unsigned int len)
Harald Welted8a003d2017-02-27 20:31:09 +010028{
Harald Welte32852bc2017-02-28 00:21:45 +010029 uint32_t addr;
30 int rc;
31
Harald Weltee8eea292017-03-03 00:35:51 +010032 printf("dnload(altif=%u, offset=%u, len=%u)\n\r", altif, offset, len);
Harald Welte32852bc2017-02-28 00:21:45 +010033
34 switch (altif) {
35 case ALTIF_RAM:
36 addr = RAM_ADDR(offset);
37 if (addr > IRAM_ADDR + IRAM_SIZE) {
Harald Welteadba0ce2017-02-28 01:02:24 +010038 g_dfu->state = DFU_STATE_dfuERROR;
39 g_dfu->status = DFU_STATUS_errADDRESS;
Harald Welte32852bc2017-02-28 00:21:45 +010040 return DFU_RET_STALL;
41 }
42 memcpy((void *)addr, data, len);
43 return DFU_RET_ZLP;
44 case ALTIF_FLASH:
45 addr = FLASH_ADDR(offset);
46 if (addr > IFLASH_ADDR + IFLASH_SIZE) {
Harald Welteadba0ce2017-02-28 01:02:24 +010047 g_dfu->state = DFU_STATE_dfuERROR;
48 g_dfu->status = DFU_STATUS_errADDRESS;
Harald Welte32852bc2017-02-28 00:21:45 +010049 return DFU_RET_STALL;
50 }
51 rc = FLASHD_Write(addr, data, len);
52 if (rc != 0) {
53 /* FIXME: set error codes */
54 return DFU_RET_STALL;
55 }
56 return DFU_RET_ZLP;
57 default:
58 /* FIXME: set error codes */
59 TRACE_ERROR("DFU download for unknown AltIf %d\n\r", altif);
60 return DFU_RET_STALL;
61 }
Harald Welted8a003d2017-02-27 20:31:09 +010062}
Harald Welte32852bc2017-02-28 00:21:45 +010063
64/* incoming call-back: Host has requested to read back 'req_len' bytes
65 * starting from 'offset' of the firmware * associated with partition
66 * 'altif' */
67int USBDFU_handle_upload(uint8_t altif, unsigned int offset,
68 uint8_t *data, unsigned int req_len)
69{
70 uint32_t addr;
71
72 printf("upload(altif=%u, offset=%u, len=%u)", altif, offset, req_len);
73
74 switch (altif) {
75 case ALTIF_RAM:
76 addr = RAM_ADDR(offset);
77 if (addr > IRAM_ADDR + IRAM_SIZE) {
Harald Welteadba0ce2017-02-28 01:02:24 +010078 g_dfu->state = DFU_STATE_dfuERROR;
79 g_dfu->status = DFU_STATUS_errADDRESS;
Harald Welte32852bc2017-02-28 00:21:45 +010080 return -1;
81 }
82 if ((uint8_t *)addr + req_len > IRAM_END)
83 req_len = IRAM_END - (uint8_t *)addr;
84 memcpy(data, (void *)addr, req_len);
85 break;
86 case ALTIF_FLASH:
87 addr = FLASH_ADDR(offset);
88 if (addr > IFLASH_ADDR + IFLASH_SIZE) {
Harald Welteadba0ce2017-02-28 01:02:24 +010089 g_dfu->state = DFU_STATE_dfuERROR;
90 g_dfu->status = DFU_STATUS_errADDRESS;
Harald Welte32852bc2017-02-28 00:21:45 +010091 return -1;
92 }
93 if ((uint8_t *)addr + req_len > IFLASH_END)
94 req_len = IFLASH_END - (uint8_t *)addr;
95 memcpy(data, (void *)addr, req_len);
96 break;
97 default:
98 TRACE_ERROR("DFU upload for unknown AltIf %d\n\r", altif);
99 /* FIXME: set error codes */
100 return -1;
101 }
Harald Welteec0837c2017-03-03 01:33:24 +0100102 printf("=%u\n\r", req_len);
Harald Welte32852bc2017-02-28 00:21:45 +0100103 return req_len;
104}
105
Harald Welted8a003d2017-02-27 20:31:09 +0100106
107/* returns '1' in case we should break any endless loop */
108static void check_exec_dbg_cmd(void)
109{
110 int ch;
111
112 if (!UART_IsRxReady())
113 return;
114
115 ch = UART_GetChar();
116
117 //board_exec_dbg_cmd(ch);
118}
119
120/*------------------------------------------------------------------------------
121 * Main
122 *------------------------------------------------------------------------------*/
123#define MAX_USB_ITER BOARD_MCK/72 // This should be around a second
124extern int main(void)
125{
126 uint8_t isUsbConnected = 0;
127 unsigned int i = 0;
128
129 LED_Configure(LED_NUM_RED);
130 LED_Configure(LED_NUM_GREEN);
131 LED_Set(LED_NUM_RED);
132
133 /* Disable watchdog */
134 WDT_Disable(WDT);
135
136 //req_ctx_init();
137
138 PIO_InitializeInterrupts(0);
139
140 EEFC_ReadUniqueID(g_unique_id);
141
Harald Welteec0837c2017-03-03 01:33:24 +0100142 printf("\n\r\n\r"
143 "=============================================================================\n\r"
144 "DFU bootloader %s for board %s (C) 2010-2017 by Harald Welte\n\r"
145 "=============================================================================\n\r",
Harald Welted8a003d2017-02-27 20:31:09 +0100146 manifest_revision, manifest_board);
147
Harald Welteec0837c2017-03-03 01:33:24 +0100148 TRACE_INFO("Chip ID: 0x%08x (Ext 0x%08x)\n\r", CHIPID->CHIPID_CIDR, CHIPID->CHIPID_EXID);
149 TRACE_INFO("Serial Nr. %08x-%08x-%08x-%08x\n\r",
Harald Welted8a003d2017-02-27 20:31:09 +0100150 g_unique_id[0], g_unique_id[1],
151 g_unique_id[2], g_unique_id[3]);
152
153 //board_main_top();
154
Harald Welteec0837c2017-03-03 01:33:24 +0100155 TRACE_INFO("USB init...\n\r");
Harald Welted8a003d2017-02-27 20:31:09 +0100156 USBDFU_Initialize(&dfu_descriptors);
157
158 while (USBD_GetState() < USBD_STATE_CONFIGURED) {
159 check_exec_dbg_cmd();
Harald Welte32852bc2017-02-28 00:21:45 +0100160#if 1
Harald Welted8a003d2017-02-27 20:31:09 +0100161 if (i >= MAX_USB_ITER * 3) {
162 TRACE_ERROR("Resetting board (USB could "
Harald Welteec0837c2017-03-03 01:33:24 +0100163 "not be configured)\n\r");
Harald Weltef415d712017-03-03 01:51:43 +0100164 USBD_Disconnect();
Harald Welted8a003d2017-02-27 20:31:09 +0100165 NVIC_SystemReset();
166 }
167#endif
168 i++;
169 }
170
Harald Welteec9b5ff2017-03-02 23:18:40 +0100171 FLASHD_Initialize(BOARD_MCK, 1);
Harald Welteec0837c2017-03-03 01:33:24 +0100172 TRACE_INFO("entering main loop...\n\r");
Harald Welted8a003d2017-02-27 20:31:09 +0100173 while (1) {
174#if TRACE_LEVEL >= TRACE_LEVEL_DEBUG
175 const char rotor[] = { '-', '\\', '|', '/' };
176 putchar('\b');
177 putchar(rotor[i++ % ARRAY_SIZE(rotor)]);
178#endif
179 check_exec_dbg_cmd();
180 //osmo_timers_prepare();
181 //osmo_timers_update();
182
183 if (USBD_GetState() < USBD_STATE_CONFIGURED) {
184
185 if (isUsbConnected) {
186 isUsbConnected = 0;
187 }
188 } else if (isUsbConnected == 0) {
Harald Welteec0837c2017-03-03 01:33:24 +0100189 TRACE_INFO("USB is now configured\n\r");
Harald Welted8a003d2017-02-27 20:31:09 +0100190 LED_Set(LED_NUM_GREEN);
191 LED_Clear(LED_NUM_RED);
192
193 isUsbConnected = 1;
194 }
195 }
196}