tools.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. /*
  2. *
  3. * Sebastien L. 2023, sle118@hotmail.com
  4. * Philippe G. 2023, philippe_44@outlook.com
  5. *
  6. * This software is released under the MIT License.
  7. * https://opensource.org/licenses/MIT
  8. *
  9. * License Overview:
  10. * ----------------
  11. * The MIT License is a permissive open source license. As a user of this software, you are free to:
  12. * - Use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of this software.
  13. * - Use the software for private, commercial, or any other purposes.
  14. *
  15. * Conditions:
  16. * - You must include the above copyright notice and this permission notice in all
  17. * copies or substantial portions of the Software.
  18. *
  19. * The MIT License offers a high degree of freedom and is well-suited for both open source and
  20. * commercial applications. It places minimal restrictions on how the software can be used,
  21. * modified, and redistributed. For more details on the MIT License, please refer to the link above.
  22. */
  23. #pragma once
  24. #include "esp_log.h"
  25. #include "cJSON.h"
  26. #include "freertos/FreeRTOS.h"
  27. #include "freertos/task.h"
  28. #ifdef __cplusplus
  29. #include "cpp_tools.h"
  30. extern "C" {
  31. #endif
  32. /**
  33. * @def QUOTE
  34. * Converts the argument name into a string.
  35. *
  36. * @param name The name to be converted into a string.
  37. */
  38. #ifndef QUOTE
  39. #define QUOTE(name) #name
  40. #endif
  41. /**
  42. * @def STR
  43. * Converts a macro value into a string.
  44. *
  45. * @param macro The macro whose value is to be converted into a string.
  46. */
  47. #ifndef STR
  48. #define STR(macro) QUOTE(macro)
  49. #endif
  50. /**
  51. * @def STR_OR_ALT
  52. * Returns the string 'str' or an alternative 'alt' if 'str' is NULL.
  53. *
  54. * @param str The string to check.
  55. * @param alt The alternative string to return if 'str' is NULL.
  56. */
  57. #ifndef STR_OR_ALT
  58. #ifdef __cplusplus
  59. #define STR_OR_ALT(str, alt) (str!=nullptr ? str : alt)
  60. #else
  61. #define STR_OR_ALT(str, alt) (str ? str : alt)
  62. #endif
  63. #endif
  64. /**
  65. * @def STR_OR_BLANK
  66. * Returns the string 'p' or an empty string if 'p' is NULL.
  67. *
  68. * @param p The string to check.
  69. */
  70. #ifndef STR_OR_BLANK
  71. #ifdef __cplusplus
  72. #define STR_OR_BLANK(p) p == nullptr ? "" : p
  73. #else
  74. #define STR_OR_BLANK(p) p == NULL ? "" : p
  75. #endif
  76. #endif
  77. /**
  78. * @def ESP_LOG_DEBUG_EVENT
  79. * Logs a debug event with a given tag.
  80. *
  81. * @param tag The tag associated with the log message.
  82. * @param e The event to log.
  83. */
  84. #define ESP_LOG_DEBUG_EVENT(tag, e) ESP_LOGD(tag, "evt: " e)
  85. /**
  86. * @def FREE_AND_NULL
  87. * Frees a pointer if it's not NULL and sets it to NULL.
  88. *
  89. * @param x The pointer to free and nullify.
  90. */
  91. #ifndef FREE_AND_NULL
  92. #define FREE_AND_NULL(x) \
  93. if (x) { \
  94. free(x); \
  95. x = NULL; \
  96. }
  97. #endif
  98. /**
  99. * @def CASE_TO_STR
  100. * A switch case that returns the string representation of a case constant.
  101. *
  102. * @param x The case constant.
  103. */
  104. #ifndef CASE_TO_STR
  105. #define CASE_TO_STR(x) \
  106. case x: \
  107. return STR(x); \
  108. break;
  109. #endif
  110. /**
  111. * @def ENUM_TO_STRING
  112. * A switch case for an enum that returns the string representation of the enum value.
  113. *
  114. * @param g The enum value.
  115. */
  116. #define ENUM_TO_STRING(g) \
  117. case g: \
  118. return STR(g); \
  119. break;
  120. /**
  121. * @fn void utf8_decode(char* src)
  122. * Decodes a string encoded in UTF-8 to CP1252 in place.
  123. * The result is stored in the same buffer as the input.
  124. *
  125. * @param src A pointer to the null-terminated string to be decoded.
  126. */
  127. void utf8_decode(char* src);
  128. /**
  129. * @fn void* malloc_init_external(size_t sz)
  130. * Allocates memory in PSRAM and initializes it to zero.
  131. *
  132. * This function attempts to allocate a block of memory of size 'sz' in PSRAM.
  133. * If allocation is successful, the memory is initialized to zero.
  134. *
  135. * @param sz The size of memory to allocate in bytes.
  136. * @return A pointer to the allocated memory block or NULL if allocation fails.
  137. */
  138. void* malloc_init_external(size_t sz);
  139. /**
  140. * @fn void* clone_obj_psram(void* source, size_t source_sz)
  141. * Clones an object into PSRAM.
  142. *
  143. * This function allocates memory in PSRAM and copies the contents of the
  144. * source object into the newly allocated memory. The size of the source
  145. * object is specified by 'source_sz'.
  146. *
  147. * @param source A pointer to the source object to clone.
  148. * @param source_sz The size of the source object in bytes.
  149. * @return A pointer to the cloned object in PSRAM or NULL if allocation fails.
  150. */
  151. void* clone_obj_psram(void* source, size_t source_sz);
  152. /**
  153. * @fn char* strdup_psram(const char* source)
  154. * Duplicates a string into PSRAM.
  155. *
  156. * This function allocates enough memory in PSRAM to hold a copy of 'source',
  157. * including the null terminator, and then copies the string into the new memory.
  158. *
  159. * @param source A pointer to the null-terminated string to duplicate.
  160. * @return A pointer to the duplicated string in PSRAM or NULL if allocation fails.
  161. */
  162. char* strdup_psram(const char* source);
  163. /**
  164. * @fn const char* get_mem_flag_desc(int flags)
  165. * Returns a string describing memory allocation flags.
  166. *
  167. * This function takes memory allocation flags as an argument and returns
  168. * a string representation of these flags. The description includes various
  169. * memory capabilities like EXEC, DMA, SPIRAM, etc.
  170. *
  171. * @param flags The memory allocation flags to describe.
  172. * @return A string representation of the memory allocation flags.
  173. */
  174. const char* get_mem_flag_desc(int flags);
  175. /**
  176. * @fn const char* get_mac_str()
  177. * Returns a string representation of the device's MAC address.
  178. *
  179. * This function reads the MAC address of the device and returns a string
  180. * representation of the last three bytes of the MAC address. It uses a static
  181. * buffer to store the MAC address string.
  182. *
  183. * @return A string representation of the device's MAC address.
  184. */
  185. const char* get_mac_str();
  186. /**
  187. * @fn char* alloc_get_string_with_mac(const char* val)
  188. * Allocates a new string which is a concatenation of the provided value and the device's MAC address.
  189. *
  190. * This function concatenates 'val' with the device's MAC address string and returns
  191. * a pointer to the newly allocated string.
  192. *
  193. * @param val The string to concatenate with the MAC address.
  194. * @return A pointer to the newly allocated string, or NULL if allocation fails.
  195. */
  196. char* alloc_get_string_with_mac(const char* val);
  197. /**
  198. * @fn char* alloc_get_fallback_unique_name()
  199. * Allocates a unique name for the device based on predefined configurations or fallbacks.
  200. *
  201. * This function creates a unique name by concatenating a base name defined by
  202. * configuration macros (CONFIG_LWIP_LOCAL_HOSTNAME, CONFIG_FW_PLATFORM_NAME)
  203. * or a fallback prefix "squeezelite-" with the device's MAC address.
  204. *
  205. * @return A pointer to the newly allocated unique name, or NULL if allocation fails.
  206. */
  207. char* alloc_get_fallback_unique_name();
  208. /**
  209. * @fn char* alloc_get_formatted_mac_string(uint8_t mac[6])
  210. * Allocates and returns a formatted MAC address string.
  211. *
  212. * This function formats a given MAC address into a human-readable string,
  213. * allocating the necessary memory dynamically.
  214. *
  215. * @param mac An array containing the 6-byte MAC address.
  216. * @return A pointer to the dynamically allocated MAC address string, or NULL if allocation fails.
  217. */
  218. char* alloc_get_formatted_mac_string(uint8_t mac[6]);
  219. /**
  220. * @fn const char* str_or_unknown(const char* str)
  221. * Returns the input string or a placeholder for unknown string.
  222. *
  223. * If the input string is not NULL, it returns the string. Otherwise,
  224. * it returns a predefined placeholder for unknown strings.
  225. *
  226. * @param str The input string.
  227. * @return The input string or a placeholder if the input is NULL.
  228. */
  229. const char* str_or_unknown(const char* str);
  230. /**
  231. * @fn const char* str_or_null(const char* str)
  232. * Returns the input string or a placeholder for null string.
  233. *
  234. * If the input string is not NULL, it returns the string. Otherwise,
  235. * it returns a predefined placeholder for null strings.
  236. *
  237. * @param str The input string.
  238. * @return The input string or a placeholder if the input is NULL.
  239. */
  240. const char* str_or_null(const char* str);
  241. /**
  242. * @fn esp_log_level_t get_log_level_from_char(char* level)
  243. * Converts a string representation of a log level to its corresponding `esp_log_level_t` enum value.
  244. *
  245. * The function supports log level strings like "NONE", "ERROR", "WARN", "INFO", "DEBUG", and "VERBOSE".
  246. *
  247. * @param level The log level as a string.
  248. * @return The corresponding `esp_log_level_t` value, or ESP_LOG_WARN if the string does not match any log level.
  249. */
  250. esp_log_level_t get_log_level_from_char(char* level);
  251. /**
  252. * @fn const char* get_log_level_desc(esp_log_level_t level)
  253. * Returns the string description of a given `esp_log_level_t` log level.
  254. *
  255. * @param level The log level as an `esp_log_level_t` enum.
  256. * @return The string description of the log level.
  257. */
  258. const char* get_log_level_desc(esp_log_level_t level);
  259. /**
  260. * @fn void set_log_level(char* tag, char* level)
  261. * Sets the log level for a specific tag.
  262. *
  263. * The log level is specified as a string and converted internally to the corresponding `esp_log_level_t` value.
  264. *
  265. * @param tag The tag for which to set the log level.
  266. * @param level The log level as a string.
  267. */
  268. void set_log_level(char* tag, char* level);
  269. /**
  270. * @fn BaseType_t xTaskCreateEXTRAM(TaskFunction_t pvTaskCode, const char* const pcName, configSTACK_DEPTH_TYPE usStackDepth, void* pvParameters,
  271. * UBaseType_t uxPriority, TaskHandle_t* pxCreatedTask) Creates a new task with its stack allocated in external RAM (PSRAM). Use these to dynamically
  272. * create tasks whose stack is on EXTRAM. Be aware that it requires configNUM_THREAD_LOCAL_STORAGE_POINTERS to bet set to 2 at least (index 0 is used
  273. * by pthread and this uses index 1, obviously
  274. *
  275. * @param pvTaskCode Pointer to the task entry function.
  276. * @param pcName Task name.
  277. * @param usStackDepth Stack depth in words.
  278. * @param pvParameters Pointer to the task's parameters.
  279. * @param uxPriority Task priority.
  280. * @param pxCreatedTask Pointer to a variable to store the task's handle.
  281. * @return pdPASS on success, pdFAIL on failure.
  282. */
  283. BaseType_t xTaskCreateEXTRAM(TaskFunction_t pvTaskCode, const char* const pcName, configSTACK_DEPTH_TYPE usStackDepth, void* pvParameters,
  284. UBaseType_t uxPriority, TaskHandle_t* pxCreatedTask);
  285. /**
  286. * @fn void vTaskDeleteEXTRAM(TaskHandle_t xTask)
  287. * Deletes a task and cleans up its allocated resources.
  288. * This function should be used in place of vTaskDelete for tasks created with xTaskCreateEXTRAM.
  289. *
  290. * @param xTask Handle to the task to be deleted.
  291. */
  292. void vTaskDeleteEXTRAM(TaskHandle_t xTask);
  293. /**
  294. * @fn void dump_json_content(const char* prefix, cJSON* json, int level)
  295. * Dumps the content of a cJSON object to the log.
  296. *
  297. * @param prefix Prefix for the log message.
  298. * @param json Pointer to the cJSON object to dump.
  299. * @param level Logging level for the output.
  300. */
  301. void dump_json_content(const char* prefix, cJSON* json, int level);
  302. #ifndef gettime_ms
  303. // body is provided somewhere else...
  304. uint32_t _gettime_ms_(void);
  305. #define gettime_ms _gettime_ms_
  306. #endif
  307. /**
  308. * @def TRACE_DEBUG
  309. * A macro for debug tracing.
  310. *
  311. * This macro prints a formatted debug message to the standard output.
  312. * The message includes the name of the function and the line number from which
  313. * the macro is called, followed by a custom formatted message.
  314. *
  315. * @param msgformat A printf-style format string for the custom message.
  316. * @param ... Variable arguments for the format string.
  317. */
  318. #ifndef TRACE_DEBUG
  319. #define TRACE_DEBUG(msgformat, ...) printf("%-30s%-5d" msgformat "\n", __FUNCTION__, __LINE__, ##__VA_ARGS__)
  320. #endif
  321. #ifdef __cplusplus
  322. }
  323. #endif