monitor.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  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/timers.h"
  13. #include "esp_system.h"
  14. #include "esp_log.h"
  15. #include "esp_task.h"
  16. #include "monitor.h"
  17. #include "driver/gpio.h"
  18. #include "buttons.h"
  19. #include "led.h"
  20. #include "globdefs.h"
  21. // #include "Configurator.h"
  22. // TODO: Add support for the commented code: search for TODO in the code below")
  23. #include "accessors.h"
  24. #include "messaging.h"
  25. #include "cJSON.h"
  26. #include "tools.h"
  27. #define PSEUDO_IDLE_STACK_SIZE (6*1024)
  28. #define MONITOR_TIMER (10*1000)
  29. #define SCRATCH_SIZE 256
  30. static const char *TAG = "monitor";
  31. void (*pseudo_idle_svc)(uint32_t now);
  32. void (*jack_handler_svc)(bool inserted);
  33. bool jack_inserted_svc(void);
  34. void (*spkfault_handler_svc)(bool inserted);
  35. bool spkfault_svc(void);
  36. static bool monitor_stats;
  37. /****************************************************************************************
  38. *
  39. */
  40. static void task_stats( cJSON* top ) {
  41. #ifdef CONFIG_FREERTOS_USE_TRACE_FACILITY
  42. #pragma message("Compiled with trace facility")
  43. static struct {
  44. TaskStatus_t *tasks;
  45. uint32_t total, n;
  46. } current, previous;
  47. cJSON * tlist=cJSON_CreateArray();
  48. current.n = uxTaskGetNumberOfTasks();
  49. current.tasks = malloc_init_external( current.n * sizeof( TaskStatus_t ) );
  50. current.n = uxTaskGetSystemState( current.tasks, current.n, &current.total );
  51. cJSON_AddNumberToObject(top,"ntasks",current.n);
  52. char scratch[SCRATCH_SIZE] = { };
  53. #ifdef CONFIG_FREERTOS_GENERATE_RUN_TIME_STATS
  54. #pragma message("Compiled with runtime stats")
  55. uint32_t elapsed = current.total - previous.total;
  56. for(int i = 0, n = 0; i < current.n; i++ ) {
  57. for (int j = 0; j < previous.n; j++) {
  58. if (current.tasks[i].xTaskNumber == previous.tasks[j].xTaskNumber) {
  59. n += snprintf(scratch + n, SCRATCH_SIZE - n, "%16s (%u) %2u%% s:%5u", current.tasks[i].pcTaskName,
  60. current.tasks[i].eCurrentState,
  61. 100 * (current.tasks[i].ulRunTimeCounter - previous.tasks[j].ulRunTimeCounter) / elapsed,
  62. current.tasks[i].usStackHighWaterMark);
  63. cJSON * t=cJSON_CreateObject();
  64. cJSON_AddNumberToObject(t,"cpu",100 * (current.tasks[i].ulRunTimeCounter - previous.tasks[j].ulRunTimeCounter) / elapsed);
  65. cJSON_AddNumberToObject(t,"minstk",current.tasks[i].usStackHighWaterMark);
  66. cJSON_AddNumberToObject(t,"bprio",current.tasks[i].uxBasePriority);
  67. cJSON_AddNumberToObject(t,"cprio",current.tasks[i].uxCurrentPriority);
  68. cJSON_AddStringToObject(t,"nme",current.tasks[i].pcTaskName);
  69. cJSON_AddNumberToObject(t,"st",current.tasks[i].eCurrentState);
  70. cJSON_AddNumberToObject(t,"num",current.tasks[i].xTaskNumber);
  71. cJSON_AddItemToArray(tlist,t);
  72. if (i % 3 == 2 || i == current.n - 1) {
  73. ESP_LOGI(TAG, "%s", scratch);
  74. n = 0;
  75. }
  76. break;
  77. }
  78. }
  79. }
  80. #else
  81. #pragma message("Compiled WITHOUT runtime stats")
  82. for (int i = 0, n = 0; i < current.n; i ++) {
  83. n += sprintf(scratch + n, "%16s s:%5u\t", current.tasks[i].pcTaskName, current.tasks[i].usStackHighWaterMark);
  84. cJSON * t=cJSON_CreateObject();
  85. cJSON_AddNumberToObject(t,"minstk",current.tasks[i].usStackHighWaterMark);
  86. cJSON_AddNumberToObject(t,"bprio",current.tasks[i].uxBasePriority);
  87. cJSON_AddNumberToObject(t,"cprio",current.tasks[i].uxCurrentPriority);
  88. cJSON_AddStringToObject(t,"nme",current.tasks[i].pcTaskName);
  89. cJSON_AddNumberToObject(t,"st",current.tasks[i].eCurrentState);
  90. cJSON_AddNumberToObject(t,"num",current.tasks[i].xTaskNumber);
  91. cJSON_AddItemToArray(tlist,t);
  92. if (i % 3 == 2 || i == current.n - 1) {
  93. ESP_LOGI(TAG, "%s", scratch);
  94. n = 0;
  95. }
  96. }
  97. #endif
  98. cJSON_AddItemToObject(top,"tasks",tlist);
  99. if (previous.tasks) free(previous.tasks);
  100. previous = current;
  101. #else
  102. #pragma message("Compiled WITHOUT trace facility")
  103. #endif
  104. }
  105. /****************************************************************************************
  106. *
  107. */
  108. static void monitor_trace(uint32_t now) {
  109. static uint32_t last;
  110. if (now < last + MONITOR_TIMER) return;
  111. last = now;
  112. cJSON * top=cJSON_CreateObject();
  113. cJSON_AddNumberToObject(top,"free_iram",heap_caps_get_free_size(MALLOC_CAP_INTERNAL));
  114. cJSON_AddNumberToObject(top,"min_free_iram",heap_caps_get_minimum_free_size(MALLOC_CAP_INTERNAL));
  115. cJSON_AddNumberToObject(top,"free_spiram",heap_caps_get_free_size(MALLOC_CAP_SPIRAM));
  116. cJSON_AddNumberToObject(top,"min_free_spiram",heap_caps_get_minimum_free_size(MALLOC_CAP_SPIRAM));
  117. ESP_LOGI(TAG, "Heap internal:%zu (min:%zu) external:%zu (min:%zu) dma:%zu (min:%zu)",
  118. heap_caps_get_free_size(MALLOC_CAP_INTERNAL),
  119. heap_caps_get_minimum_free_size(MALLOC_CAP_INTERNAL),
  120. heap_caps_get_free_size(MALLOC_CAP_SPIRAM),
  121. heap_caps_get_minimum_free_size(MALLOC_CAP_SPIRAM),
  122. heap_caps_get_free_size(MALLOC_CAP_DMA),
  123. heap_caps_get_minimum_free_size(MALLOC_CAP_DMA));
  124. task_stats(top);
  125. char * top_a= cJSON_PrintUnformatted(top);
  126. if(top_a){
  127. messaging_post_message(MESSAGING_INFO, MESSAGING_CLASS_STATS,top_a);
  128. FREE_AND_NULL(top_a);
  129. }
  130. cJSON_Delete(top);
  131. }
  132. /****************************************************************************************
  133. *
  134. */
  135. static void jack_handler_default(void *id, button_event_e event, button_press_e mode, bool long_press) {
  136. ESP_LOGI(TAG, "Jack %s", event == BUTTON_PRESSED ? "inserted" : "removed");
  137. if (jack_handler_svc) (*jack_handler_svc)(event == BUTTON_PRESSED);
  138. }
  139. /****************************************************************************************
  140. *
  141. */
  142. bool jack_inserted_svc (void) {
  143. sys_GPIO * jack=NULL;
  144. if(SYS_GPIOS_NAME(jack,jack)){
  145. return button_is_pressed(jack->pin, NULL);
  146. }
  147. return false;
  148. }
  149. /****************************************************************************************
  150. *
  151. */
  152. static void spkfault_handler_default(void *id, button_event_e event, button_press_e mode, bool long_press) {
  153. ESP_LOGD(TAG, "Speaker status %s", event == BUTTON_PRESSED ? "faulty" : "normal");
  154. if (event == BUTTON_PRESSED) led_on(LED_RED);
  155. else led_off(LED_RED);
  156. if (spkfault_handler_svc) (*spkfault_handler_svc)(event == BUTTON_PRESSED);
  157. }
  158. /****************************************************************************************
  159. *
  160. */
  161. bool spkfault_svc (void) {
  162. sys_GPIO * spkfault=NULL;
  163. if(SYS_GPIOS_NAME(spkfault,spkfault)){
  164. return button_is_pressed(spkfault->pin, NULL);
  165. }
  166. return false;
  167. }
  168. /****************************************************************************************
  169. *
  170. */
  171. static void pseudo_idle(void *arg) {
  172. while (1) {
  173. vTaskDelay(pdMS_TO_TICKS(1000));
  174. uint32_t now = pdTICKS_TO_MS(xTaskGetTickCount());
  175. if (monitor_stats) monitor_trace(now);
  176. if (pseudo_idle_svc) pseudo_idle_svc(now);
  177. }
  178. }
  179. /****************************************************************************************
  180. *
  181. */
  182. void monitor_svc_init(void) {
  183. ESP_LOGI(TAG, "Initializing monitoring");
  184. sys_Services * services = NULL;
  185. sys_GPIO * gpio=NULL;
  186. if(SYS_GPIOS_NAME(jack,gpio) && gpio->pin>=0){
  187. ESP_LOGI(TAG,"Adding jack (%s) detection GPIO %d", gpio->level ? "high" : "low", gpio->pin);
  188. button_create(NULL, gpio->pin, gpio->level ? BUTTON_HIGH : BUTTON_LOW, false, 250, jack_handler_default, 0, -1);
  189. }
  190. if(SYS_GPIOS_NAME(spkfault,gpio) && gpio->pin>=0){
  191. ESP_LOGI(TAG,"Adding speaker fault (%s) detection GPIO %d", gpio->level ? "high" : "low", gpio->pin);
  192. button_create(NULL, gpio->pin, gpio->level ? BUTTON_HIGH : BUTTON_LOW, false, 0, spkfault_handler_default, 0, -1);
  193. }
  194. // do we want stats
  195. monitor_stats = SYS_SERVICES(services) && services->statistics;
  196. ESP_LOGI(TAG, "Heap internal:%zu (min:%zu) external:%zu (min:%zu) dma:%zu (min:%zu)",
  197. heap_caps_get_free_size(MALLOC_CAP_INTERNAL),
  198. heap_caps_get_minimum_free_size(MALLOC_CAP_INTERNAL),
  199. heap_caps_get_free_size(MALLOC_CAP_SPIRAM),
  200. heap_caps_get_minimum_free_size(MALLOC_CAP_SPIRAM),
  201. heap_caps_get_free_size(MALLOC_CAP_DMA),
  202. heap_caps_get_minimum_free_size(MALLOC_CAP_DMA));
  203. // pseudo-idle callback => don't use FreeRTOS idle callbacks so we can block (should not but ...)
  204. StaticTask_t* xTaskBuffer = (StaticTask_t*) heap_caps_malloc(sizeof(StaticTask_t), MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT);
  205. static EXT_RAM_ATTR StackType_t xStack[PSEUDO_IDLE_STACK_SIZE] __attribute__ ((aligned (4)));
  206. xTaskCreateStatic( (TaskFunction_t) pseudo_idle, "pseudo_idle", PSEUDO_IDLE_STACK_SIZE,
  207. NULL, ESP_TASK_PRIO_MIN, xStack, xTaskBuffer );
  208. }