| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400 | 
							- /* 
 
-  *  Control of LED strip within squeezelite-esp32 
 
-  *     
 
-  *  (c) Wizmo 2021
 
-  *
 
-  *  Loosely based on code by 
 
-  *     Chuck Rohs 2020, chuck@zethus.ca
 
-  * 
 
-  *  This software is released under the MIT License.
 
-  *  https://opensource.org/licenses/MIT
 
-  *
 
-  * ToDo:
 
-  * Driver does support other led device. Maybe look at supporting in future. 
 
-  * The VU refresh rate has been decreaced (100->75) to optimize animation of spin dial.  Could make
 
-  *   configurable like text scrolling (or use the same value) 
 
-  * Artwork function, but not released as very buggy and not really practical
 
-  */
 
- #include <ctype.h>
 
- #include <math.h>
 
- #include "esp_log.h"
 
- #include "globdefs.h"
 
- #include "monitor.h"
 
- #include "led_strip.h"
 
- #include "platform_config.h"
 
- #include "led_vu.h"
 
- static const char *TAG = "led_vu";
 
- static void (*battery_handler_chain)(float value, int cells);
 
- static void battery_svc(float value, int cells);
 
- static int battery_status = 0;
 
- #define LED_VU_STACK_SIZE (3*1024)
 
- #define LED_VU_PEAK_HOLD 6U
 
- #define LED_VU_DEFAULT_GPIO 22
 
- #define LED_VU_DEFAULT_LENGTH 19
 
- #define LED_VU_MAX_LENGTH 255
 
- #define LED_VU_STATUS_GREEN 75
 
- #define LED_VU_STATUS_RED 25
 
- #define max(a,b) (((a) > (b)) ? (a) : (b))
 
- struct led_strip_t* led_display = NULL;
 
- static EXT_RAM_ATTR struct led_strip_t  led_strip_config;
 
- static EXT_RAM_ATTR struct {
 
-     int gpio;
 
-     int length;
 
-     int vu_length;
 
-     int vu_start_l;
 
-     int vu_start_r;
 
-     int vu_status;
 
- } strip;
 
- static int led_addr(int pos ) {
 
-     if (pos < 0) return pos + strip.length;
 
-     if (pos >= strip.length) return pos - strip.length;
 
-     return pos;
 
- }
 
- static void battery_svc(float value, int cells) {
 
- 	battery_status = battery_level_svc(); 
 
- 	ESP_LOGI(TAG, "Called for battery service with volt:%f cells:%d status:%d", value, cells, battery_status);
 
- 	if (battery_handler_chain) battery_handler_chain(value, cells);
 
- }
 
- /****************************************************************************************
 
-  * Initialize the led vu strip if configured.
 
-  * 
 
-  */
 
- void led_vu_init()
 
