1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-17 23:12:45 +01:00
RIOT/pkg/tinyusb/dfu/tinyusb_dfu.c

154 lines
4.1 KiB
C

/*
* Copyright (C) 2022 Gunar Schorcht
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup pkg_tinyusb_dfu
* @{
* @file TinyUSB DFU implementation
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @}
*/
#include "log.h"
#include "periph/pm.h"
#include "riotboot/magic.h"
#include "riotboot/slot.h"
#ifdef MODULE_RIOTBOOT_TINYUSB_DFU
#include "ztimer.h"
#endif
#include "device/usbd.h"
#include "class/dfu/dfu_device.h"
#include "tinyusb_dfu.h"
#define ENABLE_DEBUG 0
#include "debug.h"
#ifdef MODULE_RIOTBOOT_TINYUSB_DFU
static void _reboot(void *arg);
static ztimer_t scheduled_reboot = { .callback=_reboot };
#endif
/* there is only one instance of DFU devices */
tinyusb_dfu_device_t _tusb_dfu_dev = { .skip_signature = true };
void tinyusb_dfu_init(void)
{
_tusb_dfu_dev.skip_signature = true;
_tusb_dfu_dev.slot = 0;
}
#ifdef MODULE_RIOTBOOT_TINYUSB_DFU
uint32_t tud_dfu_get_timeout_cb(uint8_t alt, uint8_t state)
{
(void)alt;
(void)state;
/* Invoked before tud_dfu_download_cb() or tud_dfu_manifest_cb() is called
* to determine the poll timeout for download and manifest operation */
return CONFIG_TUSBD_DFU_POLL_TIMEOUT;
}
void tud_dfu_download_cb(uint8_t alt, uint16_t block_num, uint8_t const *data, uint16_t length)
{
/* Invoked when a DFU_DNLOAD (wLength > 0) followed by a DFU_GETSTATUS
* (state = DFU_DNBUSY) requests was received. */
(void)block_num;
int ret = 0;
if (_tusb_dfu_dev.skip_signature) {
/* Avoid underflow condition */
if (length < RIOTBOOT_FLASHWRITE_SKIPLEN) {
return;
}
riotboot_flashwrite_init(&_tusb_dfu_dev.writer, alt);
length -= RIOTBOOT_FLASHWRITE_SKIPLEN;
DEBUG("[tinyusb_dfu] starting the download with %u bytes for slot %u "
"with block %u\n", length, alt, block_num);
_tusb_dfu_dev.skip_signature = false;
_tusb_dfu_dev.slot = alt;
ret = riotboot_flashwrite_putbytes(&_tusb_dfu_dev.writer,
&data[RIOTBOOT_FLASHWRITE_SKIPLEN],
length, true);
}
else {
assert(alt == _tusb_dfu_dev.slot);
DEBUG("[tinyusb_dfu] continue the download with %u bytes for slot %u "
"with block %u\n", length, alt, block_num);
ret = riotboot_flashwrite_putbytes(&_tusb_dfu_dev.writer,
data, length, true);
}
if (ret < 0) {
LOG_ERROR("[tinyusb_dfu] error on writing %d bytes for slot %u\n",
length, alt);
tud_dfu_finish_flashing(DFU_STATUS_ERR_WRITE);
}
else {
tud_dfu_finish_flashing(DFU_STATUS_OK);
}
}
void tud_dfu_manifest_cb(uint8_t alt)
{
/* Invoked when the download process is complete and
* DFU_DNLOAD (wLength = 0) followed by a DFU_GETSTATUS (state = Manifest)
* was received. */
(void)alt;
assert(alt == _tusb_dfu_dev.slot);
DEBUG("[tinyusb_dfu] download for slot %u complete, "
"enter manifestation phase\n", alt);
/* the host indicates that the download process is complete */
riotboot_flashwrite_flush(&_tusb_dfu_dev.writer);
riotboot_flashwrite_finish(&_tusb_dfu_dev.writer);
/* indicate that flashing is finished */
tud_dfu_finish_flashing(DFU_STATUS_OK);
/* scheduled reboot after CONFIG_TUSBD_DFU_RESET_DELAY seconds to give
* enough time to finish manifestation */
ztimer_set(ZTIMER_SEC, &scheduled_reboot, CONFIG_TUSBD_DFU_RESET_DELAY);
}
static void _reboot(void *arg)
{
DEBUG("[tinyusb_dfu] reboot\n");
(void)arg;
pm_reboot();
}
#endif /* MODULE_RIOTBOOT_TINYUSB_DFU */
TU_ATTR_WEAK void tud_dfu_detach_cb(void)
{
/* the host sent a DFU_DETACH request */
DEBUG("[tinyusb_dfu] DFU_DETACH request received\n");
uint32_t *reset_addr = (uint32_t *)RIOTBOOT_MAGIC_ADDR;
*reset_addr = RIOTBOOT_MAGIC_NUMBER;
pm_reboot();
}
void tud_dfu_runtime_reboot_to_dfu_cb(void)
{
/* the host sent a DFU_DETACH request */
tud_dfu_detach_cb();
}