From ea69e7497c2261afdbb00116742fbc513db924fb Mon Sep 17 00:00:00 2001 From: MX <10697207+xMasterX@users.noreply.github.com> Date: Wed, 20 Dec 2023 00:46:14 +0300 Subject: [PATCH] add new app --- ReadMe.md | 1 + non_catalog_apps/ws2812b_tester/README.md | 35 ++ non_catalog_apps/ws2812b_tester/app.c | 566 ++++++++++++++++++ non_catalog_apps/ws2812b_tester/app.png | Bin 0 -> 2557 bytes .../ws2812b_tester/application.fam | 17 + non_catalog_apps/ws2812b_tester/led_driver.c | 283 +++++++++ non_catalog_apps/ws2812b_tester/led_driver.h | 13 + 7 files changed, 915 insertions(+) create mode 100644 non_catalog_apps/ws2812b_tester/README.md create mode 100644 non_catalog_apps/ws2812b_tester/app.c create mode 100644 non_catalog_apps/ws2812b_tester/app.png create mode 100644 non_catalog_apps/ws2812b_tester/application.fam create mode 100644 non_catalog_apps/ws2812b_tester/led_driver.c create mode 100644 non_catalog_apps/ws2812b_tester/led_driver.h diff --git a/ReadMe.md b/ReadMe.md index bd8052fe3ce..79e59602bd5 100644 --- a/ReadMe.md +++ b/ReadMe.md @@ -199,6 +199,7 @@ The Flipper and its community wouldn't be as rich as it is without your contribu | NRF24 Mouse Jacker MS | ![GPIO Badge] | [by coded-with-claws](https://github.com/coded-with-claws/flipperzero-tools) | read more details in original repo | ![None Badge] | | SD-SPI | ![GPIO Badge] | [by Gl1tchub](https://github.com/Gl1tchub/Flipperzero-SD-SPI) | read more details in original repo | ![None Badge] | | Motion Mouse | ![GPIO Badge] | [by nminaylov](https://github.com/flipperdevices/flipperzero-good-faps/pull/83/files) | read more details in [original repo](https://github.com/flipperdevices/flipperzero-good-faps/pull/83/files) | ![None Badge] | +| WS2812B LED Tester | ![GPIO Badge] | [by jamisonderek](https://github.com/jamisonderek/flipper-zero-tutorials/tree/main/gpio/ws2812b_tester) | read more details in [original repo](https://github.com/jamisonderek/flipper-zero-tutorials/tree/main/gpio/ws2812b_tester) | ![None Badge] | | IR Remote | ![IR Badge] | [by Hong5489](https://github.com/Hong5489/ir_remote) | improvements [by friebel](https://github.com/RogueMaster/flipperzero-firmware-wPlugins/pull/535) - Hold Option, RAW support [by d4ve10](https://github.com/d4ve10/ir_remote/tree/infrared_hold_option) | ![None Badge] | | IR Intervalometer | ![IR Badge] | [by Nitepone](https://github.com/Nitepone/flipper-intervalometer) | | [![UFW Badge]](https://lab.flipper.net/apps/sony_intervalometer) | | IR Xbox Controller | ![IR Badge] | [by gebeto](https://github.com/gebeto/flipper-xbox-controller) | | [![Author Badge]](https://lab.flipper.net/apps/xbox_controller) | diff --git a/non_catalog_apps/ws2812b_tester/README.md b/non_catalog_apps/ws2812b_tester/README.md new file mode 100644 index 00000000000..128c3cbf045 --- /dev/null +++ b/non_catalog_apps/ws2812b_tester/README.md @@ -0,0 +1,35 @@ +# WS2812B LED Tester +This application is used to test WS2812B LEDs. You can connect the WS2812B LEDs to any available GPIO pin. If you are powering the LEDs using the Flipper Zero, be sure to consider the power requirements of the LEDs. The 3V3 pin has a 1200mA max current (~4 watts). 5V pin has a 1000mA max current (5 watts). + +Please let me know any feedback! +- Discord - https://discord.com/invite/NsjCvqwPAd (@CodeAllNight) +- YouTube - https://youtube.com/@MrDerekJamison/playlists +- GitHub - https://github.com/jamisonderek/flipper-zero-tutorials +- Wiki - https://github.com/jamisonderek/flipper-zero-tutorials/wiki + +# Overview +This application has three submenu items: +* Test WS2812B +* About + +# Test WS2812B +- LED Pin : Select the GPIO pin that the WS2812B LED data wire is connected to. +- LED Count : select the total number of LEDs in the chain. +- LED Pattern : select the pattern to display on the LEDs. +- LED Brightness : Select the brightness of the LEDs. The brighter the LEDs, the more power they will consume. +- Enable +5V pin : Select "Yes" if you are using pin 1 to power the LEDs (note, there is a 1000mA max current on this pin). + +# About +The "About" menu item contains information about the application. + +# Updates +- Version 1.8 + - Renamed app to fit on Flipper Zero main menu screen + - Improved initialization to only happen when configuring the LEDs +- Version 1.7 + - Added blanking TRESET (LOW) signal before sending data to LEDs. + - Increased timer_buffer to uint16 to support blanking signal duration. Maybe there is a better way to do the initial low & save memory? + - Bug fix: Turn off remaining LEDs when reducing the number of LEDs. +- Version 1.6 + - Added support for up to 1000 LEDs (max set in led_driver.h) + - Added "dirty flag" to get rid of flicker when not updating the LEDs diff --git a/non_catalog_apps/ws2812b_tester/app.c b/non_catalog_apps/ws2812b_tester/app.c new file mode 100644 index 00000000000..53f5fda354a --- /dev/null +++ b/non_catalog_apps/ws2812b_tester/app.c @@ -0,0 +1,566 @@ +#include +#include +#include +#include +#include +#include +#include +#include + +#include "led_driver.h" + +#define TAG "WS2812B-Tester" + +// Our application menu has 2 items. +typedef enum { + LedTesterSubmenuIndexLeds, + LedTesterSubmenuIndexAbout, +} LedTesterSubmenuIndex; + +// Custom events that describe why the view dispatcher is calling our custom event callback. +typedef enum { + LedTesterEventInitialized, + LedTesterEventDeinit, + LedTesterEventTimer, + LedTesterEventConfigChange, + LedTesterEventClicked, +} LedTesterEventId; + +// Each view is a screen we show the user. +typedef enum { + LedTesterViewSubmenu, // The menu when the app starts + LedTesterViewLeds, // The LED screen + LedTesterViewAbout, // The about screen with directions, link to social channel, etc. +} LedTesterView; + +// Our model is the data we use to control the LEDs. +typedef struct { + uint8_t led_pin_index; // The index of the pin we are using to control the LEDs + uint32_t led_count; // The number of LEDs + uint8_t led_pattern_index; // The pattern index + int8_t led_divisor; // The speed divisor (1-4, -1 = stopped) + uint8_t led_max_brightness; // The maximum brightness (0-100) + uint8_t timer_counter; // The current timer counter (used for auto-scrolling) + bool enable_5v; // Enable 5V output +} LedTesterModel; + +// The application object. +typedef struct { + ViewDispatcher* view_dispatcher; // Switches between our views + Submenu* submenu; // The application menu + VariableItemList* variable_item_list; // The WS2812B settings + Widget* widget_about; // The about screen + FuriTimer* timer; // Timer for automatic updating the LEDs + LedTesterModel* model; // The model + LedDriver* led_driver; // The LED driver +} LedTesterApp; + +// Hack so that we can access the application object from a variable_item_list on_enter/exit callback. +static LedTesterApp* global_app = NULL; + +/** + * @brief Callback for exiting the application. + * @details This function is called when user press back button. We return VIEW_NONE to + * indicate that we want to exit the application. + * @param _context The context - unused + * @return next view id +*/ +static uint32_t led_tester_navigation_exit_callback(void* _context) { + UNUSED(_context); + return VIEW_NONE; +} + +/** + * @brief Callback for the application's menu. + * @details This function is called when user press back button. We return LedTesterViewSubmenu to + * indicate that we want to return to the application menu. + * @param _context The context - unused + * @return next view id +*/ +static uint32_t led_tester_navigation_submenu_callback(void* _context) { + UNUSED(_context); + return LedTesterViewSubmenu; +} + +/** + * @brief Handle submenu item selection. + * @details This function is called when user selects an item from the submenu. + * @param context The context - LedTesterApp object. + * @param index The LedTesterSubmenuIndex item that was clicked. +*/ +static void led_tester_submenu_callback(void* context, uint32_t index) { + LedTesterApp* app = (LedTesterApp*)context; + switch(index) { + case LedTesterSubmenuIndexLeds: + view_dispatcher_switch_to_view(app->view_dispatcher, LedTesterViewLeds); + break; + case LedTesterSubmenuIndexAbout: + view_dispatcher_switch_to_view(app->view_dispatcher, LedTesterViewAbout); + break; + default: + break; + } +} + +/** + * @brief Callback for timer elapsed. + * @details This function is called when the timer is elapsed. + * @param context The context - LedTesterApp object. +*/ +static void led_tester_timer_callback(void* context) { + LedTesterApp* app = (LedTesterApp*)context; + view_dispatcher_send_custom_event(app->view_dispatcher, LedTesterEventTimer); +} + +/** + * @brief Callback for when the LED configuration settings are updated. + * @details This function is called when the LED configuration settings are changed by the user. + * @param context The context - LedTesterApp object. +*/ +static void led_tester_settings_updated(LedTesterApp* app) { + view_dispatcher_send_custom_event(app->view_dispatcher, LedTesterEventConfigChange); +} + +/** + * @brief Callback for when the LED configuration settings are clicked. + * @details This function is called when the user presses OK button while in the LED configuration settings. + * @param context The context - LedTesterApp object. +*/ +static void led_tester_item_clicked(void* context, uint32_t index) { + UNUSED(index); + LedTesterApp* app = (LedTesterApp*)context; + view_dispatcher_send_custom_event(app->view_dispatcher, LedTesterEventClicked); +} + +/** + * @brief Enable 5V power. + * @details This function enables the 5V power output on pin 1. +*/ +static void led_tester_5v_power_on() { + uint8_t attempts = 5; + while(--attempts > 0) { + if(furi_hal_power_enable_otg()) break; + } +} + +/** + * @brief Disable 5V power. + * @details This function disables the 5V power output on pin 1. +*/ +static void led_tester_5v_power_off() { + if(furi_hal_power_is_otg_enabled()) { + furi_hal_power_disable_otg(); + } +} + +// Settings for configuring which GPIO pin to use for controlling the WS2812B LEDs. +static const char* setting_led_pin_config_label = "LED Pin"; +static const GpioPin* setting_led_pin_values[] = + {&gpio_ext_pa7, &gpio_ext_pa6, &gpio_ext_pa4, &gpio_ext_pb3, &gpio_ext_pb2, &gpio_ext_pc3}; +static char* setting_led_pin_names[] = {"A7", "A6", "A4", "B3", "B2", "C3"}; +static void led_tester_setting_led_pin_change(VariableItem* item) { + LedTesterApp* app = variable_item_get_context(item); + LedTesterModel* model = app->model; + uint8_t index = variable_item_get_current_value_index(item); + variable_item_set_current_value_text(item, setting_led_pin_names[index]); + model->led_pin_index = index; +} + +// Settings for configuring how many LEDs to enable. +static const char* setting_led_count_config_label = "LED Count"; +static uint16_t setting_led_count_values[] = {1, 2, 3, 4, 5, 6, 7, 8, 16, 32, 64, 128, 256, 512}; +static char* setting_led_count_names[] = + {"1", "2", "3", "4", "5", "6", "7", "8", "16", "32", "64", "128", "256", "512"}; +static uint8_t setting_led_count_default_index = 3; // 4 LEDs +static void led_tester_setting_led_count_change(VariableItem* item) { + LedTesterApp* app = variable_item_get_context(item); + LedTesterModel* model = app->model; + uint8_t index = variable_item_get_current_value_index(item); + variable_item_set_current_value_text(item, setting_led_count_names[index]); + model->led_count = setting_led_count_values[index]; + led_tester_settings_updated(app); +} + +// Settings for configuring which LED pattern to use. +static const char* setting_led_pattern_config_label = "LED Pattern"; +static char* setting_led_pattern_names[] = + {"Red", "Green", "Blue", "White", "RGBW", "GBWR", "BWRG", "WRGB"}; +static void led_tester_setting_led_pattern_change(VariableItem* item) { + LedTesterApp* app = variable_item_get_context(item); + LedTesterModel* model = app->model; + uint8_t index = variable_item_get_current_value_index(item); + variable_item_set_current_value_text(item, setting_led_pattern_names[index]); + model->led_pattern_index = index; + led_tester_settings_updated(app); +} + +// Settings for configuring which LED speed to use. +static const char* setting_led_speed_config_label = "LED Speed"; +static int16_t setting_led_speed_values[] = {-1, 3, 2, 1}; +static char* setting_led_speed_names[] = {"Stopped", "Slow", "Medium", "Fast"}; +static void led_tester_setting_led_speed_change(VariableItem* item) { + LedTesterApp* app = variable_item_get_context(item); + LedTesterModel* model = app->model; + uint8_t index = variable_item_get_current_value_index(item); + variable_item_set_current_value_text(item, setting_led_speed_names[index]); + model->led_divisor = setting_led_speed_values[index]; + led_tester_settings_updated(app); +} + +// Settings for configuring the LED brightness. +static const char* setting_led_brightness_config_label = "LED Brightness"; +static uint16_t setting_led_brightness_values[] = {1, 2, 3, 5, 10, 20, 25, 50, 75, 100}; +static char* setting_led_brightness_names[] = + {"1%", "2%", "3%", "5%", "10%", "20%", "25%", "50%", "75%", "100%"}; +static void led_tester_setting_led_brightness_change(VariableItem* item) { + LedTesterApp* app = variable_item_get_context(item); + LedTesterModel* model = app->model; + uint8_t index = variable_item_get_current_value_index(item); + variable_item_set_current_value_text(item, setting_led_brightness_names[index]); + model->led_max_brightness = setting_led_brightness_values[index]; + led_tester_settings_updated(app); +} + +// Settings for configuring the 5V output. +static const char* setting_5v_config_label = "Enable +5V pin"; +static bool setting_5v_values[] = {true, false}; +static char* setting_5v_names[] = {"Yes", "No"}; +static void led_tester_setting_5v_change(VariableItem* item) { + LedTesterApp* app = variable_item_get_context(item); + LedTesterModel* model = app->model; + uint8_t index = variable_item_get_current_value_index(item); + variable_item_set_current_value_text(item, setting_5v_names[index]); + model->enable_5v = setting_5v_values[index]; + if(app->model->enable_5v) { + led_tester_5v_power_on(); + } else { + led_tester_5v_power_off(); + } +} + +/** + * @brief Callback for custom events. + * @details This function is called when a custom event is sent to the view dispatcher. + * @param context The context - LedTesterApp object. + * @param event The event id - LedTesterEventId value. +*/ +static bool led_tester_custom_event_callback(void* context, uint32_t event) { + LedTesterApp* app = (LedTesterApp*)context; + const GpioPin* pin = setting_led_pin_values[app->model->led_pin_index]; + + if(!app->led_driver) { + FURI_LOG_E(TAG, "led_driver is NULL. Custom event %lu ignored.", event); + return false; + } + + if(event == LedTesterEventTimer) { + app->model->timer_counter++; + } + + uint8_t offset = (app->model->led_divisor == -1) ? + 0 : + (app->model->timer_counter / app->model->led_divisor) & 0x3; + + uint32_t rgb[4] = {0}; + switch(app->model->led_pattern_index) { + case 0: // RED + for(size_t i = 0; i < COUNT_OF(rgb); i++) { + rgb[i] = 0xFF0000; + } + break; + case 1: // GREEN + for(size_t i = 0; i < COUNT_OF(rgb); i++) { + rgb[i] = 0x00FF00; + } + break; + case 2: // BLUE + for(size_t i = 0; i < COUNT_OF(rgb); i++) { + rgb[i] = 0x0000FF; + } + break; + case 3: // WHITE + for(size_t i = 0; i < COUNT_OF(rgb); i++) { + rgb[i] = 0xFFFFFF; + } + break; + case 4: // RGBW + rgb[0] = 0xFF0000; + rgb[1] = 0x00FF00; + rgb[2] = 0x0000FF; + rgb[3] = 0xFFFFFF; + break; + case 5: // GBWR + rgb[3] = 0xFF0000; + rgb[0] = 0x00FF00; + rgb[1] = 0x0000FF; + rgb[2] = 0xFFFFFF; + break; + case 6: // BWRG + rgb[2] = 0xFF0000; + rgb[3] = 0x00FF00; + rgb[0] = 0x0000FF; + rgb[1] = 0xFFFFFF; + break; + case 7: // WRGB + rgb[0] = 0xFFFFFF; + rgb[1] = 0xFF0000; + rgb[2] = 0x00FF00; + rgb[3] = 0x0000FF; + break; + default: + break; + } + + // Rotate the pattern + for(size_t i = 0; i < offset; i++) { + uint32_t tmp = rgb[0]; + for(size_t j = 0; j < COUNT_OF(rgb) - 1; j++) { + rgb[j] = rgb[j + 1]; + } + rgb[COUNT_OF(rgb) - 1] = tmp; + } + + // If deinit, turn off the LEDs + if(event == LedTesterEventDeinit) { + for(size_t i = 0; i < COUNT_OF(rgb); i++) { + rgb[i] = 0; + } + } + + // Scale the brightness + for(size_t i = 0; i < COUNT_OF(rgb); i++) { + rgb[i] = (((rgb[i] >> 16) & 0xFF) * app->model->led_max_brightness / 100) << 16 | + (((rgb[i] >> 8) & 0xFF) * app->model->led_max_brightness / 100) << 8 | + ((rgb[i] & 0xFF) * app->model->led_max_brightness / 100); + // rgb[i] = (rgb[i] & 0x0F0F0F); // For Debugging, just use lower 4-bits. + } + + led_driver_set_pin(app->led_driver, pin); + + // Set the LEDs to the pattern + for(size_t i = 0; i < app->model->led_count; i++) { + led_driver_set_led(app->led_driver, i, rgb[i % COUNT_OF(rgb)]); + } + // Turn off any remaining LEDs + for(size_t i = app->model->led_count; i < MAX_LED_COUNT; i++) { + led_driver_set_led(app->led_driver, i, 0); + } + + led_driver_transmit(app->led_driver, true); + + if(event == LedTesterEventDeinit) { + led_driver_free(app->led_driver); + app->led_driver = NULL; + } + + return true; +} + +/** + * @brief Callback for entering the LED configuration settings. + * @details This function is called when the user enters the LED configuration settings. We + * start the periodic timer to update the LEDs & we signal initialized so the LEDs + * turn on to their default state. + * @param _context The context - unused +*/ +void led_tester_enter_leds_callback(void* _context) { + // _context is variable_item_list, but it doesn't expose the item, so we can't get the context. + UNUSED(_context); + // Hack, we use a global to access the app object. + LedTesterApp* app = (LedTesterApp*)global_app; + app->timer = furi_timer_alloc(led_tester_timer_callback, FuriTimerTypePeriodic, app); + furi_timer_start(app->timer, 250); + view_dispatcher_send_custom_event(app->view_dispatcher, LedTesterEventInitialized); + app->led_driver = + led_driver_alloc(MAX_LED_COUNT, setting_led_pin_values[app->model->led_pin_index]); +} + +/** + * @brief Callback for exiting the LED configuration settings. + * @details This function is called when the user exits the LED configuration settings. We + * stop the periodic timer to update the LEDs. + * @param _context The context - unused +*/ +void led_tester_exit_leds_callback(void* _context) { + // _context is variable_item_list, but it doesn't expose the item, so we can't get the context. + UNUSED(_context); + // Hack, we use a global to access the app object. + LedTesterApp* app = (LedTesterApp*)global_app; + furi_timer_stop(app->timer); + furi_timer_free(app->timer); + app->timer = NULL; + view_dispatcher_send_custom_event(app->view_dispatcher, LedTesterEventDeinit); +} + +/** + * @brief Allocate the led tester application. + * @details This function allocates the led tester application resources. + * @return LedTesterApp object. +*/ +static LedTesterApp* led_tester_app_alloc() { + LedTesterApp* app = (LedTesterApp*)malloc(sizeof(LedTesterApp)); + global_app = app; + + app->model = (LedTesterModel*)malloc(sizeof(LedTesterModel)); + + Gui* gui = furi_record_open(RECORD_GUI); + + app->view_dispatcher = view_dispatcher_alloc(); + view_dispatcher_enable_queue(app->view_dispatcher); + view_dispatcher_attach_to_gui(app->view_dispatcher, gui, ViewDispatcherTypeFullscreen); + view_dispatcher_set_event_callback_context(app->view_dispatcher, app); + + app->submenu = submenu_alloc(); + submenu_add_item( + app->submenu, "Test WS2812B", LedTesterSubmenuIndexLeds, led_tester_submenu_callback, app); + submenu_add_item( + app->submenu, "About", LedTesterSubmenuIndexAbout, led_tester_submenu_callback, app); + view_set_previous_callback( + submenu_get_view(app->submenu), led_tester_navigation_exit_callback); + view_dispatcher_add_view( + app->view_dispatcher, LedTesterViewSubmenu, submenu_get_view(app->submenu)); + view_dispatcher_set_event_callback_context(app->view_dispatcher, app); + view_dispatcher_set_custom_event_callback( + app->view_dispatcher, led_tester_custom_event_callback); + view_dispatcher_switch_to_view(app->view_dispatcher, LedTesterViewSubmenu); + + app->variable_item_list = variable_item_list_alloc(); + variable_item_list_set_enter_callback(app->variable_item_list, led_tester_item_clicked, app); + view_set_enter_callback( + variable_item_list_get_view(app->variable_item_list), led_tester_enter_leds_callback); + view_set_exit_callback( + variable_item_list_get_view(app->variable_item_list), led_tester_exit_leds_callback); + variable_item_list_reset(app->variable_item_list); + + // Pin + VariableItem* item = variable_item_list_add( + app->variable_item_list, + setting_led_pin_config_label, + COUNT_OF(setting_led_pin_names), + led_tester_setting_led_pin_change, + app); + uint8_t setting_led_pin_index = 0; + variable_item_set_current_value_index(item, setting_led_pin_index); + variable_item_set_current_value_text(item, setting_led_pin_names[setting_led_pin_index]); + app->model->led_pin_index = setting_led_pin_index; + + // Count + item = variable_item_list_add( + app->variable_item_list, + setting_led_count_config_label, + COUNT_OF(setting_led_count_names), + led_tester_setting_led_count_change, + app); + uint8_t setting_led_count_index = setting_led_count_default_index; + variable_item_set_current_value_index(item, setting_led_count_index); + variable_item_set_current_value_text(item, setting_led_count_names[setting_led_count_index]); + app->model->led_count = setting_led_count_values[setting_led_count_index]; + + // Pattern + item = variable_item_list_add( + app->variable_item_list, + setting_led_pattern_config_label, + COUNT_OF(setting_led_pattern_names), + led_tester_setting_led_pattern_change, + app); + uint8_t setting_led_pattern_index = 0; + variable_item_set_current_value_index(item, setting_led_pattern_index); + variable_item_set_current_value_text( + item, setting_led_pattern_names[setting_led_pattern_index]); + app->model->led_pattern_index = setting_led_pattern_index; + + // Speed + item = variable_item_list_add( + app->variable_item_list, + setting_led_speed_config_label, + COUNT_OF(setting_led_speed_names), + led_tester_setting_led_speed_change, + app); + uint8_t setting_led_speed_index = 0; + variable_item_set_current_value_index(item, setting_led_speed_index); + variable_item_set_current_value_text(item, setting_led_speed_names[setting_led_speed_index]); + app->model->led_divisor = setting_led_speed_values[setting_led_speed_index]; + + // Brightness + item = variable_item_list_add( + app->variable_item_list, + setting_led_brightness_config_label, + COUNT_OF(setting_led_brightness_names), + led_tester_setting_led_brightness_change, + app); + uint8_t setting_led_brightness_index = 3; + variable_item_set_current_value_index(item, setting_led_brightness_index); + variable_item_set_current_value_text( + item, setting_led_brightness_names[setting_led_brightness_index]); + app->model->led_max_brightness = setting_led_brightness_values[setting_led_brightness_index]; + + // 5-volt pin + item = variable_item_list_add( + app->variable_item_list, + setting_5v_config_label, + COUNT_OF(setting_5v_names), + led_tester_setting_5v_change, + app); + uint8_t setting_5v_index = 0; + variable_item_set_current_value_index(item, setting_5v_index); + variable_item_set_current_value_text(item, setting_5v_names[setting_5v_index]); + app->model->enable_5v = setting_5v_values[setting_5v_index]; + if(app->model->enable_5v) { + led_tester_5v_power_on(); + } + + view_set_previous_callback( + variable_item_list_get_view(app->variable_item_list), + led_tester_navigation_submenu_callback); + view_dispatcher_add_view( + app->view_dispatcher, + LedTesterViewLeds, + variable_item_list_get_view(app->variable_item_list)); + + app->widget_about = widget_alloc(); + widget_add_text_scroll_element( + app->widget_about, + 0, + 0, + 128, + 64, + "This is a WS2812B LED tester\nVersion 1.7\nConnect WS2812B LED data\nwire to GPIO pin on Flipper.\n\nThe 3V3 pin has a 1200mA\nmax current (~4 watts). The\n5V pin has a 1000mA max\ncurrent (5 watts).\n\nauthors: @codeallnight and\nZ3BRO!\n\nhttps://discord.com/invite/NsjCvqwPAd\nhttps://youtube.com/@MrDerekJamison\n\n"); + view_set_previous_callback( + widget_get_view(app->widget_about), led_tester_navigation_submenu_callback); + view_dispatcher_add_view( + app->view_dispatcher, LedTesterViewAbout, widget_get_view(app->widget_about)); + + return app; +} + +/** + * @brief Free the led tester application. + * @details This function frees the led tester application resources. + * @param app The led tester application object. +*/ +static void led_tester_app_free(LedTesterApp* app) { + view_dispatcher_remove_view(app->view_dispatcher, LedTesterViewAbout); + widget_free(app->widget_about); + view_dispatcher_remove_view(app->view_dispatcher, LedTesterViewLeds); + variable_item_list_free(app->variable_item_list); + view_dispatcher_remove_view(app->view_dispatcher, LedTesterViewSubmenu); + submenu_free(app->submenu); + view_dispatcher_free(app->view_dispatcher); + furi_record_close(RECORD_GUI); + free(app); +} + +/** + * @brief Main function for led tester application. + * @details This function is the entry point for the led tester application. It should be defined in + * application.fam as the entry_point setting. + * @param _p Input parameter - unused + * @return 0 - Success +*/ +int32_t ws2812b_led_tester_app(void* _p) { + UNUSED(_p); + LedTesterApp* app = led_tester_app_alloc(); + view_dispatcher_run(app->view_dispatcher); + led_tester_app_free(app); + return 0; +} \ No newline at end of file diff --git a/non_catalog_apps/ws2812b_tester/app.png b/non_catalog_apps/ws2812b_tester/app.png new file mode 100644 index 0000000000000000000000000000000000000000..8048c799e857e0f04581aaca2cbe2b80bb4e5abb GIT binary patch literal 2557 zcmcImZEO=|9Di!J0mDfJd5tmWE$G;|-fOSD%euF|xCwR#)6XsAlGN7?~6@xHXW{a3Wct`8D{`+Jg)@;(XK!MiyFGBB_RL}*X6!G zCSz?e;QYS6OIMp#)Ee3D-Mja8_qARhyEt|Q)abkj;@Gzze2->xGMyVu0Ol^RzYcKt z$Pxf%{h`G+SQ{d%MOja|B}I=zcP^Dd)~NQaxr`)lh87lw2`%l%e?EN{$27%{_jn>y zBol!BTIV(cuHUvMCU4s;3ku%aQdirY6A?lRS`wB^CDW#u^Wz0z5zXyw62}TA)@DE6 zZhOQwM50(gHz4M9(}YY>6h_m$+b1YID^V*khGJ-vrb)&}FtkVsBFkc>4~`sSShJz1 zVs|iHa)?%byx+1iB1vYmS$CFo>qdg41wkMwhGZB5SrF!6+LCfa+N>{vlwg9;lnpIo zX?hy7VWqgf)$-#g=3+9b%p`W&EL9BEj?76Jl6F&ea%ICvWO~z7YSP-YLIcQ4Dd;JI z&DdZDlHJhMw;D1G4ZyTjUv`%1*Dc-b*Z(E9$;XdGsK}EJGh2;hp+<^K!X!)~Y!lhi zh4?aJz<`pa8!=r^mRlArd&C&VO<@f^T3XSw<_bHa(g7HhEa=Bk_ZWf_2%3vAjK~Ng z=WC)^k)n#G5yS{ZvZTKmdtwwNaaVN_W2@#62q!VRIBT^$LD49;p zL>(L~$fc;Af@s#mEfrfO2_!YFz(-M%Z5jA++3r77@Y1$6MPX-_LJ>CG4-x1zyeOKP zx@u)51GXnn6CTeyat4W*(!lg-uwF)$8C3r`4_QK$IF%5*h$%A1sa~I#<(asrT!#NA zjbiR{zm!fu^r}S){7({_8S$s=&p)L&)BboW-wV?x)=b(j+e$h%lT$Od`JoQVT53ls zx9M$GP@q8X4k=NwHA(N5spU)wKFA9+MEX^Df + +#include "led_driver.h" + +// We store the HIGH/LOW durations (2 values) for each color bit (24 bits per LED) +#define LED_DRIVER_BUFFER_SIZE (MAX_LED_COUNT * 2 * 24) +// We use a setinel value to figure out when the timer is complete. +#define LED_DRIVER_TIMER_SETINEL 0xFFFFU + +/** 64 transitions per us @ 64MHz. Our timing is in NANO_SECONDS */ +#define LED_DRIVER_TIMER_NANOSECOND (1000U / (SystemCoreClock / 1000000U)) +// Timings for WS2812B +#define LED_DRIVER_T0H 400U +#define LED_DRIVER_T1H 800U +#define LED_DRIVER_T0L 850U +#define LED_DRIVER_T1L 450U +#define LED_DRIVER_TRESETL 55 * 1000U + +// Wait for 35ms for the DMA to complete. NOTE: 1000 leds*(850ns+450ns)*24 = 32ms +#define LED_DRIVER_SETINEL_WAIT_MS 35 + +struct LedDriver { + LL_DMA_InitTypeDef dma_gpio_update; + LL_DMA_InitTypeDef dma_led_transition_timer; + + const GpioPin* gpio; + uint32_t gpio_buf[2]; // On/Off for GPIO + + uint16_t timer_buffer[LED_DRIVER_BUFFER_SIZE + 2]; + uint32_t write_pos; + uint32_t read_pos; + bool dirty; + + uint32_t count_leds; + uint32_t* led_data; +}; + +static void led_driver_init_dma_gpio_update(LedDriver* led_driver, const GpioPin* gpio) { + led_driver->gpio = gpio; + + // Memory to Peripheral + led_driver->dma_gpio_update.Direction = LL_DMA_DIRECTION_MEMORY_TO_PERIPH; + // Peripheral (GPIO - We populate GPIO port's BSRR register) + led_driver->dma_gpio_update.PeriphOrM2MSrcAddress = (uint32_t)&gpio->port->BSRR; + led_driver->dma_gpio_update.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; + led_driver->dma_gpio_update.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_WORD; + // Memory (State to set GPIO) + led_driver->dma_gpio_update.MemoryOrM2MDstAddress = (uint32_t)led_driver->gpio_buf; + led_driver->dma_gpio_update.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT; + led_driver->dma_gpio_update.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_WORD; + // Data + led_driver->dma_gpio_update.Mode = LL_DMA_MODE_CIRCULAR; + led_driver->dma_gpio_update.NbData = 2; // We cycle between two (HIGH/LOW)values + // When to perform data exchange + led_driver->dma_gpio_update.PeriphRequest = LL_DMAMUX_REQ_TIM2_UP; + led_driver->dma_gpio_update.Priority = LL_DMA_PRIORITY_VERYHIGH; +} + +static void led_driver_init_dma_led_transition_timer(LedDriver* led_driver) { + // Timer that triggers based on user data. + led_driver->dma_led_transition_timer.Direction = LL_DMA_DIRECTION_MEMORY_TO_PERIPH; + // Peripheral (Timer - We populate TIM2's ARR register) + led_driver->dma_led_transition_timer.PeriphOrM2MSrcAddress = (uint32_t)&TIM2->ARR; + led_driver->dma_led_transition_timer.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; + led_driver->dma_led_transition_timer.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_WORD; + // Memory (Timings) + led_driver->dma_led_transition_timer.MemoryOrM2MDstAddress = + (uint32_t)led_driver->timer_buffer; + led_driver->dma_led_transition_timer.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT; + led_driver->dma_led_transition_timer.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_HALFWORD; + // Data + led_driver->dma_led_transition_timer.Mode = LL_DMA_MODE_NORMAL; + led_driver->dma_led_transition_timer.NbData = LED_DRIVER_BUFFER_SIZE; + // When to perform data exchange + led_driver->dma_led_transition_timer.PeriphRequest = LL_DMAMUX_REQ_TIM2_UP; + led_driver->dma_led_transition_timer.Priority = LL_DMA_PRIORITY_HIGH; +} + +LedDriver* led_driver_alloc(int count_leds, const GpioPin* gpio) { + furi_assert(gpio); + furi_assert(count_leds && count_leds <= MAX_LED_COUNT); + + LedDriver* led_driver = malloc(sizeof(LedDriver)); + led_driver_init_dma_gpio_update(led_driver, gpio); + led_driver_init_dma_led_transition_timer(led_driver); + led_driver->led_data = malloc(MAX_LED_COUNT * sizeof(uint32_t)); + led_driver->dirty = true; + + led_driver->count_leds = count_leds; + + return led_driver; +} + +void led_driver_free(LedDriver* led_driver) { + furi_assert(led_driver); + + free(led_driver->led_data); + free(led_driver); +} + +void led_driver_set_pin(LedDriver* led_driver, const GpioPin* gpio) { + if(led_driver->gpio == gpio) { + return; + } + + led_driver_init_dma_gpio_update(led_driver, gpio); + led_driver->dirty = true; +} + +uint32_t led_driver_set_led(LedDriver* led_driver, uint32_t index, uint32_t rrggbb) { + furi_assert(led_driver); + if(index >= led_driver->count_leds) { + return 0xFFFFFFFF; + } + + uint32_t previous = led_driver->led_data[index]; + led_driver->led_data[index] = rrggbb; + led_driver->dirty |= previous != rrggbb; + + return previous; +} + +uint32_t led_driver_get_led(LedDriver* led_driver, uint32_t index) { + furi_assert(led_driver); + if(index >= led_driver->count_leds) { + return 0xFFFFFFFF; + } + + return led_driver->led_data[index]; +} + +static void led_driver_start_dma(LedDriver* led_driver) { + furi_assert(led_driver); + + LL_DMA_Init(DMA1, LL_DMA_CHANNEL_1, &led_driver->dma_gpio_update); + LL_DMA_Init(DMA1, LL_DMA_CHANNEL_2, &led_driver->dma_led_transition_timer); + + LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_1); + LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_2); +} + +static void led_driver_start_timer() { + furi_hal_bus_enable(FuriHalBusTIM2); + + LL_TIM_SetCounterMode(TIM2, LL_TIM_COUNTERMODE_UP); + LL_TIM_SetClockDivision(TIM2, LL_TIM_CLOCKDIVISION_DIV1); + LL_TIM_SetPrescaler(TIM2, 0); + // Updated by led_driver->dma_led_transition_timer.PeriphOrM2MSrcAddress + LL_TIM_SetAutoReload(TIM2, LED_DRIVER_TIMER_SETINEL); + LL_TIM_SetCounter(TIM2, 0); + + LL_TIM_EnableCounter(TIM2); + LL_TIM_EnableUpdateEvent(TIM2); + LL_TIM_EnableDMAReq_UPDATE(TIM2); + LL_TIM_GenerateEvent_UPDATE(TIM2); +} + +static void led_driver_stop_timer() { + LL_TIM_DisableCounter(TIM2); + LL_TIM_DisableUpdateEvent(TIM2); + LL_TIM_DisableDMAReq_UPDATE(TIM2); + furi_hal_bus_disable(FuriHalBusTIM2); +} + +static void led_driver_stop_dma() { + LL_DMA_DisableChannel(DMA1, LL_DMA_CHANNEL_1); + LL_DMA_DisableChannel(DMA1, LL_DMA_CHANNEL_2); + LL_DMA_ClearFlag_TC1(DMA1); + LL_DMA_ClearFlag_TC2(DMA1); +} + +static void led_driver_spin_lock(LedDriver* led_driver) { + const uint32_t prev_timer = DWT->CYCCNT; + const uint32_t wait_time = LED_DRIVER_SETINEL_WAIT_MS * SystemCoreClock / 1000; + + do { + /* Make sure it's started (allow 100 ticks), but then check for sentinel value. */ + if(TIM2->ARR == LED_DRIVER_TIMER_SETINEL && DWT->CYCCNT - prev_timer > 100) { + break; + } + + // 0xFF is fairly quick, make sure we didn't miss it. + if((DWT->CYCCNT - prev_timer > wait_time)) { + FURI_LOG_D( + "Demo", "0xFF not found (ARR 0x%08lx, read %lu)", TIM2->ARR, led_driver->read_pos); + led_driver->read_pos = led_driver->write_pos - 1; + break; + } + } while(true); +} + +static void led_driver_add_period_length(LedDriver* led_driver, uint32_t length) { + led_driver->timer_buffer[led_driver->write_pos++] = length; + led_driver->timer_buffer[led_driver->write_pos] = LED_DRIVER_TIMER_SETINEL; +} + +static void led_driver_add_period(LedDriver* led_driver, uint16_t duration_ns) { + furi_assert(led_driver); + + uint32_t reload_value = duration_ns / LED_DRIVER_TIMER_NANOSECOND; + + if(reload_value > 255) { + FURI_LOG_E("Demo", "reload_value: %ld", reload_value); + } + furi_check(reload_value > 0); + furi_check(reload_value < 256 * 256); + + led_driver_add_period_length(led_driver, reload_value - 1); +} + +static void led_driver_add_color(LedDriver* led_driver, uint32_t rrggbb) { + UNUSED(rrggbb); + + uint32_t ggrrbb = (rrggbb & 0xFF) | ((rrggbb & 0xFF00) << 8) | ((rrggbb & 0xFF0000) >> 8); + + for(int i = 23; i >= 0; i--) { + if(ggrrbb & (1 << i)) { + led_driver_add_period(led_driver, LED_DRIVER_T0L); + led_driver_add_period(led_driver, LED_DRIVER_T1L); + } else { + led_driver_add_period(led_driver, LED_DRIVER_T0H); + led_driver_add_period(led_driver, LED_DRIVER_T1H); + } + } +} + +void led_driver_transmit(LedDriver* led_driver, bool transmit_if_clean) { + furi_assert(led_driver); + + furi_assert(!led_driver->read_pos); + furi_assert(!led_driver->write_pos); + + if(!transmit_if_clean && !led_driver->dirty) { + FURI_LOG_D("LED_DRIVER", "Skipping transmit"); + return; + } + FURI_LOG_D("LED_DRIVER", "Transmit"); + + furi_hal_gpio_init(led_driver->gpio, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh); + furi_hal_gpio_write(led_driver->gpio, false); + + const uint32_t bit_set = led_driver->gpio->pin << GPIO_BSRR_BS0_Pos; + const uint32_t bit_reset = led_driver->gpio->pin << GPIO_BSRR_BR0_Pos; + + // Always start with LOW (reset) + led_driver->gpio_buf[0] = bit_reset; + led_driver->gpio_buf[1] = bit_set; + + for(size_t i = 0; i < LED_DRIVER_BUFFER_SIZE; i++) { + led_driver->timer_buffer[i] = LED_DRIVER_TIMER_SETINEL; + } + + led_driver_add_period(led_driver, LED_DRIVER_TRESETL); + for(size_t i = 0; i < led_driver->count_leds; i++) { + led_driver_add_color(led_driver, led_driver->led_data[i]); + } + led_driver->dma_led_transition_timer.NbData = led_driver->write_pos + 1; + + FURI_CRITICAL_ENTER(); + + led_driver_start_dma(led_driver); + led_driver_start_timer(); + + led_driver_spin_lock(led_driver); + + led_driver_stop_timer(); + led_driver_stop_dma(); + + FURI_CRITICAL_EXIT(); + + memset(led_driver->timer_buffer, LED_DRIVER_TIMER_SETINEL, LED_DRIVER_BUFFER_SIZE); + led_driver->read_pos = 0; + led_driver->write_pos = 0; + led_driver->dirty = false; +} diff --git a/non_catalog_apps/ws2812b_tester/led_driver.h b/non_catalog_apps/ws2812b_tester/led_driver.h new file mode 100644 index 00000000000..9dfba6aa9b1 --- /dev/null +++ b/non_catalog_apps/ws2812b_tester/led_driver.h @@ -0,0 +1,13 @@ +#include +#include + +#define MAX_LED_COUNT 512 + +typedef struct LedDriver LedDriver; + +LedDriver* led_driver_alloc(int count_leds, const GpioPin* gpio); +void led_driver_free(LedDriver* led_driver); +void led_driver_set_pin(LedDriver* led_driver, const GpioPin* gpio); +uint32_t led_driver_set_led(LedDriver* led_driver, uint32_t index, uint32_t rrggbb); +uint32_t led_driver_get_led(LedDriver* led_driver, uint32_t index); +void led_driver_transmit(LedDriver* led_driver, bool transmit_if_clean);