flipbip_file.c 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. #include "flipbip_file.h"
  2. #include <storage/storage.h>
  3. #include "../helpers/flipbip_string.h"
  4. // From: lib/crypto
  5. #include <memzero.h>
  6. #include <rand.h>
  7. #define FLIPBIP_APP_BASE_FOLDER EXT_PATH("apps_data/flipbip")
  8. #define FLIPBIP_DAT_FILE_NAME ".flipbip.dat"
  9. // #define FLIPBIP_DAT_FILE_NAME ".flipbip.dat.txt"
  10. #define FLIPBIP_DAT_FILE_NAME_BAK ".flipbip.dat.bak"
  11. #define FLIPBIP_KEY_FILE_NAME ".flipbip.key"
  12. // #define FLIPBIP_KEY_FILE_NAME ".flipbip.key.txt"
  13. #define FLIPBIP_KEY_FILE_NAME_BAK ".flipbip.key.bak"
  14. #define FLIPBIP_DAT_PATH FLIPBIP_APP_BASE_FOLDER "/" FLIPBIP_DAT_FILE_NAME
  15. #define FLIPBIP_DAT_PATH_BAK FLIPBIP_APP_BASE_FOLDER "/" FLIPBIP_DAT_FILE_NAME_BAK
  16. #define FLIPBIP_KEY_PATH FLIPBIP_APP_BASE_FOLDER "/" FLIPBIP_KEY_FILE_NAME
  17. #define FLIPBIP_KEY_PATH_BAK FLIPBIP_APP_BASE_FOLDER "/" FLIPBIP_KEY_FILE_NAME_BAK
  18. const size_t FILE_HLEN = 4;
  19. const size_t FILE_KLEN = 256;
  20. const size_t FILE_SLEN = 512;
  21. const char* FILE_HSTR = "fb01";
  22. const char* FILE_K1 = "fb0131d5cf688221c109163908ebe51debb46227c6cc8b37641910833222772a"
  23. "baefe6d9ceb651842260e0d1e05e3b90d15e7d5ffaaabc0207bf200a117793a2";
  24. bool flipbip_load_settings(char* settings, bool key_file) {
  25. bool ret = false;
  26. const char* path;
  27. if(key_file) {
  28. path = FLIPBIP_KEY_PATH;
  29. } else {
  30. path = FLIPBIP_DAT_PATH;
  31. }
  32. Storage* fs_api = furi_record_open(RECORD_STORAGE);
  33. File* settings_file = storage_file_alloc(fs_api);
  34. if(storage_file_open(settings_file, path, FSAM_READ, FSOM_OPEN_EXISTING)) {
  35. char chr;
  36. int i = 0;
  37. while((storage_file_read(settings_file, &chr, 1) == 1) &&
  38. !storage_file_eof(settings_file) && !isspace(chr)) {
  39. settings[i] = chr;
  40. i++;
  41. }
  42. ret = true;
  43. } else {
  44. memzero(settings, strlen(settings));
  45. settings[0] = '\0';
  46. ret = false;
  47. }
  48. storage_file_close(settings_file);
  49. storage_file_free(settings_file);
  50. furi_record_close(RECORD_STORAGE);
  51. if(strlen(settings) > 0) {
  52. Storage* fs_api = furi_record_open(RECORD_STORAGE);
  53. FileInfo layout_file_info;
  54. FS_Error file_check_err = storage_common_stat(fs_api, path, &layout_file_info);
  55. furi_record_close(RECORD_STORAGE);
  56. if(file_check_err != FSE_OK) {
  57. memzero(settings, strlen(settings));
  58. settings[0] = '\0';
  59. ret = false;
  60. }
  61. // if(layout_file_info.size != 256) {
  62. // memzero(settings, strlen(settings));
  63. // settings[0] = '\0';
  64. // }
  65. }
  66. return ret;
  67. }
  68. bool flipbip_has_settings(bool key_file) {
  69. bool ret = false;
  70. const char* path;
  71. if(key_file) {
  72. path = FLIPBIP_KEY_PATH;
  73. } else {
  74. path = FLIPBIP_DAT_PATH;
  75. }
  76. Storage* fs_api = furi_record_open(RECORD_STORAGE);
  77. if(storage_file_exists(fs_api, path)) {
  78. ret = true;
  79. }
  80. furi_record_close(RECORD_STORAGE);
  81. return ret;
  82. }
  83. bool flipbip_save_settings(const char* settings, bool key_file, bool append) {
  84. bool ret = false;
  85. const char* path;
  86. const char* path_bak;
  87. if(key_file) {
  88. path = FLIPBIP_KEY_PATH;
  89. path_bak = FLIPBIP_KEY_PATH_BAK;
  90. } else {
  91. path = FLIPBIP_DAT_PATH;
  92. path_bak = FLIPBIP_DAT_PATH_BAK;
  93. }
  94. int open_mode = FSOM_OPEN_ALWAYS;
  95. if(append) {
  96. open_mode = FSOM_OPEN_APPEND;
  97. }
  98. Storage* fs_api = furi_record_open(RECORD_STORAGE);
  99. // // if the key file exists, we don't want to overwrite it
  100. // if (key_file && storage_file_exists(fs_api, path)) {
  101. // furi_record_close(RECORD_STORAGE);
  102. // ret = true;
  103. // return ret;
  104. // }
  105. // try to create the folder
  106. storage_common_mkdir(fs_api, FLIPBIP_APP_BASE_FOLDER);
  107. File* settings_file = storage_file_alloc(fs_api);
  108. if(storage_file_open(settings_file, path, FSAM_WRITE, open_mode)) {
  109. storage_file_write(settings_file, settings, strlen(settings));
  110. storage_file_write(settings_file, "\n", 1);
  111. ret = true;
  112. }
  113. storage_file_close(settings_file);
  114. storage_file_free(settings_file);
  115. File* settings_file_bak = storage_file_alloc(fs_api);
  116. if(storage_file_open(settings_file_bak, path_bak, FSAM_WRITE, open_mode)) {
  117. storage_file_write(settings_file_bak, settings, strlen(settings));
  118. storage_file_write(settings_file_bak, "\n", 1);
  119. }
  120. storage_file_close(settings_file_bak);
  121. storage_file_free(settings_file_bak);
  122. furi_record_close(RECORD_STORAGE);
  123. return ret;
  124. }
  125. bool flipbip_load_settings_secure(char* settings) {
  126. const size_t dlen = FILE_HLEN + FILE_SLEN + 1;
  127. // allocate memory for key/data
  128. char* data = malloc(dlen);
  129. memzero(data, dlen);
  130. // load k2 from file
  131. if(!flipbip_load_settings(data, true)) return false;
  132. // check header
  133. if(data[0] != FILE_HSTR[0] || data[1] != FILE_HSTR[1] || data[2] != FILE_HSTR[2] ||
  134. data[3] != FILE_HSTR[3]) {
  135. memzero(data, dlen);
  136. free(data);
  137. return false;
  138. }
  139. // seek --> header
  140. data += FILE_HLEN;
  141. // prepare k1
  142. uint8_t k1[64];
  143. flipbip_xtob(FILE_K1, k1, strlen(FILE_K1) / 2);
  144. // load k2 from file buffer (secured by k1)
  145. flipbip_cipher(k1, strlen(FILE_K1) / 2, data, data, FILE_KLEN);
  146. uint8_t k2[128];
  147. flipbip_xtob(data, k2, FILE_KLEN / 2);
  148. // zero k2 buffer
  149. memzero(data, FILE_KLEN);
  150. // seek <-- header
  151. data -= FILE_HLEN;
  152. // load data from file
  153. if(!flipbip_load_settings(data, false)) return false;
  154. // check header
  155. if(data[0] != FILE_HSTR[0] || data[1] != FILE_HSTR[1] || data[2] != FILE_HSTR[2] ||
  156. data[3] != FILE_HSTR[3]) {
  157. memzero(data, dlen);
  158. free(data);
  159. memzero(k1, strlen(FILE_K1) / 2);
  160. memzero(k2, FILE_KLEN / 2);
  161. return false;
  162. }
  163. // seek --> header
  164. data += FILE_HLEN;
  165. // load settings from file buffer (secured by k2)
  166. flipbip_cipher(k2, FILE_KLEN / 2, data, data, FILE_SLEN);
  167. flipbip_xtob(data, (unsigned char*)data, FILE_SLEN / 2);
  168. // copy to output
  169. strcpy(settings, data);
  170. // seek <-- header
  171. data -= FILE_HLEN;
  172. // clear memory
  173. memzero(data, dlen);
  174. free(data);
  175. memzero(k1, strlen(FILE_K1) / 2);
  176. memzero(k2, FILE_KLEN / 2);
  177. return true;
  178. }
  179. bool flipbip_save_settings_secure(const char* settings) {
  180. const size_t dlen = FILE_HLEN + FILE_SLEN + 1;
  181. // cap settings to 256 bytes
  182. size_t len = strlen(settings);
  183. if(len > (FILE_SLEN / 2)) len = FILE_SLEN / 2;
  184. // allocate memory for key/data
  185. char* data = malloc(dlen);
  186. memzero(data, dlen);
  187. // write header
  188. strncpy(data, FILE_HSTR, FILE_HLEN);
  189. // seek --> header
  190. data += FILE_HLEN;
  191. // prepare k1
  192. uint8_t k1[64];
  193. flipbip_xtob(FILE_K1, k1, strlen(FILE_K1) / 2);
  194. // generate k2
  195. uint8_t k2[128];
  196. random_buffer(k2, FILE_KLEN / 2);
  197. // write k2 to file buffer (secured by k1)
  198. flipbip_btox(k2, FILE_KLEN / 2, data);
  199. flipbip_cipher(k1, strlen(FILE_K1) / 2, data, data, FILE_KLEN);
  200. // seek <-- header
  201. data -= FILE_HLEN;
  202. // save k2 to file
  203. flipbip_save_settings(data, true, false);
  204. // seek --> header
  205. data += FILE_HLEN;
  206. // zero k2 memory
  207. memzero(data, FILE_KLEN);
  208. // write settings to file buffer (secured by k2)
  209. flipbip_btox((uint8_t*)settings, len, data);
  210. flipbip_cipher(k2, FILE_KLEN / 2, data, data, FILE_SLEN);
  211. // seek <-- header
  212. data -= FILE_HLEN;
  213. // save data to file
  214. flipbip_save_settings(data, false, false);
  215. // clear memory
  216. memzero(data, dlen);
  217. free(data);
  218. memzero(k1, strlen(FILE_K1) / 2);
  219. memzero(k2, FILE_KLEN / 2);
  220. return true;
  221. }