fcb.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. /*
  2. * Copyright (c) 2017-2020 Nordic Semiconductor ASA
  3. * Copyright (c) 2015 Runtime Inc
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #include <limits.h>
  8. #include <stdlib.h>
  9. #include <fs/fcb.h>
  10. #include "fcb_priv.h"
  11. #include "string.h"
  12. #include <errno.h>
  13. #include <device.h>
  14. #include <drivers/flash.h>
  15. uint8_t
  16. fcb_get_align(const struct fcb *fcb)
  17. {
  18. uint8_t align;
  19. if (fcb->fap == NULL) {
  20. return 0;
  21. }
  22. align = flash_area_align(fcb->fap);
  23. return align;
  24. }
  25. int fcb_flash_read(const struct fcb *fcb, const struct flash_sector *sector,
  26. off_t off, void *dst, size_t len)
  27. {
  28. int rc;
  29. if (off + len > sector->fs_size) {
  30. return -EINVAL;
  31. }
  32. if (fcb->fap == NULL) {
  33. return -EIO;
  34. }
  35. rc = flash_area_read(fcb->fap, sector->fs_off + off, dst, len);
  36. if (rc != 0) {
  37. return -EIO;
  38. }
  39. return 0;
  40. }
  41. int fcb_flash_write(const struct fcb *fcb, const struct flash_sector *sector,
  42. off_t off, const void *src, size_t len)
  43. {
  44. int rc;
  45. if (off + len > sector->fs_size) {
  46. return -EINVAL;
  47. }
  48. if (fcb->fap == NULL) {
  49. return -EIO;
  50. }
  51. rc = flash_area_write(fcb->fap, sector->fs_off + off, src, len);
  52. if (rc != 0) {
  53. return -EIO;
  54. }
  55. return 0;
  56. }
  57. int
  58. fcb_erase_sector(const struct fcb *fcb, const struct flash_sector *sector)
  59. {
  60. int rc;
  61. if (fcb->fap == NULL) {
  62. return -EIO;
  63. }
  64. rc = flash_area_erase(fcb->fap, sector->fs_off, sector->fs_size);
  65. if (rc != 0) {
  66. return -EIO;
  67. }
  68. return 0;
  69. }
  70. int
  71. fcb_init(int f_area_id, struct fcb *fcb)
  72. {
  73. struct flash_sector *sector;
  74. int rc;
  75. int i;
  76. uint8_t align;
  77. int oldest = -1, newest = -1;
  78. struct flash_sector *oldest_sector = NULL, *newest_sector = NULL;
  79. struct fcb_disk_area fda;
  80. const struct device *dev = NULL;
  81. const struct flash_parameters *fparam;
  82. if (!fcb->f_sectors || fcb->f_sector_cnt - fcb->f_scratch_cnt < 1) {
  83. return -EINVAL;
  84. }
  85. rc = flash_area_open(f_area_id, &fcb->fap);
  86. if (rc != 0) {
  87. return -EINVAL;
  88. }
  89. dev = device_get_binding(fcb->fap->fa_dev_name);
  90. fparam = flash_get_parameters(dev);
  91. fcb->f_erase_value = fparam->erase_value;
  92. align = fcb_get_align(fcb);
  93. if (align == 0U) {
  94. return -EINVAL;
  95. }
  96. /* Fill last used, first used */
  97. for (i = 0; i < fcb->f_sector_cnt; i++) {
  98. sector = &fcb->f_sectors[i];
  99. rc = fcb_sector_hdr_read(fcb, sector, &fda);
  100. if (rc < 0) {
  101. return rc;
  102. }
  103. if (rc == 0) {
  104. continue;
  105. }
  106. if (oldest < 0) {
  107. oldest = newest = fda.fd_id;
  108. oldest_sector = newest_sector = sector;
  109. continue;
  110. }
  111. if (FCB_ID_GT(fda.fd_id, newest)) {
  112. newest = fda.fd_id;
  113. newest_sector = sector;
  114. } else if (FCB_ID_GT(oldest, fda.fd_id)) {
  115. oldest = fda.fd_id;
  116. oldest_sector = sector;
  117. }
  118. }
  119. if (oldest < 0) {
  120. /*
  121. * No initialized areas.
  122. */
  123. oldest_sector = newest_sector = &fcb->f_sectors[0];
  124. rc = fcb_sector_hdr_init(fcb, oldest_sector, 0);
  125. if (rc) {
  126. return rc;
  127. }
  128. newest = oldest = 0;
  129. }
  130. fcb->f_align = align;
  131. fcb->f_oldest = oldest_sector;
  132. fcb->f_active.fe_sector = newest_sector;
  133. fcb->f_active.fe_elem_off = sizeof(struct fcb_disk_area);
  134. fcb->f_active_id = newest;
  135. while (1) {
  136. rc = fcb_getnext_in_sector(fcb, &fcb->f_active);
  137. if (rc == -ENOTSUP) {
  138. rc = 0;
  139. break;
  140. }
  141. if (rc != 0) {
  142. break;
  143. }
  144. }
  145. k_mutex_init(&fcb->f_mtx);
  146. return rc;
  147. }
  148. int
  149. fcb_free_sector_cnt(struct fcb *fcb)
  150. {
  151. int i;
  152. struct flash_sector *fa;
  153. fa = fcb->f_active.fe_sector;
  154. for (i = 0; i < fcb->f_sector_cnt; i++) {
  155. fa = fcb_getnext_sector(fcb, fa);
  156. if (fa == fcb->f_oldest) {
  157. break;
  158. }
  159. }
  160. return i;
  161. }
  162. int
  163. fcb_is_empty(struct fcb *fcb)
  164. {
  165. return (fcb->f_active.fe_sector == fcb->f_oldest &&
  166. fcb->f_active.fe_elem_off == sizeof(struct fcb_disk_area));
  167. }
  168. /**
  169. * Length of an element is encoded in 1 or 2 bytes.
  170. * 1 byte for lengths < 128 bytes, and 2 bytes for < 16384.
  171. *
  172. * The storage of length has been originally designed to work with 0xff erasable
  173. * flash devices and gives length 0xffff special meaning: that there is no value
  174. * written; this is smart way to utilize value in non-written flash to figure
  175. * out where data ends. Additionally it sets highest bit of first byte of
  176. * the length to 1, to mark that there is second byte to be read.
  177. * Above poses some problems when non-0xff erasable flash is used. To solve
  178. * the problem all length values are xored with not of erase value for given
  179. * flash:
  180. * len' = len ^ ~erase_value;
  181. * To obtain original value, the logic is reversed:
  182. * len = len' ^ ~erase_value;
  183. *
  184. * In case of 0xff erased flash this does not modify data that is written to
  185. * flash; in case of other flash devices, e.g. that erase to 0x00, it allows
  186. * to correctly use the first bit of byte to figure out how many bytes are there
  187. * and if there is any data at all or both bytes are equal to erase value.
  188. */
  189. int
  190. fcb_put_len(const struct fcb *fcb, uint8_t *buf, uint16_t len)
  191. {
  192. if (len < 0x80) {
  193. buf[0] = len ^ ~fcb->f_erase_value;
  194. return 1;
  195. } else if (len < FCB_MAX_LEN) {
  196. buf[0] = (len | 0x80) ^ ~fcb->f_erase_value;
  197. buf[1] = (len >> 7) ^ ~fcb->f_erase_value;
  198. return 2;
  199. } else {
  200. return -EINVAL;
  201. }
  202. }
  203. int
  204. fcb_get_len(const struct fcb *fcb, uint8_t *buf, uint16_t *len)
  205. {
  206. int rc;
  207. if ((buf[0] ^ ~fcb->f_erase_value) & 0x80) {
  208. if ((buf[0] == fcb->f_erase_value) &&
  209. (buf[1] == fcb->f_erase_value)) {
  210. return -ENOTSUP;
  211. }
  212. *len = ((buf[0] ^ ~fcb->f_erase_value) & 0x7f) |
  213. ((uint8_t)(buf[1] ^ ~fcb->f_erase_value) << 7);
  214. rc = 2;
  215. } else {
  216. *len = (uint8_t)(buf[0] ^ ~fcb->f_erase_value);
  217. rc = 1;
  218. }
  219. return rc;
  220. }
  221. /**
  222. * Initialize erased sector for use.
  223. */
  224. int
  225. fcb_sector_hdr_init(struct fcb *fcb, struct flash_sector *sector, uint16_t id)
  226. {
  227. struct fcb_disk_area fda;
  228. int rc;
  229. fda.fd_magic = fcb_flash_magic(fcb);
  230. fda.fd_ver = fcb->f_version;
  231. fda._pad = fcb->f_erase_value;
  232. fda.fd_id = id;
  233. rc = fcb_flash_write(fcb, sector, 0, &fda, sizeof(fda));
  234. if (rc != 0) {
  235. return -EIO;
  236. }
  237. return 0;
  238. }
  239. /**
  240. * Checks whether FCB sector contains data or not.
  241. * Returns <0 in error.
  242. * Returns 0 if sector is unused;
  243. * Returns 1 if sector has data.
  244. */
  245. int fcb_sector_hdr_read(struct fcb *fcb, struct flash_sector *sector,
  246. struct fcb_disk_area *fdap)
  247. {
  248. struct fcb_disk_area fda;
  249. int rc;
  250. if (!fdap) {
  251. fdap = &fda;
  252. }
  253. rc = fcb_flash_read(fcb, sector, 0, fdap, sizeof(*fdap));
  254. if (rc) {
  255. return -EIO;
  256. }
  257. if (fdap->fd_magic == MK32(fcb->f_erase_value)) {
  258. return 0;
  259. }
  260. if (fdap->fd_magic != fcb_flash_magic(fcb)) {
  261. return -ENOMSG;
  262. }
  263. return 1;
  264. }
  265. /**
  266. * Finds the fcb entry that gives back upto n entries at the end.
  267. * @param0 ptr to fcb
  268. * @param1 n number of fcb entries the user wants to get
  269. * @param2 ptr to the fcb_entry to be returned
  270. * @return 0 on there are any fcbs aviable; -ENOENT otherwise
  271. */
  272. int
  273. fcb_offset_last_n(struct fcb *fcb, uint8_t entries,
  274. struct fcb_entry *last_n_entry)
  275. {
  276. struct fcb_entry loc;
  277. int i;
  278. int rc;
  279. /* assure a minimum amount of entries */
  280. if (!entries) {
  281. entries = 1U;
  282. }
  283. i = 0;
  284. (void)memset(&loc, 0, sizeof(loc));
  285. while (!fcb_getnext(fcb, &loc)) {
  286. if (i == 0) {
  287. /* Start from the beginning of fcb entries */
  288. *last_n_entry = loc;
  289. }
  290. /* Update last_n_entry after n entries and keep updating */
  291. else if (i > (entries - 1)) {
  292. rc = fcb_getnext(fcb, last_n_entry);
  293. if (rc) {
  294. /* A fcb history must have been erased,
  295. * wanted entry doesn't exist anymore.
  296. */
  297. return -ENOENT;
  298. }
  299. }
  300. i++;
  301. }
  302. return (i == 0) ? -ENOENT : 0;
  303. }
  304. /**
  305. * Clear fcb
  306. * @param fcb
  307. * @return 0 on success; non-zero on failure
  308. */
  309. int
  310. fcb_clear(struct fcb *fcb)
  311. {
  312. int rc;
  313. rc = 0;
  314. while (!fcb_is_empty(fcb)) {
  315. rc = fcb_rotate(fcb);
  316. if (rc) {
  317. break;
  318. }
  319. }
  320. return rc;
  321. }