cmd_squeezelite.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #include "../cmd_system.h"
  2. #include "Config.h"
  3. #include "application_name.h"
  4. #include "argtable3/argtable3.h"
  5. #include "esp_app_format.h"
  6. #include "esp_console.h"
  7. #include "esp_log.h"
  8. #include "freertos/FreeRTOS.h"
  9. #include "freertos/event_groups.h"
  10. #include "messaging.h"
  11. #include "platform_esp32.h"
  12. #include "pthread.h"
  13. #include "tools.h"
  14. #include <stdio.h>
  15. #include <string.h>
  16. #include "bootstate.h"
  17. extern esp_err_t process_recovery_ota(const char* bin_url, char* bin_buffer, uint32_t length);
  18. extern int squeezelite_main_start();
  19. extern sys_dac_default_sets* default_dac_sets;
  20. static const char* TAG = "squeezelite_cmd";
  21. #define SQUEEZELITE_THREAD_STACK_SIZE (8 * 1024)
  22. #define ADDITIONAL_SQUEEZELITE_ARGS 5
  23. const __attribute__((section(".rodata_desc"))) esp_app_desc_t esp_app_desc = {
  24. .magic_word = ESP_APP_DESC_MAGIC_WORD,
  25. .version = PROJECT_VER,
  26. .project_name = CONFIG_PROJECT_NAME,
  27. .idf_ver = IDF_VER,
  28. #ifdef CONFIG_BOOTLOADER_APP_SECURE_VERSION
  29. .secure_version = CONFIG_BOOTLOADER_APP_SECURE_VERSION,
  30. #else
  31. .secure_version = 0,
  32. #endif
  33. #ifdef CONFIG_APP_COMPILE_TIME_DATE
  34. .time = __TIME__,
  35. .date = __DATE__,
  36. #else
  37. .time = "",
  38. .date = "",
  39. #endif
  40. };
  41. extern void register_audio_config(void);
  42. // extern void register_rotary_config(void);
  43. extern void register_nvs();
  44. extern cJSON* get_gpio_list_handler(bool refresh);
  45. cJSON* get_gpio_list(bool refresh) {
  46. #if CONFIG_WITH_CONFIG_UI
  47. return get_gpio_list_handler(refresh);
  48. #else
  49. return cJSON_CreateArray();
  50. #endif
  51. }
  52. void register_optional_cmd(void) {
  53. // #if CONFIG_WITH_CONFIG_UI
  54. // register_rotary_config();
  55. // #endif
  56. register_audio_config();
  57. // register_ledvu_config();
  58. }
  59. static struct {
  60. int argc;
  61. char** argv;
  62. } thread_parms;
  63. static void squeezelite_thread(void* arg) {
  64. ESP_LOGI(TAG, "Calling squeezelite");
  65. int wait = 60;
  66. int ret = squeezelite_main_start();
  67. if (ret == -2) {
  68. cmd_send_messaging("cfg-audio-tmpl", MESSAGING_ERROR,
  69. "Squeezelite not configured. Rebooting to factory in %d sec\n", wait);
  70. vTaskDelay(pdMS_TO_TICKS(wait * 1000));
  71. guided_factory();
  72. } else {
  73. cmd_send_messaging("cfg-audio-tmpl", ret > 1 ? MESSAGING_ERROR : MESSAGING_WARNING,
  74. "squeezelite exited with error code %d\n", ret);
  75. if (ret <= 1) {
  76. cmd_send_messaging(
  77. "cfg-audio-tmpl", MESSAGING_ERROR, "Rebooting to factory in %d sec\n", wait);
  78. vTaskDelay(pdMS_TO_TICKS(wait * 1000));
  79. guided_factory();
  80. } else {
  81. cmd_send_messaging(
  82. "cfg-audio-tmpl", MESSAGING_ERROR, "Correct command line and reboot\n");
  83. vTaskSuspend(NULL);
  84. }
  85. }
  86. ESP_LOGV(TAG, "Exited from squeezelite's main(). Freeing argv structure.");
  87. for (int i = 0; i < thread_parms.argc; i++)
  88. free(thread_parms.argv[i]);
  89. free(thread_parms.argv);
  90. }
  91. static int launchsqueezelite(int argc, char** argv) {
  92. static DRAM_ATTR StaticTask_t xTaskBuffer __attribute__((aligned(4)));
  93. static EXT_RAM_ATTR StackType_t xStack[SQUEEZELITE_THREAD_STACK_SIZE]
  94. __attribute__((aligned(4)));
  95. static bool isRunning = false;
  96. if (isRunning) {
  97. ESP_LOGE(TAG, "Squeezelite already running. Exiting!");
  98. return -1;
  99. }
  100. ESP_LOGV(TAG, "Begin");
  101. isRunning = true;
  102. // load the presets here, as the squeezelite cannot access the
  103. // spiffs storage
  104. // proto_load_file("/spiffs/presets/default_sets.bin", &sys_dac_default_sets_msg, default_dac_sets, false);
  105. ESP_LOGD(TAG, "Starting Squeezelite Thread");
  106. xTaskCreateStaticPinnedToCore(squeezelite_thread, "squeezelite", SQUEEZELITE_THREAD_STACK_SIZE,
  107. NULL, CONFIG_ESP32_PTHREAD_TASK_PRIO_DEFAULT, xStack, &xTaskBuffer,
  108. CONFIG_PTHREAD_TASK_CORE_DEFAULT);
  109. return 0;
  110. }
  111. void start_squeezelite() { launchsqueezelite(0, NULL); }
  112. // void register_squeezelite() {
  113. // squeezelite_args.parameters = arg_str0(NULL, NULL, "<parms>", "command line for squeezelite. -h
  114. // for help, --defaults to launch with default values."); squeezelite_args.end = arg_end(1); const
  115. // esp_console_cmd_t launch_squeezelite = { .command = "squeezelite", .help = "Starts squeezelite",
  116. // .hint = NULL,
  117. // .func = &launchsqueezelite,
  118. // .argtable = &squeezelite_args
  119. // };
  120. // ESP_ERROR_CHECK( esp_console_cmd_register(&launch_squeezelite) );
  121. // }
  122. esp_err_t start_ota(const char* bin_url, char* bin_buffer, uint32_t length) {
  123. if (!bin_url || strlen(bin_url) == 0) {
  124. ESP_LOGE(TAG, "missing URL parameter. Unable to start OTA");
  125. return ESP_ERR_INVALID_ARG;
  126. }
  127. ESP_LOGW(TAG, "Called to update the firmware from url: %s", bin_url);
  128. system_set_string(&sys_state_data_msg, sys_state_data_ota_url_tag, sys_state, bin_url);
  129. config_raise_state_changed();
  130. if (!config_waitcommit()) {
  131. ESP_LOGW(TAG, "Unable to commit configuration. ");
  132. }
  133. ESP_LOGW(TAG, "Rebooting to recovery to complete the installation");
  134. return guided_factory();
  135. return ESP_OK;
  136. }