fcb_append.c 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  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 <stddef.h>
  8. #include <string.h>
  9. #include <fs/fcb.h>
  10. #include "fcb_priv.h"
  11. static struct flash_sector *
  12. fcb_new_sector(struct fcb *fcb, int cnt)
  13. {
  14. struct flash_sector *prev;
  15. struct flash_sector *cur;
  16. int i;
  17. prev = NULL;
  18. i = 0;
  19. cur = fcb->f_active.fe_sector;
  20. do {
  21. cur = fcb_getnext_sector(fcb, cur);
  22. if (!prev) {
  23. prev = cur;
  24. }
  25. if (cur == fcb->f_oldest) {
  26. return NULL;
  27. }
  28. } while (i++ < cnt);
  29. return prev;
  30. }
  31. /*
  32. * Take one of the scratch blocks into use, if at all possible.
  33. */
  34. int
  35. fcb_append_to_scratch(struct fcb *fcb)
  36. {
  37. struct flash_sector *sector;
  38. int rc;
  39. sector = fcb_new_sector(fcb, 0);
  40. if (!sector) {
  41. return -ENOSPC;
  42. }
  43. rc = fcb_sector_hdr_init(fcb, sector, fcb->f_active_id + 1);
  44. if (rc) {
  45. return rc;
  46. }
  47. fcb->f_active.fe_sector = sector;
  48. fcb->f_active.fe_elem_off = sizeof(struct fcb_disk_area);
  49. fcb->f_active_id++;
  50. return 0;
  51. }
  52. int
  53. fcb_append(struct fcb *fcb, uint16_t len, struct fcb_entry *append_loc)
  54. {
  55. struct flash_sector *sector;
  56. struct fcb_entry *active;
  57. int cnt;
  58. int rc;
  59. uint8_t tmp_str[8];
  60. cnt = fcb_put_len(fcb, tmp_str, len);
  61. if (cnt < 0) {
  62. return cnt;
  63. }
  64. cnt = fcb_len_in_flash(fcb, cnt);
  65. len = fcb_len_in_flash(fcb, len) + fcb_len_in_flash(fcb, FCB_CRC_SZ);
  66. __ASSERT_NO_MSG(cnt <= sizeof(tmp_str));
  67. rc = k_mutex_lock(&fcb->f_mtx, K_FOREVER);
  68. if (rc) {
  69. return -EINVAL;
  70. }
  71. active = &fcb->f_active;
  72. if (active->fe_elem_off + len + cnt > active->fe_sector->fs_size) {
  73. sector = fcb_new_sector(fcb, fcb->f_scratch_cnt);
  74. if (!sector || (sector->fs_size <
  75. sizeof(struct fcb_disk_area) + len + cnt)) {
  76. rc = -ENOSPC;
  77. goto err;
  78. }
  79. rc = fcb_sector_hdr_init(fcb, sector, fcb->f_active_id + 1);
  80. if (rc) {
  81. goto err;
  82. }
  83. fcb->f_active.fe_sector = sector;
  84. fcb->f_active.fe_elem_off = sizeof(struct fcb_disk_area);
  85. fcb->f_active_id++;
  86. }
  87. rc = fcb_flash_write(fcb, active->fe_sector, active->fe_elem_off, tmp_str, cnt);
  88. if (rc) {
  89. rc = -EIO;
  90. goto err;
  91. }
  92. append_loc->fe_sector = active->fe_sector;
  93. append_loc->fe_elem_off = active->fe_elem_off;
  94. append_loc->fe_data_off = active->fe_elem_off + cnt;
  95. active->fe_elem_off = append_loc->fe_data_off + len;
  96. k_mutex_unlock(&fcb->f_mtx);
  97. return 0;
  98. err:
  99. k_mutex_unlock(&fcb->f_mtx);
  100. return rc;
  101. }
  102. int
  103. fcb_append_finish(struct fcb *fcb, struct fcb_entry *loc)
  104. {
  105. int rc;
  106. uint8_t crc8[fcb->f_align];
  107. off_t off;
  108. (void)memset(crc8, 0xFF, sizeof(crc8));
  109. rc = fcb_elem_crc8(fcb, loc, &crc8[0]);
  110. if (rc) {
  111. return rc;
  112. }
  113. off = loc->fe_data_off + fcb_len_in_flash(fcb, loc->fe_data_len);
  114. rc = fcb_flash_write(fcb, loc->fe_sector, off, crc8, fcb->f_align);
  115. if (rc) {
  116. return -EIO;
  117. }
  118. return 0;
  119. }