nvram_storage.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief NVRAM storage
  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/flash.h>
  18. #include <string.h>
  19. #include <board_cfg.h>
  20. #include "nvram_storage.h"
  21. #define LOG_LEVEL 2
  22. #include <logging/log.h>
  23. LOG_MODULE_REGISTER(nvram_stor);
  24. #else
  25. #include "nvram_test_simulate.h"
  26. #endif
  27. #ifdef CONFIG_NVRAM_STORAGE_RAMSIM
  28. #define NVRAM_MEM_SIZE (CONFIG_NVRAM_FACTORY_REGION_SIZE + \
  29. CONFIG_NVRAM_FACTORY_RW_REGION_SIZE + CONFIG_NVRAM_USER_REGION_SIZE)
  30. #define NVRAM_MEM_OFFS CONFIG_NVRAM_FACTORY_REGION_BASE_ADDR
  31. uint8_t nvram_mem_buf[NVRAM_MEM_SIZE];
  32. #endif
  33. #if defined(CONFIG_BOARD_NANDBOOT) || defined(CONFIG_BOARD_EMMCBOOT)
  34. #define NV_NAND_PAGE_SIZE 2048
  35. #define NV_NAND_SECTOR_SIZE 512
  36. #define NV_CACHE_SECT_NUM (NV_NAND_PAGE_SIZE/NV_NAND_SECTOR_SIZE)
  37. #define NV_CACEH_NUM_ITEM 2
  38. struct nv_nand_cache_item {
  39. u16_t cache_valid:1;
  40. u16_t write_valid:1;
  41. u16_t reserved:14;
  42. u16_t seq_index;
  43. u32_t cache_start_sector;
  44. u8_t cache_data[NV_NAND_PAGE_SIZE];
  45. };
  46. static struct nv_nand_cache_item __act_s2_notsave nv_nand_cache[NV_CACEH_NUM_ITEM];
  47. static u16_t g_seq_index;
  48. static void _nv_nand_cache_init(void)
  49. {
  50. int i;
  51. for(i = 0; i < NV_CACEH_NUM_ITEM; i++) {
  52. nv_nand_cache[i].cache_valid = 0;
  53. nv_nand_cache[i].write_valid = 0;
  54. }
  55. }
  56. static int _nv_flush_cache(struct device *dev, struct nv_nand_cache_item *item)
  57. {
  58. int err;
  59. if(item->write_valid && item->cache_valid){
  60. item->write_valid = 0;
  61. LOG_DBG("nand flush startsec 0x%x, buf %p", item->cache_start_sector, item->cache_data);
  62. err = flash_write(dev, (item->cache_start_sector<<9), item->cache_data, NV_CACHE_SECT_NUM<<9);
  63. if (err < 0) {
  64. LOG_ERR("nand write error %d, offsec 0x%x, buf %p",
  65. err, item->cache_start_sector, item->cache_data);
  66. return -EIO;
  67. }
  68. }
  69. return 0;
  70. }
  71. static int _nv_update_cache(struct device *dev, struct nv_nand_cache_item *item, u32_t sec_off)
  72. {
  73. int err;
  74. u32_t page_off = sec_off/NV_CACHE_SECT_NUM;
  75. item->cache_start_sector = page_off * NV_CACHE_SECT_NUM;
  76. item->cache_valid = 1;
  77. item->write_valid = 0;
  78. item->seq_index = g_seq_index++;
  79. LOG_DBG("nand update start_sec 0x%x, buf %p, sec_off=0x%x,0x%x",
  80. item->cache_start_sector, item->cache_data, sec_off, sec_off*NV_NAND_SECTOR_SIZE);
  81. err = flash_read(dev, (item->cache_start_sector)<<9, item->cache_data, NV_CACHE_SECT_NUM<<9);
  82. if (err < 0) {
  83. LOG_ERR("nand read error %d, offsec 0x%x, buf %p",
  84. err, item->cache_start_sector, item->cache_data);
  85. return -EIO;
  86. }
  87. return 0;
  88. }
  89. static struct nv_nand_cache_item * _nv_find_cache_item(u32_t sec_off)
  90. {
  91. for (int i = 0; i < NV_CACEH_NUM_ITEM; i++) {
  92. if (nv_nand_cache[i].cache_valid && (nv_nand_cache[i].cache_start_sector <= sec_off)
  93. && (sec_off < nv_nand_cache[i].cache_start_sector+NV_CACHE_SECT_NUM)) {
  94. return &nv_nand_cache[i];
  95. }
  96. }
  97. //LOG_INF("sect_off 0x%x not in cache", sec_off);
  98. return NULL;
  99. }
  100. /*
  101. if have invaid ,use invalid, else use same
  102. */
  103. static struct nv_nand_cache_item * _nv_new_cache_item(struct device *dev, u32_t sec_off)
  104. {
  105. int i;
  106. struct nv_nand_cache_item *invalid, *valid, *new_item;
  107. valid = invalid = NULL;
  108. for(i = 0; i < NV_CACEH_NUM_ITEM; i++) {
  109. if(nv_nand_cache[i].cache_valid){
  110. if(valid == NULL){
  111. valid = &nv_nand_cache[i];
  112. }else{
  113. if(valid->seq_index > nv_nand_cache[i].seq_index)
  114. valid = &nv_nand_cache[i];
  115. }
  116. }else{
  117. invalid = &nv_nand_cache[i];
  118. break;
  119. }
  120. }
  121. if(invalid){
  122. new_item = invalid;
  123. }else{
  124. new_item = valid;
  125. }
  126. //LOG_INF("new item sect_off 0x%x, 0x%x is_write=%d", sec_off, sec_off*NV_NAND_SECTOR_SIZE, is_write);
  127. _nv_flush_cache(dev, new_item);
  128. _nv_update_cache(dev, new_item, sec_off);
  129. return new_item;
  130. }
  131. void nvram_storage_flush(struct device *dev)
  132. {
  133. int i;
  134. for(i = 0; i < NV_CACEH_NUM_ITEM; i++)
  135. _nv_flush_cache(dev, &nv_nand_cache[i]);
  136. }
  137. /*brwe = 0 (read), 1(write), 2(erase)*/
  138. static void _nand_storage_read_write(struct device *dev, uint32_t addr,
  139. uint8_t *buf, int32_t size, int brwe)
  140. {
  141. int32_t wsize, len;
  142. uint32_t sector_off, addr_in_sec,cache_sec_index;
  143. static struct nv_nand_cache_item * item;
  144. wsize = size;
  145. sector_off = addr / NV_NAND_SECTOR_SIZE;
  146. addr_in_sec = addr - (sector_off*NV_NAND_SECTOR_SIZE);
  147. //LOG_INF("sect_off 0x%x, size 0x%x iswrite %d", sector_off, size, brwe);
  148. while(wsize > 0) {
  149. item = _nv_find_cache_item(sector_off);
  150. if(item == NULL)
  151. item = _nv_new_cache_item(dev, sector_off);
  152. cache_sec_index = sector_off -item->cache_start_sector;
  153. if(addr_in_sec){//not sector algin
  154. len = NV_NAND_SECTOR_SIZE - addr_in_sec;
  155. if(len > wsize)
  156. len = wsize;
  157. if(brwe == 1)
  158. memcpy(&item->cache_data[cache_sec_index*NV_NAND_SECTOR_SIZE+addr_in_sec],buf, len);
  159. else if(brwe == 0)
  160. memcpy(buf, &item->cache_data[cache_sec_index*NV_NAND_SECTOR_SIZE+addr_in_sec], len);
  161. else
  162. memset(&item->cache_data[cache_sec_index*NV_NAND_SECTOR_SIZE+addr_in_sec], 0xff, len);
  163. buf += len;
  164. wsize -= len;
  165. cache_sec_index++;
  166. addr_in_sec = 0; // next align
  167. }
  168. for( ; wsize && (cache_sec_index < NV_CACHE_SECT_NUM); cache_sec_index++){
  169. if(wsize < NV_NAND_SECTOR_SIZE)
  170. len = wsize;
  171. else
  172. len = NV_NAND_SECTOR_SIZE;
  173. if(brwe == 1)
  174. memcpy(&item->cache_data[cache_sec_index*NV_NAND_SECTOR_SIZE],buf, len);
  175. else if(brwe == 0)
  176. memcpy(buf, &item->cache_data[cache_sec_index*NV_NAND_SECTOR_SIZE], len);
  177. else
  178. memset(&item->cache_data[cache_sec_index*NV_NAND_SECTOR_SIZE], 0xff, len);
  179. buf += len;
  180. wsize -= len;
  181. }
  182. if(brwe)
  183. item->write_valid = 1; //dirty
  184. sector_off = item->cache_start_sector + NV_CACHE_SECT_NUM;
  185. }
  186. }
  187. static int _nand_storage_write(struct device *dev, uint32_t addr,
  188. uint8_t *buf, int32_t size)
  189. {
  190. _nand_storage_read_write(dev, addr, buf, size, 1);
  191. //LOG_INF("nand write end 0x%x len 0x%x", addr, size);
  192. return 0;
  193. }
  194. static int _nand_storage_read(struct device *dev, uint32_t addr,
  195. void *buf, int32_t size)
  196. {
  197. _nand_storage_read_write(dev, addr, buf, size, 0);
  198. //LOG_INF("nand read end 0x%x len 0x%x", addr, size);
  199. return 0;
  200. }
  201. static int _nand_storage_erase(struct device *dev, uint32_t addr,
  202. int32_t size)
  203. {
  204. uint8_t buf[1]; // not use;
  205. _nand_storage_read_write(dev, addr, buf, size, 2);
  206. //LOG_INF("nand erase end 0x%x len 0x%x", addr, size);
  207. return 0;
  208. }
  209. #endif
  210. #define NVRAM_STORAGE_READ_SEGMENT_SIZE 256
  211. #define NVRAM_STORAGE_WRITE_SEGMENT_SIZE 32
  212. int nvram_storage_write(struct device *dev, uint64_t addr,
  213. const void *buf, int32_t size)
  214. {
  215. LOG_INF("write addr 0x%x len 0x%x", addr, size);
  216. #ifdef CONFIG_NVRAM_STORAGE_RAMSIM
  217. memcpy(nvram_mem_buf + addr - NVRAM_MEM_OFFS, buf, size);
  218. return 0;
  219. #else
  220. #if defined(CONFIG_BOARD_NANDBOOT) || defined(CONFIG_BOARD_EMMCBOOT)
  221. _nand_storage_write(dev, addr, (uint8_t *)buf, size);
  222. #else
  223. int err;
  224. int wlen = NVRAM_STORAGE_WRITE_SEGMENT_SIZE;
  225. while (size > 0) {
  226. if (size < NVRAM_STORAGE_WRITE_SEGMENT_SIZE)
  227. wlen = size;
  228. err = flash_write(dev, addr, buf, wlen);
  229. if (err < 0) {
  230. LOG_ERR("write error %d, addr 0x%x, buf %p, size %d",
  231. err, addr, buf, size);
  232. return -EIO;
  233. }
  234. addr += wlen;
  235. buf = (void*)((uint8_t *)buf + wlen);
  236. size -= wlen;
  237. }
  238. #endif
  239. return 0;
  240. #endif
  241. }
  242. int nvram_storage_read(struct device *dev, uint64_t addr,
  243. void *buf, int32_t size)
  244. {
  245. LOG_DBG("read addr 0x%x len 0x%x", addr, size);
  246. #ifdef CONFIG_NVRAM_STORAGE_RAMSIM
  247. memcpy(buf, nvram_mem_buf + addr - NVRAM_MEM_OFFS, size);
  248. return 0;
  249. #else
  250. #if defined(CONFIG_BOARD_NANDBOOT) || defined(CONFIG_BOARD_EMMCBOOT)
  251. _nand_storage_read(dev, addr, (uint8_t *)buf, size);
  252. #else
  253. int err;
  254. int rlen = NVRAM_STORAGE_READ_SEGMENT_SIZE;
  255. while (size > 0) {
  256. if (size < NVRAM_STORAGE_READ_SEGMENT_SIZE)
  257. rlen = size;
  258. err = flash_read(dev, addr, buf, rlen);
  259. if (err < 0) {
  260. LOG_ERR("read error %d, addr 0x%x, buf %p, size %d", err, addr, buf, size);
  261. return -EIO;
  262. }
  263. addr += rlen;
  264. buf = (void*)((uint8_t *)buf + rlen);
  265. size -= rlen;
  266. }
  267. #endif
  268. return 0;
  269. #endif
  270. }
  271. int nvram_storage_erase(struct device *dev, uint64_t addr,
  272. int32_t size)
  273. {
  274. LOG_INF("erase addr 0x%x size 0x%x", addr, size);
  275. #ifdef CONFIG_NVRAM_STORAGE_RAMSIM
  276. memset(nvram_mem_buf + addr - NVRAM_MEM_OFFS, 0xff, size);
  277. return 0;
  278. #else
  279. #if defined(CONFIG_BOARD_NANDBOOT) || defined(CONFIG_BOARD_EMMCBOOT)
  280. return _nand_storage_erase(dev, addr, size);
  281. #else
  282. return flash_erase(dev, (off_t)addr, (size_t)size);
  283. #endif
  284. #endif
  285. }
  286. struct device *nvram_storage_init(void)
  287. {
  288. struct device *dev;
  289. printk("init nvram storage\n");
  290. #ifdef CONFIG_NVRAM_STORAGE_RAMSIM
  291. LOG_INF("init simulated nvram storage: addr 0x%x size 0x%x",
  292. NVRAM_MEM_OFFS, NVRAM_MEM_SIZE);
  293. // memset(nvram_mem_buf, 0xff, NVRAM_MEM_SIZE);
  294. /* dummy device pointer */
  295. dev = (struct device *)-1;
  296. #else
  297. #if defined(CONFIG_BOARD_NANDBOOT)
  298. #define FLASH_DEV_NAME "spinand"
  299. _nv_nand_cache_init();
  300. #elif defined(CONFIG_BOARD_EMMCBOOT)
  301. #define FLASH_DEV_NAME CONFIG_SD_NAME
  302. _nv_nand_cache_init();
  303. #else
  304. #define FLASH_DEV_NAME CONFIG_SPI_FLASH_NAME
  305. #endif
  306. dev = (struct device *)device_get_binding(FLASH_DEV_NAME);
  307. if (!dev) {
  308. LOG_ERR("cannot found device \'%s\'",
  309. FLASH_DEV_NAME);
  310. return NULL;
  311. }
  312. #endif
  313. return dev;
  314. }