tools.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. /*
  2. * (c) Philippe G. 20201, philippe_44@outlook.com
  3. * see other copyrights below
  4. *
  5. * This software is released under the MIT License.
  6. * https://opensource.org/licenses/MIT
  7. *
  8. */
  9. #define LOG_LOCAL_LEVEL ESP_LOG_INFO
  10. #include "tools.h"
  11. #include "esp_heap_caps.h"
  12. #include "esp_log.h"
  13. #include "esp_task.h"
  14. #include <ctype.h>
  15. #include <stdint.h>
  16. #include <string.h>
  17. #if CONFIG_FREERTOS_THREAD_LOCAL_STORAGE_POINTERS < 2
  18. #error CONFIG_FREERTOS_THREAD_LOCAL_STORAGE_POINTERS must be at least 2
  19. #endif
  20. const static char TAG[] = "tools";
  21. const char unknown_string_placeholder[] = "unknown";
  22. const char null_string_placeholder[] = "null";
  23. /****************************************************************************************
  24. * UTF-8 tools
  25. */
  26. // Copyright (c) 2008-2009 Bjoern Hoehrmann <bjoern@hoehrmann.de>
  27. // See http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ for details.
  28. // Copyright (c) 2017 ZephRay <zephray@outlook.com>
  29. //
  30. // utf8to1252 - almost equivalent to iconv -f utf-8 -t windows-1252, but better
  31. #define UTF8_ACCEPT 0
  32. #define UTF8_REJECT 1
  33. static const uint8_t utf8d[] = {
  34. 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  35. 0, // 00..1f
  36. 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  37. 0, // 20..3f
  38. 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  39. 0, // 40..5f
  40. 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
  41. 0, // 60..7f
  42. 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
  43. 9, // 80..9f
  44. 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
  45. 7, // a0..bf
  46. 8, 8, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
  47. 2, // c0..df
  48. 0xa, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x4, 0x3, 0x3, // e0..ef
  49. 0xb, 0x6, 0x6, 0x6, 0x5, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, // f0..ff
  50. 0x0, 0x1, 0x2, 0x3, 0x5, 0x8, 0x7, 0x1, 0x1, 0x1, 0x4, 0x6, 0x1, 0x1, 0x1, 0x1, // s0..s0
  51. 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 1, 0, 1, 1, 1, 1, 1,
  52. 1, // s1..s2
  53. 1, 2, 1, 1, 1, 1, 1, 2, 1, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1,
  54. 1, // s3..s4
  55. 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 3, 1, 1, 1, 1, 1,
  56. 1, // s5..s6
  57. 1, 3, 1, 1, 1, 1, 1, 3, 1, 3, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
  58. 1, // s7..s8
  59. };
  60. /**
  61. * @fn static uint32_t decode(uint32_t* state, uint32_t* codep, uint32_t byte)
  62. * Decodes a single UTF-8 encoded byte.
  63. *
  64. * @param state A pointer to the current state of the UTF-8 decoder.
  65. * @param codep A pointer to the code point being built from the UTF-8 bytes.
  66. * @param byte The current byte to be decoded.
  67. * @return The new state after processing the byte.
  68. */
  69. static uint32_t decode(uint32_t* state, uint32_t* codep, uint32_t byte) {
  70. uint32_t type = utf8d[byte];
  71. *codep = (*state != UTF8_ACCEPT) ? (byte & 0x3fu) | (*codep << 6) : (0xff >> type) & (byte);
  72. *state = utf8d[256 + *state * 16 + type];
  73. return *state;
  74. }
  75. /**
  76. * @fn static uint8_t UNICODEtoCP1252(uint16_t chr)
  77. * Converts a Unicode character to its corresponding CP1252 character.
  78. *
  79. * @param chr The Unicode character to be converted.
  80. * @return The corresponding CP1252 character or 0x00 if there is no direct mapping.
  81. */
  82. static uint8_t UNICODEtoCP1252(uint16_t chr) {
  83. if (chr <= 0xff)
  84. return (chr & 0xff);
  85. else {
  86. ESP_LOGI(TAG, "some multi-byte %hx", chr);
  87. switch (chr) {
  88. case 0x20ac:
  89. return 0x80;
  90. break;
  91. case 0x201a:
  92. return 0x82;
  93. break;
  94. case 0x0192:
  95. return 0x83;
  96. break;
  97. case 0x201e:
  98. return 0x84;
  99. break;
  100. case 0x2026:
  101. return 0x85;
  102. break;
  103. case 0x2020:
  104. return 0x86;
  105. break;
  106. case 0x2021:
  107. return 0x87;
  108. break;
  109. case 0x02c6:
  110. return 0x88;
  111. break;
  112. case 0x2030:
  113. return 0x89;
  114. break;
  115. case 0x0160:
  116. return 0x8a;
  117. break;
  118. case 0x2039:
  119. return 0x8b;
  120. break;
  121. case 0x0152:
  122. return 0x8c;
  123. break;
  124. case 0x017d:
  125. return 0x8e;
  126. break;
  127. case 0x2018:
  128. return 0x91;
  129. break;
  130. case 0x2019:
  131. return 0x92;
  132. break;
  133. case 0x201c:
  134. return 0x93;
  135. break;
  136. case 0x201d:
  137. return 0x94;
  138. break;
  139. case 0x2022:
  140. return 0x95;
  141. break;
  142. case 0x2013:
  143. return 0x96;
  144. break;
  145. case 0x2014:
  146. return 0x97;
  147. break;
  148. case 0x02dc:
  149. return 0x98;
  150. break;
  151. case 0x2122:
  152. return 0x99;
  153. break;
  154. case 0x0161:
  155. return 0x9a;
  156. break;
  157. case 0x203a:
  158. return 0x9b;
  159. break;
  160. case 0x0153:
  161. return 0x9c;
  162. break;
  163. case 0x017e:
  164. return 0x9e;
  165. break;
  166. case 0x0178:
  167. return 0x9f;
  168. break;
  169. default:
  170. return 0x00;
  171. break;
  172. }
  173. }
  174. }
  175. void utf8_decode(char* src) {
  176. uint32_t codep = 0, state = UTF8_ACCEPT;
  177. char* dst = src;
  178. while (src && *src) {
  179. if (!decode(&state, &codep, *src++)) *dst++ = UNICODEtoCP1252(codep);
  180. }
  181. *dst = '\0';
  182. }
  183. /****************************************************************************************
  184. * Memory tools
  185. */
  186. void* malloc_init_external(size_t sz) {
  187. void* ptr = NULL;
  188. ptr = heap_caps_malloc(sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  189. if (ptr == NULL) {
  190. ESP_LOGE(TAG, "malloc_init_external: unable to allocate %d bytes of PSRAM!", sz);
  191. } else {
  192. memset(ptr, 0x00, sz);
  193. }
  194. return ptr;
  195. }
  196. void* clone_obj_psram(void* source, size_t source_sz) {
  197. void* ptr = NULL;
  198. ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  199. if (ptr == NULL) {
  200. ESP_LOGE(TAG, "clone_obj_psram: unable to allocate %d bytes of PSRAM!", source_sz);
  201. } else {
  202. memcpy(ptr, source, source_sz);
  203. }
  204. return ptr;
  205. }
  206. char* strdup_psram(const char* source) {
  207. void* ptr = NULL;
  208. size_t source_sz = strlen(source) + 1;
  209. ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  210. if (ptr == NULL) {
  211. ESP_LOGE(TAG, "strdup_psram: unable to allocate %d bytes of PSRAM! Cannot clone string %s", source_sz, source);
  212. } else {
  213. memset(ptr, 0x00, source_sz);
  214. strcpy(ptr, source);
  215. }
  216. return ptr;
  217. }
  218. /****************************************************************************************
  219. * Task manager
  220. */
  221. #define TASK_TLS_INDEX 1
  222. /**
  223. * @struct task_context_t
  224. * Structure to hold the context of a task including its task buffer and stack.
  225. *
  226. * @var StaticTask_t* xTaskBuffer
  227. * Pointer to the task's control block.
  228. *
  229. * @var StackType_t* xStack
  230. * Pointer to the task's stack.
  231. */
  232. typedef struct {
  233. StaticTask_t* xTaskBuffer;
  234. StackType_t* xStack;
  235. } task_context_t;
  236. /**
  237. * @fn static void task_cleanup(int index, task_context_t* context)
  238. * Cleans up the resources allocated for a task.
  239. * This function is intended to be used as a callback for task deletion.
  240. *
  241. * @param index The TLS index where the task's context is stored.
  242. * @param context Pointer to the task's context to be cleaned up.
  243. */
  244. static void task_cleanup(int index, task_context_t* context) {
  245. free(context->xTaskBuffer);
  246. free(context->xStack);
  247. free(context);
  248. }
  249. BaseType_t xTaskCreateEXTRAM(TaskFunction_t pvTaskCode, const char* const pcName, configSTACK_DEPTH_TYPE usStackDepth, void* pvParameters,
  250. UBaseType_t uxPriority, TaskHandle_t* pxCreatedTask) {
  251. // create the worker task as a static
  252. task_context_t* context = calloc(1, sizeof(task_context_t));
  253. context->xTaskBuffer = (StaticTask_t*)heap_caps_malloc(sizeof(StaticTask_t), (MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT));
  254. context->xStack = heap_caps_malloc(usStackDepth, (MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT));
  255. TaskHandle_t handle = xTaskCreateStatic(pvTaskCode, pcName, usStackDepth, pvParameters, uxPriority, context->xStack, context->xTaskBuffer);
  256. // store context in TCB or free everything in case of failure
  257. if (!handle) {
  258. free(context->xTaskBuffer);
  259. free(context->xStack);
  260. free(context);
  261. } else {
  262. vTaskSetThreadLocalStoragePointerAndDelCallback(handle, TASK_TLS_INDEX, context, (TlsDeleteCallbackFunction_t)task_cleanup);
  263. }
  264. if (pxCreatedTask) *pxCreatedTask = handle;
  265. return handle != NULL ? pdPASS : pdFAIL;
  266. }
  267. void vTaskDeleteEXTRAM(TaskHandle_t xTask) {
  268. /* At this point we leverage FreeRTOS extension to have callbacks on task deletion.
  269. * If not, we need to have here our own deletion implementation that include delayed
  270. * free for when this is called with NULL (self-deletion)
  271. */
  272. vTaskDelete(xTask);
  273. }
  274. void dump_json_content(const char* prefix, cJSON* json, int level) {
  275. if (!json) {
  276. ESP_LOG_LEVEL(level, TAG, "%s: empty!", prefix);
  277. return;
  278. }
  279. char* output = cJSON_Print(json);
  280. if (output) {
  281. ESP_LOG_LEVEL(level, TAG, "%s: \n%s", prefix, output);
  282. }
  283. FREE_AND_NULL(output);
  284. }
  285. const char* get_mem_flag_desc(int flags) {
  286. static char flagString[101];
  287. memset(flagString, 0x00, sizeof(flagString));
  288. if (flags & MALLOC_CAP_EXEC) strcat(flagString, "EXEC ");
  289. if (flags & MALLOC_CAP_32BIT) strcat(flagString, "32BIT ");
  290. if (flags & MALLOC_CAP_8BIT) strcat(flagString, "8BIT ");
  291. if (flags & MALLOC_CAP_DMA) strcat(flagString, "DMA ");
  292. if (flags & MALLOC_CAP_PID2) strcat(flagString, "PID2 ");
  293. if (flags & MALLOC_CAP_PID3) strcat(flagString, "PID3 ");
  294. if (flags & MALLOC_CAP_PID4) strcat(flagString, "PID4 ");
  295. if (flags & MALLOC_CAP_PID5) strcat(flagString, "PID5 ");
  296. if (flags & MALLOC_CAP_PID6) strcat(flagString, "PID6 ");
  297. if (flags & MALLOC_CAP_PID7) strcat(flagString, "PID7 ");
  298. if (flags & MALLOC_CAP_SPIRAM) strcat(flagString, "SPIRAM ");
  299. if (flags & MALLOC_CAP_INTERNAL) strcat(flagString, "INTERNAL ");
  300. if (flags & MALLOC_CAP_DEFAULT) strcat(flagString, "DEFAULT ");
  301. if (flags & MALLOC_CAP_IRAM_8BIT) strcat(flagString, "IRAM_8BIT ");
  302. if (flags & MALLOC_CAP_RETENTION) strcat(flagString, "RETENTION ");
  303. return flagString;
  304. }
  305. #define LOCAL_MAC_SIZE 10
  306. const char* get_mac_str() {
  307. uint8_t mac[6];
  308. static char macStr[LOCAL_MAC_SIZE + 1] = {0};
  309. if (macStr[0] == 0) {
  310. ESP_LOGD(TAG, "calling esp_read_mac");
  311. esp_read_mac((uint8_t*)&mac, ESP_MAC_WIFI_STA);
  312. ESP_LOGD(TAG, "Writing mac to string");
  313. snprintf(macStr, sizeof(macStr), "%x%x%x", mac[3], mac[4], mac[5]);
  314. ESP_LOGD(TAG, "Determined mac string: %s", macStr);
  315. }
  316. return macStr;
  317. }
  318. char* alloc_get_string_with_mac(const char* val) {
  319. const char* macstr = get_mac_str();
  320. char* fullvalue = (char*)malloc_init_external(strlen(val) + sizeof(macstr) + 1);
  321. if (fullvalue) {
  322. strcpy(fullvalue, val);
  323. strcat(fullvalue, macstr);
  324. } else {
  325. ESP_LOGE(TAG, "malloc failed for value %s", val);
  326. }
  327. return fullvalue;
  328. }
  329. char* alloc_get_fallback_unique_name() {
  330. #ifdef CONFIG_LWIP_LOCAL_HOSTNAME
  331. return alloc_get_string_with_mac(CONFIG_LWIP_LOCAL_HOSTNAME "-");
  332. #elif defined(CONFIG_FW_PLATFORM_NAME)
  333. return alloc_get_string_with_mac(CONFIG_FW_PLATFORM_NAME "-");
  334. #else
  335. return alloc_get_string_with_mac("squeezelite-");
  336. #endif
  337. }
  338. #define LOCAL_MAC_SIZE 20
  339. char* alloc_get_formatted_mac_string(uint8_t mac[6]) {
  340. char* macStr = malloc_init_external(LOCAL_MAC_SIZE);
  341. if (macStr) {
  342. snprintf(macStr, LOCAL_MAC_SIZE, MACSTR, MAC2STR(mac));
  343. }
  344. return macStr;
  345. }
  346. const char* str_or_unknown(const char* str) { return (str ? str : unknown_string_placeholder); }
  347. const char* str_or_null(const char* str) { return (str ? str : null_string_placeholder); }
  348. esp_log_level_t get_log_level_from_char(char* level) {
  349. if (!strcasecmp(level, "NONE")) {
  350. return ESP_LOG_NONE;
  351. }
  352. if (!strcasecmp(level, "ERROR")) {
  353. return ESP_LOG_ERROR;
  354. }
  355. if (!strcasecmp(level, "WARN")) {
  356. return ESP_LOG_WARN;
  357. }
  358. if (!strcasecmp(level, "INFO")) {
  359. return ESP_LOG_INFO;
  360. }
  361. if (!strcasecmp(level, "DEBUG")) {
  362. return ESP_LOG_DEBUG;
  363. }
  364. if (!strcasecmp(level, "VERBOSE")) {
  365. return ESP_LOG_VERBOSE;
  366. }
  367. return ESP_LOG_WARN;
  368. }
  369. const char* get_log_level_desc(esp_log_level_t level) {
  370. switch (level) {
  371. case ESP_LOG_NONE:
  372. return "NONE";
  373. break;
  374. case ESP_LOG_ERROR:
  375. return "ERROR";
  376. break;
  377. case ESP_LOG_WARN:
  378. return "WARN";
  379. break;
  380. case ESP_LOG_INFO:
  381. return "INFO";
  382. break;
  383. case ESP_LOG_DEBUG:
  384. return "DEBUG";
  385. break;
  386. case ESP_LOG_VERBOSE:
  387. return "VERBOSE";
  388. break;
  389. default:
  390. break;
  391. }
  392. return "UNKNOWN";
  393. }
  394. void set_log_level(char* tag, char* level) { esp_log_level_set(tag, get_log_level_from_char(level)); }