1 /*
2 * Copyright (c) 2020 NXP
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/kernel.h>
8 #include <zephyr/device.h>
9 #include <string.h>
10 #include <zephyr/drivers/flash.h>
11 #include <errno.h>
12 #include <zephyr/init.h>
13 #include <soc.h>
14 #include "flash_priv.h"
15
16 #include "fsl_common.h"
17 #include "fsl_flashiap.h"
18
19
20 #if DT_NODE_HAS_STATUS_OKAY(DT_INST(0, nxp_iap_fmc11))
21 #define DT_DRV_COMPAT nxp_iap_fmc11
22 #elif DT_NODE_HAS_STATUS_OKAY(DT_INST(0, nxp_iap_fmc54))
23 #define DT_DRV_COMPAT nxp_iap_fmc54
24 #else
25 #error No matching compatible for soc_flash_lpc.c
26 #endif
27
28 #define SOC_NV_FLASH_NODE DT_INST(0, soc_nv_flash)
29
30 struct flash_priv {
31 /* HACK: flash write protection is managed in software. */
32 struct k_sem write_lock;
33 uint32_t pflash_block_base;
34 uint32_t sector_size;
35 };
36
37 static const struct flash_parameters flash_lpc_parameters = {
38 #if DT_NODE_HAS_PROP(SOC_NV_FLASH_NODE, write_block_size)
39 .write_block_size = DT_PROP(SOC_NV_FLASH_NODE, write_block_size),
40 #else
41 .write_block_size = FSL_FEATURE_FLASH_PFLASH_BLOCK_WRITE_UNIT_SIZE,
42 #endif
43 .erase_value = 0xff,
44 };
45
prepare_erase_write(off_t offset,size_t len,uint32_t sector_size)46 static inline void prepare_erase_write(off_t offset, size_t len,
47 uint32_t sector_size)
48 {
49 uint32_t start;
50 uint32_t stop;
51
52 start = offset / sector_size;
53 stop = (offset+len-1) / sector_size;
54 FLASHIAP_PrepareSectorForWrite(start, stop);
55 }
56
flash_lpc_erase(const struct device * dev,off_t offset,size_t len)57 static int flash_lpc_erase(const struct device *dev, off_t offset, size_t len)
58 {
59 struct flash_priv *priv = dev->data;
60 status_t rc;
61 unsigned int key;
62 uint32_t start;
63 uint32_t stop;
64 uint32_t page_size;
65
66 if (k_sem_take(&priv->write_lock, K_FOREVER)) {
67 return -EACCES;
68 }
69
70 key = irq_lock();
71 prepare_erase_write(offset, len, priv->sector_size);
72 page_size = flash_lpc_parameters.write_block_size;
73 start = offset / page_size;
74 stop = (offset+len-1) / page_size;
75 rc = FLASHIAP_ErasePage(start, stop,
76 CONFIG_SYS_CLOCK_HW_CYCLES_PER_SEC);
77 irq_unlock(key);
78
79 k_sem_give(&priv->write_lock);
80
81 return (rc == kStatus_FLASHIAP_Success) ? 0 : -EINVAL;
82 }
83
flash_lpc_read(const struct device * dev,off_t offset,void * data,size_t len)84 static int flash_lpc_read(const struct device *dev, off_t offset,
85 void *data, size_t len)
86 {
87 struct flash_priv *priv = dev->data;
88 uint32_t addr;
89
90 addr = offset + priv->pflash_block_base;
91
92 memcpy(data, (void *) addr, len);
93
94 return 0;
95 }
96
flash_lpc_write(const struct device * dev,off_t offset,const void * data,size_t len)97 static int flash_lpc_write(const struct device *dev, off_t offset,
98 const void *data, size_t len)
99 {
100 struct flash_priv *priv = dev->data;
101 uint32_t addr;
102 status_t rc;
103 unsigned int key;
104
105 if (k_sem_take(&priv->write_lock, K_FOREVER)) {
106 return -EACCES;
107 }
108
109 addr = offset + priv->pflash_block_base;
110
111 key = irq_lock();
112 prepare_erase_write(offset, len, priv->sector_size);
113 rc = FLASHIAP_CopyRamToFlash(addr, (uint32_t *) data, len,
114 CONFIG_SYS_CLOCK_HW_CYCLES_PER_SEC);
115 irq_unlock(key);
116
117 k_sem_give(&priv->write_lock);
118
119 return (rc == kStatus_FLASHIAP_Success) ? 0 : -EINVAL;
120 }
121
122 #if defined(CONFIG_FLASH_PAGE_LAYOUT)
123 static const struct flash_pages_layout dev_layout = {
124 .pages_count = DT_REG_SIZE(SOC_NV_FLASH_NODE) /
125 DT_PROP(SOC_NV_FLASH_NODE, erase_block_size),
126 .pages_size = DT_PROP(SOC_NV_FLASH_NODE, erase_block_size),
127 };
128
flash_lpc_pages_layout(const struct device * dev,const struct flash_pages_layout ** layout,size_t * layout_size)129 static void flash_lpc_pages_layout(const struct device *dev,
130 const struct flash_pages_layout **layout,
131 size_t *layout_size)
132 {
133 *layout = &dev_layout;
134 *layout_size = 1;
135 }
136 #endif /* CONFIG_FLASH_PAGE_LAYOUT */
137
138 static const struct flash_parameters *
flash_lpc_get_parameters(const struct device * dev)139 flash_lpc_get_parameters(const struct device *dev)
140 {
141 ARG_UNUSED(dev);
142
143 return &flash_lpc_parameters;
144 }
145
146 static struct flash_priv flash_data;
147
148 static DEVICE_API(flash, flash_lpc_api) = {
149 .erase = flash_lpc_erase,
150 .write = flash_lpc_write,
151 .read = flash_lpc_read,
152 .get_parameters = flash_lpc_get_parameters,
153 #if defined(CONFIG_FLASH_PAGE_LAYOUT)
154 .page_layout = flash_lpc_pages_layout,
155 #endif
156 };
157
flash_lpc_init(const struct device * dev)158 static int flash_lpc_init(const struct device *dev)
159 {
160 struct flash_priv *priv = dev->data;
161
162 k_sem_init(&priv->write_lock, 1, 1);
163
164 priv->pflash_block_base = DT_REG_ADDR(SOC_NV_FLASH_NODE);
165
166 #if defined(FSL_FEATURE_SYSCON_FLASH_SECTOR_SIZE_BYTES)
167 priv->sector_size = FSL_FEATURE_SYSCON_FLASH_SECTOR_SIZE_BYTES;
168 #else
169 #error "Sector size not set"
170 #endif
171
172 return 0;
173 }
174
175 DEVICE_DT_INST_DEFINE(0, flash_lpc_init, NULL,
176 &flash_data, NULL, POST_KERNEL,
177 CONFIG_FLASH_INIT_PRIORITY, &flash_lpc_api);
178