| /* |
| * 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. |
| * |
| */ |
| |
| #include <stdlib.h> |
| #include <stdio.h> |
| #include <string.h> |
| |
| #include "bsp/board_api.h" |
| #include "tusb.h" |
| #include "usb_descriptors.h" |
| |
| //--------------------------------------------------------------------+ |
| // MACRO CONSTANT TYPEDEF PROTYPES |
| //--------------------------------------------------------------------+ |
| |
| /* 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; |
| |
| void led_blinking_task(void* param); |
| void usb_device_task(void *param); |
| void video_task(void* param); |
| |
| #if CFG_TUSB_OS == OPT_OS_FREERTOS |
| void freertos_init(void); |
| #endif |
| |
| #if !defined(CFG_EXAMPLE_VIDEO_READONLY) || defined(CFG_EXAMPLE_VIDEO_BUFFERLESS) |
| /* EBU color bars: https://stackoverflow.com/questions/6939422 */ |
| static uint8_t const bar_color[8][4] = { |
| /* Y, U, Y, V */ |
| { 235, 128, 235, 128}, /* 100% White */ |
| { 219, 16, 219, 138}, /* Yellow */ |
| { 188, 154, 188, 16}, /* Cyan */ |
| { 173, 42, 173, 26}, /* Green */ |
| { 78, 214, 78, 230}, /* Magenta */ |
| { 63, 102, 63, 240}, /* Red */ |
| { 32, 240, 32, 118}, /* Blue */ |
| { 16, 128, 16, 128}, /* Black */ |
| }; |
| #endif |
| |
| //--------------------------------------------------------------------+ |
| // Main |
| //--------------------------------------------------------------------+ |
| int main(void) { |
| board_init(); |
| |
| // If using FreeRTOS: create blinky, tinyusb device, video task |
| #if CFG_TUSB_OS == OPT_OS_FREERTOS |
| freertos_init(); |
| #else |
| // init device stack on configured roothub port |
| tusb_rhport_init_t dev_init = { |
| .role = TUSB_ROLE_DEVICE, |
| .speed = TUSB_SPEED_AUTO |
| }; |
| tusb_init(BOARD_TUD_RHPORT, &dev_init); |
| |
| board_init_after_tusb(); |
| |
| while (1) { |
| tud_task(); // tinyusb device task |
| led_blinking_task(NULL); |
| video_task(NULL); |
| } |
| #endif |
| } |
| |
| //--------------------------------------------------------------------+ |
| // Device callbacks |
| //--------------------------------------------------------------------+ |
| |
| // 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; |
| } |
| |
| // Invoked when usb bus is suspended |
| // remote_wakeup_en : if host allow us to perform remote wakeup |
| // Within 7ms, device must draw an average of current less than 2.5 mA from bus |
| void tud_suspend_cb(bool remote_wakeup_en) { |
| (void) remote_wakeup_en; |
| blink_interval_ms = BLINK_SUSPENDED; |
| } |
| |
| // Invoked when usb bus is resumed |
| void tud_resume_cb(void) { |
| blink_interval_ms = tud_mounted() ? BLINK_MOUNTED : BLINK_NOT_MOUNTED; |
| } |
| |
| #ifdef CFG_EXAMPLE_VIDEO_BUFFERLESS |
| |
| #ifndef CFG_EXAMPLE_VIDEO_DISABLE_MJPEG |
| #error Demo only supports YUV2 please define CFG_EXAMPLE_VIDEO_DISABLE_MJPEG |
| #endif |
| |
| void tud_video_prepare_payload_cb(uint_fast8_t ctl_idx, uint_fast8_t stm_idx, tud_video_payload_request_t* request) |
| { |
| static uint32_t frame_counter = 0; |
| (void)ctl_idx; |
| (void)stm_idx; |
| |
| /* Offset will be zero at the start of a new frame */ |
| if (!request->offset) frame_counter++; |
| |
| for (size_t buf_pos = 0; buf_pos < request->length; buf_pos += 2) { |
| |
| /* Position within the current line (pixel relative) */ |
| int line_pos = ((request->offset + buf_pos)>>1) % FRAME_WIDTH; |
| |
| /* Choose color based on the position and change the table offset every 4 frames */ |
| const uint8_t* color = bar_color[(line_pos/(FRAME_WIDTH / 8) + (frame_counter>>2)) % 8]; |
| |
| /* Copy pixel data for odd or even pixels */ |
| memcpy(&((uint8_t*)request->buf)[buf_pos], &color[(line_pos & 1) ? 2 : 0], 2); |
| } |
| |
| } |
| #endif |
| |
| //--------------------------------------------------------------------+ |
| // USB Video |
| //--------------------------------------------------------------------+ |
| static unsigned frame_num = 0; |
| static unsigned tx_busy = 0; |
| static unsigned interval_ms = 1000 / FRAME_RATE; |
| #ifndef CFG_EXAMPLE_VIDEO_BUFFERLESS |
| |
| #ifdef CFG_EXAMPLE_VIDEO_READONLY |
| // For mcus that does not have enough SRAM for frame buffer, we use fixed frame data. |
| // To further reduce the size, we use MJPEG format instead of YUY2. |
| #include "images.h" |
| |
| #if !defined(CFG_EXAMPLE_VIDEO_DISABLE_MJPEG) |
| static struct { |
| uint32_t size; |
| uint8_t const *buffer; |
| } const frames[] = { |
| {color_bar_0_jpg_len, color_bar_0_jpg}, |
| {color_bar_1_jpg_len, color_bar_1_jpg}, |
| {color_bar_2_jpg_len, color_bar_2_jpg}, |
| {color_bar_3_jpg_len, color_bar_3_jpg}, |
| {color_bar_4_jpg_len, color_bar_4_jpg}, |
| {color_bar_5_jpg_len, color_bar_5_jpg}, |
| {color_bar_6_jpg_len, color_bar_6_jpg}, |
| {color_bar_7_jpg_len, color_bar_7_jpg}, |
| }; |
| #endif |
| |
| #else |
| |
| // YUY2 frame buffer |
| static uint8_t frame_buffer[FRAME_WIDTH * FRAME_HEIGHT * 16 / 8]; |
| |
| static void fill_color_bar(uint8_t* buffer, unsigned start_position) { |
| uint8_t* p; |
| |
| /* Generate the 1st line */ |
| uint8_t* end = &buffer[FRAME_WIDTH * 2]; |
| unsigned idx = (FRAME_WIDTH / 2 - 1) - (start_position % (FRAME_WIDTH / 2)); |
| p = &buffer[idx * 4]; |
| for (unsigned i = 0; i < 8; ++i) { |
| for (int j = 0; j < FRAME_WIDTH / (2 * 8); ++j) { |
| memcpy(p, &bar_color[i], 4); |
| p += 4; |
| if (end <= p) { |
| p = buffer; |
| } |
| } |
| } |
| |
| /* Duplicate the 1st line to the others */ |
| p = &buffer[FRAME_WIDTH * 2]; |
| for (unsigned i = 1; i < FRAME_HEIGHT; ++i) { |
| memcpy(p, buffer, FRAME_WIDTH * 2); |
| p += FRAME_WIDTH * 2; |
| } |
| } |
| |
| #endif |
| |
| #endif /* NDEF CFG_EXAMPLE_VIDEO_BUFFERLESS */ |
| |
| static void video_send_frame(void) { |
| static unsigned start_ms = 0; |
| static unsigned already_sent = 0; |
| |
| if (!tud_video_n_streaming(0, 0)) { |
| already_sent = 0; |
| frame_num = 0; |
| return; |
| } |
| |
| if (!already_sent) { |
| already_sent = 1; |
| tx_busy = 1; |
| start_ms = tusb_time_millis_api(); |
| #if defined(CFG_EXAMPLE_VIDEO_BUFFERLESS) |
| tud_video_n_frame_xfer(0, 0, NULL, FRAME_WIDTH * FRAME_HEIGHT * 16 / 8); |
| #elif defined (CFG_EXAMPLE_VIDEO_READONLY) |
| #if defined(CFG_EXAMPLE_VIDEO_DISABLE_MJPEG) |
| tud_video_n_frame_xfer(0, 0, (void*)(uintptr_t)&frame_buffer[(frame_num % (FRAME_WIDTH / 2)) * 4], |
| FRAME_WIDTH * FRAME_HEIGHT * 16/8); |
| #else |
| tud_video_n_frame_xfer(0, 0, (void*)(uintptr_t)frames[frame_num % 8].buffer, frames[frame_num % 8].size); |
| #endif |
| #else |
| fill_color_bar(frame_buffer, frame_num); |
| tud_video_n_frame_xfer(0, 0, (void*) frame_buffer, FRAME_WIDTH * FRAME_HEIGHT * 16 / 8); |
| #endif |
| } |
| |
| unsigned cur = tusb_time_millis_api(); |
| if (cur - start_ms < interval_ms) { |
| return; // not enough time |
| } |
| if (tx_busy) { |
| return; |
| } |
| start_ms += interval_ms; |
| tx_busy = 1; |
| |
| #if defined(CFG_EXAMPLE_VIDEO_BUFFERLESS) |
| tud_video_n_frame_xfer(0, 0, NULL, FRAME_WIDTH * FRAME_HEIGHT * 16 / 8); |
| #elif defined(CFG_EXAMPLE_VIDEO_READONLY) |
| #if defined(CFG_EXAMPLE_VIDEO_DISABLE_MJPEG) |
| tud_video_n_frame_xfer(0, 0, (void*)(uintptr_t)&frame_buffer[(frame_num % (FRAME_WIDTH / 2)) * 4], |
| FRAME_WIDTH * FRAME_HEIGHT * 16/8); |
| #else |
| tud_video_n_frame_xfer(0, 0, (void*)(uintptr_t)frames[frame_num % 8].buffer, frames[frame_num % 8].size); |
| #endif |
| #else |
| fill_color_bar(frame_buffer, frame_num); |
| tud_video_n_frame_xfer(0, 0, (void*) frame_buffer, FRAME_WIDTH * FRAME_HEIGHT * 16 / 8); |
| #endif |
| } |
| |
| |
| void video_task(void* param) { |
| (void) param; |
| |
| while(1) { |
| video_send_frame(); |
| |
| #if CFG_TUSB_OS == OPT_OS_FREERTOS |
| vTaskDelay(interval_ms / portTICK_PERIOD_MS); |
| #else |
| return; |
| #endif |
| } |
| } |
| |
| void tud_video_frame_xfer_complete_cb(uint_fast8_t ctl_idx, uint_fast8_t stm_idx) { |
| (void) ctl_idx; |
| (void) stm_idx; |
| tx_busy = 0; |
| /* flip buffer */ |
| ++frame_num; |
| } |
| |
| int tud_video_commit_cb(uint_fast8_t ctl_idx, uint_fast8_t stm_idx, |
| video_probe_and_commit_control_t const* parameters) { |
| (void) ctl_idx; |
| (void) stm_idx; |
| /* convert unit to ms from 100 ns */ |
| interval_ms = parameters->dwFrameInterval / 10000; |
| return VIDEO_ERROR_NONE; |
| } |
| |
| //--------------------------------------------------------------------+ |
| // Blinking Task |
| //--------------------------------------------------------------------+ |
| void led_blinking_task(void* param) { |
| (void) param; |
| static uint32_t start_ms = 0; |
| static bool led_state = false; |
| |
| while (1) { |
| #if CFG_TUSB_OS == OPT_OS_FREERTOS |
| vTaskDelay(blink_interval_ms / portTICK_PERIOD_MS); |
| #else |
| if (tusb_time_millis_api() - start_ms < blink_interval_ms) { |
| return; // not enough time |
| } |
| #endif |
| |
| start_ms += blink_interval_ms; |
| board_led_write(led_state); |
| led_state = 1 - led_state; // toggle |
| } |
| } |
| |
| //--------------------------------------------------------------------+ |
| // FreeRTOS |
| //--------------------------------------------------------------------+ |
| #if CFG_TUSB_OS == OPT_OS_FREERTOS |
| |
| #define BLINKY_STACK_SIZE configMINIMAL_STACK_SIZE |
| #define VIDEO_STACK_SIZE (configMINIMAL_STACK_SIZE*4) |
| |
| #ifdef ESP_PLATFORM |
| #define USBD_STACK_SIZE 4096 |
| int main(void); |
| void app_main(void) { |
| main(); |
| } |
| #else |
| // Increase stack size when debug log is enabled |
| #define USBD_STACK_SIZE (3*configMINIMAL_STACK_SIZE/2) * (CFG_TUSB_DEBUG ? 2 : 1) |
| #endif |
| |
| // static task |
| #if configSUPPORT_STATIC_ALLOCATION |
| StackType_t blinky_stack[BLINKY_STACK_SIZE]; |
| StaticTask_t blinky_taskdef; |
| |
| StackType_t usb_device_stack[USBD_STACK_SIZE]; |
| StaticTask_t usb_device_taskdef; |
| |
| StackType_t video_stack[VIDEO_STACK_SIZE]; |
| StaticTask_t video_taskdef; |
| #endif |
| |
| // USB Device Driver task |
| // This top level thread process all usb events and invoke callbacks |
| void usb_device_task(void *param) { |
| (void) param; |
| |
| // init device stack on configured roothub port |
| // This should be called after scheduler/kernel is started. |
| // Otherwise, it could cause kernel issue since USB IRQ handler does use RTOS queue API. |
| tusb_rhport_init_t dev_init = { |
| .role = TUSB_ROLE_DEVICE, |
| .speed = TUSB_SPEED_AUTO |
| }; |
| tusb_init(BOARD_TUD_RHPORT, &dev_init); |
| |
| board_init_after_tusb(); |
| |
| // RTOS forever loop |
| while (1) { |
| // put this thread to waiting state until there is new events |
| tud_task(); |
| } |
| } |
| |
| void freertos_init(void) { |
| #if configSUPPORT_STATIC_ALLOCATION |
| xTaskCreateStatic(led_blinking_task, "blinky", BLINKY_STACK_SIZE, NULL, 1, blinky_stack, &blinky_taskdef); |
| xTaskCreateStatic(usb_device_task, "usbd", USBD_STACK_SIZE, NULL, configMAX_PRIORITIES-1, usb_device_stack, &usb_device_taskdef); |
| xTaskCreateStatic(video_task, "cdc", VIDEO_STACK_SIZE, NULL, configMAX_PRIORITIES - 2, video_stack, &video_taskdef); |
| #else |
| xTaskCreate(led_blinking_task, "blinky", BLINKY_STACK_SIZE, NULL, 1, NULL); |
| xTaskCreate(usb_device_task, "usbd", USBD_STACK_SIZE, NULL, configMAX_PRIORITIES - 1, NULL); |
| xTaskCreate(video_task, "video", VIDEO_STACK_SZIE, NULL, configMAX_PRIORITIES - 2, NULL); |
| #endif |
| |
| // only start scheduler for non-espressif mcu |
| #ifndef ESP_PLATFORM |
| vTaskStartScheduler(); |
| #endif |
| } |
| #endif |