This port is meant to grow to encompass all existing boards. For now, it is a port while we transition over. It is named `zephyr-cp` to differentiate it from the MicroPython `zephyr` port. They are separate implementations.
235 lines
6 KiB
C
235 lines
6 KiB
C
// This file is part of the CircuitPython project: https://circuitpython.org
|
|
//
|
|
// SPDX-FileCopyrightText: Copyright (c) 2018 hathach for Adafruit Industries
|
|
//
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
#include "py/objstr.h"
|
|
#include "supervisor/background_callback.h"
|
|
#include "supervisor/linker.h"
|
|
#include "supervisor/shared/tick.h"
|
|
#include "supervisor/usb.h"
|
|
#include "shared/readline/readline.h"
|
|
|
|
#if CIRCUITPY_STATUS_BAR
|
|
#include "supervisor/shared/status_bar.h"
|
|
#endif
|
|
|
|
#if CIRCUITPY_STORAGE
|
|
#include "shared-module/storage/__init__.h"
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_DEVICE
|
|
#include "shared-bindings/supervisor/__init__.h"
|
|
|
|
#if CIRCUITPY_USB_CDC
|
|
#include "shared-module/usb_cdc/__init__.h"
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_HID
|
|
#include "shared-module/usb_hid/__init__.h"
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_MIDI
|
|
#include "shared-module/usb_midi/__init__.h"
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_VIDEO
|
|
#include "shared-module/usb_video/__init__.h"
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#include "tusb.h"
|
|
|
|
#if CFG_TUSB_OS == OPT_OS_ZEPHYR
|
|
#include <zephyr/kernel.h>
|
|
int CFG_TUSB_DEBUG_PRINTF(const char *format, ...) {
|
|
va_list args;
|
|
va_start(args, format);
|
|
printk(format, args);
|
|
va_end(args);
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CFG_TUSB_DEBUG
|
|
#define USBD_STACK_SIZE (5 * CONFIG_IDLE_STACK_SIZE)
|
|
#else
|
|
#define USBD_STACK_SIZE (5 * CONFIG_IDLE_STACK_SIZE / 2)
|
|
#endif
|
|
|
|
struct k_thread tinyusb_thread_data;
|
|
K_THREAD_STACK_DEFINE(tinyusb_thread_stack, USBD_STACK_SIZE);
|
|
k_tid_t _tinyusb_tid;
|
|
|
|
// USB Device Driver task
|
|
// This top level thread process all usb events and invoke callbacks
|
|
static void tinyusb_thread(void *unused0, void *unused1, void *unused2) {
|
|
(void)unused0;
|
|
(void)unused1;
|
|
(void)unused2;
|
|
|
|
// RTOS forever loop
|
|
while (1) {
|
|
// tinyusb device task
|
|
if (tusb_inited()) {
|
|
tud_task();
|
|
tud_cdc_write_flush();
|
|
} else {
|
|
k_sleep(K_MSEC(1000));
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
bool usb_enabled(void) {
|
|
return tusb_inited();
|
|
}
|
|
|
|
MP_WEAK void post_usb_init(void) {
|
|
}
|
|
|
|
void usb_init(void) {
|
|
#if CIRCUITPY_USB_DEVICE
|
|
usb_identification_t defaults;
|
|
usb_identification_t *identification;
|
|
#if CIRCUITPY_USB_IDENTIFICATION
|
|
if (custom_usb_identification != NULL) {
|
|
identification = custom_usb_identification;
|
|
} else {
|
|
#endif
|
|
// This compiles to less code than using a struct initializer.
|
|
defaults.vid = USB_VID;
|
|
defaults.pid = USB_PID;
|
|
strcpy(defaults.manufacturer_name, USB_MANUFACTURER);
|
|
strcpy(defaults.product_name, USB_PRODUCT);
|
|
identification = &defaults;
|
|
// This memory only needs to be live through the end of usb_build_descriptors.
|
|
#if CIRCUITPY_USB_IDENTIFICATION
|
|
}
|
|
#endif
|
|
if (!usb_build_descriptors(identification)) {
|
|
return;
|
|
}
|
|
init_usb_hardware();
|
|
#if CIRCUITPY_USB_HID
|
|
usb_hid_build_report_descriptor();
|
|
#endif
|
|
|
|
// Only init device. Host gets inited by the `usb_host` module common-hal.
|
|
tud_init(TUD_OPT_RHPORT);
|
|
#endif
|
|
|
|
post_usb_init();
|
|
|
|
#if MICROPY_KBD_EXCEPTION && CIRCUITPY_USB_DEVICE && CIRCUITPY_USB_CDC
|
|
// Set Ctrl+C as wanted char, tud_cdc_rx_wanted_cb() usb_callback will be invoked when Ctrl+C is received
|
|
// This usb_callback always got invoked regardless of mp_interrupt_char value since we only set it once here
|
|
|
|
// Don't watch for ctrl-C if there is no REPL.
|
|
if (usb_cdc_console_enabled()) {
|
|
// Console will always be itf 0.
|
|
tud_cdc_set_wanted_char(CHAR_CTRL_C);
|
|
}
|
|
#endif
|
|
|
|
#if CFG_TUSB_OS == OPT_OS_ZEPHYR
|
|
_tinyusb_tid = k_thread_create(&tinyusb_thread_data, tinyusb_thread_stack,
|
|
K_THREAD_STACK_SIZEOF(tinyusb_thread_stack),
|
|
tinyusb_thread,
|
|
NULL, NULL, NULL,
|
|
CONFIG_MAIN_THREAD_PRIORITY - 1, 0, K_NO_WAIT);
|
|
k_thread_name_set(_tinyusb_tid, "tinyusb");
|
|
#endif
|
|
}
|
|
|
|
// Set up USB defaults before any USB changes are made in boot.py
|
|
void usb_set_defaults(void) {
|
|
#if CIRCUITPY_USB_DEVICE
|
|
#if CIRCUITPY_STORAGE && CIRCUITPY_USB_MSC
|
|
storage_usb_set_defaults();
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_CDC
|
|
usb_cdc_set_defaults();
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_HID
|
|
usb_hid_set_defaults();
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_MIDI
|
|
usb_midi_set_defaults();
|
|
#endif
|
|
#endif
|
|
};
|
|
|
|
// Call this when ready to run code.py or a REPL, and a VM has been started.
|
|
void usb_setup_with_vm(void) {
|
|
#if CIRCUITPY_USB_DEVICE
|
|
#if CIRCUITPY_USB_HID
|
|
usb_hid_setup_devices();
|
|
#endif
|
|
|
|
#if CIRCUITPY_USB_MIDI
|
|
usb_midi_setup_ports();
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
void usb_background(void) {
|
|
if (usb_enabled()) {
|
|
#if CFG_TUSB_OS == OPT_OS_NONE || CFG_TUSB_OS == OPT_OS_PICO
|
|
tud_task();
|
|
#if CIRCUITPY_USB_HOST || CIRCUITPY_MAX3421E
|
|
tuh_task();
|
|
#endif
|
|
#elif CFG_TUSB_OS == OPT_OS_FREERTOS
|
|
// Yield to FreeRTOS in case TinyUSB runs in a separate task. Don't use
|
|
// port_yield() because it has a longer delay.
|
|
vTaskDelay(0);
|
|
#endif
|
|
// No need to flush if there's no REPL.
|
|
#if CIRCUITPY_USB_DEVICE && CIRCUITPY_USB_CDC
|
|
if (usb_cdc_console_enabled()) {
|
|
// Console will always be itf 0.
|
|
tud_cdc_write_flush();
|
|
}
|
|
#endif
|
|
#if CIRCUITPY_USB_DEVICE && CIRCUITPY_USB_VIDEO
|
|
usb_video_task();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
uint32_t tusb_time_millis_api(void) {
|
|
return supervisor_ticks_ms32();
|
|
}
|
|
|
|
static background_callback_t usb_callback;
|
|
static void usb_background_do(void *unused) {
|
|
usb_background();
|
|
}
|
|
|
|
void PLACE_IN_ITCM(usb_background_schedule)(void) {
|
|
background_callback_add(&usb_callback, usb_background_do, NULL);
|
|
}
|
|
|
|
void PLACE_IN_ITCM(usb_irq_handler)(int instance) {
|
|
#if CFG_TUSB_MCU != OPT_MCU_RP2040
|
|
#if CIRCUITPY_USB_DEVICE
|
|
// For rp2040, IRQ handler is already installed and invoked automatically
|
|
if (instance == CIRCUITPY_USB_DEVICE_INSTANCE) {
|
|
tud_int_handler(instance);
|
|
}
|
|
#endif
|
|
#if CIRCUITPY_USB_HOST
|
|
if (instance == CIRCUITPY_USB_HOST_INSTANCE) {
|
|
tuh_int_handler(instance);
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
usb_background_schedule();
|
|
}
|