tools.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  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. #include <stdio.h>
  10. #include <stdlib.h>
  11. #include <stdint.h>
  12. #include <string.h>
  13. #include <ctype.h>
  14. #include "freertos/FreeRTOS.h"
  15. #include "freertos/task.h"
  16. #include "esp_task.h"
  17. #include "esp_tls.h"
  18. #include "esp_http_client.h"
  19. #include "esp_heap_caps.h"
  20. #include "esp_log.h"
  21. #include "tools.h"
  22. #if CONFIG_FREERTOS_THREAD_LOCAL_STORAGE_POINTERS < 2
  23. #error CONFIG_FREERTOS_THREAD_LOCAL_STORAGE_POINTERS must be at least 2
  24. #endif
  25. #include "cJSON.h"
  26. const static char TAG[] = "tools";
  27. /****************************************************************************************
  28. * UTF-8 tools
  29. */
  30. // Copyright (c) 2008-2009 Bjoern Hoehrmann <bjoern@hoehrmann.de>
  31. // See http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ for details.
  32. // Copyright (c) 2017 ZephRay <zephray@outlook.com>
  33. //
  34. // utf8to1252 - almost equivalent to iconv -f utf-8 -t windows-1252, but better
  35. #define UTF8_ACCEPT 0
  36. #define UTF8_REJECT 1
  37. static const uint8_t utf8d[] = {
  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,0, // 00..1f
  39. 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
  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,0, // 40..5f
  41. 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
  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,9, // 80..9f
  43. 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
  44. 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
  45. 0xa,0x3,0x3,0x3,0x3,0x3,0x3,0x3,0x3,0x3,0x3,0x3,0x3,0x4,0x3,0x3, // e0..ef
  46. 0xb,0x6,0x6,0x6,0x5,0x8,0x8,0x8,0x8,0x8,0x8,0x8,0x8,0x8,0x8,0x8, // f0..ff
  47. 0x0,0x1,0x2,0x3,0x5,0x8,0x7,0x1,0x1,0x1,0x4,0x6,0x1,0x1,0x1,0x1, // s0..s0
  48. 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
  49. 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
  50. 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
  51. 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
  52. };
  53. static uint32_t decode(uint32_t* state, uint32_t* codep, uint32_t byte) {
  54. uint32_t type = utf8d[byte];
  55. *codep = (*state != UTF8_ACCEPT) ?
  56. (byte & 0x3fu) | (*codep << 6) :
  57. (0xff >> type) & (byte);
  58. *state = utf8d[256 + *state*16 + type];
  59. return *state;
  60. }
  61. static uint8_t UNICODEtoCP1252(uint16_t chr) {
  62. if (chr <= 0xff)
  63. return (chr&0xff);
  64. else {
  65. ESP_LOGI(TAG, "some multi-byte %hx", chr);
  66. switch(chr) {
  67. case 0x20ac: return 0x80; break;
  68. case 0x201a: return 0x82; break;
  69. case 0x0192: return 0x83; break;
  70. case 0x201e: return 0x84; break;
  71. case 0x2026: return 0x85; break;
  72. case 0x2020: return 0x86; break;
  73. case 0x2021: return 0x87; break;
  74. case 0x02c6: return 0x88; break;
  75. case 0x2030: return 0x89; break;
  76. case 0x0160: return 0x8a; break;
  77. case 0x2039: return 0x8b; break;
  78. case 0x0152: return 0x8c; break;
  79. case 0x017d: return 0x8e; break;
  80. case 0x2018: return 0x91; break;
  81. case 0x2019: return 0x92; break;
  82. case 0x201c: return 0x93; break;
  83. case 0x201d: return 0x94; break;
  84. case 0x2022: return 0x95; break;
  85. case 0x2013: return 0x96; break;
  86. case 0x2014: return 0x97; break;
  87. case 0x02dc: return 0x98; break;
  88. case 0x2122: return 0x99; break;
  89. case 0x0161: return 0x9a; break;
  90. case 0x203a: return 0x9b; break;
  91. case 0x0153: return 0x9c; break;
  92. case 0x017e: return 0x9e; break;
  93. case 0x0178: return 0x9f; break;
  94. default: return 0x00; break;
  95. }
  96. }
  97. }
  98. void utf8_decode(char *src) {
  99. uint32_t codep = 0, state = UTF8_ACCEPT;
  100. char *dst = src;
  101. while (src && *src) {
  102. if (!decode(&state, &codep, *src++)) *dst++ = UNICODEtoCP1252(codep);
  103. }
  104. *dst = '\0';
  105. }
  106. /****************************************************************************************
  107. * URL tools
  108. */
  109. static inline char from_hex(char ch) {
  110. return isdigit(ch) ? ch - '0' : tolower(ch) - 'a' + 10;
  111. }
  112. void url_decode(char *url) {
  113. char *p, *src = strdup(url);
  114. for (p = src; *src; url++) {
  115. *url = *src++;
  116. if (*url == '%') {
  117. *url = from_hex(*src++) << 4;
  118. *url |= from_hex(*src++);
  119. } else if (*url == '+') {
  120. *url = ' ';
  121. }
  122. }
  123. *url = '\0';
  124. free(p);
  125. }
  126. /****************************************************************************************
  127. * Memory tools
  128. */
  129. void * malloc_init_external(size_t sz){
  130. void * ptr=NULL;
  131. ptr = heap_caps_malloc(sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  132. if(ptr==NULL){
  133. ESP_LOGE(TAG,"malloc_init_external: unable to allocate %d bytes of PSRAM!",sz);
  134. }
  135. else {
  136. memset(ptr,0x00,sz);
  137. }
  138. return ptr;
  139. }
  140. void * clone_obj_psram(void * source, size_t source_sz){
  141. void * ptr=NULL;
  142. ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  143. if(ptr==NULL){
  144. ESP_LOGE(TAG,"clone_obj_psram: unable to allocate %d bytes of PSRAM!",source_sz);
  145. }
  146. else {
  147. memcpy(ptr,source,source_sz);
  148. }
  149. return ptr;
  150. }
  151. char * strdup_psram(const char * source){
  152. void * ptr=NULL;
  153. size_t source_sz = strlen(source)+1;
  154. ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  155. if(ptr==NULL){
  156. ESP_LOGE(TAG,"strdup_psram: unable to allocate %d bytes of PSRAM! Cannot clone string %s",source_sz,source);
  157. }
  158. else {
  159. memset(ptr,0x00,source_sz);
  160. strcpy(ptr,source);
  161. }
  162. return ptr;
  163. }
  164. /****************************************************************************************
  165. * Task manager
  166. */
  167. #define TASK_TLS_INDEX 1
  168. typedef struct {
  169. StaticTask_t *xTaskBuffer;
  170. StackType_t *xStack;
  171. } task_context_t;
  172. static void task_cleanup(int index, task_context_t *context) {
  173. free(context->xTaskBuffer);
  174. free(context->xStack);
  175. free(context);
  176. }
  177. BaseType_t xTaskCreateEXTRAM( TaskFunction_t pvTaskCode,
  178. const char * const pcName,
  179. configSTACK_DEPTH_TYPE usStackDepth,
  180. void *pvParameters,
  181. UBaseType_t uxPriority,
  182. TaskHandle_t *pxCreatedTask) {
  183. // create the worker task as a static
  184. task_context_t *context = calloc(1, sizeof(task_context_t));
  185. context->xTaskBuffer = (StaticTask_t*) heap_caps_malloc(sizeof(StaticTask_t), (MALLOC_CAP_INTERNAL|MALLOC_CAP_8BIT));
  186. context->xStack = heap_caps_malloc(usStackDepth,(MALLOC_CAP_SPIRAM|MALLOC_CAP_8BIT));
  187. TaskHandle_t handle = xTaskCreateStatic(pvTaskCode, pcName, usStackDepth, pvParameters, uxPriority, context->xStack, context->xTaskBuffer);
  188. // store context in TCB or free everything in case of failure
  189. if (!handle) {
  190. free(context->xTaskBuffer);
  191. free(context->xStack);
  192. free(context);
  193. } else {
  194. vTaskSetThreadLocalStoragePointerAndDelCallback( handle, TASK_TLS_INDEX, context, (TlsDeleteCallbackFunction_t) task_cleanup);
  195. }
  196. if (pxCreatedTask) *pxCreatedTask = handle;
  197. return handle != NULL ? pdPASS : pdFAIL;
  198. }
  199. void vTaskDeleteEXTRAM(TaskHandle_t xTask) {
  200. /* At this point we leverage FreeRTOS extension to have callbacks on task deletion.
  201. * If not, we need to have here our own deletion implementation that include delayed
  202. * free for when this is called with NULL (self-deletion)
  203. */
  204. vTaskDelete(xTask);
  205. }
  206. /****************************************************************************************
  207. * URL download
  208. */
  209. typedef struct {
  210. void *user_context;
  211. http_download_cb_t callback;
  212. size_t max, bytes;
  213. bool abort;
  214. uint8_t *data;
  215. esp_http_client_handle_t client;
  216. } http_context_t;
  217. static void http_downloader(void *arg);
  218. static esp_err_t http_event_handler(esp_http_client_event_t *evt);
  219. void http_download(char *url, size_t max, http_download_cb_t callback, void *context) {
  220. http_context_t *http_context = (http_context_t*) heap_caps_calloc(sizeof(http_context_t), 1, MALLOC_CAP_SPIRAM);
  221. esp_http_client_config_t config = {
  222. .url = url,
  223. .event_handler = http_event_handler,
  224. .user_data = http_context,
  225. };
  226. http_context->callback = callback;
  227. http_context->user_context = context;
  228. http_context->max = max;
  229. http_context->client = esp_http_client_init(&config);
  230. xTaskCreateEXTRAM(http_downloader, "downloader", 8*1024, http_context, ESP_TASK_PRIO_MIN + 1, NULL);
  231. }
  232. static void http_downloader(void *arg) {
  233. http_context_t *http_context = (http_context_t*) arg;
  234. esp_http_client_perform(http_context->client);
  235. esp_http_client_cleanup(http_context->client);
  236. free(http_context);
  237. vTaskDeleteEXTRAM(NULL);
  238. }
  239. static esp_err_t http_event_handler(esp_http_client_event_t *evt) {
  240. http_context_t *http_context = (http_context_t*) evt->user_data;
  241. if (http_context->abort) return ESP_FAIL;
  242. switch(evt->event_id) {
  243. case HTTP_EVENT_ERROR:
  244. http_context->callback(NULL, 0, http_context->user_context);
  245. http_context->abort = true;
  246. break;
  247. case HTTP_EVENT_ON_HEADER:
  248. if (!strcasecmp(evt->header_key, "Content-Length")) {
  249. size_t len = atoi(evt->header_value);
  250. if (!len || len > http_context->max) {
  251. ESP_LOGI(TAG, "content-length null or too large %zu / %zu", len, http_context->max);
  252. http_context->abort = true;
  253. }
  254. }
  255. break;
  256. case HTTP_EVENT_ON_DATA: {
  257. size_t len = esp_http_client_get_content_length(evt->client);
  258. if (!http_context->data) {
  259. if ((http_context->data = (uint8_t*) malloc(len)) == NULL) {
  260. http_context->abort = true;
  261. ESP_LOGE(TAG, "failed to allocate memory for output buffer %zu", len);
  262. return ESP_FAIL;
  263. }
  264. }
  265. memcpy(http_context->data + http_context->bytes, evt->data, evt->data_len);
  266. http_context->bytes += evt->data_len;
  267. break;
  268. }
  269. case HTTP_EVENT_ON_FINISH:
  270. http_context->callback(http_context->data, http_context->bytes, http_context->user_context);
  271. break;
  272. case HTTP_EVENT_DISCONNECTED: {
  273. int mbedtls_err = 0;
  274. esp_err_t err = esp_tls_get_and_clear_last_error(evt->data, &mbedtls_err, NULL);
  275. if (err != ESP_OK) {
  276. ESP_LOGE(TAG, "HTTP download disconnect %d", err);
  277. if (http_context->data) free(http_context->data);
  278. http_context->callback(NULL, 0, http_context->user_context);
  279. return ESP_FAIL;
  280. }
  281. break;
  282. }
  283. default:
  284. break;
  285. }
  286. return ESP_OK;
  287. }
  288. time_t millis() {
  289. struct timeval tv;
  290. gettimeofday(&tv, NULL);
  291. return (tv.tv_sec * 1000) + (tv.tv_usec / 1000);
  292. }
  293. void dump_json_content(const char* prefix, cJSON* json, int level) {
  294. if (!json) {
  295. ESP_LOG_LEVEL(level,TAG, "%s: empty!", prefix);
  296. return;
  297. }
  298. char* output = cJSON_Print(json);
  299. if (output) {
  300. ESP_LOG_LEVEL(level,TAG, "%s: \n%s", prefix, output);
  301. }
  302. FREE_AND_NULL(output);
  303. }