tools.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754
  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_DEBUG
  10. #include "tools.h"
  11. #include "esp_heap_caps.h"
  12. #include "esp_http_client.h"
  13. #include "esp_log.h"
  14. #include "esp_task.h"
  15. #include "esp_tls.h"
  16. #include <ctype.h>
  17. #include <stdint.h>
  18. #include <stdlib.h>
  19. #include <string.h>
  20. #include <dirent.h>
  21. #include "esp_http_server.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. static bool initialized = false;
  26. const static char TAG[] = "tools";
  27. static esp_vfs_spiffs_conf_t* spiffs_conf = NULL;
  28. /****************************************************************************************
  29. * UTF-8 tools
  30. */
  31. // Copyright (c) 2008-2009 Bjoern Hoehrmann <bjoern@hoehrmann.de>
  32. // See http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ for details.
  33. // Copyright (c) 2017 ZephRay <zephray@outlook.com>
  34. //
  35. // utf8to1252 - almost equivalent to iconv -f utf-8 -t windows-1252, but better
  36. #define UTF8_ACCEPT 0
  37. #define UTF8_REJECT 1
  38. static const uint8_t utf8d[] = {
  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,
  40. 0, // 00..1f
  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,
  42. 0, // 20..3f
  43. 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,
  44. 0, // 40..5f
  45. 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,
  46. 0, // 60..7f
  47. 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,
  48. 9, // 80..9f
  49. 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,
  50. 7, // a0..bf
  51. 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,
  52. 2, // c0..df
  53. 0xa, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x4, 0x3, 0x3, // e0..ef
  54. 0xb, 0x6, 0x6, 0x6, 0x5, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, // f0..ff
  55. 0x0, 0x1, 0x2, 0x3, 0x5, 0x8, 0x7, 0x1, 0x1, 0x1, 0x4, 0x6, 0x1, 0x1, 0x1, 0x1, // s0..s0
  56. 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,
  57. 1, // s1..s2
  58. 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,
  59. 1, // s3..s4
  60. 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,
  61. 1, // s5..s6
  62. 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,
  63. 1, // s7..s8
  64. };
  65. static uint32_t decode(uint32_t* state, uint32_t* codep, uint32_t byte) {
  66. uint32_t type = utf8d[byte];
  67. *codep = (*state != UTF8_ACCEPT) ? (byte & 0x3fu) | (*codep << 6) : (0xff >> type) & (byte);
  68. *state = utf8d[256 + *state * 16 + type];
  69. return *state;
  70. }
  71. static uint8_t UNICODEtoCP1252(uint16_t chr) {
  72. if (chr <= 0xff)
  73. return (chr & 0xff);
  74. else {
  75. ESP_LOGI(TAG, "some multi-byte %hx", chr);
  76. switch (chr) {
  77. case 0x20ac:
  78. return 0x80;
  79. break;
  80. case 0x201a:
  81. return 0x82;
  82. break;
  83. case 0x0192:
  84. return 0x83;
  85. break;
  86. case 0x201e:
  87. return 0x84;
  88. break;
  89. case 0x2026:
  90. return 0x85;
  91. break;
  92. case 0x2020:
  93. return 0x86;
  94. break;
  95. case 0x2021:
  96. return 0x87;
  97. break;
  98. case 0x02c6:
  99. return 0x88;
  100. break;
  101. case 0x2030:
  102. return 0x89;
  103. break;
  104. case 0x0160:
  105. return 0x8a;
  106. break;
  107. case 0x2039:
  108. return 0x8b;
  109. break;
  110. case 0x0152:
  111. return 0x8c;
  112. break;
  113. case 0x017d:
  114. return 0x8e;
  115. break;
  116. case 0x2018:
  117. return 0x91;
  118. break;
  119. case 0x2019:
  120. return 0x92;
  121. break;
  122. case 0x201c:
  123. return 0x93;
  124. break;
  125. case 0x201d:
  126. return 0x94;
  127. break;
  128. case 0x2022:
  129. return 0x95;
  130. break;
  131. case 0x2013:
  132. return 0x96;
  133. break;
  134. case 0x2014:
  135. return 0x97;
  136. break;
  137. case 0x02dc:
  138. return 0x98;
  139. break;
  140. case 0x2122:
  141. return 0x99;
  142. break;
  143. case 0x0161:
  144. return 0x9a;
  145. break;
  146. case 0x203a:
  147. return 0x9b;
  148. break;
  149. case 0x0153:
  150. return 0x9c;
  151. break;
  152. case 0x017e:
  153. return 0x9e;
  154. break;
  155. case 0x0178:
  156. return 0x9f;
  157. break;
  158. default:
  159. return 0x00;
  160. break;
  161. }
  162. }
  163. }
  164. void utf8_decode(char* src) {
  165. uint32_t codep = 0, state = UTF8_ACCEPT;
  166. char* dst = src;
  167. while (src && *src) {
  168. if (!decode(&state, &codep, *src++)) *dst++ = UNICODEtoCP1252(codep);
  169. }
  170. *dst = '\0';
  171. }
  172. /****************************************************************************************
  173. * URL tools
  174. */
  175. static inline char from_hex(char ch) { return isdigit(ch) ? ch - '0' : tolower(ch) - 'a' + 10; }
  176. void url_decode(char* url) {
  177. char *p, *src = strdup(url);
  178. for (p = src; *src; url++) {
  179. *url = *src++;
  180. if (*url == '%') {
  181. *url = from_hex(*src++) << 4;
  182. *url |= from_hex(*src++);
  183. } else if (*url == '+') {
  184. *url = ' ';
  185. }
  186. }
  187. *url = '\0';
  188. free(p);
  189. }
  190. /****************************************************************************************
  191. * Memory tools
  192. */
  193. void* malloc_init_external(size_t sz) {
  194. void* ptr = NULL;
  195. ptr = heap_caps_malloc(sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  196. if (ptr == NULL) {
  197. ESP_LOGE(TAG, "malloc_init_external: unable to allocate %d bytes of PSRAM!", sz);
  198. } else {
  199. memset(ptr, 0x00, sz);
  200. }
  201. return ptr;
  202. }
  203. void* clone_obj_psram(void* source, size_t source_sz) {
  204. void* ptr = NULL;
  205. ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  206. if (ptr == NULL) {
  207. ESP_LOGE(TAG, "clone_obj_psram: unable to allocate %d bytes of PSRAM!", source_sz);
  208. } else {
  209. memcpy(ptr, source, source_sz);
  210. }
  211. return ptr;
  212. }
  213. char* strdup_psram(const char* source) {
  214. void* ptr = NULL;
  215. size_t source_sz = strlen(source) + 1;
  216. ptr = heap_caps_malloc(source_sz, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
  217. if (ptr == NULL) {
  218. ESP_LOGE(TAG, "strdup_psram: unable to allocate %d bytes of PSRAM! Cannot clone string %s",
  219. source_sz, source);
  220. } else {
  221. memset(ptr, 0x00, source_sz);
  222. strcpy(ptr, source);
  223. }
  224. return ptr;
  225. }
  226. /****************************************************************************************
  227. * Task manager
  228. */
  229. #define TASK_TLS_INDEX 1
  230. typedef struct {
  231. StaticTask_t* xTaskBuffer;
  232. StackType_t* xStack;
  233. } task_context_t;
  234. static void task_cleanup(int index, task_context_t* context) {
  235. free(context->xTaskBuffer);
  236. free(context->xStack);
  237. free(context);
  238. }
  239. BaseType_t xTaskCreateEXTRAM(TaskFunction_t pvTaskCode, const char* const pcName,
  240. configSTACK_DEPTH_TYPE usStackDepth, void* pvParameters, UBaseType_t uxPriority,
  241. TaskHandle_t* pxCreatedTask) {
  242. // create the worker task as a static
  243. task_context_t* context = calloc(1, sizeof(task_context_t));
  244. context->xTaskBuffer = (StaticTask_t*)heap_caps_malloc(
  245. sizeof(StaticTask_t), (MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT));
  246. context->xStack = heap_caps_malloc(usStackDepth, (MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT));
  247. TaskHandle_t handle = xTaskCreateStatic(pvTaskCode, pcName, usStackDepth, pvParameters,
  248. uxPriority, context->xStack, context->xTaskBuffer);
  249. // store context in TCB or free everything in case of failure
  250. if (!handle) {
  251. free(context->xTaskBuffer);
  252. free(context->xStack);
  253. free(context);
  254. } else {
  255. vTaskSetThreadLocalStoragePointerAndDelCallback(
  256. handle, TASK_TLS_INDEX, context, (TlsDeleteCallbackFunction_t)task_cleanup);
  257. }
  258. if (pxCreatedTask) *pxCreatedTask = handle;
  259. return handle != NULL ? pdPASS : pdFAIL;
  260. }
  261. void vTaskDeleteEXTRAM(TaskHandle_t xTask) {
  262. /* At this point we leverage FreeRTOS extension to have callbacks on task deletion.
  263. * If not, we need to have here our own deletion implementation that include delayed
  264. * free for when this is called with NULL (self-deletion)
  265. */
  266. vTaskDelete(xTask);
  267. }
  268. /****************************************************************************************
  269. * URL download
  270. */
  271. typedef struct {
  272. void* user_context;
  273. http_download_cb_t callback;
  274. size_t max, bytes;
  275. bool abort;
  276. uint8_t* data;
  277. esp_http_client_handle_t client;
  278. } http_context_t;
  279. static void http_downloader(void* arg);
  280. static esp_err_t http_event_handler(esp_http_client_event_t* evt);
  281. void http_download(char* url, size_t max, http_download_cb_t callback, void* context) {
  282. http_context_t* http_context =
  283. (http_context_t*)heap_caps_calloc(sizeof(http_context_t), 1, MALLOC_CAP_SPIRAM);
  284. esp_http_client_config_t config = {
  285. .url = url,
  286. .event_handler = http_event_handler,
  287. .user_data = http_context,
  288. };
  289. http_context->callback = callback;
  290. http_context->user_context = context;
  291. http_context->max = max;
  292. http_context->client = esp_http_client_init(&config);
  293. xTaskCreateEXTRAM(
  294. http_downloader, "downloader", 8 * 1024, http_context, ESP_TASK_PRIO_MIN + 1, NULL);
  295. }
  296. static void http_downloader(void* arg) {
  297. http_context_t* http_context = (http_context_t*)arg;
  298. esp_http_client_perform(http_context->client);
  299. esp_http_client_cleanup(http_context->client);
  300. free(http_context);
  301. vTaskDeleteEXTRAM(NULL);
  302. }
  303. static esp_err_t http_event_handler(esp_http_client_event_t* evt) {
  304. http_context_t* http_context = (http_context_t*)evt->user_data;
  305. if (http_context->abort) return ESP_FAIL;
  306. switch (evt->event_id) {
  307. case HTTP_EVENT_ERROR:
  308. http_context->callback(NULL, 0, http_context->user_context);
  309. http_context->abort = true;
  310. break;
  311. case HTTP_EVENT_ON_HEADER:
  312. if (!strcasecmp(evt->header_key, "Content-Length")) {
  313. size_t len = atoi(evt->header_value);
  314. if (!len || len > http_context->max) {
  315. ESP_LOGI(TAG, "content-length null or too large %zu / %zu", len, http_context->max);
  316. http_context->abort = true;
  317. }
  318. }
  319. break;
  320. case HTTP_EVENT_ON_DATA: {
  321. size_t len = esp_http_client_get_content_length(evt->client);
  322. if (!http_context->data) {
  323. if ((http_context->data = (uint8_t*)malloc(len)) == NULL) {
  324. http_context->abort = true;
  325. ESP_LOGE(TAG, "failed to allocate memory for output buffer %zu", len);
  326. return ESP_FAIL;
  327. }
  328. }
  329. memcpy(http_context->data + http_context->bytes, evt->data, evt->data_len);
  330. http_context->bytes += evt->data_len;
  331. break;
  332. }
  333. case HTTP_EVENT_ON_FINISH:
  334. http_context->callback(http_context->data, http_context->bytes, http_context->user_context);
  335. break;
  336. case HTTP_EVENT_DISCONNECTED: {
  337. int mbedtls_err = 0;
  338. esp_err_t err = esp_tls_get_and_clear_last_error(evt->data, &mbedtls_err, NULL);
  339. if (err != ESP_OK) {
  340. ESP_LOGE(TAG, "HTTP download disconnect %d", err);
  341. if (http_context->data) free(http_context->data);
  342. http_context->callback(NULL, 0, http_context->user_context);
  343. return ESP_FAIL;
  344. }
  345. break;
  346. }
  347. default:
  348. break;
  349. }
  350. return ESP_OK;
  351. }
  352. void dump_json_content(const char* prefix, cJSON* json, int level) {
  353. if (!json) {
  354. ESP_LOG_LEVEL(level, TAG, "%s: empty!", prefix);
  355. return;
  356. }
  357. char* output = cJSON_Print(json);
  358. if (output) {
  359. ESP_LOG_LEVEL(level, TAG, "%s: \n%s", prefix, output);
  360. }
  361. FREE_AND_NULL(output);
  362. }
  363. void init_spiffs() {
  364. if (initialized) {
  365. ESP_LOGD(TAG, "SPIFFS already initialized. returning");
  366. return;
  367. }
  368. ESP_LOGI(TAG, "Initializing the SPI File system");
  369. spiffs_conf = (esp_vfs_spiffs_conf_t*)malloc(sizeof(esp_vfs_spiffs_conf_t));
  370. spiffs_conf->base_path = "/spiffs";
  371. spiffs_conf->partition_label = NULL;
  372. spiffs_conf->max_files = 5;
  373. spiffs_conf->format_if_mount_failed = true;
  374. // Use settings defined above to initialize and mount SPIFFS filesystem.
  375. // Note: esp_vfs_spiffs_register is an all-in-one convenience function.
  376. esp_err_t ret = esp_vfs_spiffs_register(spiffs_conf);
  377. if (ret != ESP_OK) {
  378. if (ret == ESP_FAIL) {
  379. ESP_LOGE(TAG, "Failed to mount or format filesystem");
  380. } else if (ret == ESP_ERR_NOT_FOUND) {
  381. ESP_LOGE(TAG, "Failed to find SPIFFS partition");
  382. } else {
  383. ESP_LOGE(TAG, "Failed to initialize SPIFFS (%s)", esp_err_to_name(ret));
  384. }
  385. return;
  386. }
  387. size_t total = 0, used = 0;
  388. ret = esp_spiffs_info(spiffs_conf->partition_label, &total, &used);
  389. if (ret != ESP_OK) {
  390. ESP_LOGW(TAG, "Failed to get SPIFFS partition information (%s). Formatting...",
  391. esp_err_to_name(ret));
  392. esp_spiffs_format(spiffs_conf->partition_label);
  393. } else {
  394. ESP_LOGI(TAG, "Partition size: total: %d, used: %d", total, used);
  395. }
  396. initialized = true;
  397. }
  398. // Function to safely append a path part with '/' if needed
  399. void append_path_part(char** dest, const char* part) {
  400. if ((*dest)[-1] != '/' && part[0] != '/') {
  401. strcat(*dest, "/");
  402. *dest += 1; // Move the pointer past the '/'
  403. }
  404. strcat(*dest, part);
  405. *dest += strlen(part);
  406. }
  407. // Function to calculate the total length needed for the new path
  408. size_t calculate_total_length(const char* base_path, va_list args) {
  409. ESP_LOGV(TAG, "%s, Starting with base path: %s", "calculate_total_length", base_path);
  410. size_t length = strlen(base_path) + 1; // +1 for null terminator
  411. const char* part;
  412. va_list args_copy;
  413. va_copy(args_copy, args);
  414. while ((part = va_arg(args_copy, const char*)) != NULL) {
  415. ESP_LOGV(TAG, "Adding length of %s", part);
  416. length += strlen(part) + 1; // +1 for potential '/'
  417. }
  418. ESP_LOGV(TAG, "Done looping. calculated length: %d", length);
  419. va_end(args_copy);
  420. return length;
  421. }
  422. // Main function to join paths
  423. char* __alloc_join_path(const char* base_path, va_list args) {
  424. size_t count = 0;
  425. ESP_LOGD(TAG, "Getting path length starting with %s", base_path);
  426. size_t total_length = calculate_total_length(base_path, args);
  427. // Allocate memory
  428. char* full_path = malloc_init_external(total_length);
  429. if (!full_path) {
  430. ESP_LOGE(TAG, "Unable to allocate memory for path");
  431. return NULL;
  432. }
  433. // Start constructing the path
  434. strcpy(full_path, base_path);
  435. char* current_position = full_path + strlen(full_path);
  436. // Append each path part
  437. const char* part;
  438. while ((part = va_arg(args, const char*)) != NULL) {
  439. append_path_part(&current_position, part);
  440. }
  441. *current_position = '\0'; // Null-terminate the string
  442. return full_path;
  443. }
  444. char* _alloc_join_path(const char* base_path, ...) {
  445. va_list args;
  446. ESP_LOGD(TAG, "%s", "join_path_var_parms");
  447. va_start(args, base_path);
  448. char* result = __alloc_join_path(base_path, args);
  449. va_end(args);
  450. return result;
  451. }
  452. FILE* __open_file(const char* mode, va_list args) {
  453. FILE* file = NULL;
  454. char* fullfilename = __alloc_join_path(spiffs_conf->base_path, args);
  455. if (!fullfilename) {
  456. ESP_LOGE(TAG, "Open file failed: unable to determine name");
  457. } else {
  458. ESP_LOGI(TAG, "Opening file %s in mode %s ", fullfilename, mode);
  459. file = fopen(fullfilename, mode);
  460. }
  461. if (file == NULL) {
  462. ESP_LOGE(TAG, "Open file failed");
  463. }
  464. if (fullfilename) free(fullfilename);
  465. return file;
  466. }
  467. FILE* _open_file(const char* mode, ...) {
  468. va_list args;
  469. FILE* file = NULL;
  470. va_start(args, mode);
  471. file = __open_file(mode, args);
  472. va_end(args);
  473. return file;
  474. }
  475. bool _write_file(uint8_t* data, size_t sz, ...) {
  476. bool result = true;
  477. FILE* file = NULL;
  478. va_list args;
  479. if (data == NULL) {
  480. ESP_LOGE(TAG, "Cannot write file. Data not received");
  481. return false;
  482. }
  483. if (sz == 0) {
  484. ESP_LOGE(TAG, "Cannot write file. Data length 0");
  485. return false;
  486. }
  487. va_start(args, sz);
  488. file = __open_file("w+", args);
  489. va_end(args);
  490. if (file == NULL) {
  491. return false;
  492. }
  493. size_t written = fwrite(data, 1, sz, file);
  494. if (written != sz) {
  495. ESP_LOGE(TAG, "Write error. Wrote %d bytes of %d.", written, sz);
  496. result = false;
  497. }
  498. fclose(file);
  499. return result;
  500. }
  501. const char* get_mem_flag_desc(int flags) {
  502. static char flagString[101];
  503. memset(flagString,0x00,sizeof(flagString));
  504. if (flags & MALLOC_CAP_EXEC) strcat(flagString, "EXEC ");
  505. if (flags & MALLOC_CAP_32BIT) strcat(flagString, "32BIT ");
  506. if (flags & MALLOC_CAP_8BIT) strcat(flagString, "8BIT ");
  507. if (flags & MALLOC_CAP_DMA) strcat(flagString, "DMA ");
  508. if (flags & MALLOC_CAP_PID2) strcat(flagString, "PID2 ");
  509. if (flags & MALLOC_CAP_PID3) strcat(flagString, "PID3 ");
  510. if (flags & MALLOC_CAP_PID4) strcat(flagString, "PID4 ");
  511. if (flags & MALLOC_CAP_PID5) strcat(flagString, "PID5 ");
  512. if (flags & MALLOC_CAP_PID6) strcat(flagString, "PID6 ");
  513. if (flags & MALLOC_CAP_PID7) strcat(flagString, "PID7 ");
  514. if (flags & MALLOC_CAP_SPIRAM) strcat(flagString, "SPIRAM ");
  515. if (flags & MALLOC_CAP_INTERNAL) strcat(flagString, "INTERNAL ");
  516. if (flags & MALLOC_CAP_DEFAULT) strcat(flagString, "DEFAULT ");
  517. if (flags & MALLOC_CAP_IRAM_8BIT) strcat(flagString, "IRAM_8BIT ");
  518. if (flags & MALLOC_CAP_RETENTION) strcat(flagString, "RETENTION ");
  519. return flagString;
  520. }
  521. void* _load_file(uint32_t memflags,size_t* sz, ...) {
  522. void* data = NULL;
  523. FILE* file = NULL;
  524. size_t fsz = 0;
  525. va_list args;
  526. va_start(args, sz);
  527. file = __open_file("rb", args);
  528. va_end(args);
  529. if (file == NULL) {
  530. return data;
  531. }
  532. fseek(file, 0, SEEK_END);
  533. fsz = ftell(file);
  534. fseek(file, 0, SEEK_SET);
  535. if (fsz > 0) {
  536. ESP_LOGD(TAG, "Allocating %d bytes to load file content with flags: %s ", fsz,get_mem_flag_desc(memflags));
  537. data = (void*)heap_caps_calloc(1, fsz, memflags);
  538. if (data == NULL) {
  539. ESP_LOGE(TAG, "Failed to allocate %d bytes to load file", fsz);
  540. } else {
  541. fread(data, 1, fsz, file);
  542. if (sz) {
  543. *sz = fsz;
  544. }
  545. }
  546. } else {
  547. ESP_LOGW(TAG, "File is empty. Nothing to read");
  548. }
  549. fclose(file);
  550. return data;
  551. }
  552. bool _get_file_info(struct stat* pfileInfo, ...) {
  553. va_list args;
  554. struct stat fileInfo;
  555. va_start(args, pfileInfo);
  556. char* fullfilename = __alloc_join_path(spiffs_conf->base_path, args);
  557. va_end(args);
  558. ESP_LOGD(TAG, "Getting file info for %s", fullfilename);
  559. if (!fullfilename) {
  560. ESP_LOGE(TAG, "Failed to construct full file path");
  561. return false;
  562. }
  563. bool result = false;
  564. // Use stat to fill the fileInfo structure
  565. if (stat(fullfilename, &fileInfo) != 0) {
  566. ESP_LOGD(TAG, "File %s not found", fullfilename);
  567. } else {
  568. result = true;
  569. if (pfileInfo) {
  570. memcpy(pfileInfo, &fileInfo, sizeof(fileInfo));
  571. }
  572. ESP_LOGD(TAG, "File %s has %lu bytes", fullfilename, fileInfo.st_size);
  573. }
  574. free(fullfilename);
  575. return result;
  576. }
  577. #define LOCAL_MAC_SIZE 10
  578. const char* get_mac_str() {
  579. uint8_t mac[6];
  580. static char macStr[LOCAL_MAC_SIZE + 1] = {0};
  581. if (macStr[0] == 0) {
  582. ESP_LOGD(TAG, "calling esp_read_mac");
  583. esp_read_mac((uint8_t*)&mac, ESP_MAC_WIFI_STA);
  584. ESP_LOGD(TAG, "Writing mac to string");
  585. snprintf(macStr, sizeof(macStr), "%x%x%x", mac[3], mac[4], mac[5]);
  586. ESP_LOGD(TAG, "Determined mac string: %s", macStr);
  587. }
  588. return macStr;
  589. }
  590. char* alloc_get_string_with_mac(const char* val) {
  591. uint8_t mac[6];
  592. char macStr[LOCAL_MAC_SIZE + 1];
  593. char* fullvalue = NULL;
  594. esp_read_mac((uint8_t*)&mac, ESP_MAC_WIFI_STA);
  595. snprintf(macStr, LOCAL_MAC_SIZE - 1, "-%x%x%x", mac[3], mac[4], mac[5]);
  596. fullvalue = (char*)malloc_init_external(strlen(val) + sizeof(macStr) + 1);
  597. if (fullvalue) {
  598. strcpy(fullvalue, val);
  599. strcat(fullvalue, macStr);
  600. } else {
  601. ESP_LOGE(TAG, "malloc failed for value %s", val);
  602. }
  603. return fullvalue;
  604. }
  605. void listFiles(const char *path_requested) {
  606. DIR *dir = NULL;
  607. char * sep="---------------------------------------------------------\n";
  608. struct dirent *ent;
  609. char type;
  610. char size[21];
  611. char tpath[255];
  612. struct stat sb;
  613. struct tm *tm_info;
  614. char *lpath = NULL;
  615. int statok;
  616. char * path= alloc_join_path(spiffs_conf->base_path,path_requested);
  617. printf("\nList of Directory [%s]\n", path);
  618. printf(sep);
  619. // Open directory
  620. dir = opendir(path);
  621. if (!dir) {
  622. printf("Error opening directory\n");
  623. free(path);
  624. return;
  625. }
  626. // Read directory entries
  627. uint64_t total = 0;
  628. int nfiles = 0;
  629. printf("T Size Name\n");
  630. printf(sep);
  631. while ((ent = readdir(dir)) != NULL) {
  632. sprintf(tpath, path);
  633. if (path[strlen(path)-1] != '/') strcat(tpath,"/");
  634. strcat(tpath,ent->d_name);
  635. // Get file stat
  636. statok = stat(tpath, &sb);
  637. if (ent->d_type == DT_REG) {
  638. type = 'f';
  639. nfiles++;
  640. if (statok) strcpy(size, " ?");
  641. else {
  642. total += sb.st_size;
  643. if (sb.st_size < (1024*1024)) sprintf(size,"%8d", (int)sb.st_size);
  644. else if ((sb.st_size/1024) < (1024*1024)) sprintf(size,"%6dKB", (int)(sb.st_size / 1024));
  645. else sprintf(size,"%6dMB", (int)(sb.st_size / (1024 * 1024)));
  646. }
  647. }
  648. else {
  649. type = 'd';
  650. strcpy(size, " -");
  651. }
  652. printf("%c %s %s\r\n",
  653. type,
  654. size,
  655. ent->d_name
  656. );
  657. }
  658. if (total) {
  659. printf(sep);
  660. if (total < (1024*1024)) printf(" %8d", (int)total);
  661. else if ((total/1024) < (1024*1024)) printf(" %6dKB", (int)(total / 1024));
  662. else printf(" %6dMB", (int)(total / (1024 * 1024)));
  663. printf(" in %d file(s)\n", nfiles);
  664. }
  665. printf(sep);
  666. closedir(dir);
  667. free(lpath);
  668. free(path);
  669. uint32_t tot=0, used=0;
  670. esp_spiffs_info(NULL, &tot, &used);
  671. printf("SPIFFS: free %d KB of %d KB\n", (tot-used) / 1024, tot / 1024);
  672. printf(sep);
  673. }
  674. bool out_file_binding(pb_ostream_t* stream, const uint8_t* buf, size_t count) {
  675. FILE* file = (FILE*)stream->state;
  676. ESP_LOGD(TAG, "Writing %d bytes to file", count);
  677. return fwrite(buf, 1, count, file) == count;
  678. }
  679. bool in_file_binding(pb_istream_t* stream, pb_byte_t *buf, size_t count) {
  680. FILE* file = (FILE*)stream->state;
  681. ESP_LOGD(TAG, "Reading %d bytes from file", count);
  682. return fread(buf, 1, count, file) == count;
  683. }
  684. bool out_http_binding(pb_ostream_t* stream, const uint8_t* buf, size_t count) {
  685. httpd_req_t* req = (httpd_req_t*)stream->state;
  686. ESP_LOGD(TAG, "Writing %d bytes to file", count);
  687. return httpd_resp_send_chunk(req, (const char*)buf, count) == ESP_OK;
  688. }