2023-03-17 16:12:49 +07:00
|
|
|
/*
|
2019-12-22 18:18:54 -06:00
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2019 Ha Thach (tinyusb.org)
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2025-09-26 16:45:54 +08:00
|
|
|
#include <rtthread.h>
|
|
|
|
#include "bsp_init.h"
|
|
|
|
#include "stm32f4xx.h"
|
|
|
|
#include "dcd.h"
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
|
2019-12-22 18:18:54 -06:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
2023-08-03 15:44:05 +07:00
|
|
|
#include "bsp/board_api.h"
|
2019-12-22 18:18:54 -06:00
|
|
|
#include "tusb.h"
|
|
|
|
|
2024-03-11 22:00:21 +07:00
|
|
|
/* Blink pattern
|
|
|
|
* - 250 ms : device not mounted
|
|
|
|
* - 1000 ms : device mounted
|
|
|
|
* - 2500 ms : device is suspended
|
|
|
|
*/
|
|
|
|
enum {
|
|
|
|
BLINK_NOT_MOUNTED = 250,
|
|
|
|
BLINK_MOUNTED = 1000,
|
|
|
|
BLINK_SUSPENDED = 2500,
|
|
|
|
};
|
|
|
|
|
|
|
|
static uint32_t blink_interval_ms = BLINK_NOT_MOUNTED;
|
|
|
|
|
|
|
|
static void led_blinking_task(void);
|
2019-12-22 18:18:54 -06:00
|
|
|
static void cdc_task(void);
|
|
|
|
|
2025-09-26 16:45:54 +08:00
|
|
|
|
|
|
|
|
|
|
|
int tusb_board_init(void) {
|
|
|
|
GPIO_InitTypeDef GPIO_InitStructure;
|
|
|
|
|
|
|
|
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);
|
|
|
|
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_OTG_HS, ENABLE);
|
|
|
|
|
|
|
|
GPIO_PinAFConfig(GPIOB, GPIO_PinSource14, GPIO_AF_OTG2_FS);
|
|
|
|
GPIO_PinAFConfig(GPIOB, GPIO_PinSource15, GPIO_AF_OTG2_FS);
|
|
|
|
|
|
|
|
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
|
|
|
|
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
|
|
|
|
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14 | GPIO_Pin_15;
|
|
|
|
GPIO_Init(GPIOB, &GPIO_InitStructure);
|
|
|
|
|
|
|
|
// no need enable irq, it will be enabled in dwc2_dcd_int_enable()
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void usb_irq_enable(){
|
|
|
|
NVIC_InitTypeDef NVIC_InitStructure;
|
|
|
|
|
|
|
|
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannel = OTG_HS_IRQn;
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
|
|
|
|
NVIC_Init(&NVIC_InitStructure);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tusb_thread_entry(void *parameter)
|
|
|
|
{
|
|
|
|
(void) parameter;
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
#if CFG_TUH_ENABLED
|
|
|
|
tuh_task();
|
|
|
|
#endif
|
|
|
|
#if CFG_TUD_ENABLED
|
|
|
|
tud_task();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-12-22 18:18:54 -06:00
|
|
|
/*------------- MAIN -------------*/
|
2025-09-26 16:45:54 +08:00
|
|
|
int main_task(void) {
|
2019-12-22 18:18:54 -06:00
|
|
|
board_init();
|
|
|
|
|
2022-06-06 22:41:04 +07:00
|
|
|
// init device stack on configured roothub port
|
2024-10-11 12:58:18 +07:00
|
|
|
tusb_rhport_init_t dev_init = {
|
|
|
|
.role = TUSB_ROLE_DEVICE,
|
|
|
|
.speed = TUSB_SPEED_AUTO
|
|
|
|
};
|
2024-10-14 18:27:52 +07:00
|
|
|
tusb_init(BOARD_TUD_RHPORT, &dev_init);
|
2019-12-22 18:18:54 -06:00
|
|
|
|
2023-07-31 19:09:40 +07:00
|
|
|
if (board_init_after_tusb) {
|
|
|
|
board_init_after_tusb();
|
|
|
|
}
|
|
|
|
|
2024-03-11 22:00:21 +07:00
|
|
|
while (1) {
|
2019-12-22 18:18:54 -06:00
|
|
|
tud_task(); // tinyusb device task
|
|
|
|
cdc_task();
|
2024-03-11 22:00:21 +07:00
|
|
|
led_blinking_task();
|
2019-12-22 18:18:54 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-11 12:59:02 +07:00
|
|
|
// echo to either Serial0 or Serial1
|
|
|
|
// with Serial0 as all lower case, Serial1 as all upper case
|
2024-03-11 22:00:21 +07:00
|
|
|
static void echo_serial_port(uint8_t itf, uint8_t buf[], uint32_t count) {
|
2022-06-28 16:27:44 +07:00
|
|
|
uint8_t const case_diff = 'a' - 'A';
|
|
|
|
|
2024-03-11 22:00:21 +07:00
|
|
|
for (uint32_t i = 0; i < count; i++) {
|
|
|
|
if (itf == 0) {
|
2019-12-22 18:18:54 -06:00
|
|
|
// echo back 1st port as lower case
|
2022-06-28 16:27:44 +07:00
|
|
|
if (isupper(buf[i])) buf[i] += case_diff;
|
2024-03-11 22:00:21 +07:00
|
|
|
} else {
|
2021-03-18 15:23:08 +07:00
|
|
|
// echo back 2nd port as upper case
|
2022-06-28 16:27:44 +07:00
|
|
|
if (islower(buf[i])) buf[i] -= case_diff;
|
2019-12-22 18:18:54 -06:00
|
|
|
}
|
2020-01-11 12:59:02 +07:00
|
|
|
|
2019-12-22 18:18:54 -06:00
|
|
|
tud_cdc_n_write_char(itf, buf[i]);
|
|
|
|
}
|
|
|
|
tud_cdc_n_write_flush(itf);
|
|
|
|
}
|
|
|
|
|
2024-03-11 22:00:21 +07:00
|
|
|
// Invoked when device is mounted
|
|
|
|
void tud_mount_cb(void) {
|
|
|
|
blink_interval_ms = BLINK_MOUNTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Invoked when device is unmounted
|
|
|
|
void tud_umount_cb(void) {
|
|
|
|
blink_interval_ms = BLINK_NOT_MOUNTED;
|
|
|
|
}
|
|
|
|
|
2019-12-22 18:18:54 -06:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// USB CDC
|
|
|
|
//--------------------------------------------------------------------+
|
2024-03-11 22:00:21 +07:00
|
|
|
static void cdc_task(void) {
|
2025-07-03 13:42:05 +07:00
|
|
|
for (uint8_t itf = 0; itf < CFG_TUD_CDC; itf++) {
|
2020-11-24 00:06:59 +07:00
|
|
|
// connected() check for DTR bit
|
|
|
|
// Most but not all terminal client set this when making connection
|
2021-03-18 15:23:08 +07:00
|
|
|
// if ( tud_cdc_n_connected(itf) )
|
2019-12-22 18:18:54 -06:00
|
|
|
{
|
2024-03-11 22:00:21 +07:00
|
|
|
if (tud_cdc_n_available(itf)) {
|
2020-11-18 09:47:39 +01:00
|
|
|
uint8_t buf[64];
|
|
|
|
uint32_t count = tud_cdc_n_read(itf, buf, sizeof(buf));
|
2019-12-22 18:18:54 -06:00
|
|
|
|
2020-11-18 09:47:39 +01:00
|
|
|
// echo back to both serial ports
|
|
|
|
echo_serial_port(0, buf, count);
|
|
|
|
echo_serial_port(1, buf, count);
|
|
|
|
}
|
2024-04-12 01:17:50 +02:00
|
|
|
|
|
|
|
// Press on-board button to send Uart status notification
|
|
|
|
static uint32_t btn_prev = 0;
|
2025-07-03 13:42:05 +07:00
|
|
|
static cdc_notify_uart_state_t uart_state = { .value = 0 };
|
|
|
|
const uint32_t btn = board_button_read();
|
2024-04-12 01:17:50 +02:00
|
|
|
if (!btn_prev && btn) {
|
2025-07-03 13:42:05 +07:00
|
|
|
uart_state.dsr ^= 1;
|
|
|
|
tud_cdc_notify_uart_state(&uart_state);
|
2024-04-12 01:17:50 +02:00
|
|
|
}
|
|
|
|
btn_prev = btn;
|
2019-12-22 18:18:54 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-03-11 22:00:21 +07:00
|
|
|
|
2024-07-05 15:40:02 +07:00
|
|
|
// Invoked when cdc when line state changed e.g connected/disconnected
|
|
|
|
// Use to reset to DFU when disconnect with 1200 bps
|
|
|
|
void tud_cdc_line_state_cb(uint8_t instance, bool dtr, bool rts) {
|
|
|
|
(void)rts;
|
|
|
|
|
|
|
|
// DTR = false is counted as disconnected
|
|
|
|
if (!dtr) {
|
|
|
|
// touch1200 only with first CDC instance (Serial)
|
|
|
|
if (instance == 0) {
|
|
|
|
cdc_line_coding_t coding;
|
|
|
|
tud_cdc_get_line_coding(&coding);
|
|
|
|
if (coding.bit_rate == 1200) {
|
|
|
|
if (board_reset_to_bootloader) {
|
|
|
|
board_reset_to_bootloader();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-11 22:00:21 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// BLINKING TASK
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
void led_blinking_task(void) {
|
|
|
|
static uint32_t start_ms = 0;
|
|
|
|
static bool led_state = false;
|
|
|
|
|
|
|
|
// Blink every interval ms
|
|
|
|
if (board_millis() - start_ms < blink_interval_ms) return; // not enough time
|
|
|
|
start_ms += blink_interval_ms;
|
|
|
|
|
|
|
|
board_led_write(led_state);
|
|
|
|
led_state = 1 - led_state; // toggle
|
|
|
|
}
|