fcb.h 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  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. #ifndef ZEPHYR_INCLUDE_FS_FCB_H_
  8. #define ZEPHYR_INCLUDE_FS_FCB_H_
  9. /*
  10. * Flash circular buffer.
  11. */
  12. #include <inttypes.h>
  13. #include <limits.h>
  14. #include <storage/flash_map.h>
  15. #include <kernel.h>
  16. #ifdef __cplusplus
  17. extern "C" {
  18. #endif
  19. /**
  20. * @defgroup fcb Flash Circular Buffer (FCB)
  21. * @ingroup file_system_storage
  22. * @{
  23. * @}
  24. */
  25. /**
  26. * @defgroup fcb_data_structures Flash Circular Buffer Data Structures
  27. * @ingroup fcb
  28. * @{
  29. */
  30. #define FCB_MAX_LEN (CHAR_MAX | CHAR_MAX << 7) /**< Max length of element */
  31. /**
  32. * @brief FCB entry info structure. This data structure describes the element
  33. * location in the flash.
  34. *
  35. * You would use it to figure out what parameters to pass to flash_area_read()
  36. * to read element contents. Or to flash_area_write() when adding a new element.
  37. * Entry location is pointer to area (within fcb->f_sectors), and offset
  38. * within that area.
  39. */
  40. struct fcb_entry {
  41. struct flash_sector *fe_sector;
  42. /**< Pointer to info about sector where data are placed */
  43. uint32_t fe_elem_off;
  44. /**< Offset from the start of the sector to beginning of element. */
  45. uint32_t fe_data_off;
  46. /**< Offset from the start of the sector to the start of element. */
  47. uint16_t fe_data_len; /**< Size of data area in fcb entry*/
  48. };
  49. /**
  50. * @brief Helper macro for calculating the data offset related to
  51. * the fcb flash_area start offset.
  52. *
  53. * @param entry fcb entry structure
  54. */
  55. #define FCB_ENTRY_FA_DATA_OFF(entry) (entry.fe_sector->fs_off +\
  56. entry.fe_data_off)
  57. /**
  58. * @brief Structure for transferring complete information about FCB entry
  59. * location within flash memory.
  60. */
  61. struct fcb_entry_ctx {
  62. struct fcb_entry loc; /**< FCB entry info */
  63. const struct flash_area *fap;
  64. /**< Flash area where the entry is placed */
  65. };
  66. /**
  67. * @brief FCB instance structure
  68. *
  69. * The following data structure describes the FCB itself. First part should
  70. * be filled in by the user before calling @ref fcb_init. The second part is
  71. * used by FCB for its internal bookkeeping.
  72. */
  73. struct fcb {
  74. /* Caller of fcb_init fills this in */
  75. uint32_t f_magic;
  76. /**< Magic value, should not be 0xFFFFFFFF.
  77. * It is xored with inversion of f_erase_value and placed in
  78. * the beginning of FCB flash sector. FCB uses this when determining
  79. * whether sector contains valid data or not.
  80. * Giving it value of 0xFFFFFFFF means leaving bytes of the filed
  81. * in "erased" state.
  82. */
  83. uint8_t f_version; /**< Current version number of the data */
  84. uint8_t f_sector_cnt; /**< Number of elements in sector array */
  85. uint8_t f_scratch_cnt;
  86. /**< Number of sectors to keep empty. This can be used if you need
  87. * to have scratch space for garbage collecting when FCB fills up.
  88. */
  89. struct flash_sector *f_sectors;
  90. /**< Array of sectors, must be contiguous */
  91. /* Flash circular buffer internal state */
  92. struct k_mutex f_mtx;
  93. /**< Locking for accessing the FCB data, internal state */
  94. struct flash_sector *f_oldest;
  95. /**< Pointer to flash sector containing the oldest data,
  96. * internal state
  97. */
  98. struct fcb_entry f_active; /**< internal state */
  99. uint16_t f_active_id;
  100. /**< Flash location where the newest data is, internal state */
  101. uint8_t f_align;
  102. /**< writes to flash have to aligned to this, internal state */
  103. const struct flash_area *fap;
  104. /**< Flash area used by the fcb instance, , internal state.
  105. * This can be transfer to FCB user
  106. */
  107. uint8_t f_erase_value;
  108. /**< The value flash takes when it is erased. This is read from
  109. * flash parameters and initialized upon call to fcb_init.
  110. */
  111. };
  112. /**
  113. * @}
  114. */
  115. /**
  116. * @brief Flash Circular Buffer APIs
  117. * @defgroup fcb_api fcb API
  118. * @ingroup fcb
  119. * @{
  120. */
  121. /**
  122. * Initialize FCB instance.
  123. *
  124. * @param[in] f_area_id ID of flash area where fcb storage resides.
  125. * @param[in,out] fcb FCB instance structure.
  126. *
  127. * @return 0 on success, non-zero on failure.
  128. */
  129. int fcb_init(int f_area_id, struct fcb *fcb);
  130. /**
  131. * Appends an entry to circular buffer.
  132. *
  133. * When writing the
  134. * contents for the entry, use loc->fe_sector and loc->fe_data_off with
  135. * flash_area_write() to fcb flash_area.
  136. * When you're finished, call fcb_append_finish() with loc as argument.
  137. *
  138. * @param[in] fcb FCB instance structure.
  139. * @param[in] len Length of data which are expected to be written as the entry
  140. * payload.
  141. * @param[out] loc entry location information
  142. *
  143. * @return 0 on success, non-zero on failure.
  144. */
  145. int fcb_append(struct fcb *fcb, uint16_t len, struct fcb_entry *loc);
  146. /**
  147. * Finishes entry append operation.
  148. *
  149. * @param[in] fcb FCB instance structure.
  150. * @param[in] append_loc entry location information
  151. *
  152. * @return 0 on success, non-zero on failure.
  153. */
  154. int fcb_append_finish(struct fcb *fcb, struct fcb_entry *append_loc);
  155. /**
  156. * FCB Walk callback function type.
  157. *
  158. * Type of function which is expected to be called while walking over fcb
  159. * entries thanks to a @ref fcb_walk call.
  160. *
  161. * Entry data can be read using flash_area_read(), using
  162. * loc_ctx fields as arguments.
  163. * If cb wants to stop the walk, it should return non-zero value.
  164. *
  165. * @param[in] loc_ctx entry location information (full context)
  166. * @param[in,out] arg callback context, transferred from @ref fcb_walk.
  167. *
  168. * @return 0 continue walking, non-zero stop walking.
  169. */
  170. typedef int (*fcb_walk_cb)(struct fcb_entry_ctx *loc_ctx, void *arg);
  171. /**
  172. * Walk over all entries in the FCB sector
  173. *
  174. * @param[in] sector fcb sector to be walked. If null, traverse entire
  175. * storage.
  176. * @param[in] fcb FCB instance structure.
  177. * @param[in] cb pointer to the function which gets called for every
  178. * entry. If cb wants to stop the walk, it should return
  179. * non-zero value.
  180. * @param[in,out] cb_arg callback context, transferred to the callback
  181. * implementation.
  182. *
  183. * @return 0 on success, negative on failure (or transferred form callback
  184. * return-value), positive transferred form callback return-value
  185. */
  186. int fcb_walk(struct fcb *fcb, struct flash_sector *sector, fcb_walk_cb cb,
  187. void *cb_arg);
  188. /**
  189. * Get next fcb entry location.
  190. *
  191. * Function to obtain fcb entry location in relation to entry pointed by
  192. * <p> loc.
  193. * If loc->fe_sector is set and loc->fe_elem_off is not 0 function fetches next
  194. * fcb entry location.
  195. * If loc->fe_sector is NULL function fetches the oldest entry location within
  196. * FCB storage. loc->fe_sector is set and loc->fe_elem_off is 0 function fetches
  197. * the first entry location in the fcb sector.
  198. *
  199. * @param[in] fcb FCB instance structure.
  200. * @param[in,out] loc entry location information
  201. *
  202. * @return 0 on success, non-zero on failure.
  203. */
  204. int fcb_getnext(struct fcb *fcb, struct fcb_entry *loc);
  205. /*
  206. * Rotate fcb sectors
  207. *
  208. * Function erases the data from oldest sector. Upon that the next sector
  209. * becomes the oldest. Active sector is also switched if needed.
  210. *
  211. * @param[in] fcb FCB instance structure.
  212. */
  213. int fcb_rotate(struct fcb *fcb);
  214. /*
  215. * Start using the scratch block.
  216. *
  217. * Take one of the scratch blocks into use. So a scratch sector becomes
  218. * active sector to which entries can be appended.
  219. *
  220. * @param[in] fcb FCB instance structure.
  221. *
  222. * @return 0 on success, non-zero on failure.
  223. */
  224. int fcb_append_to_scratch(struct fcb *fcb);
  225. /**
  226. * Get free sector count.
  227. *
  228. * @param[in] fcb FCB instance structure.
  229. *
  230. * @return Number of free sectors.
  231. */
  232. int fcb_free_sector_cnt(struct fcb *fcb);
  233. /**
  234. * Check whether FCB has any data.
  235. *
  236. * @param[in] fcb FCB instance structure.
  237. *
  238. * @return Positive value if fcb is empty, otherwise 0.
  239. */
  240. int fcb_is_empty(struct fcb *fcb);
  241. /**
  242. * Finds the fcb entry that gives back up to n entries at the end.
  243. *
  244. * @param[in] fcb FCB instance structure.
  245. * @param[in] entries number of fcb entries the user wants to get
  246. * @param[out] last_n_entry last_n_entry the fcb_entry to be returned
  247. *
  248. * @return 0 on there are any fcbs available; -ENOENT otherwise
  249. */
  250. int fcb_offset_last_n(struct fcb *fcb, uint8_t entries,
  251. struct fcb_entry *last_n_entry);
  252. /**
  253. * Clear fcb instance storage.
  254. *
  255. * @param[in] fcb FCB instance structure.
  256. *
  257. * @return 0 on success; non-zero on failure
  258. */
  259. int fcb_clear(struct fcb *fcb);
  260. /**
  261. * @}
  262. */
  263. /**
  264. * @brief Flash Circular internal
  265. * @defgroup fcb_internall fcb non-API prototypes
  266. * @ingroup fcb
  267. * @{
  268. */
  269. /**
  270. * Read raw data from the fcb flash sector.
  271. *
  272. * @param[in] fcb FCB instance structure.
  273. * @param[in] sector FCB sector.
  274. * @param[in] off Read offset form sector begin.
  275. * @param[out] dst Destination buffer.
  276. * @param[in] len Read-out size.
  277. *
  278. * @return 0 on success, negative errno code on fail.
  279. */
  280. int fcb_flash_read(const struct fcb *fcb, const struct flash_sector *sector,
  281. off_t off, void *dst, size_t len);
  282. /**
  283. * Write raw data to the fcb flash sector.
  284. *
  285. * @param[in] fcb FCB instance structure.
  286. * @param[in] sector FCB sector.
  287. * @param[in] off Write offset form sector begin.
  288. * @param[out] src Source buffer.
  289. * @param[in] len Read-out size.
  290. *
  291. * @return 0 on success, negative errno code on fail.
  292. */
  293. int fcb_flash_write(const struct fcb *fcb, const struct flash_sector *sector,
  294. off_t off, const void *src, size_t len);
  295. /**
  296. * @}
  297. */
  298. #ifdef __cplusplus
  299. }
  300. #endif
  301. #endif /* ZEPHYR_INCLUDE_FS_FCB_H_ */