1 /* Copyright (c) 2011,2012 Simon Schubert <2@0x2c.org>.
2 * Modifications by Jacob Alexander 2014-2015 <haata@kiibohd.com>
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 // ----- Local Includes -----
26 // ----- Functions -----
28 void dfu_write_done( enum dfu_status err, struct dfu_ctx *ctx )
31 if (ctx->status == DFU_STATUS_OK) {
33 case DFU_STATE_dfuDNBUSY:
34 ctx->state = DFU_STATE_dfuDNLOAD_IDLE;
40 ctx->state = DFU_STATE_dfuERROR;
44 static void dfu_dnload_complete( void *buf, ssize_t len, void *cbdata )
46 struct dfu_ctx *ctx = cbdata;
49 ctx->state = DFU_STATE_dfuDNBUSY;
51 ctx->state = DFU_STATE_dfuMANIFEST;
52 ctx->status = ctx->finish_write(buf, ctx->off, len);
56 if (ctx->status != DFU_STATUS_async)
57 dfu_write_done(ctx->status, ctx);
59 usb_handle_control_status(ctx->state == DFU_STATE_dfuERROR);
62 static void dfu_reset_system( void *buf, ssize_t len, void *cbdata )
67 static int dfu_handle_control( struct usb_ctrl_req_t *req, void *data )
69 struct dfu_ctx *ctx = data;
72 switch ((enum dfu_ctrl_req_code)req->bRequest)
74 case USB_CTRL_REQ_DFU_DNLOAD: {
78 case DFU_STATE_dfuIDLE:
81 case DFU_STATE_dfuDNLOAD_IDLE:
88 * XXX we are not allowed to STALL here, and we need to eat all transferred data.
89 * better not allow setup_write to break the protocol.
91 ctx->status = ctx->setup_write(ctx->off, req->wLength, &buf);
92 if (ctx->status != DFU_STATUS_OK) {
93 ctx->state = DFU_STATE_dfuERROR;
98 usb_ep0_rx(buf, req->wLength, dfu_dnload_complete, ctx);
100 dfu_dnload_complete(NULL, 0, ctx);
103 case USB_CTRL_REQ_DFU_UPLOAD: {
107 switch ( ctx->state )
109 case DFU_STATE_dfuIDLE:
111 case DFU_STATE_dfuUPLOAD_IDLE:
117 // Find which sector to read
118 ctx->status = ctx->setup_read(ctx->off, &len, &buf);
119 print("UPLOAD off:");
120 printHex( ctx->off );
124 printHex( (uint32_t)buf );
127 if ( ctx->status != DFU_STATUS_OK || len > req->wLength )
129 ctx->state = DFU_STATE_dfuERROR;
130 goto err_have_status;
133 // Send bytes to Host
136 usb_ep0_rx( buf, len, NULL, NULL );
140 ctx->state = DFU_STATE_dfuIDLE;
145 case USB_CTRL_REQ_DFU_GETSTATUS: {
146 struct dfu_status_t st;
148 st.bState = ctx->state;
149 st.bStatus = ctx->status;
150 st.bwPollTimeout = 1000; /* XXX */
153 if ( ctx->state == DFU_STATE_dfuMANIFEST )
155 uint8_t data[] = { 0x10, 0x20, 0x30, 0x40 };
156 flash_program_longword((uintptr_t)&_app_rom, data);
160 uint32_t *position = &_app_rom + 0x100;
161 for ( ; position < &_app_rom + 0x200; position++ )
162 //for ( ; position < &_app_rom + 0x800; position++ )
164 if ( *position != 0xFFFFFFFF )
168 GPIOA_PTOR |= (1<<5);
169 for (uint32_t d = 0; d < 7200000; d++ );
174 // Check to see if vector table was flashed correctly
175 // Return a flash error if it was not
176 if (_app_rom == 0xffffffff && ctx->state == DFU_STATE_dfuMANIFEST)
177 st.bStatus = DFU_STATUS_errPROG;
180 if (ctx->state == DFU_STATE_dfuMANIFEST)
182 uint8_t *addr = (uint8_t*)_app_rom;
183 //while (*(addr++) != 0x80);
184 //st.bStatus = DFU_STATUS_errPROG;
185 st.bStatus = (uint8_t)((uint32_t)(&_app_rom) >> 16);
190 * If we're in DFU_STATE_dfuMANIFEST, we just finished
191 * the download, and we're just about to send our last
192 * status report. Once the report has been sent, go
193 * and reset the system to put the new firmware into
196 usb_ep0_tx_cp(&st, sizeof(st), req->wLength, NULL, NULL);
197 if (ctx->state == DFU_STATE_dfuMANIFEST) {
198 usb_handle_control_status_cb(dfu_reset_system);
203 case USB_CTRL_REQ_DFU_CLRSTATUS:
204 ctx->state = DFU_STATE_dfuIDLE;
205 ctx->status = DFU_STATUS_OK;
207 case USB_CTRL_REQ_DFU_GETSTATE: {
208 uint8_t st = ctx->state;
209 usb_ep0_tx_cp(&st, sizeof(st), req->wLength, NULL, NULL);
212 case USB_CTRL_REQ_DFU_ABORT:
213 switch (ctx->state) {
214 case DFU_STATE_dfuIDLE:
215 case DFU_STATE_dfuDNLOAD_IDLE:
216 case DFU_STATE_dfuUPLOAD_IDLE:
217 ctx->state = DFU_STATE_dfuIDLE;
231 ctx->status = DFU_STATUS_errSTALLEDPKT;
233 ctx->state = DFU_STATE_dfuERROR;
235 usb_handle_control_status(fail);
240 void dfu_init( dfu_setup_read_t setup_read, dfu_setup_write_t setup_write, dfu_finish_write_t finish_write, struct dfu_ctx *ctx )
242 ctx->state = DFU_STATE_dfuIDLE;
243 ctx->setup_read = setup_read;
244 ctx->setup_write = setup_write;
245 ctx->finish_write = finish_write;
246 usb_attach_function(&dfu_function, &ctx->header);
249 const struct usbd_function dfu_function = {
250 .control = dfu_handle_control,
251 .interface_count = USB_FUNCTION_DFU_IFACE_COUNT,