123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432 |
- /*
- * (c) Philippe G. 20201, philippe_44@outlook.com
- * see other copyrights below
- *
- * This software is released under the MIT License.
- * https://opensource.org/licenses/MIT
- *
- */
- #define LOG_LOCAL_LEVEL ESP_LOG_INFO
- #include "tools.h"
- #include "esp_heap_caps.h"
- #include "esp_log.h"
- #include "esp_task.h"
- #include <ctype.h>
- #include <stdint.h>
- #include <string.h>
- #if CONFIG_FREERTOS_THREAD_LOCAL_STORAGE_POINTERS < 2
- #error CONFIG_FREERTOS_THREAD_LOCAL_STORAGE_POINTERS must be at least 2
- #endif
- const static char TAG[] = "tools";
- const char unknown_string_placeholder[] = "unknown";
- const char null_string_placeholder[] = "null";
- /****************************************************************************************
- * UTF-8 tools
- */
- // Copyright (c) 2008-2009 Bjoern Hoehrmann <bjoern@hoehrmann.de>
- // See http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ for details.
- // Copyright (c) 2017 ZephRay <zephray@outlook.com>
- //
- // utf8to1252 - almost equivalent to iconv -f utf-8 -t windows-1252, but better
- #define UTF8_ACCEPT 0
- #define UTF8_REJECT 1
- static const uint8_t utf8d[] = {
- 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,
- 0, // 00..1f
- 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,
- 0, // 20..3f
- 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,
- 0, // 40..5f
- 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,
- 0, // 60..7f
- 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,
- 9, // 80..9f
- 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,
- 7, // a0..bf
- 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,
- 2, // c0..df
- 0xa, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x4, 0x3, 0x3, // e0..ef
- 0xb, 0x6, 0x6, 0x6, 0x5, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, // f0..ff
- 0x0, 0x1, 0x2, 0x3, 0x5, 0x8, 0x7, 0x1, 0x1, 0x1, 0x4, 0x6, 0x1, 0x1, 0x1, 0x1, // s0..s0
- 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,
- 1, // s1..s2
- 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,
- 1, // s3..s4
- 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,
- 1, // s5..s6
- 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,
- 1, // s7..s8
- };
- /**
- * @fn static uint32_t decode(uint32_t* state, uint32_t* codep, uint32_t byte)
- * Decodes a single UTF-8 encoded byte.
- *
- * @param state A pointer to the current state of the UTF-8 decoder.
- * @param codep A pointer to the code point being built from the UTF-8 bytes.
- * @param byte The current byte to be decoded.
- * @return The new state after processing the byte.
- */
- static uint32_t decode(uint32_t* state, uint32_t* codep, uint32_t byte) {
- uint32_t type = utf8d[byte];
- *codep = (*state != UTF8_ACCEPT) ? (byte & 0x3fu) | (*codep << 6) : (0xff >> type) & (byte);
- *state = utf8d[256 + *state * 16 + type];
- return *state;
- }
- /**
- * @fn static uint8_t UNICODEtoCP1252(uint16_t chr)
- * Converts a Unicode character to its corresponding CP1252 character.
- *
- * @param chr The Unicode character to be converted.
- * @return The corresponding CP1252 character or 0x00 if there is no direct mapping.
- */
- static uint8_t UNICODEtoCP1252(uint16_t chr) {
- if (chr <= 0xff)
- return (chr & 0xff);
- else {
- ESP_LOGI(TAG, "some multi-byte %hx", chr);
- switch (chr) {
- case 0x20ac:
- return 0x80;
- break;
- case 0x201a:
- return 0x82;
- break;
- case 0x0192:
- return 0x83;
- break;
- case 0x201e:
- return 0x84;
- break;
- case 0x2026:
- return 0x85;
- break;
- case 0x2020:
- return 0x86;
- break;
- case 0x2021:
- return 0x87;
- break;
- case 0x02c6:
- return 0x88;
- break;
- case 0x2030:
- return 0x89;
- break;
- case 0x0160:
- return 0x8a;
- break;
- case 0x2039:
- return 0x8b;
- break;
- case 0x0152:
- return 0x8c;
- break;
- case 0x017d:
- return 0x8e;
- break;
- case 0x2018:
- return 0x91;
- break;
- case 0x2019:
- return 0x92;
- break;
- case 0x201c:
- return 0x93;
- break;
- case 0x201d:
- return 0x94;
- break;
- case 0x2022:
- return 0x95;
- break;
- case 0x2013:
- return 0x96;
- break;
- case 0x2014:
- return 0x97;
- break;
- case 0x02dc:
- return 0x98;
- break;
- case 0x2122:
- return 0x99;
- break;
- case 0x0161:
- return 0x9a;
- break;
- case 0x203a:
- return 0x9b;
- break;
- case 0x0153:
- return 0x9c;
- break;
- case 0x017e:
- return 0x9e;
- break;
- case 0x0178:
- return 0x9f;
- break;
- default:
- return 0x00;
- break;
- }
- }
- }
- void utf8_decode(char* src) {
- uint32_t codep = 0, state = UTF8_ACCEPT;
- char* dst = src;
- while (src && *src) {
- if (!decode(&state, &codep, *src++)) *dst++ = UNICODEtoCP1252(codep);
- }
- *dst = '\0';
- }
- /****************************************************************************************
- * Memory tools
- */
- void* malloc_init_external(size_t sz) {
- void* ptr = NULL;
- ptr = heap_caps_malloc(sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
- if (ptr == NULL) {
- ESP_LOGE(TAG, "malloc_init_external: unable to allocate %d bytes of PSRAM!", sz);
- } else {
- memset(ptr, 0x00, sz);
- }
- return ptr;
- }
- void* clone_obj_psram(void* source, size_t source_sz) {
- void* ptr = NULL;
- ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
- if (ptr == NULL) {
- ESP_LOGE(TAG, "clone_obj_psram: unable to allocate %d bytes of PSRAM!", source_sz);
- } else {
- memcpy(ptr, source, source_sz);
- }
- return ptr;
- }
- char* strdup_psram(const char* source) {
- void* ptr = NULL;
- size_t source_sz = strlen(source) + 1;
- ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
- if (ptr == NULL) {
- ESP_LOGE(TAG, "strdup_psram: unable to allocate %d bytes of PSRAM! Cannot clone string %s", source_sz, source);
- } else {
- memset(ptr, 0x00, source_sz);
- strcpy(ptr, source);
- }
- return ptr;
- }
- /****************************************************************************************
- * Task manager
- */
- #define TASK_TLS_INDEX 1
- /**
- * @struct task_context_t
- * Structure to hold the context of a task including its task buffer and stack.
- *
- * @var StaticTask_t* xTaskBuffer
- * Pointer to the task's control block.
- *
- * @var StackType_t* xStack
- * Pointer to the task's stack.
- */
- typedef struct {
- StaticTask_t* xTaskBuffer;
- StackType_t* xStack;
- } task_context_t;
- /**
- * @fn static void task_cleanup(int index, task_context_t* context)
- * Cleans up the resources allocated for a task.
- * This function is intended to be used as a callback for task deletion.
- *
- * @param index The TLS index where the task's context is stored.
- * @param context Pointer to the task's context to be cleaned up.
- */
- static void task_cleanup(int index, task_context_t* context) {
- free(context->xTaskBuffer);
- free(context->xStack);
- free(context);
- }
- BaseType_t xTaskCreateEXTRAM(TaskFunction_t pvTaskCode, const char* const pcName, configSTACK_DEPTH_TYPE usStackDepth, void* pvParameters,
- UBaseType_t uxPriority, TaskHandle_t* pxCreatedTask) {
- // create the worker task as a static
- task_context_t* context = calloc(1, sizeof(task_context_t));
- context->xTaskBuffer = (StaticTask_t*)heap_caps_malloc(sizeof(StaticTask_t), (MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT));
- context->xStack = heap_caps_malloc(usStackDepth, (MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT));
- TaskHandle_t handle = xTaskCreateStatic(pvTaskCode, pcName, usStackDepth, pvParameters, uxPriority, context->xStack, context->xTaskBuffer);
- // store context in TCB or free everything in case of failure
- if (!handle) {
- free(context->xTaskBuffer);
- free(context->xStack);
- free(context);
- } else {
- vTaskSetThreadLocalStoragePointerAndDelCallback(handle, TASK_TLS_INDEX, context, (TlsDeleteCallbackFunction_t)task_cleanup);
- }
- if (pxCreatedTask) *pxCreatedTask = handle;
- return handle != NULL ? pdPASS : pdFAIL;
- }
- void vTaskDeleteEXTRAM(TaskHandle_t xTask) {
- /* At this point we leverage FreeRTOS extension to have callbacks on task deletion.
- * If not, we need to have here our own deletion implementation that include delayed
- * free for when this is called with NULL (self-deletion)
- */
- vTaskDelete(xTask);
- }
- void dump_json_content(const char* prefix, cJSON* json, int level) {
- if (!json) {
- ESP_LOG_LEVEL(level, TAG, "%s: empty!", prefix);
- return;
- }
- char* output = cJSON_Print(json);
- if (output) {
- ESP_LOG_LEVEL(level, TAG, "%s: \n%s", prefix, output);
- }
- FREE_AND_NULL(output);
- }
- const char* get_mem_flag_desc(int flags) {
- static char flagString[101];
- memset(flagString, 0x00, sizeof(flagString));
- if (flags & MALLOC_CAP_EXEC) strcat(flagString, "EXEC ");
- if (flags & MALLOC_CAP_32BIT) strcat(flagString, "32BIT ");
- if (flags & MALLOC_CAP_8BIT) strcat(flagString, "8BIT ");
- if (flags & MALLOC_CAP_DMA) strcat(flagString, "DMA ");
- if (flags & MALLOC_CAP_PID2) strcat(flagString, "PID2 ");
- if (flags & MALLOC_CAP_PID3) strcat(flagString, "PID3 ");
- if (flags & MALLOC_CAP_PID4) strcat(flagString, "PID4 ");
- if (flags & MALLOC_CAP_PID5) strcat(flagString, "PID5 ");
- if (flags & MALLOC_CAP_PID6) strcat(flagString, "PID6 ");
- if (flags & MALLOC_CAP_PID7) strcat(flagString, "PID7 ");
- if (flags & MALLOC_CAP_SPIRAM) strcat(flagString, "SPIRAM ");
- if (flags & MALLOC_CAP_INTERNAL) strcat(flagString, "INTERNAL ");
- if (flags & MALLOC_CAP_DEFAULT) strcat(flagString, "DEFAULT ");
- if (flags & MALLOC_CAP_IRAM_8BIT) strcat(flagString, "IRAM_8BIT ");
- if (flags & MALLOC_CAP_RETENTION) strcat(flagString, "RETENTION ");
- return flagString;
- }
- #define LOCAL_MAC_SIZE 10
- const char* get_mac_str() {
- uint8_t mac[6];
- static char macStr[LOCAL_MAC_SIZE + 1] = {0};
- if (macStr[0] == 0) {
- ESP_LOGD(TAG, "calling esp_read_mac");
- esp_read_mac((uint8_t*)&mac, ESP_MAC_WIFI_STA);
- ESP_LOGD(TAG, "Writing mac to string");
- snprintf(macStr, sizeof(macStr), "%x%x%x", mac[3], mac[4], mac[5]);
- ESP_LOGD(TAG, "Determined mac string: %s", macStr);
- }
- return macStr;
- }
- char* alloc_get_string_with_mac(const char* val) {
- const char* macstr = get_mac_str();
- char* fullvalue = (char*)malloc_init_external(strlen(val) + sizeof(macstr) + 1);
- if (fullvalue) {
- strcpy(fullvalue, val);
- strcat(fullvalue, macstr);
- } else {
- ESP_LOGE(TAG, "malloc failed for value %s", val);
- }
- return fullvalue;
- }
- char* alloc_get_fallback_unique_name() {
- #ifdef CONFIG_LWIP_LOCAL_HOSTNAME
- return alloc_get_string_with_mac(CONFIG_LWIP_LOCAL_HOSTNAME "-");
- #elif defined(CONFIG_FW_PLATFORM_NAME)
- return alloc_get_string_with_mac(CONFIG_FW_PLATFORM_NAME "-");
- #else
- return alloc_get_string_with_mac("squeezelite-");
- #endif
- }
- #define LOCAL_MAC_SIZE 20
- char* alloc_get_formatted_mac_string(uint8_t mac[6]) {
- char* macStr = malloc_init_external(LOCAL_MAC_SIZE);
- if (macStr) {
- snprintf(macStr, LOCAL_MAC_SIZE, MACSTR, MAC2STR(mac));
- }
- return macStr;
- }
- const char* str_or_unknown(const char* str) { return (str ? str : unknown_string_placeholder); }
- const char* str_or_null(const char* str) { return (str ? str : null_string_placeholder); }
- esp_log_level_t get_log_level_from_char(char* level) {
- if (!strcasecmp(level, "NONE")) {
- return ESP_LOG_NONE;
- }
- if (!strcasecmp(level, "ERROR")) {
- return ESP_LOG_ERROR;
- }
- if (!strcasecmp(level, "WARN")) {
- return ESP_LOG_WARN;
- }
- if (!strcasecmp(level, "INFO")) {
- return ESP_LOG_INFO;
- }
- if (!strcasecmp(level, "DEBUG")) {
- return ESP_LOG_DEBUG;
- }
- if (!strcasecmp(level, "VERBOSE")) {
- return ESP_LOG_VERBOSE;
- }
- return ESP_LOG_WARN;
- }
- const char* get_log_level_desc(esp_log_level_t level) {
- switch (level) {
- case ESP_LOG_NONE:
- return "NONE";
- break;
- case ESP_LOG_ERROR:
- return "ERROR";
- break;
- case ESP_LOG_WARN:
- return "WARN";
- break;
- case ESP_LOG_INFO:
- return "INFO";
- break;
- case ESP_LOG_DEBUG:
- return "DEBUG";
- break;
- case ESP_LOG_VERBOSE:
- return "VERBOSE";
- break;
- default:
- break;
- }
- return "UNKNOWN";
- }
- void set_log_level(char* tag, char* level) { esp_log_level_set(tag, get_log_level_from_char(level)); }
|