nvram_config.c 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Non-volatile memory driver
  9. */
  10. #ifndef PC
  11. #include <errno.h>
  12. #include <sys/__assert.h>
  13. #include <stdbool.h>
  14. #include <kernel.h>
  15. #include <device.h>
  16. #include <init.h>
  17. #include <drivers/nvram_config.h>
  18. #include <string.h>
  19. #include <storage/flash_map.h>
  20. #include <partition/partition.h>
  21. #include "nvram_storage.h"
  22. #define LOG_LEVEL 2
  23. #include <logging/log.h>
  24. LOG_MODULE_REGISTER(nvram_cfg);
  25. #else
  26. #include "nvram_test_simulate.h"
  27. #include <ffs.h>
  28. #endif
  29. #define CONFIG_NVRAM_FAST_SEARCH
  30. #define NVRAM_REGION_SEG_VERSION 0x1
  31. /* name size + item size <= buffer size */
  32. #define NVRAM_ITEM_ALIGN_SIZE 0x10
  33. #define NVRAM_ERASE_ALIGN_SIZE 0x1000
  34. #define NVRAM_BUFFER_SIZE 128
  35. enum {
  36. ITEM_STATUS_EMPTY = 0,
  37. ITEM_STATUS_VALID,
  38. ITEM_STATUS_OBSOLETE,
  39. ITEM_STATUS_INVALID,
  40. ITEM_STATUS_CRCERR,
  41. };
  42. struct __packed region_seg_header
  43. {
  44. uint32_t magic;
  45. uint8_t state;
  46. uint8_t crc;
  47. uint8_t version;
  48. uint8_t reserved;
  49. uint8_t seq_id;
  50. uint8_t head_size;
  51. uint16_t seg_size;
  52. uint8_t reserved2[4];
  53. };
  54. struct __packed nvram_item {
  55. uint8_t magic;
  56. uint8_t state;
  57. uint8_t crc;
  58. uint8_t hash;
  59. uint8_t reserved;
  60. uint8_t name_size;
  61. uint16_t data_size;
  62. char data[0];
  63. };
  64. struct region_info
  65. {
  66. struct device *storage;
  67. char name[16];
  68. uint32_t flag;
  69. /* write regions base address */
  70. uint32_t base_addr;
  71. /* total write regions size, aligned to erase size */
  72. int32_t total_size;
  73. /* current segment info */
  74. int32_t seg_size;
  75. uint32_t seg_offset;
  76. uint32_t seg_write_offset;
  77. uint8_t seg_seq_id;
  78. #ifdef CONFIG_NVRAM_FAST_SEARCH
  79. uint32_t *seg_item_map;
  80. int seg_item_map_size;
  81. #endif
  82. };
  83. /* region segment magic: 'NVRS' */
  84. #define NVRAM_REGION_SEG_MAGIC 0x5253564E
  85. /* region item magic: 'I' */
  86. #define NVRAM_REGION_ITEM_MAGIC 0x49
  87. #define NVRAM_REGION_SEG_HEADER_CRC_OFFSET 6
  88. #define NVRAM_REGION_ITEM_CRC_OFFSET 3
  89. #define NVRAM_REGION_SEG_STATE_VALID 0xff
  90. #define NVRAM_REGION_SEG_STATE_OBSOLETE 0x5a
  91. #define NVRAM_ITEM_STATE_VALID 0xff
  92. #define NVRAM_ITEM_STATE_OBSOLETE 0x5a
  93. #define NVRAM_SEG_ITEM_START_OFFSET (ROUND_UP(sizeof(struct region_seg_header), NVRAM_ITEM_ALIGN_SIZE))
  94. #ifdef CONFIG_NVRAM_FAST_SEARCH
  95. uint32_t user_region_item_map[CONFIG_NVRAM_USER_REGION_SEGMENT_SIZE / NVRAM_ITEM_ALIGN_SIZE / 32];
  96. #endif
  97. /* user config region */
  98. struct region_info user_nvram_region = {
  99. .name = "User Config",
  100. .base_addr = 0, /*init by nvram_config_init, load from partition table*/
  101. .total_size = 0, /*init by nvram_config_init, load from partition table*/
  102. .seg_size = CONFIG_NVRAM_USER_REGION_SEGMENT_SIZE,
  103. #ifdef CONFIG_NVRAM_FAST_SEARCH
  104. .seg_item_map = user_region_item_map,
  105. .seg_item_map_size = sizeof(user_region_item_map),
  106. #endif
  107. };
  108. /* factory config region */
  109. struct region_info factory_nvram_region = {
  110. .name = "Factory Config",
  111. .base_addr = 0, /*init by nvram_config_init, load from partition table*/
  112. .total_size = 0, /*init by nvram_config_init, load from partition table*/
  113. .seg_size = CONFIG_NVRAM_FACTORY_REGION_SEGMENT_SIZE,
  114. #ifdef CONFIG_NVRAM_FAST_SEARCH
  115. .seg_item_map = NULL,
  116. .seg_item_map_size = 0,
  117. #endif
  118. };
  119. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  120. /* factory writeable config region
  121. * used to store machine specific configs, like SN,MAC...
  122. */
  123. struct region_info factory_rw_nvram_region = {
  124. .name = "Factory RW",
  125. .base_addr = 0, /*init by nvram_config_init, load from partition table*/
  126. .total_size = 0, /*init by nvram_config_init, load from partition table*/
  127. .seg_size = CONFIG_NVRAM_FACTORY_RW_REGION_SEGMENT_SIZE,
  128. #ifdef CONFIG_NVRAM_FAST_SEARCH
  129. .seg_item_map = NULL,
  130. .seg_item_map_size = 0,
  131. #endif
  132. };
  133. #endif
  134. static uint8_t __act_s2_notsave nvram_buf[NVRAM_BUFFER_SIZE];
  135. #ifndef PC
  136. static K_SEM_DEFINE(nvram_lock, 1, 1);
  137. #else
  138. int32_t nvram_lock;
  139. #endif
  140. static int region_read(struct region_info *region, uint32_t offset, uint8_t *buf, int len);
  141. static int region_write(struct region_info *region, uint32_t offset, const uint8_t *buf, int len);
  142. static int region_erase(struct region_info *region, uint32_t offset, int size);
  143. static int region_copy(struct region_info *region, uint32_t src_offset, uint32_t dest_offset, int len);
  144. static int region_is_empy(struct region_info *region, uint32_t offset, int32_t size);
  145. static uint8_t calc_hash(const uint8_t *key, int len)
  146. {
  147. uint8_t hash = 0;
  148. while(len--)
  149. hash += *key++;
  150. return (hash ^ 0xa5);
  151. }
  152. /*
  153. * Name: CRC-8/MAXIM x8+x5+x4+1
  154. * Poly: 0x31
  155. * Init: 0x00
  156. * Refin: True
  157. * Refout: True
  158. * Xorout: 0x00
  159. * Alias: DOW-CRC,CRC-8/IBUTTON
  160. * Use: Maxim(Dallas)'s some devices,e.g. DS18B20
  161. */
  162. static const unsigned char crc8_table[32] = {
  163. 0x00, 0x5E, 0xBC, 0xE2, 0x61, 0x3F, 0xDD, 0x83,
  164. 0xC2, 0x9C, 0x7E, 0x20, 0xA3, 0xFD, 0x1F, 0x41,
  165. 0x00, 0x9D, 0x23, 0xBE, 0x46, 0xDB, 0x65, 0xF8,
  166. 0x8C, 0x11, 0xAF, 0x32, 0xCA, 0x57, 0xE9, 0x74
  167. };
  168. static unsigned char calc_crc8(const unsigned char *addr, int len, uint8_t initial_crc)
  169. {
  170. unsigned char crc = initial_crc;
  171. while (len--) {
  172. crc = *addr++ ^ crc;
  173. crc = crc8_table[crc & 0x0f] ^ crc8_table[16 + ((crc >> 4) & 0x0f)];
  174. }
  175. return crc;
  176. }
  177. static int region_read(struct region_info *region, uint32_t offset, uint8_t *buf, int len)
  178. {
  179. LOG_DBG("offset 0x%x len 0x%x, buf 0x%x", offset, len, (uint32_t)buf);
  180. if (!region->storage || (offset + len > region->total_size)) {
  181. LOG_ERR("invalid param storage %p, offset 0x%x, len 0x%x",
  182. region->storage, offset, len);
  183. return -EINVAL;
  184. }
  185. return nvram_storage_read(region->storage, region->base_addr + offset,
  186. buf, len);
  187. }
  188. static int region_write(struct region_info *region, uint32_t offset,
  189. const uint8_t *buf, int len)
  190. {
  191. LOG_DBG("offset 0x%x len 0x%x, buf 0x%x", offset, len, (uint32_t)buf);
  192. if (!region->storage || (offset + len > region->total_size)) {
  193. LOG_ERR("invalid param storage %p, offset 0x%x, len 0x%x",
  194. region->storage, offset, len);
  195. return -EINVAL;
  196. }
  197. return nvram_storage_write(region->storage, region->base_addr + offset,
  198. buf, len);
  199. }
  200. static int region_copy(struct region_info *region, uint32_t src_offset,
  201. uint32_t dest_offset, int len)
  202. {
  203. int err, write_size;
  204. uint8_t *buf;
  205. LOG_DBG("src_offset 0x%x dest_offset 0x%x len 0x%x", src_offset, dest_offset, len);
  206. buf = nvram_buf;
  207. write_size = NVRAM_BUFFER_SIZE;
  208. while (len > 0) {
  209. if (len < write_size)
  210. write_size = len;
  211. err = region_read(region, src_offset, buf, write_size);
  212. if (err) {
  213. return err;
  214. }
  215. region_write(region, dest_offset, buf, write_size);
  216. if (err) {
  217. return err;
  218. }
  219. src_offset += write_size;
  220. dest_offset += write_size;
  221. len -= write_size;
  222. }
  223. return 0;
  224. }
  225. static int region_is_empy(struct region_info *region, uint32_t offset, int32_t size)
  226. {
  227. int32_t read_size, i;
  228. uint32_t *pdat;
  229. read_size = NVRAM_BUFFER_SIZE;
  230. while (size > 0) {
  231. if (size < read_size)
  232. read_size = size;
  233. nvram_storage_read(region->storage, region->base_addr + offset,
  234. nvram_buf, read_size);
  235. pdat = (uint32_t *)nvram_buf;
  236. for (i = 0; i < read_size / 4; i++) {
  237. if (*pdat++ != 0xffffffff)
  238. return false;
  239. }
  240. size -= read_size;
  241. offset += read_size;
  242. }
  243. return true;
  244. }
  245. static int region_erase(struct region_info *region, uint32_t offset, int size)
  246. {
  247. LOG_DBG("offset 0x%x len 0x%x", offset, size);
  248. if (!region->storage || (offset + size > region->total_size) ||
  249. (offset & (NVRAM_ERASE_ALIGN_SIZE - 1) ||
  250. size & (NVRAM_ERASE_ALIGN_SIZE - 1))) {
  251. LOG_ERR("invalid param storage %p, offset 0x%x, len 0x%x",
  252. region->storage, offset, size);
  253. return -EINVAL;
  254. }
  255. while (size > 0) {
  256. if (!region_is_empy(region, offset, NVRAM_ERASE_ALIGN_SIZE)) {
  257. nvram_storage_erase(region->storage, region->base_addr + offset,
  258. NVRAM_ERASE_ALIGN_SIZE);
  259. }
  260. offset += NVRAM_ERASE_ALIGN_SIZE;
  261. size -= NVRAM_ERASE_ALIGN_SIZE;
  262. }
  263. return 0;
  264. }
  265. static int region_clear(struct region_info *region, int clear_size)
  266. {
  267. uint32_t offset, erase_size;
  268. clear_size = ROUND_UP(clear_size, region->seg_size);
  269. if (clear_size > (region->total_size - region->seg_size)) {
  270. LOG_ERR("clear_size 0x%x is too large", clear_size);
  271. clear_size = (region->total_size - region->seg_size);
  272. }
  273. offset = region->seg_offset + region->seg_size;
  274. if ((offset + clear_size) > region->total_size) {
  275. erase_size = region->total_size - offset;
  276. region_erase(region, offset, erase_size);
  277. /* wrap to region start pos */
  278. offset = 0;
  279. clear_size -= erase_size;
  280. }
  281. region_erase(region, offset, clear_size);
  282. return 0;
  283. }
  284. #ifdef CONFIG_NVRAM_FAST_SEARCH
  285. //#define BITS_PER_LONG 32
  286. #define BITOP_WORD(nr) ((nr) / BITS_PER_LONG)
  287. /*
  288. * Find the first set bit in a memory region.
  289. */
  290. static unsigned int find_first_bit(const unsigned int *addr, unsigned int size)
  291. {
  292. const unsigned int *p = addr;
  293. unsigned int result = 0;
  294. unsigned int tmp;
  295. while (size & ~(BITS_PER_LONG - 1)) {
  296. if ((tmp = *(p++)))
  297. goto found;
  298. result += BITS_PER_LONG;
  299. size -= BITS_PER_LONG;
  300. }
  301. if (!size)
  302. return result;
  303. tmp = (*p) & (~0UL >> (BITS_PER_LONG - size));
  304. if (tmp == 0UL) /* Are any bits set? */
  305. return result + size; /* Nope. */
  306. found:
  307. return result + (find_lsb_set(tmp) - 1);
  308. }
  309. /*
  310. * Find the next set bit in a memory region.
  311. */
  312. static unsigned int find_next_bit(const unsigned int *addr, unsigned int size,
  313. unsigned int offset)
  314. {
  315. const unsigned int *p = addr + BITOP_WORD(offset);
  316. unsigned int result = offset & ~(BITS_PER_LONG - 1);
  317. unsigned int tmp;
  318. if (offset >= size)
  319. return size;
  320. size -= result;
  321. offset %= BITS_PER_LONG;
  322. if (offset) {
  323. tmp = *(p++);
  324. tmp &= (~0UL << offset);
  325. if (size < BITS_PER_LONG)
  326. goto found_first;
  327. if (tmp)
  328. goto found_middle;
  329. size -= BITS_PER_LONG;
  330. result += BITS_PER_LONG;
  331. }
  332. while (size & ~(BITS_PER_LONG - 1)) {
  333. if ((tmp = *(p++)))
  334. goto found_middle;
  335. result += BITS_PER_LONG;
  336. size -= BITS_PER_LONG;
  337. }
  338. if (!size)
  339. return result;
  340. tmp = *p;
  341. found_first:
  342. tmp &= (~0UL >> (BITS_PER_LONG - size));
  343. if (tmp == 0UL) /* Are any bits set? */
  344. return result + size; /* Nope. */
  345. found_middle:
  346. return result + (find_lsb_set(tmp) - 1);
  347. }
  348. #define ITEM_BITMAP_BIT_TO_OFFSET(bit) ((bit) * NVRAM_ITEM_ALIGN_SIZE)
  349. #define ITEM_BITMAP_OFFSET_TO_BIT(offset) ((offset) / NVRAM_ITEM_ALIGN_SIZE)
  350. static int item_bitmap_first_offset(const unsigned int *seg_item_map, int seg_size)
  351. {
  352. int bit;
  353. if (!seg_item_map)
  354. return NVRAM_SEG_ITEM_START_OFFSET;
  355. bit = find_first_bit(seg_item_map, ITEM_BITMAP_OFFSET_TO_BIT(seg_size));
  356. return (ITEM_BITMAP_BIT_TO_OFFSET(bit));
  357. }
  358. static int item_bitmap_next_offset(const unsigned int *seg_item_map, int seg_size, int offset)
  359. {
  360. int bit;
  361. if (!seg_item_map)
  362. return offset;
  363. bit = find_next_bit(seg_item_map, ITEM_BITMAP_OFFSET_TO_BIT(seg_size),
  364. ITEM_BITMAP_OFFSET_TO_BIT(offset));
  365. return (ITEM_BITMAP_BIT_TO_OFFSET(bit));
  366. }
  367. static void item_bitmap_update(unsigned int *seg_item_map, int offset, int is_set)
  368. {
  369. int bit, w, b;
  370. if (!seg_item_map)
  371. return;
  372. bit = ITEM_BITMAP_OFFSET_TO_BIT(offset);
  373. w = bit / 32;
  374. b = bit % 32;
  375. if (is_set)
  376. seg_item_map[w] |= 1u << b;
  377. else
  378. seg_item_map[w] &= ~(1u << b);
  379. }
  380. static void item_bitmap_clear_all(unsigned int *seg_item_map, int item_map_size)
  381. {
  382. if (!seg_item_map)
  383. return;
  384. memset(seg_item_map, 0, item_map_size);
  385. }
  386. #endif
  387. static int item_is_empty(struct nvram_item *item)
  388. {
  389. uint8_t *buf = (uint8_t *)item;
  390. int size = sizeof(struct nvram_item);
  391. while (size > 0) {
  392. if (*buf++ != 0xff)
  393. return false;
  394. size--;
  395. }
  396. return true;
  397. }
  398. static int item_calc_real_size(uint8_t name_size, uint16_t data_size)
  399. {
  400. return (sizeof(struct nvram_item) + name_size + data_size);
  401. }
  402. static int item_calc_aligned_size(uint8_t name_size, uint16_t data_size)
  403. {
  404. return ROUND_UP(item_calc_real_size(name_size, data_size), NVRAM_ITEM_ALIGN_SIZE);
  405. }
  406. static int item_get_real_size(struct nvram_item *item)
  407. {
  408. return item_calc_real_size(item->name_size, item->data_size);
  409. }
  410. static int item_get_aligned_size(struct nvram_item *item)
  411. {
  412. return item_calc_aligned_size(item->name_size, item->data_size);
  413. }
  414. static void item_update_state(struct region_info *region, int item_offset, uint8_t state)
  415. {
  416. int state_offs;
  417. state_offs = item_offset + (int)(&((struct nvram_item *)0)->state);
  418. LOG_DBG("set state_offs 0x%x state to 0x%x", state_offs, state);
  419. region_write(region, state_offs, &state, sizeof(state));
  420. }
  421. static uint8_t item_calc_data_crc(struct region_info *region, uint32_t offset, int32_t size, uint8_t crc)
  422. {
  423. int32_t read_size;
  424. read_size = NVRAM_BUFFER_SIZE;
  425. while (size > 0) {
  426. if (size < read_size)
  427. read_size = size;
  428. region_read(region, offset, nvram_buf, read_size);
  429. crc = calc_crc8(nvram_buf, read_size, crc);
  430. offset += read_size;
  431. size -= read_size;
  432. }
  433. return crc;
  434. }
  435. static int item_get_item_name(struct region_info *region, int item_offs, struct nvram_item *item, uint8_t *buf, uint8_t len)
  436. {
  437. uint32_t name_offset;
  438. if (!buf){
  439. return -1;
  440. }
  441. if (item->magic != NVRAM_REGION_ITEM_MAGIC) {
  442. if (!item_is_empty(item)) {
  443. LOG_ERR("invalid item maigc 0x%x", item->magic);
  444. return -1;
  445. }
  446. return -1;
  447. }
  448. if (item->state != NVRAM_ITEM_STATE_VALID) {
  449. if (item->state != NVRAM_ITEM_STATE_OBSOLETE) {
  450. LOG_ERR("invalid item status 0x%x", item->state);
  451. return -1;
  452. }
  453. return -1;
  454. }
  455. if (len < item->name_size){
  456. return -1;
  457. }else{
  458. len = item->name_size;
  459. }
  460. name_offset = item_offs + sizeof(struct nvram_item);
  461. /* check config name */
  462. region_read(region, name_offset, buf, len);
  463. return 0;
  464. }
  465. static int item_check_validity(struct region_info *region, int item_offs, struct nvram_item *item,
  466. int check_crc)
  467. {
  468. uint8_t name_hash, crc;
  469. uint32_t name_offset;
  470. if (item->magic != NVRAM_REGION_ITEM_MAGIC) {
  471. if (!item_is_empty(item)) {
  472. LOG_ERR("invalid item maigc 0x%x", item->magic);
  473. return ITEM_STATUS_INVALID;
  474. }
  475. return ITEM_STATUS_EMPTY;
  476. }
  477. if (item->state != NVRAM_ITEM_STATE_VALID) {
  478. if (item->state != NVRAM_ITEM_STATE_OBSOLETE) {
  479. LOG_ERR("invalid item status 0x%x", item->state);
  480. return ITEM_STATUS_INVALID;
  481. }
  482. return ITEM_STATUS_OBSOLETE;
  483. }
  484. name_offset = item_offs + sizeof(struct nvram_item);
  485. /* check config name */
  486. region_read(region, name_offset, nvram_buf, item->name_size);
  487. name_hash = calc_hash(nvram_buf, item->name_size);
  488. if (item->hash != name_hash) {
  489. return ITEM_STATUS_INVALID;
  490. }
  491. if (check_crc) {
  492. crc = calc_crc8((uint8_t *)item + NVRAM_REGION_ITEM_CRC_OFFSET,
  493. sizeof(struct nvram_item) - NVRAM_REGION_ITEM_CRC_OFFSET, 0);
  494. crc = calc_crc8(nvram_buf, item->name_size, crc);
  495. crc = item_calc_data_crc(region, name_offset + item->name_size, item->data_size, crc);
  496. if (item->crc != crc) {
  497. LOG_ERR("item crc error! offset 0x%x, crc 0x%x != item->crc 0x%x",
  498. item_offs, crc, item->crc);
  499. item_update_state(region, item_offs - region->seg_offset, NVRAM_ITEM_STATE_OBSOLETE);
  500. return ITEM_STATUS_INVALID;
  501. }
  502. }
  503. return ITEM_STATUS_VALID;
  504. }
  505. static int region_find_item(struct region_info *region, const char *name,
  506. struct nvram_item *item)
  507. {
  508. uint32_t item_offs;
  509. uint16_t hash;
  510. int32_t offs;
  511. if (!name || !item)
  512. return -EINVAL;
  513. hash = calc_hash((const uint8_t*)name, strlen(name) + 1);
  514. #ifdef CONFIG_NVRAM_FAST_SEARCH
  515. offs = item_bitmap_first_offset(region->seg_item_map, region->seg_size);
  516. #else
  517. offs = NVRAM_SEG_ITEM_START_OFFSET;
  518. #endif
  519. while (offs < region->seg_size) {
  520. item_offs = region->seg_offset + offs;
  521. /* read item header */
  522. region_read(region, item_offs, (uint8_t *)item, sizeof(struct nvram_item));
  523. if (item->magic != NVRAM_REGION_ITEM_MAGIC)
  524. break;
  525. if (item->state != NVRAM_ITEM_STATE_VALID) {
  526. goto next;
  527. }
  528. if (item->hash == hash) {
  529. /* read config name */
  530. region_read(region, item_offs + sizeof(struct nvram_item),
  531. nvram_buf, item->name_size);
  532. if (!memcmp(name, (const char *)nvram_buf, item->name_size)) {
  533. /* TODO: check data crc? */
  534. /* founded! */
  535. return item_offs;
  536. }
  537. }
  538. next:
  539. offs += item_get_aligned_size(item);
  540. #ifdef CONFIG_NVRAM_FAST_SEARCH
  541. offs = item_bitmap_next_offset(region->seg_item_map,
  542. region->seg_size, offs);
  543. #endif
  544. }
  545. return -ENOENT;
  546. }
  547. static void region_seg_update_state(struct region_info *region, uint32_t seg_offset, uint8_t state)
  548. {
  549. int state_offs;
  550. state_offs = seg_offset + (int)(&((struct region_seg_header *)0)->state);
  551. LOG_DBG("set seg_offset 0x%x state to 0x%x", region->seg_offset, state);
  552. region_write(region, state_offs, &state, sizeof(state));
  553. }
  554. static int region_write_data(struct region_info *region, uint32_t offset,
  555. const uint8_t *data, int32_t len)
  556. {
  557. int32_t size, wsize;
  558. uint32_t woffs;
  559. wsize = NVRAM_BUFFER_SIZE;
  560. woffs = 0;
  561. size = len;
  562. while (size > 0) {
  563. if (size < wsize)
  564. wsize = size;
  565. memcpy(nvram_buf, data + woffs, wsize);
  566. region_write(region, offset + woffs, nvram_buf, wsize);
  567. woffs += wsize;
  568. size -= wsize;
  569. }
  570. return 0;
  571. }
  572. static int region_write_item(struct region_info *region, uint32_t offset,
  573. const char *name, const uint8_t *data, int32_t len)
  574. {
  575. struct nvram_item *item;
  576. uint32_t item_data_offs, item_size;
  577. item = (struct nvram_item *)nvram_buf;
  578. memset(item, 0x0, sizeof(struct nvram_item));
  579. item->magic = NVRAM_REGION_ITEM_MAGIC;
  580. item->state = NVRAM_ITEM_STATE_VALID;
  581. item->name_size = strlen(name) + 1;
  582. item->hash = calc_hash((const uint8_t*)name, item->name_size);
  583. item->data_size = (uint16_t)len;
  584. item_data_offs = sizeof(struct nvram_item) + item->name_size;
  585. item_size = item_data_offs + item->data_size;
  586. item->crc = calc_crc8((uint8_t *)item + NVRAM_REGION_ITEM_CRC_OFFSET,
  587. sizeof(struct nvram_item) - NVRAM_REGION_ITEM_CRC_OFFSET, 0);
  588. item->crc = calc_crc8((uint8_t *)name, item->name_size, item->crc);
  589. item->crc = calc_crc8(data, item->data_size, item->crc);
  590. /* write item header & name */
  591. if (item_data_offs > NVRAM_BUFFER_SIZE) {
  592. LOG_ERR("BUG! invalid item name size %d", item->name_size);
  593. return -EINVAL;
  594. }
  595. memcpy(&item->data[0], name, item->name_size);
  596. region_write(region, offset, (uint8_t *)item, item_data_offs);
  597. /* write item data */
  598. #if 0
  599. /* use original data buffer to write data */
  600. region_write(region, offset + item_data_offs, data, len);
  601. #else
  602. /* use nvram buffer to write data to avoid cache-miss when access spinor mapping memory */
  603. region_write_data(region, offset + item_data_offs, data, len);
  604. #endif
  605. return item_size;
  606. }
  607. int region_copy_seg(struct region_info *region, uint32_t new_seg_offset,
  608. uint32_t old_seg_offset, int copy_size, int check_crc)
  609. {
  610. struct nvram_item item;
  611. int item_offs, new_item_offs;
  612. int item_size, item_total_size;
  613. int status;
  614. #ifdef CONFIG_NVRAM_FAST_SEARCH
  615. /* clear item bitmap for new segment */
  616. item_bitmap_clear_all(region->seg_item_map, region->seg_item_map_size);
  617. #endif
  618. item_offs = old_seg_offset + NVRAM_SEG_ITEM_START_OFFSET;
  619. new_item_offs = new_seg_offset + NVRAM_SEG_ITEM_START_OFFSET;
  620. while (item_offs < (old_seg_offset + copy_size)) {
  621. /* read item header */
  622. region_read(region, item_offs, (uint8_t *)&item, sizeof(struct nvram_item));
  623. /* check item */
  624. status = item_check_validity(region, item_offs, &item, check_crc);
  625. LOG_DBG("item_offs: status 0x%x", status);
  626. if (status == ITEM_STATUS_INVALID || status == ITEM_STATUS_EMPTY) {
  627. /* invalid */
  628. LOG_ERR("BUG! invalid nvram region item 0x%x, status 0x%x",
  629. item_offs, status);
  630. break;
  631. }
  632. item_total_size = item_get_aligned_size(&item);
  633. if (status == ITEM_STATUS_VALID) {
  634. item_size = item_get_real_size(&item);
  635. LOG_DBG("valid item: copy from 0x%x to 0x%x, len 0x%x",
  636. item_offs, new_item_offs, item_size);
  637. region_copy(region, item_offs, new_item_offs, item_size);
  638. #ifdef CONFIG_NVRAM_FAST_SEARCH
  639. item_bitmap_update(region->seg_item_map,
  640. new_item_offs - new_seg_offset, 1);
  641. #endif
  642. new_item_offs += item_total_size;
  643. }
  644. item_offs += item_total_size;
  645. }
  646. region->seg_write_offset = new_item_offs;
  647. LOG_DBG("new_seg write offset 0x%x", region->seg_write_offset);
  648. return 0;
  649. }
  650. static int region_init_new_seg(struct region_info *region, uint32_t seg_offset)
  651. {
  652. struct region_seg_header seg_hdr;
  653. memset(&seg_hdr, 0x0, sizeof(struct region_seg_header));
  654. seg_hdr.magic = NVRAM_REGION_SEG_MAGIC;
  655. seg_hdr.state = NVRAM_REGION_SEG_STATE_VALID;
  656. seg_hdr.version = NVRAM_REGION_SEG_VERSION;
  657. seg_hdr.head_size = sizeof(struct region_seg_header);
  658. seg_hdr.seg_size = region->seg_size;
  659. seg_hdr.seq_id = region->seg_seq_id + 1;
  660. seg_hdr.crc = calc_crc8(((uint8_t *)&seg_hdr) + NVRAM_REGION_SEG_HEADER_CRC_OFFSET,
  661. sizeof(struct region_seg_header) - NVRAM_REGION_SEG_HEADER_CRC_OFFSET, 0);
  662. region_erase(region, seg_offset, region->seg_size);
  663. region_write(region, seg_offset, (uint8_t *)&seg_hdr, sizeof(struct region_seg_header));
  664. region->seg_seq_id = seg_hdr.seq_id;
  665. region->seg_offset = seg_offset;
  666. region->seg_write_offset = region->seg_offset + NVRAM_SEG_ITEM_START_OFFSET;
  667. #ifdef CONFIG_NVRAM_FAST_SEARCH
  668. item_bitmap_clear_all(region->seg_item_map, region->seg_item_map_size);
  669. #endif
  670. return 0;
  671. }
  672. static int region_purge_seg(struct region_info *region, int check_crc)
  673. {
  674. uint32_t new_seg_offset, old_seg_offset, seg_copy_size;
  675. LOG_DBG("purge seg offset 0x%x", region->seg_offset);
  676. new_seg_offset = region->seg_offset + region->seg_size;
  677. if (new_seg_offset >= region->total_size) {
  678. /* wrap to begin of region */
  679. new_seg_offset = 0;
  680. }
  681. /* check new seg */
  682. if (!region_is_empy(region, new_seg_offset, region->seg_size)) {
  683. /* sorry, must erase in this context */
  684. LOG_WRN("new region seg 0x%x is not empty, need erase", new_seg_offset);
  685. region_erase(region, new_seg_offset, region->seg_size);
  686. }
  687. old_seg_offset = region->seg_offset;
  688. seg_copy_size = region->seg_write_offset - region->seg_offset;
  689. /* init new seg */
  690. region_init_new_seg(region, new_seg_offset);
  691. /* copy valid data to new seg */
  692. region_copy_seg(region, new_seg_offset, old_seg_offset, seg_copy_size, check_crc);
  693. /* current seg set to obsolete */
  694. region_seg_update_state(region, old_seg_offset, NVRAM_REGION_SEG_STATE_OBSOLETE);
  695. return 0;
  696. }
  697. static int region_prepare_write_item(struct region_info *region, int item_size)
  698. {
  699. if ((region->seg_write_offset + item_size) > (region->seg_offset + region->seg_size)) {
  700. region_purge_seg(region, 0);
  701. }
  702. return 0;
  703. }
  704. static int region_is_same_config_data(struct region_info *region, const char *name,
  705. const void *data, int len)
  706. {
  707. struct nvram_item item;
  708. int old_item_offs, offset;
  709. int32_t pos, read_size;
  710. old_item_offs = region_find_item(region, name, &item);
  711. if(old_item_offs > 0 && len == item.data_size){
  712. pos = 0;
  713. read_size = NVRAM_BUFFER_SIZE;
  714. offset = old_item_offs + sizeof(struct nvram_item) + item.name_size;
  715. while (len > 0) {
  716. if (len < read_size)
  717. read_size = len;
  718. region_read(region, offset + pos, nvram_buf, read_size);
  719. if(memcmp(nvram_buf ,(uint8_t *)data + pos, read_size) != 0){
  720. return false;
  721. }
  722. pos += read_size;
  723. len -= read_size;
  724. }
  725. if(len == 0){
  726. return true;
  727. }
  728. }
  729. return false;
  730. }
  731. static int region_set(struct region_info *region, const char *name,
  732. const void *data, int len)
  733. {
  734. struct nvram_item item;
  735. int32_t name_len, new_item_size, item_len;
  736. int old_item_offs;
  737. if (!name || (!data && len) || len > NVRAM_MAX_DATA_SIZE)
  738. return -EINVAL;
  739. LOG_DBG("set config '%s', len %d", name, len);
  740. name_len = strlen(name) + 1;
  741. if (name_len > NVRAM_MAX_NAME_SIZE)
  742. return -EINVAL;
  743. if (len > 0) {
  744. if(region_is_same_config_data(region, name, data, len)){
  745. return 0;
  746. }
  747. /* write new config */
  748. new_item_size = item_calc_aligned_size(name_len, len);
  749. region_prepare_write_item(region, new_item_size);
  750. old_item_offs = region_find_item(region, name, &item);
  751. item_len = region_write_item(region, region->seg_write_offset, name, data, len);
  752. if (item_len > new_item_size) {
  753. LOG_ERR("BUG! new_item_size 0x%x, write item_len 0x%x",
  754. new_item_size, item_len);
  755. }
  756. #ifdef CONFIG_NVRAM_FAST_SEARCH
  757. item_bitmap_update(region->seg_item_map,
  758. region->seg_write_offset - region->seg_offset, 1);
  759. #endif
  760. region->seg_write_offset += new_item_size;
  761. }
  762. else {
  763. old_item_offs = region_find_item(region, name, &item);
  764. }
  765. LOG_DBG("name %s, old_item_offs %d", name, old_item_offs);
  766. if (old_item_offs > 0) {
  767. /* set the old item state to obsolete */
  768. item_update_state(region, old_item_offs, NVRAM_ITEM_STATE_OBSOLETE);
  769. #ifdef CONFIG_NVRAM_FAST_SEARCH
  770. item_bitmap_update(region->seg_item_map, old_item_offs - region->seg_offset, 0);
  771. #endif
  772. }
  773. return 0;
  774. }
  775. static int region_get(struct region_info *region, const char *name,
  776. void *data, int max_len)
  777. {
  778. struct nvram_item item;
  779. uint32_t item_offs, data_offs, len;
  780. if (!name || !data || !max_len)
  781. return -EINVAL;
  782. LOG_DBG("get config '%s', max_len %d", name, max_len);
  783. /* search write region firstly */
  784. item_offs = region_find_item(region, name, &item);
  785. if ((int32_t)item_offs < 0)
  786. return -ENOENT;
  787. data_offs = item_offs + sizeof(struct nvram_item) + item.name_size;
  788. if (max_len < item.data_size)
  789. len = max_len;
  790. else
  791. len = item.data_size;
  792. region_read(region, data_offs, data, len);
  793. return len;
  794. }
  795. extern void print_buffer(const struct shell *shell,const char *addr, int width,
  796. int count, int linelen, unsigned int disp_addr);
  797. static void region_dump_data(const struct shell *shell, struct region_info *region, uint32_t offset, int32_t size)
  798. {
  799. int32_t pos, read_size;
  800. pos = 0;
  801. read_size = NVRAM_BUFFER_SIZE;
  802. while (size > 0) {
  803. if (size < read_size)
  804. read_size = size;
  805. region_read(region, offset + pos, nvram_buf, read_size);
  806. #ifdef CONFIG_SHELL_DBG
  807. print_buffer(shell, (const char *)nvram_buf, 1, read_size, 16, pos);
  808. #endif
  809. pos += read_size;
  810. size -= read_size;
  811. /* sleep a while to avoid print buffer overflow */
  812. k_busy_wait(1000);
  813. }
  814. }
  815. static void region_dump(const struct shell *shell, struct region_info *region, int detailed)
  816. {
  817. struct nvram_item item;
  818. uint32_t offs, item_offs;
  819. int i;
  820. printk("region %s: base addr 0x%x total size 0x%x\n",
  821. region->name, region->base_addr, region->total_size);
  822. printk("region segment offs 0x%x, size 0x%x, seq_id 0x%x, write offset 0x%x\n",
  823. region->seg_offset, region->seg_size, region->seg_seq_id,
  824. region->seg_write_offset);
  825. if (!detailed)
  826. return;
  827. i = 0;
  828. offs = 0;
  829. #ifdef CONFIG_NVRAM_FAST_SEARCH
  830. offs = item_bitmap_first_offset(region->seg_item_map, region->seg_size);
  831. #else
  832. offs = NVRAM_SEG_ITEM_START_OFFSET;
  833. #endif
  834. while (offs < region->seg_size) {
  835. item_offs = region->seg_offset + offs;
  836. /* read item header */
  837. region_read(region, item_offs, (uint8_t *)&item, sizeof(struct nvram_item));
  838. if (item.magic != NVRAM_REGION_ITEM_MAGIC)
  839. break;
  840. if (item.state != NVRAM_ITEM_STATE_VALID) {
  841. goto next;
  842. }
  843. printk("[%2d] config item_offs 0x%x size 0x%x data size 0x%x\n",
  844. i, item_offs, item.name_size, item.data_size);
  845. /* read config name */
  846. region_read(region, item_offs + sizeof(struct nvram_item),
  847. nvram_buf, item.name_size);
  848. printk(" config name: %s\n", nvram_buf);
  849. printk(" config data:\n");
  850. region_dump_data(shell, region, item_offs + sizeof(struct nvram_item) + item.name_size,
  851. item.data_size);
  852. next:
  853. offs += item_get_aligned_size(&item);
  854. #ifdef CONFIG_NVRAM_FAST_SEARCH
  855. offs = item_bitmap_next_offset(region->seg_item_map,
  856. region->seg_size, offs);
  857. #endif
  858. i++;
  859. }
  860. }
  861. static bool is_valid_region_seg_header(struct region_seg_header *hdr)
  862. {
  863. uint32_t crc;
  864. /* validate magic number and state*/
  865. if (hdr->magic != NVRAM_REGION_SEG_MAGIC ||
  866. hdr->state != NVRAM_REGION_SEG_STATE_VALID)
  867. return false;
  868. /* validate header crc */
  869. crc = calc_crc8((uint8_t *)hdr + NVRAM_REGION_SEG_HEADER_CRC_OFFSET,
  870. sizeof(struct region_seg_header) - NVRAM_REGION_SEG_HEADER_CRC_OFFSET, 0);
  871. if (hdr->crc != crc) {
  872. LOG_ERR("invalid header crc 0x%x != hdr->crc 0x%x", crc, hdr->crc);
  873. return false;
  874. }
  875. return true;
  876. }
  877. static int region_seg_scan(struct region_info *region)
  878. {
  879. struct nvram_item item;
  880. int err, offs, item_offs, status;
  881. int need_purge = 0;
  882. uint8_t item_name[NVRAM_MAX_NAME_SIZE];
  883. int old_item_offs;
  884. struct nvram_item old_item;
  885. #ifdef CONFIG_NVRAM_FAST_SEARCH
  886. item_bitmap_clear_all(region->seg_item_map, region->seg_item_map_size);
  887. #endif
  888. offs = NVRAM_SEG_ITEM_START_OFFSET;
  889. item_offs = region->seg_offset + offs;
  890. #ifdef PC
  891. nvram_count = 0;
  892. memset(nvram_data, 0, sizeof(nvram_data));
  893. #endif
  894. while (offs < region->seg_size) {
  895. /* read item header */
  896. err = region_read(region, item_offs, (uint8_t *)&item,
  897. sizeof(struct nvram_item));
  898. if (err) {
  899. LOG_ERR("read item_offs 0x%x error", item_offs);
  900. return err;
  901. }
  902. /* read item header */
  903. status = item_check_validity(region, item_offs, &item, 1);
  904. LOG_DBG("item_offs 0x%x: status 0x%x", item_offs, status);
  905. if (status == ITEM_STATUS_VALID) {
  906. memset(item_name, 0, sizeof(item_name));
  907. if(item_get_item_name(region, item_offs, &item, item_name, sizeof(item_name)) == 0){
  908. old_item_offs = region_find_item(region, (const char*)item_name, &old_item);
  909. if(old_item_offs > 0 && (old_item_offs != item_offs)){
  910. LOG_WRN("same item %s found new:0x%x old 0x%x", item_name, item_offs, old_item_offs);
  911. /* set the old item state to obsolete */
  912. item_update_state(region, old_item_offs, NVRAM_ITEM_STATE_OBSOLETE);
  913. #ifdef CONFIG_NVRAM_FAST_SEARCH
  914. item_bitmap_update(region->seg_item_map, old_item_offs - region->seg_offset, 0);
  915. #endif
  916. }
  917. }
  918. #ifdef CONFIG_NVRAM_FAST_SEARCH
  919. item_bitmap_update(region->seg_item_map, offs, 1);
  920. #endif
  921. #ifdef PC
  922. memcpy(nvram_data[nvram_count].name, item_name, sizeof(item_name));
  923. nvram_data[nvram_count].name[item.name_size] = 0;
  924. nvram_count++;
  925. #endif
  926. } else if (status == ITEM_STATUS_EMPTY) {
  927. break;
  928. } else if (status != ITEM_STATUS_OBSOLETE) {
  929. /* invalid */
  930. LOG_ERR("found invalid item, item_offs 0x%x status %d", item_offs, status);
  931. // need_purge = 1;
  932. // break;
  933. }
  934. offs += item_get_aligned_size(&item);
  935. item_offs = region->seg_offset + offs;
  936. }
  937. region->seg_write_offset = item_offs;
  938. LOG_DBG("seg_offset 0x%x, write_offset 0x%x", region->seg_offset,
  939. region->seg_write_offset);
  940. if (offs >= region->seg_size) {
  941. /* region is full */
  942. need_purge = 1;
  943. } else {
  944. if (!region_is_empy(region, item_offs, region->seg_size - offs)) {
  945. LOG_ERR("region seg is not clean after write_offset 0x%x, need purge",
  946. item_offs);
  947. need_purge = 1;
  948. }
  949. }
  950. if (need_purge) {
  951. region_purge_seg(region, 1);
  952. }
  953. #ifdef PC
  954. if (nvram_count > 0)
  955. {
  956. Nvram_ValidData_Save(region->name, region->seg_offset);
  957. }
  958. #endif
  959. return 0;
  960. }
  961. static int region_scan(struct region_info *region, bool b_init_seg)
  962. {
  963. struct region_seg_header hdr;
  964. uint32_t offs = 0;
  965. int err, found = 0;
  966. while (offs < region->total_size) {
  967. err = region_read(region, offs, (uint8_t *)&hdr, sizeof(struct region_seg_header));
  968. if (err) {
  969. return err;
  970. }
  971. if (is_valid_region_seg_header(&hdr) &&
  972. (!found || ((int8_t)(hdr.seq_id - region->seg_seq_id)) > 0)) {
  973. region->seg_seq_id = hdr.seq_id;
  974. region->seg_offset = offs;
  975. found = 1;
  976. }
  977. offs += region->seg_size;
  978. }
  979. if (found) {
  980. LOG_DBG("found region seg offset 0x%x\n", region->seg_offset);
  981. err = region_seg_scan(region);
  982. if (err) {
  983. LOG_ERR("invalid region seg offset 0x%x\n", region->seg_offset);
  984. return -1;
  985. }
  986. }
  987. else {
  988. if(b_init_seg){
  989. LOG_DBG("region %s: first init\n", region->name);
  990. region_init_new_seg(region, 0);
  991. }else{
  992. LOG_ERR("region %s: init err\n", region->name);
  993. }
  994. }
  995. return 0;
  996. }
  997. int nvram_config_set_factory(const char *name, const void *data, int len)
  998. {
  999. int ret;
  1000. k_sem_take(&nvram_lock, K_FOREVER);
  1001. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  1002. ret = region_set(&factory_rw_nvram_region, name ,data, len);
  1003. nvram_storage_flush(factory_rw_nvram_region.storage);
  1004. #else
  1005. ret = region_set(&factory_nvram_region, name ,data, len);
  1006. nvram_storage_flush(factory_nvram_region.storage);
  1007. #endif
  1008. k_sem_give(&nvram_lock);
  1009. return ret;
  1010. }
  1011. int nvram_config_set(const char *name, const void *data, int len)
  1012. {
  1013. int ret;
  1014. k_sem_take(&nvram_lock, K_FOREVER);
  1015. ret = region_set(&user_nvram_region, name, data, len);
  1016. nvram_storage_flush(user_nvram_region.storage);
  1017. k_sem_give(&nvram_lock);
  1018. return ret;
  1019. }
  1020. int nvram_config_get_factory(const char *name, void *data, int max_len)
  1021. {
  1022. int ret;
  1023. k_sem_take(&nvram_lock, K_FOREVER);
  1024. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  1025. ret = region_get(&factory_rw_nvram_region, name, data, max_len);
  1026. if (ret >= 0) {
  1027. k_sem_give(&nvram_lock);
  1028. return ret;
  1029. }
  1030. #endif
  1031. ret = region_get(&factory_nvram_region, name, data, max_len);
  1032. k_sem_give(&nvram_lock);
  1033. return ret;
  1034. }
  1035. int nvram_config_get(const char *name, void *data, int max_len)
  1036. {
  1037. int ret;
  1038. k_sem_take(&nvram_lock, K_FOREVER);
  1039. /* config priority: user > factory rw > factory ro */
  1040. /* search user nvram region */
  1041. ret = region_get(&user_nvram_region, name, data, max_len);
  1042. if (ret >= 0){
  1043. k_sem_give(&nvram_lock);
  1044. return ret;
  1045. }
  1046. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  1047. ret = region_get(&factory_rw_nvram_region, name, data, max_len);
  1048. if (ret >= 0) {
  1049. k_sem_give(&nvram_lock);
  1050. return ret;
  1051. }
  1052. #endif
  1053. /* search factory nvram region */
  1054. ret = region_get(&factory_nvram_region, name, data, max_len);
  1055. if (ret < 0){
  1056. LOG_DBG("cannot found config %s", name);
  1057. }
  1058. k_sem_give(&nvram_lock);
  1059. return ret;
  1060. }
  1061. void nvram_config_dump(const struct shell *shell)
  1062. {
  1063. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  1064. region_dump(shell, &factory_rw_nvram_region, 1);
  1065. #endif
  1066. region_dump(shell, &factory_nvram_region, 1);
  1067. region_dump(shell, &user_nvram_region, 1);
  1068. }
  1069. int nvram_config_clear(int len)
  1070. {
  1071. int ret;
  1072. ret = region_clear(&user_nvram_region, len);
  1073. nvram_storage_flush(user_nvram_region.storage);
  1074. return ret;
  1075. }
  1076. int nvram_config_clear_all(void)
  1077. {
  1078. struct region_info *region = &user_nvram_region;
  1079. LOG_WRN("clear all user nvram config");
  1080. k_sem_take(&nvram_lock, K_FOREVER);
  1081. /* erase region */
  1082. region_erase(region, 0, region->total_size);
  1083. /* rescan region */
  1084. region_scan(region, true);
  1085. nvram_storage_flush(region->storage);
  1086. k_sem_give(&nvram_lock);
  1087. return 0;
  1088. }
  1089. int nvram_config_defrag_user(void)
  1090. {
  1091. struct region_info *region = &user_nvram_region;
  1092. LOG_WRN("defrag user nvram config");
  1093. k_sem_take(&nvram_lock, K_FOREVER);
  1094. /* copy to first region */
  1095. if (region->seg_offset > 0) {
  1096. region_erase(region, 0, region->seg_size);
  1097. region_copy(region, region->seg_offset, 0, region->seg_size);
  1098. }
  1099. /* erase other region */
  1100. region_erase(region, region->seg_size, region->total_size - region->seg_size);
  1101. /* rescan region */
  1102. region_scan(region, true);
  1103. nvram_storage_flush(region->storage);
  1104. k_sem_give(&nvram_lock);
  1105. return 0;
  1106. }
  1107. void nvram_region_info_init(void)
  1108. {
  1109. const struct partition_entry *nvram_part;
  1110. nvram_part = partition_get_part(PARTITION_FILE_ID_NVRAM_FACTORY);
  1111. if (nvram_part) {
  1112. factory_nvram_region.base_addr = nvram_part->offset;
  1113. factory_nvram_region.total_size = nvram_part->size;
  1114. LOG_INF("nvram fac:0x%x,0x%x", factory_nvram_region.base_addr, factory_nvram_region.total_size);
  1115. } else {
  1116. LOG_ERR("nvram fac partition NOT find");
  1117. }
  1118. nvram_part = partition_get_part(PARTITION_FILE_ID_NVRAM_USER);
  1119. if (nvram_part) {
  1120. user_nvram_region.base_addr = nvram_part->offset;
  1121. user_nvram_region.total_size = nvram_part->size;
  1122. LOG_INF("nvram user:0x%x,0x%x", user_nvram_region.base_addr, user_nvram_region.total_size);
  1123. } else {
  1124. LOG_ERR("nvram user partition NOT find");
  1125. }
  1126. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  1127. nvram_part = partition_get_part(PARTITION_FILE_ID_NVRAM_FACTORY_RW);
  1128. if (nvram_part) {
  1129. factory_rw_nvram_region.base_addr = nvram_part->offset;
  1130. factory_rw_nvram_region.total_size = nvram_part->size;
  1131. LOG_INF("nvram fac rw:0x%x,0x%x", factory_rw_nvram_region.base_addr, factory_rw_nvram_region.total_size);
  1132. } else {
  1133. LOG_ERR("nvram fac rw partition NOT find");
  1134. }
  1135. #endif
  1136. }
  1137. int nvram_config_init(const struct device *dev)
  1138. {
  1139. struct device *storage;
  1140. LOG_INF("init nvram config");
  1141. nvram_region_info_init();
  1142. storage = nvram_storage_init();
  1143. if (!storage) {
  1144. LOG_ERR("NVRAM storage driver was not found!");
  1145. return -ENODEV;
  1146. }
  1147. factory_nvram_region.storage = storage;
  1148. user_nvram_region.storage = storage;
  1149. #ifdef CONFIG_NVRAM_STORAGE_FACTORY_RW_REGION
  1150. factory_rw_nvram_region.storage = storage;
  1151. region_scan(&factory_rw_nvram_region, true);
  1152. #endif
  1153. region_scan(&factory_nvram_region, false);
  1154. region_scan(&user_nvram_region, true);
  1155. /* clear next write region to avoid erasing in system */
  1156. region_clear(&user_nvram_region, user_nvram_region.total_size / 2);
  1157. nvram_storage_flush(user_nvram_region.storage);
  1158. return 0;
  1159. }
  1160. #ifndef PC
  1161. SYS_INIT(nvram_config_init, POST_KERNEL, CONFIG_NVRAM_CONFIG_INIT_PRIORITY);
  1162. #else
  1163. static int nvram_item_data_restore(struct region_info *region, int item_offs, struct nvram_item *item,
  1164. struct nvram_data_t *nvram_item_data)
  1165. {
  1166. uint8_t name_hash, crc;
  1167. uint32_t name_offset;
  1168. if (item->magic != NVRAM_REGION_ITEM_MAGIC) {
  1169. if (!item_is_empty(item)) {
  1170. LOG_ERR("invalid item maigc 0x%x", item->magic);
  1171. return ITEM_STATUS_INVALID;
  1172. }
  1173. return ITEM_STATUS_EMPTY;
  1174. }
  1175. name_offset = item_offs + sizeof(struct nvram_item);
  1176. /* check config name */
  1177. region_read(region, name_offset, (uint8_t*)nvram_item_data->name, item->name_size);
  1178. nvram_item_data->name[item->name_size] = 0;
  1179. name_hash = calc_hash((const uint8_t*)nvram_item_data->name, item->name_size);
  1180. if (item->hash != name_hash) {
  1181. return ITEM_STATUS_INVALID;
  1182. }
  1183. crc = calc_crc8((uint8_t *)item + NVRAM_REGION_ITEM_CRC_OFFSET,
  1184. sizeof(struct nvram_item) - NVRAM_REGION_ITEM_CRC_OFFSET, 0);
  1185. crc = calc_crc8((uint8_t *)nvram_item_data->name, item->name_size, crc);
  1186. region_read(region, name_offset + item->name_size, nvram_item_data->data, item->data_size);
  1187. crc = calc_crc8(nvram_item_data->data, item->data_size, crc);
  1188. nvram_item_data->data_size = item->data_size;
  1189. if (item->crc != crc) {
  1190. LOG_ERR("item crc error! offset 0x%x, crc 0x%x != item->crc 0x%x",
  1191. item_offs, crc, item->crc);
  1192. return ITEM_STATUS_CRCERR;
  1193. }
  1194. return ITEM_STATUS_VALID;
  1195. }
  1196. static int nvram_section_restore(struct region_info *region)
  1197. {
  1198. struct nvram_item item;
  1199. int err, offs, item_offs, status, i;
  1200. nvram_count = 0;
  1201. i = 0;
  1202. memset(nvram_data, 0, sizeof(nvram_data));
  1203. offs = NVRAM_SEG_ITEM_START_OFFSET;
  1204. item_offs = region->seg_offset + offs;
  1205. while (offs < region->seg_size) {
  1206. /* read item header */
  1207. err = region_read(region, item_offs, (uint8_t *)&item,
  1208. sizeof(struct nvram_item));
  1209. if (err) {
  1210. LOG_ERR("read item_offs 0x%x error", item_offs);
  1211. return err;
  1212. }
  1213. /* read item header */
  1214. status = nvram_item_data_restore(region, item_offs, &item, &nvram_data[i]);
  1215. LOG_DBG("item_offs 0x%x: status 0x%x", item_offs, status);
  1216. if ((ITEM_STATUS_VALID == status) || (ITEM_STATUS_CRCERR == status))
  1217. {
  1218. nvram_data[i].state = status;
  1219. i++;
  1220. }
  1221. offs += item_get_aligned_size(&item);
  1222. item_offs = region->seg_offset + offs;
  1223. }
  1224. nvram_count = i;
  1225. return i;
  1226. }
  1227. static bool has_valid_region_seg_header(struct region_seg_header *hdr)
  1228. {
  1229. uint32_t crc;
  1230. /* validate magic number and state*/
  1231. if (hdr->magic != NVRAM_REGION_SEG_MAGIC)
  1232. return false;
  1233. /* validate header crc */
  1234. crc = calc_crc8((uint8_t *)hdr + NVRAM_REGION_SEG_HEADER_CRC_OFFSET,
  1235. sizeof(struct region_seg_header) - NVRAM_REGION_SEG_HEADER_CRC_OFFSET, 0);
  1236. if (hdr->crc != crc) {
  1237. LOG_ERR("invalid header crc 0x%x != hdr->crc 0x%x", crc, hdr->crc);
  1238. return false;
  1239. }
  1240. return true;
  1241. }
  1242. int nvram_region_restore(struct region_info *region)
  1243. {
  1244. struct region_seg_header hdr;
  1245. uint32_t offs = 0;
  1246. int err;
  1247. while (offs < region->total_size) {
  1248. err = region_read(region, offs, (uint8_t *)&hdr, sizeof(struct region_seg_header));
  1249. if (err) {
  1250. return err;
  1251. }
  1252. if (has_valid_region_seg_header(&hdr)) {
  1253. region->seg_seq_id = hdr.seq_id;
  1254. region->seg_offset = offs;
  1255. nvram_section_restore(region);
  1256. Nvram_HistoryData_Save(region->name, offs);
  1257. }
  1258. else {
  1259. LOG_ERR("Invalid nvram section in 0x%x of %s", offs, region->name);
  1260. }
  1261. offs += region->seg_size;
  1262. }
  1263. return 0;
  1264. }
  1265. #endif