led.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. /*
  2. This example code is in the Public Domain (or CC0 licensed, at your option.)
  3. Unless required by applicable law or agreed to in writing, this
  4. software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
  5. CONDITIONS OF ANY KIND, either express or implied.
  6. */
  7. #include <stdio.h>
  8. #include <stdlib.h>
  9. #include <unistd.h>
  10. #include <string.h>
  11. #include "freertos/FreeRTOS.h"
  12. #include "freertos/task.h"
  13. #include "freertos/timers.h"
  14. #include "esp_system.h"
  15. #include "esp_log.h"
  16. #include "driver/gpio.h"
  17. #include "led.h"
  18. #define MAX_LED 8
  19. #define BLOCKTIME 10 // up to portMAX_DELAY
  20. static struct led_s {
  21. gpio_num_t gpio;
  22. bool on;
  23. int onstate;
  24. int ontime, offtime;
  25. int pushedon, pushedoff;
  26. bool pushed;
  27. TimerHandle_t timer;
  28. } leds[MAX_LED];
  29. static void vCallbackFunction( TimerHandle_t xTimer ) {
  30. struct led_s *led = (struct led_s*) pvTimerGetTimerID (xTimer);
  31. if (!led->timer) return;
  32. led->on = !led->on;
  33. gpio_set_level(led->gpio, led->on ? led->onstate : !led->onstate);
  34. // was just on for a while
  35. if (!led->on && led->offtime == -1) return;
  36. // regular blinking
  37. xTimerChangePeriod(xTimer, (led->on ? led->ontime : led->offtime) / portTICK_RATE_MS, BLOCKTIME);
  38. }
  39. bool led_blink_core(int idx, int ontime, int offtime, bool pushed) {
  40. if (!leds[idx].gpio) return false;
  41. if (leds[idx].timer) {
  42. // normal requests waits if a pop is pending
  43. if (!pushed && leds[idx].pushed) {
  44. leds[idx].pushedon = ontime;
  45. leds[idx].pushedoff = offtime;
  46. return true;
  47. }
  48. xTimerStop(leds[idx].timer, BLOCKTIME);
  49. }
  50. // save current state if not already pushed
  51. if (!leds[idx].pushed) {
  52. leds[idx].pushedon = leds[idx].ontime;
  53. leds[idx].pushedoff = leds[idx].offtime;
  54. leds[idx].pushed = pushed;
  55. }
  56. // then set new one
  57. leds[idx].ontime = ontime;
  58. leds[idx].offtime = offtime;
  59. if (ontime == 0) {
  60. gpio_set_level(leds[idx].gpio, !leds[idx].onstate);
  61. } else if (offtime == 0) {
  62. gpio_set_level(leds[idx].gpio, leds[idx].onstate);
  63. } else {
  64. if (!leds[idx].timer) leds[idx].timer = xTimerCreate("ledTimer", ontime / portTICK_RATE_MS, pdFALSE, (void *)&leds[idx], vCallbackFunction);
  65. leds[idx].on = true;
  66. gpio_set_level(leds[idx].gpio, leds[idx].onstate);
  67. if (xTimerStart(leds[idx].timer, BLOCKTIME) == pdFAIL) return false;
  68. }
  69. return true;
  70. }
  71. bool led_unpush(int idx) {
  72. if (!leds[idx].gpio) return false;
  73. led_blink_core(idx, leds[idx].pushedon, leds[idx].pushedoff, true);
  74. leds[idx].pushed = false;
  75. return true;
  76. }
  77. bool led_config(int idx, gpio_num_t gpio, int onstate) {
  78. if (idx >= MAX_LED) return false;
  79. leds[idx].gpio = gpio;
  80. leds[idx].onstate = onstate;
  81. gpio_pad_select_gpio(gpio);
  82. gpio_set_direction(gpio, GPIO_MODE_OUTPUT);
  83. gpio_set_level(gpio, !onstate);
  84. return true;
  85. }
  86. bool led_unconfig(int idx) {
  87. if (idx >= MAX_LED) return false;
  88. if (leds[idx].timer) xTimerDelete(leds[idx].timer, BLOCKTIME);
  89. leds[idx].timer = NULL;
  90. return true;
  91. }