log_backend_fs.c 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. /*
  2. * Copyright (c) 2020 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <stdio.h>
  7. #include <stdlib.h>
  8. #include <logging/log_backend.h>
  9. #include <logging/log_output_dict.h>
  10. #include <logging/log_backend_std.h>
  11. #include <assert.h>
  12. #include <fs/fs.h>
  13. #define MAX_PATH_LEN 256
  14. #define MAX_FLASH_WRITE_SIZE 256
  15. #define LOG_PREFIX_LEN (sizeof(CONFIG_LOG_BACKEND_FS_FILE_PREFIX) - 1)
  16. #define MAX_FILE_NUMERAL 9999
  17. #define FILE_NUMERAL_LEN 4
  18. enum backend_fs_state {
  19. BACKEND_FS_NOT_INITIALIZED = 0,
  20. BACKEND_FS_CORRUPTED,
  21. BACKEND_FS_OK
  22. };
  23. static struct fs_file_t file;
  24. static enum backend_fs_state backend_state = BACKEND_FS_NOT_INITIALIZED;
  25. static int file_ctr, newest, oldest;
  26. static int allocate_new_file(struct fs_file_t *file);
  27. static int del_oldest_log(void);
  28. static int get_log_file_id(struct fs_dirent *ent);
  29. static int check_log_volumen_available(void)
  30. {
  31. int index = 0;
  32. char const *name;
  33. int rc = 0;
  34. while (rc == 0) {
  35. rc = fs_readmount(&index, &name);
  36. if (rc == 0) {
  37. if (strncmp(CONFIG_LOG_BACKEND_FS_DIR,
  38. name,
  39. strlen(name))
  40. == 0) {
  41. return 0;
  42. }
  43. }
  44. }
  45. return -ENOENT;
  46. }
  47. static int create_log_dir(const char *path)
  48. {
  49. const char *next;
  50. const char *last = path + (strlen(path) - 1);
  51. char w_path[MAX_PATH_LEN];
  52. int rc, len;
  53. struct fs_dir_t dir;
  54. fs_dir_t_init(&dir);
  55. /* the fist directory name is the mount point*/
  56. /* the firs path's letter might be meaningless `/`, let's skip it */
  57. next = strchr(path + 1, '/');
  58. if (!next) {
  59. return 0;
  60. }
  61. while (true) {
  62. next++;
  63. if (next > last) {
  64. return 0;
  65. }
  66. next = strchr(next, '/');
  67. if (!next) {
  68. next = last;
  69. len = last - path + 1;
  70. } else {
  71. len = next - path;
  72. }
  73. memcpy(w_path, path, len);
  74. w_path[len] = 0;
  75. rc = fs_opendir(&dir, w_path);
  76. if (rc) {
  77. /* assume directory doesn't exist */
  78. rc = fs_mkdir(w_path);
  79. if (rc) {
  80. break;
  81. }
  82. }
  83. rc = fs_closedir(&dir);
  84. if (rc) {
  85. break;
  86. }
  87. }
  88. return rc;
  89. }
  90. static int check_log_file_exist(int num)
  91. {
  92. struct fs_dir_t dir;
  93. struct fs_dirent ent;
  94. int rc;
  95. fs_dir_t_init(&dir);
  96. rc = fs_opendir(&dir, CONFIG_LOG_BACKEND_FS_DIR);
  97. if (rc) {
  98. return -EIO;
  99. }
  100. while (true) {
  101. rc = fs_readdir(&dir, &ent);
  102. if (rc < 0) {
  103. rc = -EIO;
  104. goto close_dir;
  105. }
  106. if (ent.name[0] == 0) {
  107. break;
  108. }
  109. rc = get_log_file_id(&ent);
  110. if (rc == num) {
  111. rc = 1;
  112. goto close_dir;
  113. }
  114. }
  115. rc = 0;
  116. close_dir:
  117. (void) fs_closedir(&dir);
  118. return rc;
  119. }
  120. int write_log_to_file(uint8_t *data, size_t length, void *ctx)
  121. {
  122. int rc;
  123. struct fs_file_t *f = &file;
  124. if (backend_state == BACKEND_FS_NOT_INITIALIZED) {
  125. if (check_log_volumen_available()) {
  126. return length;
  127. }
  128. rc = create_log_dir(CONFIG_LOG_BACKEND_FS_DIR);
  129. if (!rc) {
  130. rc = allocate_new_file(&file);
  131. }
  132. backend_state = (rc ? BACKEND_FS_CORRUPTED : BACKEND_FS_OK);
  133. }
  134. if (backend_state == BACKEND_FS_OK) {
  135. /* Check if new data overwrites max file size.
  136. * If so, create new log file.
  137. */
  138. int size = fs_tell(f);
  139. if (size < 0) {
  140. backend_state = BACKEND_FS_CORRUPTED;
  141. return length;
  142. } else if ((size + length) > CONFIG_LOG_BACKEND_FS_FILE_SIZE) {
  143. rc = allocate_new_file(f);
  144. if (rc < 0) {
  145. goto on_error;
  146. }
  147. }
  148. rc = fs_write(f, data, length);
  149. if (rc >= 0) {
  150. if (IS_ENABLED(CONFIG_LOG_BACKEND_FS_OVERWRITE) &&
  151. (rc != length)) {
  152. del_oldest_log();
  153. return 0;
  154. }
  155. /* If overwrite is disabled, full memory
  156. * cause the log record abandonment.
  157. */
  158. length = rc;
  159. } else {
  160. rc = check_log_file_exist(newest);
  161. if (rc == 0) {
  162. /* file was lost somehow
  163. * try to get a new one
  164. */
  165. file_ctr--;
  166. rc = allocate_new_file(f);
  167. if (rc < 0) {
  168. goto on_error;
  169. }
  170. } else if (rc < 0) {
  171. /* fs is corrupted*/
  172. goto on_error;
  173. }
  174. length = 0;
  175. }
  176. rc = fs_sync(f);
  177. if (rc < 0) {
  178. /* Something is wrong */
  179. goto on_error;
  180. }
  181. }
  182. return length;
  183. on_error:
  184. backend_state = BACKEND_FS_CORRUPTED;
  185. return length;
  186. }
  187. static int get_log_file_id(struct fs_dirent *ent)
  188. {
  189. size_t len;
  190. int num;
  191. if (ent->type != FS_DIR_ENTRY_FILE) {
  192. return -1;
  193. }
  194. len = strlen(ent->name);
  195. if (len != LOG_PREFIX_LEN + FILE_NUMERAL_LEN) {
  196. return -1;
  197. }
  198. if (memcmp(ent->name, CONFIG_LOG_BACKEND_FS_FILE_PREFIX, LOG_PREFIX_LEN) != 0) {
  199. return -1;
  200. }
  201. num = atoi(ent->name + LOG_PREFIX_LEN);
  202. if (num <= MAX_FILE_NUMERAL && num >= 0) {
  203. return num;
  204. }
  205. return -1;
  206. }
  207. static int allocate_new_file(struct fs_file_t *file)
  208. {
  209. /* In case of no log file or current file fills up
  210. * create new log file.
  211. */
  212. int rc;
  213. struct fs_statvfs stat;
  214. int curr_file_num;
  215. struct fs_dirent ent;
  216. char fname[MAX_PATH_LEN];
  217. assert(file);
  218. if (backend_state == BACKEND_FS_NOT_INITIALIZED) {
  219. /* Search for the last used log number. */
  220. struct fs_dir_t dir;
  221. int file_num = 0;
  222. fs_dir_t_init(&dir);
  223. curr_file_num = 0;
  224. int max = 0, min = MAX_FILE_NUMERAL;
  225. rc = fs_opendir(&dir, CONFIG_LOG_BACKEND_FS_DIR);
  226. while (rc >= 0) {
  227. rc = fs_readdir(&dir, &ent);
  228. if ((rc < 0) || (ent.name[0] == 0)) {
  229. break;
  230. }
  231. file_num = get_log_file_id(&ent);
  232. if (file_num >= 0) {
  233. if (file_num > max) {
  234. max = file_num;
  235. }
  236. if (file_num < min) {
  237. min = file_num;
  238. }
  239. ++file_ctr;
  240. }
  241. }
  242. oldest = min;
  243. if ((file_ctr > 1) &&
  244. ((max - min) >
  245. 2 * CONFIG_LOG_BACKEND_FS_FILES_LIMIT)) {
  246. /* oldest log is in the range around the min */
  247. newest = min;
  248. oldest = max;
  249. (void)fs_closedir(&dir);
  250. rc = fs_opendir(&dir, CONFIG_LOG_BACKEND_FS_DIR);
  251. while (rc == 0) {
  252. rc = fs_readdir(&dir, &ent);
  253. if ((rc < 0) || (ent.name[0] == 0)) {
  254. break;
  255. }
  256. file_num = get_log_file_id(&ent);
  257. if (file_num < min + CONFIG_LOG_BACKEND_FS_FILES_LIMIT) {
  258. if (newest < file_num) {
  259. newest = file_num;
  260. }
  261. }
  262. if (file_num > max - CONFIG_LOG_BACKEND_FS_FILES_LIMIT) {
  263. if (oldest > file_num) {
  264. oldest = file_num;
  265. }
  266. }
  267. }
  268. } else {
  269. newest = max;
  270. oldest = min;
  271. }
  272. (void)fs_closedir(&dir);
  273. if (rc < 0) {
  274. goto out;
  275. }
  276. curr_file_num = newest;
  277. if (file_ctr >= 1) {
  278. curr_file_num++;
  279. if (curr_file_num > MAX_FILE_NUMERAL) {
  280. curr_file_num = 0;
  281. }
  282. }
  283. backend_state = BACKEND_FS_OK;
  284. } else {
  285. fs_close(file);
  286. curr_file_num = newest;
  287. curr_file_num++;
  288. if (curr_file_num > MAX_FILE_NUMERAL) {
  289. curr_file_num = 0;
  290. }
  291. }
  292. rc = fs_statvfs(CONFIG_LOG_BACKEND_FS_DIR, &stat);
  293. /* Check if there is enough space to write file or max files number
  294. * is not exceeded.
  295. */
  296. while ((file_ctr >= CONFIG_LOG_BACKEND_FS_FILES_LIMIT) ||
  297. ((stat.f_bfree * stat.f_frsize) <=
  298. CONFIG_LOG_BACKEND_FS_FILE_SIZE)) {
  299. if (IS_ENABLED(CONFIG_LOG_BACKEND_FS_OVERWRITE)) {
  300. rc = del_oldest_log();
  301. if (rc < 0) {
  302. goto out;
  303. }
  304. rc = fs_statvfs(CONFIG_LOG_BACKEND_FS_DIR,
  305. &stat);
  306. if (rc < 0) {
  307. goto out;
  308. }
  309. } else {
  310. return -ENOSPC;
  311. }
  312. }
  313. snprintf(fname, sizeof(fname), "%s/%s%04d",
  314. CONFIG_LOG_BACKEND_FS_DIR,
  315. CONFIG_LOG_BACKEND_FS_FILE_PREFIX, curr_file_num);
  316. rc = fs_open(file, fname, FS_O_CREATE | FS_O_WRITE);
  317. if (rc < 0) {
  318. goto out;
  319. }
  320. ++file_ctr;
  321. newest = curr_file_num;
  322. out:
  323. return rc;
  324. }
  325. static int del_oldest_log(void)
  326. {
  327. int rc;
  328. static char dellname[MAX_PATH_LEN];
  329. while (true) {
  330. snprintf(dellname, sizeof(dellname), "%s/%s%04d",
  331. CONFIG_LOG_BACKEND_FS_DIR,
  332. CONFIG_LOG_BACKEND_FS_FILE_PREFIX, oldest);
  333. rc = fs_unlink(dellname);
  334. if ((rc == 0) || (rc == -ENOENT)) {
  335. oldest++;
  336. if (oldest > MAX_FILE_NUMERAL) {
  337. oldest = 0;
  338. }
  339. if (rc == 0) {
  340. --file_ctr;
  341. break;
  342. }
  343. } else {
  344. break;
  345. }
  346. }
  347. return rc;
  348. }
  349. BUILD_ASSERT(!IS_ENABLED(CONFIG_LOG_MODE_IMMEDIATE),
  350. "Immediate logging is not supported by LOG FS backend.");
  351. #ifndef CONFIG_LOG_BACKEND_FS_TESTSUITE
  352. static uint8_t __aligned(4) buf[MAX_FLASH_WRITE_SIZE];
  353. LOG_OUTPUT_DEFINE(log_output, write_log_to_file, buf, MAX_FLASH_WRITE_SIZE);
  354. static void put(const struct log_backend *const backend,
  355. struct log_msg *msg)
  356. {
  357. log_backend_std_put(&log_output, 0, msg);
  358. }
  359. static void log_backend_fs_init(const struct log_backend *const backend)
  360. {
  361. }
  362. static void panic(struct log_backend const *const backend)
  363. {
  364. /* In case of panic deinitialize backend. It is better to keep
  365. * current data rather than log new and risk of failure.
  366. */
  367. log_backend_deactivate(backend);
  368. }
  369. static void dropped(const struct log_backend *const backend, uint32_t cnt)
  370. {
  371. ARG_UNUSED(backend);
  372. if (IS_ENABLED(CONFIG_LOG_BACKEND_FS_OUTPUT_DICTIONARY)) {
  373. log_dict_output_dropped_process(&log_output, cnt);
  374. } else {
  375. log_backend_std_dropped(&log_output, cnt);
  376. }
  377. }
  378. static void process(const struct log_backend *const backend,
  379. union log_msg2_generic *msg)
  380. {
  381. uint32_t flags = log_backend_std_get_flags();
  382. if (IS_ENABLED(CONFIG_LOG_BACKEND_FS_OUTPUT_DICTIONARY)) {
  383. log_dict_output_msg2_process(&log_output,
  384. &msg->log, flags);
  385. } else {
  386. log_output_msg2_process(&log_output, &msg->log, flags);
  387. }
  388. }
  389. static const struct log_backend_api log_backend_fs_api = {
  390. .process = IS_ENABLED(CONFIG_LOG2) ? process : NULL,
  391. .put = put,
  392. .put_sync_string = NULL,
  393. .put_sync_hexdump = NULL,
  394. .panic = panic,
  395. .init = log_backend_fs_init,
  396. .dropped = dropped,
  397. };
  398. LOG_BACKEND_DEFINE(log_backend_fs, log_backend_fs_api, true);
  399. #endif