- {
 
-     char* p;
 
-     char* config = config_alloc_get_str("led_vu_config", NULL, "N/A");
 
-     // Initialize led VU strip 
 
-     char* drivername = strcasestr(config, "WS2812");
 
-     if ((p = strcasestr(config, "length")) != NULL) {
 
-         strip.length = atoi(strchr(p, '=') + 1);
 
-     } // else 0
 
-     if ((p = strcasestr(config, "gpio")) != NULL) {
 
-         strip.gpio = atoi(strchr(p, '=') + 1);
 
-     } else {
 
-         strip.gpio = LED_VU_DEFAULT_GPIO;
 
-     }
 
-     // check for valid configuration
 
-     if (!drivername || !strip.gpio) {
 
-         ESP_LOGI(TAG, "led_vu configuration invalid");
 
-         goto done;
 
-     }
 
-     battery_handler_chain = battery_handler_svc;
 
-     battery_handler_svc = battery_svc;
 
-     battery_status = battery_level_svc();
 
-    
 
-     if (strip.length > LED_VU_MAX_LENGTH) strip.length = LED_VU_MAX_LENGTH;
 
-     // initialize vu meter settings
 
-     if (strip.length < 10) {
 
-         // single bar for small strips
 
-         strip.vu_length = strip.length;
 
-         strip.vu_start_l  = 0;
 
-         strip.vu_start_r = strip.vu_start_l;
 
-         strip.vu_status = 0;
 
-     } else {
 
-         strip.vu_length = (strip.length  - 1) / 2;
 
-         strip.vu_start_l  = (strip.length % 2) ? strip.vu_length -1 : strip.vu_length;
 
-         strip.vu_start_r = strip.vu_length + 1;
 
-         strip.vu_status = strip.vu_length;
 
-     }
 
-     ESP_LOGI(TAG, "vu meter using length:%d left:%d right:%d status:%d", strip.vu_length, strip.vu_start_l, strip.vu_start_r, strip.vu_status);
 
-     // create driver configuration
 
-     led_strip_config.rgb_led_type = RGB_LED_TYPE_WS2812;
 
-     led_strip_config.access_semaphore = xSemaphoreCreateBinary();
 
-     led_strip_config.led_strip_length = strip.length;
 
-     led_strip_config.led_strip_working = heap_caps_malloc(strip.length * sizeof(struct led_color_t), MALLOC_CAP_8BIT);
 
-     led_strip_config.led_strip_showing = heap_caps_malloc(strip.length * sizeof(struct led_color_t), MALLOC_CAP_8BIT);
 
-     led_strip_config.gpio = strip.gpio;
 
-     led_strip_config.rmt_channel = RMT_NEXT_TX_CHANNEL();
 
-     // initialize driver 
 
-     bool led_init_ok = led_strip_init(&led_strip_config);
 
-     if (led_init_ok) {
 
-         led_display = &led_strip_config;
 
-         ESP_LOGI(TAG, "led_vu using gpio:%d length:%d on channel:%d", strip.gpio, strip.length, led_strip_config.rmt_channel);
 
-     } else {
 
-         ESP_LOGE(TAG, "led_vu init failed");
 
-         goto done;
 
-     }
 
-     // reserver max memory for remote management systems
 
-     rmt_set_mem_block_num(led_strip_config.rmt_channel, 7);
 
-     led_vu_clear(led_display);
 
-     done:
 
-         free(config);
 
-         return;
 
-     }
 
- inline bool inRange(double x, double y, double z) {
 
-     return (x > y && x < z);
 
- }
 
- /****************************************************************************************
 
-  * Returns the led strip length
 
-  */
 
- uint16_t led_vu_string_length() {
 
-     if (!led_display) return 0;
 
-     return (uint16_t)strip.length;
 
- }
 
- /****************************************************************************************
 
-  * Turns all LEDs off (Black)
 
-  */
 
- void led_vu_clear() {
 
-     if (!led_display) return;
 
-     led_strip_clear(led_display);
 
-     led_strip_show(led_display);
 
- }
 
- /****************************************************************************************
 
-  * Sets all LEDs to one color
 
-  * r = red (0-255), g = green (0-255), b - blue (0-255)
 
-  *      note - all colors are adjusted for brightness
 
-  */
 
- void led_vu_color_all(uint8_t r, uint8_t g, uint8_t b) {
 
-     if (!led_display) return;
 
-     struct led_color_t color_on = {.red = r, .green = g, .blue = b}; 
 
-     for (int i = 0 ; i < strip.length ; i ++){
 
-         led_strip_set_pixel_color(led_display, i, &color_on);
 
-     }
 
-     led_strip_show(led_display);
 
- }
 
- /****************************************************************************************
 
-  * Sets LEDs based on a data packet consiting of rgb data
 
-  * offset - starting LED,
 
-  * length - number of leds (3x rgb bytes) 
 
-  * data - array of rgb values in multiples of 3 bytes
 
-  */
 
- void led_vu_data(uint8_t* data, uint16_t offset, uint16_t length) {
 
-     if (!led_display) return;
 
- 	uint8_t* p = (uint8_t*) data;									        
 
- 	for (int i = 0; i < length; i++) {					            
 
- 		led_strip_set_pixel_rgb(led_display, i+offset, *p, *(p+1), *(p+2));
 
-         p+=3;
 
- 	} 
 
-     led_strip_show(led_display);
 
- }
 
