flipper_format.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638
  1. #include <core/check.h>
  2. #include <toolbox/stream/stream.h>
  3. #include <toolbox/stream/string_stream.h>
  4. #include <toolbox/stream/file_stream.h>
  5. #include <toolbox/stream/buffered_file_stream.h>
  6. #include "flipper_format.h"
  7. #include "flipper_format_i.h"
  8. #include "flipper_format_stream.h"
  9. #include "flipper_format_stream_i.h"
  10. /********************************** Private **********************************/
  11. struct FlipperFormat {
  12. Stream* stream;
  13. bool strict_mode;
  14. };
  15. static const char* const flipper_format_filetype_key = "Filetype";
  16. static const char* const flipper_format_version_key = "Version";
  17. Stream* flipper_format_get_raw_stream(FlipperFormat* flipper_format) {
  18. return flipper_format->stream;
  19. }
  20. /********************************** Public **********************************/
  21. FlipperFormat* flipper_format_string_alloc() {
  22. FlipperFormat* flipper_format = malloc(sizeof(FlipperFormat));
  23. flipper_format->stream = string_stream_alloc();
  24. flipper_format->strict_mode = false;
  25. return flipper_format;
  26. }
  27. FlipperFormat* flipper_format_file_alloc(Storage* storage) {
  28. FlipperFormat* flipper_format = malloc(sizeof(FlipperFormat));
  29. flipper_format->stream = file_stream_alloc(storage);
  30. flipper_format->strict_mode = false;
  31. return flipper_format;
  32. }
  33. FlipperFormat* flipper_format_buffered_file_alloc(Storage* storage) {
  34. FlipperFormat* flipper_format = malloc(sizeof(FlipperFormat));
  35. flipper_format->stream = buffered_file_stream_alloc(storage);
  36. flipper_format->strict_mode = false;
  37. return flipper_format;
  38. }
  39. bool flipper_format_file_open_existing(FlipperFormat* flipper_format, const char* path) {
  40. furi_assert(flipper_format);
  41. return file_stream_open(flipper_format->stream, path, FSAM_READ_WRITE, FSOM_OPEN_EXISTING);
  42. }
  43. bool flipper_format_buffered_file_open_existing(FlipperFormat* flipper_format, const char* path) {
  44. furi_assert(flipper_format);
  45. return buffered_file_stream_open(
  46. flipper_format->stream, path, FSAM_READ_WRITE, FSOM_OPEN_EXISTING);
  47. }
  48. bool flipper_format_file_open_append(FlipperFormat* flipper_format, const char* path) {
  49. furi_assert(flipper_format);
  50. bool result =
  51. file_stream_open(flipper_format->stream, path, FSAM_READ_WRITE, FSOM_OPEN_APPEND);
  52. // Add EOL if it is not there
  53. if(stream_size(flipper_format->stream) >= 1) {
  54. do {
  55. char last_char;
  56. result = false;
  57. if(!stream_seek(flipper_format->stream, -1, StreamOffsetFromEnd)) break;
  58. uint16_t bytes_were_read =
  59. stream_read(flipper_format->stream, (uint8_t*)&last_char, 1);
  60. if(bytes_were_read != 1) break;
  61. if(last_char != flipper_format_eoln) {
  62. if(!flipper_format_stream_write_eol(flipper_format->stream)) break;
  63. }
  64. result = true;
  65. } while(false);
  66. } else {
  67. stream_seek(flipper_format->stream, 0, StreamOffsetFromEnd);
  68. }
  69. return result;
  70. }
  71. bool flipper_format_file_open_always(FlipperFormat* flipper_format, const char* path) {
  72. furi_assert(flipper_format);
  73. return file_stream_open(flipper_format->stream, path, FSAM_READ_WRITE, FSOM_CREATE_ALWAYS);
  74. }
  75. bool flipper_format_file_open_new(FlipperFormat* flipper_format, const char* path) {
  76. furi_assert(flipper_format);
  77. return file_stream_open(flipper_format->stream, path, FSAM_READ_WRITE, FSOM_CREATE_NEW);
  78. }
  79. bool flipper_format_file_close(FlipperFormat* flipper_format) {
  80. furi_assert(flipper_format);
  81. return file_stream_close(flipper_format->stream);
  82. }
  83. bool flipper_format_buffered_file_close(FlipperFormat* flipper_format) {
  84. furi_assert(flipper_format);
  85. return buffered_file_stream_close(flipper_format->stream);
  86. }
  87. void flipper_format_free(FlipperFormat* flipper_format) {
  88. furi_assert(flipper_format);
  89. stream_free(flipper_format->stream);
  90. free(flipper_format);
  91. }
  92. void flipper_format_set_strict_mode(FlipperFormat* flipper_format, bool strict_mode) {
  93. flipper_format->strict_mode = strict_mode;
  94. }
  95. bool flipper_format_rewind(FlipperFormat* flipper_format) {
  96. furi_assert(flipper_format);
  97. return stream_rewind(flipper_format->stream);
  98. }
  99. bool flipper_format_seek_to_end(FlipperFormat* flipper_format) {
  100. furi_assert(flipper_format);
  101. return stream_seek(flipper_format->stream, 0, StreamOffsetFromEnd);
  102. }
  103. bool flipper_format_key_exist(FlipperFormat* flipper_format, const char* key) {
  104. size_t pos = stream_tell(flipper_format->stream);
  105. stream_seek(flipper_format->stream, 0, StreamOffsetFromStart);
  106. bool result = flipper_format_stream_seek_to_key(flipper_format->stream, key, false);
  107. stream_seek(flipper_format->stream, pos, StreamOffsetFromStart);
  108. return result;
  109. }
  110. bool flipper_format_read_header(
  111. FlipperFormat* flipper_format,
  112. FuriString* filetype,
  113. uint32_t* version) {
  114. furi_assert(flipper_format);
  115. return flipper_format_read_string(flipper_format, flipper_format_filetype_key, filetype) &&
  116. flipper_format_read_uint32(flipper_format, flipper_format_version_key, version, 1);
  117. }
  118. bool flipper_format_write_header(
  119. FlipperFormat* flipper_format,
  120. FuriString* filetype,
  121. const uint32_t version) {
  122. furi_assert(flipper_format);
  123. return flipper_format_write_header_cstr(
  124. flipper_format, furi_string_get_cstr(filetype), version);
  125. }
  126. bool flipper_format_write_header_cstr(
  127. FlipperFormat* flipper_format,
  128. const char* filetype,
  129. const uint32_t version) {
  130. furi_assert(flipper_format);
  131. return flipper_format_write_string_cstr(
  132. flipper_format, flipper_format_filetype_key, filetype) &&
  133. flipper_format_write_uint32(flipper_format, flipper_format_version_key, &version, 1);
  134. }
  135. bool flipper_format_get_value_count(
  136. FlipperFormat* flipper_format,
  137. const char* key,
  138. uint32_t* count) {
  139. furi_assert(flipper_format);
  140. return flipper_format_stream_get_value_count(
  141. flipper_format->stream, key, count, flipper_format->strict_mode);
  142. }
  143. bool flipper_format_read_string(FlipperFormat* flipper_format, const char* key, FuriString* data) {
  144. furi_assert(flipper_format);
  145. return flipper_format_stream_read_value_line(
  146. flipper_format->stream, key, FlipperStreamValueStr, data, 1, flipper_format->strict_mode);
  147. }
  148. bool flipper_format_write_string(FlipperFormat* flipper_format, const char* key, FuriString* data) {
  149. furi_assert(flipper_format);
  150. FlipperStreamWriteData write_data = {
  151. .key = key,
  152. .type = FlipperStreamValueStr,
  153. .data = furi_string_get_cstr(data),
  154. .data_size = 1,
  155. };
  156. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  157. return result;
  158. }
  159. bool flipper_format_write_string_cstr(
  160. FlipperFormat* flipper_format,
  161. const char* key,
  162. const char* data) {
  163. furi_assert(flipper_format);
  164. FlipperStreamWriteData write_data = {
  165. .key = key,
  166. .type = FlipperStreamValueStr,
  167. .data = data,
  168. .data_size = 1,
  169. };
  170. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  171. return result;
  172. }
  173. bool flipper_format_read_hex_uint64(
  174. FlipperFormat* flipper_format,
  175. const char* key,
  176. uint64_t* data,
  177. const uint16_t data_size) {
  178. furi_assert(flipper_format);
  179. return flipper_format_stream_read_value_line(
  180. flipper_format->stream,
  181. key,
  182. FlipperStreamValueHexUint64,
  183. data,
  184. data_size,
  185. flipper_format->strict_mode);
  186. }
  187. bool flipper_format_write_hex_uint64(
  188. FlipperFormat* flipper_format,
  189. const char* key,
  190. const uint64_t* data,
  191. const uint16_t data_size) {
  192. furi_assert(flipper_format);
  193. FlipperStreamWriteData write_data = {
  194. .key = key,
  195. .type = FlipperStreamValueHexUint64,
  196. .data = data,
  197. .data_size = data_size,
  198. };
  199. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  200. return result;
  201. }
  202. bool flipper_format_read_uint32(
  203. FlipperFormat* flipper_format,
  204. const char* key,
  205. uint32_t* data,
  206. const uint16_t data_size) {
  207. furi_assert(flipper_format);
  208. return flipper_format_stream_read_value_line(
  209. flipper_format->stream,
  210. key,
  211. FlipperStreamValueUint32,
  212. data,
  213. data_size,
  214. flipper_format->strict_mode);
  215. }
  216. bool flipper_format_write_uint32(
  217. FlipperFormat* flipper_format,
  218. const char* key,
  219. const uint32_t* data,
  220. const uint16_t data_size) {
  221. furi_assert(flipper_format);
  222. FlipperStreamWriteData write_data = {
  223. .key = key,
  224. .type = FlipperStreamValueUint32,
  225. .data = data,
  226. .data_size = data_size,
  227. };
  228. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  229. return result;
  230. }
  231. bool flipper_format_read_int32(
  232. FlipperFormat* flipper_format,
  233. const char* key,
  234. int32_t* data,
  235. const uint16_t data_size) {
  236. return flipper_format_stream_read_value_line(
  237. flipper_format->stream,
  238. key,
  239. FlipperStreamValueInt32,
  240. data,
  241. data_size,
  242. flipper_format->strict_mode);
  243. }
  244. bool flipper_format_write_int32(
  245. FlipperFormat* flipper_format,
  246. const char* key,
  247. const int32_t* data,
  248. const uint16_t data_size) {
  249. furi_assert(flipper_format);
  250. FlipperStreamWriteData write_data = {
  251. .key = key,
  252. .type = FlipperStreamValueInt32,
  253. .data = data,
  254. .data_size = data_size,
  255. };
  256. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  257. return result;
  258. }
  259. bool flipper_format_read_bool(
  260. FlipperFormat* flipper_format,
  261. const char* key,
  262. bool* data,
  263. const uint16_t data_size) {
  264. return flipper_format_stream_read_value_line(
  265. flipper_format->stream,
  266. key,
  267. FlipperStreamValueBool,
  268. data,
  269. data_size,
  270. flipper_format->strict_mode);
  271. }
  272. bool flipper_format_write_bool(
  273. FlipperFormat* flipper_format,
  274. const char* key,
  275. const bool* data,
  276. const uint16_t data_size) {
  277. furi_assert(flipper_format);
  278. FlipperStreamWriteData write_data = {
  279. .key = key,
  280. .type = FlipperStreamValueBool,
  281. .data = data,
  282. .data_size = data_size,
  283. };
  284. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  285. return result;
  286. }
  287. bool flipper_format_read_float(
  288. FlipperFormat* flipper_format,
  289. const char* key,
  290. float* data,
  291. const uint16_t data_size) {
  292. return flipper_format_stream_read_value_line(
  293. flipper_format->stream,
  294. key,
  295. FlipperStreamValueFloat,
  296. data,
  297. data_size,
  298. flipper_format->strict_mode);
  299. }
  300. bool flipper_format_write_float(
  301. FlipperFormat* flipper_format,
  302. const char* key,
  303. const float* data,
  304. const uint16_t data_size) {
  305. furi_assert(flipper_format);
  306. FlipperStreamWriteData write_data = {
  307. .key = key,
  308. .type = FlipperStreamValueFloat,
  309. .data = data,
  310. .data_size = data_size,
  311. };
  312. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  313. return result;
  314. }
  315. bool flipper_format_read_hex(
  316. FlipperFormat* flipper_format,
  317. const char* key,
  318. uint8_t* data,
  319. const uint16_t data_size) {
  320. return flipper_format_stream_read_value_line(
  321. flipper_format->stream,
  322. key,
  323. FlipperStreamValueHex,
  324. data,
  325. data_size,
  326. flipper_format->strict_mode);
  327. }
  328. bool flipper_format_write_hex(
  329. FlipperFormat* flipper_format,
  330. const char* key,
  331. const uint8_t* data,
  332. const uint16_t data_size) {
  333. furi_assert(flipper_format);
  334. FlipperStreamWriteData write_data = {
  335. .key = key,
  336. .type = FlipperStreamValueHex,
  337. .data = data,
  338. .data_size = data_size,
  339. };
  340. bool result = flipper_format_stream_write_value_line(flipper_format->stream, &write_data);
  341. return result;
  342. }
  343. bool flipper_format_write_comment(FlipperFormat* flipper_format, FuriString* data) {
  344. furi_assert(flipper_format);
  345. return flipper_format_write_comment_cstr(flipper_format, furi_string_get_cstr(data));
  346. }
  347. bool flipper_format_write_comment_cstr(FlipperFormat* flipper_format, const char* data) {
  348. furi_assert(flipper_format);
  349. return flipper_format_stream_write_comment_cstr(flipper_format->stream, data);
  350. }
  351. bool flipper_format_delete_key(FlipperFormat* flipper_format, const char* key) {
  352. furi_assert(flipper_format);
  353. FlipperStreamWriteData write_data = {
  354. .key = key,
  355. .type = FlipperStreamValueIgnore,
  356. .data = NULL,
  357. .data_size = 0,
  358. };
  359. bool result = flipper_format_stream_delete_key_and_write(
  360. flipper_format->stream, &write_data, flipper_format->strict_mode);
  361. return result;
  362. }
  363. bool flipper_format_update_string(FlipperFormat* flipper_format, const char* key, FuriString* data) {
  364. furi_assert(flipper_format);
  365. FlipperStreamWriteData write_data = {
  366. .key = key,
  367. .type = FlipperStreamValueStr,
  368. .data = furi_string_get_cstr(data),
  369. .data_size = 1,
  370. };
  371. bool result = flipper_format_stream_delete_key_and_write(
  372. flipper_format->stream, &write_data, flipper_format->strict_mode);
  373. return result;
  374. }
  375. bool flipper_format_update_string_cstr(
  376. FlipperFormat* flipper_format,
  377. const char* key,
  378. const char* data) {
  379. furi_assert(flipper_format);
  380. FlipperStreamWriteData write_data = {
  381. .key = key,
  382. .type = FlipperStreamValueStr,
  383. .data = data,
  384. .data_size = 1,
  385. };
  386. bool result = flipper_format_stream_delete_key_and_write(
  387. flipper_format->stream, &write_data, flipper_format->strict_mode);
  388. return result;
  389. }
  390. bool flipper_format_update_uint32(
  391. FlipperFormat* flipper_format,
  392. const char* key,
  393. const uint32_t* data,
  394. const uint16_t data_size) {
  395. furi_assert(flipper_format);
  396. FlipperStreamWriteData write_data = {
  397. .key = key,
  398. .type = FlipperStreamValueUint32,
  399. .data = data,
  400. .data_size = data_size,
  401. };
  402. bool result = flipper_format_stream_delete_key_and_write(
  403. flipper_format->stream, &write_data, flipper_format->strict_mode);
  404. return result;
  405. }
  406. bool flipper_format_update_int32(
  407. FlipperFormat* flipper_format,
  408. const char* key,
  409. const int32_t* data,
  410. const uint16_t data_size) {
  411. FlipperStreamWriteData write_data = {
  412. .key = key,
  413. .type = FlipperStreamValueInt32,
  414. .data = data,
  415. .data_size = data_size,
  416. };
  417. bool result = flipper_format_stream_delete_key_and_write(
  418. flipper_format->stream, &write_data, flipper_format->strict_mode);
  419. return result;
  420. }
  421. bool flipper_format_update_bool(
  422. FlipperFormat* flipper_format,
  423. const char* key,
  424. const bool* data,
  425. const uint16_t data_size) {
  426. FlipperStreamWriteData write_data = {
  427. .key = key,
  428. .type = FlipperStreamValueBool,
  429. .data = data,
  430. .data_size = data_size,
  431. };
  432. bool result = flipper_format_stream_delete_key_and_write(
  433. flipper_format->stream, &write_data, flipper_format->strict_mode);
  434. return result;
  435. }
  436. bool flipper_format_update_float(
  437. FlipperFormat* flipper_format,
  438. const char* key,
  439. const float* data,
  440. const uint16_t data_size) {
  441. FlipperStreamWriteData write_data = {
  442. .key = key,
  443. .type = FlipperStreamValueFloat,
  444. .data = data,
  445. .data_size = data_size,
  446. };
  447. bool result = flipper_format_stream_delete_key_and_write(
  448. flipper_format->stream, &write_data, flipper_format->strict_mode);
  449. return result;
  450. }
  451. bool flipper_format_update_hex(
  452. FlipperFormat* flipper_format,
  453. const char* key,
  454. const uint8_t* data,
  455. const uint16_t data_size) {
  456. FlipperStreamWriteData write_data = {
  457. .key = key,
  458. .type = FlipperStreamValueHex,
  459. .data = data,
  460. .data_size = data_size,
  461. };
  462. bool result = flipper_format_stream_delete_key_and_write(
  463. flipper_format->stream, &write_data, flipper_format->strict_mode);
  464. return result;
  465. }
  466. bool flipper_format_insert_or_update_string(
  467. FlipperFormat* flipper_format,
  468. const char* key,
  469. FuriString* data) {
  470. bool result = false;
  471. if(!flipper_format_key_exist(flipper_format, key)) {
  472. flipper_format_seek_to_end(flipper_format);
  473. result = flipper_format_write_string(flipper_format, key, data);
  474. } else {
  475. result = flipper_format_update_string(flipper_format, key, data);
  476. }
  477. return result;
  478. }
  479. bool flipper_format_insert_or_update_string_cstr(
  480. FlipperFormat* flipper_format,
  481. const char* key,
  482. const char* data) {
  483. bool result = false;
  484. if(!flipper_format_key_exist(flipper_format, key)) {
  485. flipper_format_seek_to_end(flipper_format);
  486. result = flipper_format_write_string_cstr(flipper_format, key, data);
  487. } else {
  488. result = flipper_format_update_string_cstr(flipper_format, key, data);
  489. }
  490. return result;
  491. }
  492. bool flipper_format_insert_or_update_uint32(
  493. FlipperFormat* flipper_format,
  494. const char* key,
  495. const uint32_t* data,
  496. const uint16_t data_size) {
  497. bool result = false;
  498. if(!flipper_format_key_exist(flipper_format, key)) {
  499. flipper_format_seek_to_end(flipper_format);
  500. result = flipper_format_write_uint32(flipper_format, key, data, data_size);
  501. } else {
  502. result = flipper_format_update_uint32(flipper_format, key, data, data_size);
  503. }
  504. return result;
  505. }
  506. bool flipper_format_insert_or_update_int32(
  507. FlipperFormat* flipper_format,
  508. const char* key,
  509. const int32_t* data,
  510. const uint16_t data_size) {
  511. bool result = false;
  512. if(!flipper_format_key_exist(flipper_format, key)) {
  513. flipper_format_seek_to_end(flipper_format);
  514. result = flipper_format_write_int32(flipper_format, key, data, data_size);
  515. } else {
  516. result = flipper_format_update_int32(flipper_format, key, data, data_size);
  517. }
  518. return result;
  519. }
  520. bool flipper_format_insert_or_update_bool(
  521. FlipperFormat* flipper_format,
  522. const char* key,
  523. const bool* data,
  524. const uint16_t data_size) {
  525. bool result = false;
  526. if(!flipper_format_key_exist(flipper_format, key)) {
  527. flipper_format_seek_to_end(flipper_format);
  528. result = flipper_format_write_bool(flipper_format, key, data, data_size);
  529. } else {
  530. result = flipper_format_update_bool(flipper_format, key, data, data_size);
  531. }
  532. return result;
  533. }
  534. bool flipper_format_insert_or_update_float(
  535. FlipperFormat* flipper_format,
  536. const char* key,
  537. const float* data,
  538. const uint16_t data_size) {
  539. bool result = false;
  540. if(!flipper_format_key_exist(flipper_format, key)) {
  541. flipper_format_seek_to_end(flipper_format);
  542. result = flipper_format_write_float(flipper_format, key, data, data_size);
  543. } else {
  544. result = flipper_format_update_float(flipper_format, key, data, data_size);
  545. }
  546. return result;
  547. }
  548. bool flipper_format_insert_or_update_hex(
  549. FlipperFormat* flipper_format,
  550. const char* key,
  551. const uint8_t* data,
  552. const uint16_t data_size) {
  553. bool result = false;
  554. if(!flipper_format_key_exist(flipper_format, key)) {
  555. flipper_format_seek_to_end(flipper_format);
  556. result = flipper_format_write_hex(flipper_format, key, data, data_size);
  557. } else {
  558. result = flipper_format_update_hex(flipper_format, key, data, data_size);
  559. }
  560. return result;
  561. }