- /****************************************************************************************
 
-  * Progress bar display
 
-  * data - array of gain values(0-100)
 
-  * offset - starting position
 
-  * length - size of array
 
-  */
 
- void led_vu_spectrum(uint8_t* data, int bright, int length, int style) {
 
-     if (!led_display) return;
 
-     uint8_t gain,r,g,b;
 
-     int width = strip.length / length;
 
-     int pos = 0;
 
-     uint8_t* p = (uint8_t*) data;									        
 
- 	for (int i=0; i<length; i++) {
 
- 		gain = *p;
 
-         r = gain*gain/bright;
 
-         if (!style) {
 
-             g = 0;
 
-             b = gain;
 
-         } else {
 
-             g = r;
 
-             r = 0;
 
-             b = gain * (bright-gain)/bright;
 
-         }
 
-         for (int j=0; j<width; j++) {
 
-             led_strip_set_pixel_rgb(led_display, pos, r, g, b);
 
-             pos++;
 
-         }
 
-         p++;
 
-     }
 
-     
 
-     led_strip_show(led_display);
 
-  }
 
- /****************************************************************************************
 
-  * Progress bar display
 
-  * pct - percentage complete (0-100)
 
-  */
 
- void led_vu_progress_bar(int pct, int bright) {
 
-     if (!led_display) return;
 
-     // define colors
 
-     struct led_color_t color_on   = {.red = bright, .green = 0, .blue = 0};
 
-     struct led_color_t color_off = {.red = 0, .green = bright, .blue = 0};
 
-     // calcuate led position
 
-     int led_lit = strip.length * pct / 100;
 
-     // set colors
 
-     for (int i = 0; i < strip.length; i++) {
 
-         led_strip_set_pixel_color(led_display, i, (i < led_lit) ? &color_off : &color_on);
 
-     }
 
-     led_strip_show(led_display);
 
- }
 
- /****************************************************************************************
 
-  * Spin dial display
 
-  * gain - brightness (0-100), rate - color change speed (0-100) 
 
-  * comet - alternate display mode
 
-  */
 
- void led_vu_spin_dial(int gain, int rate, int speed, bool comet) 
 
- {
 
-     if (!led_display) return;
 
-     static int led_pos = 0;
 
-     static uint8_t r = 0;
 
-     static uint8_t g = 0;
 
-     static uint8_t b = 0;
 
-     
 
-     // calculate next color
 
-     uint8_t step = rate / 2; // controls color change speed
 
-     if (r == 0 && g == 0 && b == 0) {
 
-         r = LED_VU_MAX; g = step; 
 
-     } else if (b == 0) {
 
-         g = (g > LED_VU_MAX-step) ? LED_VU_MAX : g + step;
 
-         r = (r < step) ? 0 : r - step;
 
-         if (r == 0) b = step;
 
-     } else if (r == 0) {
 
-         b = (b > LED_VU_MAX-step) ? LED_VU_MAX : b + step;
 
-         g = (g < step) ? 0 : g- step;
 
-         if (g == 0) r = step;
 
-     } else { 
 
-         r = (r > LED_VU_MAX-step) ? LED_VU_MAX : r + step;
 
-         b = (b < step) ? 0 : b - step;
 
-         if (r == 0) b = step;
 
-     }
 
-     uint8_t rp = r * gain / LED_VU_MAX; 
 
-     uint8_t gp = g * gain / LED_VU_MAX; 
 
-     uint8_t bp = b * gain / LED_VU_MAX; 
 
-     // set led color
 
-     speed++;
 
-     if (comet) {
 
-         led_strip_clear(led_display);
 
-         led_strip_set_pixel_rgb(led_display, led_addr(led_pos-1), rp/2, gp/2, bp/2);
 
-         led_strip_set_pixel_rgb(led_display, led_addr(led_pos-2), rp/4, gp/4, bp/4);
 
-         led_strip_set_pixel_rgb(led_display, led_addr(led_pos-3), rp/8, gp/8, bp/8);
 
-         //led_strip_set_pixel_rgb(led_display, led_addr(led_pos-4), 0, 0, 0);
 
-     }
 
-     for (int i = 0; i < speed; i++) {
 
-         led_strip_set_pixel_rgb(led_display, led_pos, rp, gp, bp);
 
-         led_pos = led_addr(++led_pos);
 
-     }
 
-     
 
-     led_strip_show(led_display);
 
- }
 
- /****************************************************************************************
 
-  * VU meter display
 
-  * vu_l - left response (0-100), vu_r - right response (0-100)
 
-  * comet - alternate display mode
 
-  */
 
- void led_vu_display(int vu_l, int vu_r, int bright, bool comet) {
 
-     static int peak_l = 0;
 
-     static int peak_r = 0;
 
-     static int decay_l = 0;
 
-     static int decay_r = 0;
 
-     if (!led_display) return;
 
-     // single bar
 
-     if (strip.vu_start_l == strip.vu_start_r) {
 
-         vu_r = (vu_l + vu_r) / 2;
 
-         vu_l = 0;
 
-     }
 
-     // scale vu samples to length
 
-     vu_l  = vu_l * strip.vu_length / bright;
 
-     vu_r = vu_r * strip.vu_length / bright;
 
-     // calculate hold peaks
 
-     if (peak_l > vu_l) {
 
-         if (decay_l-- < 0) {
 
-             decay_l = LED_VU_PEAK_HOLD;
 
-             peak_l--;
 
-         }
 
-     } else {
 
-         peak_l = vu_l;
 
-         decay_l = LED_VU_PEAK_HOLD;
 
-     }
 
-     if (peak_r > vu_r) {
 
-         if (decay_r-- < 0) {
 
-             decay_r = LED_VU_PEAK_HOLD;
 
-             peak_r--;
 
-         }
 
-     } else {
 
-         peak_r = vu_r;
 
-         decay_r = LED_VU_PEAK_HOLD;
 
-     }
 
-     // turn off all leds
 
-     led_strip_clear(led_display);
 
-     // set the led bar values
 
-     uint8_t step = bright / (strip.vu_length-1);
 
-     if (step < 1) step = 1; // dor low brightness or larger strips     
 
-     uint8_t g = bright * 2 / 3; // more red at top
 
-     uint8_t r = 0;
 
-     int shift = 0;
 
-     for (int i = 0; i < strip.vu_length; i++) {
 
-         // set left
 
-         if (i == peak_l) {
 
-             led_strip_set_pixel_rgb(led_display, strip.vu_start_l - i, r, g, bright);
 
-         } else if (i <= vu_l) {
 
-             shift = vu_l - i; 
 
-             if (comet)
 
-                 led_strip_set_pixel_rgb(led_display, strip.vu_start_l - i, r>>shift, g>>shift, 0);
 
-             else
 
-                 led_strip_set_pixel_rgb(led_display, strip.vu_start_l - i, r, g, 0);
 
-         }
 
-         // set right  
 
-         if (i == peak_r) {
 
-             led_strip_set_pixel_rgb(led_display, strip.vu_start_r + i, r, g, bright);
 
-         }  else if (i <= vu_r) {
 
-             shift = vu_r - i;
 
-             if (comet)
 
-                 led_strip_set_pixel_rgb(led_display, strip.vu_start_r + i, r>>shift, g>>shift, 0);
 
-             else
 
-                 led_strip_set_pixel_rgb(led_display, strip.vu_start_r + i, r, g, 0);
 
-         }
 
-         // adjust colors (with limit checks)
 
-         r = (r > bright-step) ? bright : r + step;
 
-         g = (g < step) ? 0 : g - step;
 
-     }
 
-     // show battery status
 
-     if (battery_status > LED_VU_STATUS_GREEN)
 
-         led_strip_set_pixel_rgb(led_display, strip.vu_status, 0, bright, 0);
 
-     else if (battery_status > LED_VU_STATUS_RED)
 
-         led_strip_set_pixel_rgb(led_display, strip.vu_status, bright/2, bright/2, 0);
 
-     else if (battery_status > 0)
 
-         led_strip_set_pixel_rgb(led_display, strip.vu_status, bright, 0, 0);
 
-     
 
-     led_strip_show(led_display);
 
- }
 
 
  |