1 /*
2 * Copyright (c) 2022 Intel Corporation
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6 #define DT_DRV_COMPAT cdns_qspi_nor
7
8 #include "flash_cadence_qspi_nor_ll.h"
9
10 #include <string.h>
11
12 #include <zephyr/kernel.h>
13 #include <zephyr/device.h>
14 #include <zephyr/drivers/flash.h>
15 #include <zephyr/logging/log.h>
16
17 LOG_MODULE_REGISTER(flash_cadence, CONFIG_FLASH_LOG_LEVEL);
18
19 struct flash_cad_priv {
20 DEVICE_MMIO_NAMED_RAM(qspi_reg);
21 DEVICE_MMIO_NAMED_RAM(qspi_data);
22 struct cad_qspi_params params;
23 };
24
25 struct flash_cad_config {
26 DEVICE_MMIO_NAMED_ROM(qspi_reg);
27 DEVICE_MMIO_NAMED_ROM(qspi_data);
28 };
29
30 static const struct flash_parameters flash_cad_parameters = {
31 .write_block_size = QSPI_BYTES_PER_DEV,
32 .erase_value = 0xff,
33 };
34
35 #define DEV_DATA(dev) ((struct flash_cad_priv *)((dev)->data))
36 #define DEV_CFG(dev) ((struct flash_cad_config *)((dev)->config))
37
flash_cad_read(const struct device * dev,off_t offset,void * data,size_t len)38 static int flash_cad_read(const struct device *dev, off_t offset,
39 void *data, size_t len)
40 {
41 struct flash_cad_priv *priv = dev->data;
42 struct cad_qspi_params *cad_params = &priv->params;
43 int rc;
44
45 if ((data == NULL) || (len == 0)) {
46 LOG_ERR("Invalid input parameter for QSPI Read!");
47 return -EINVAL;
48 }
49
50 rc = cad_qspi_read(cad_params, data, (uint32_t)offset, len);
51
52 if (rc < 0) {
53 LOG_ERR("Cadence QSPI Flash Read Failed");
54 return rc;
55 }
56
57 return 0;
58 }
59
flash_cad_erase(const struct device * dev,off_t offset,size_t len)60 static int flash_cad_erase(const struct device *dev, off_t offset,
61 size_t len)
62 {
63 struct flash_cad_priv *priv = dev->data;
64 struct cad_qspi_params *cad_params = &priv->params;
65 int rc;
66
67 if (len == 0) {
68 LOG_ERR("Invalid input parameter for QSPI Erase!");
69 return -EINVAL;
70 }
71
72 rc = cad_qspi_erase(cad_params, (uint32_t)offset, len);
73
74 if (rc < 0) {
75 LOG_ERR("Cadence QSPI Flash Erase Failed!");
76 return rc;
77 }
78
79 return 0;
80 }
81
flash_cad_write(const struct device * dev,off_t offset,const void * data,size_t len)82 static int flash_cad_write(const struct device *dev, off_t offset,
83 const void *data, size_t len)
84 {
85 struct flash_cad_priv *priv = dev->data;
86 struct cad_qspi_params *cad_params = &priv->params;
87 int rc;
88
89 if ((data == NULL) || (len == 0)) {
90 LOG_ERR("Invalid input parameter for QSPI Write!");
91 return -EINVAL;
92 }
93
94 rc = cad_qspi_write(cad_params, (void *)data, (uint32_t)offset, len);
95
96 if (rc < 0) {
97 LOG_ERR("Cadence QSPI Flash Write Failed!");
98 return rc;
99 }
100
101 return 0;
102 }
103
104 static const struct flash_parameters *
flash_cad_get_parameters(const struct device * dev)105 flash_cad_get_parameters(const struct device *dev)
106 {
107 ARG_UNUSED(dev);
108
109 return &flash_cad_parameters;
110 }
111
112 static DEVICE_API(flash, flash_cad_api) = {
113 .erase = flash_cad_erase,
114 .write = flash_cad_write,
115 .read = flash_cad_read,
116 .get_parameters = flash_cad_get_parameters,
117 };
118
flash_cad_init(const struct device * dev)119 static int flash_cad_init(const struct device *dev)
120 {
121 struct flash_cad_priv *priv = dev->data;
122 struct cad_qspi_params *cad_params = &priv->params;
123 int rc;
124
125 DEVICE_MMIO_NAMED_MAP(dev, qspi_reg, K_MEM_CACHE_NONE);
126 DEVICE_MMIO_NAMED_MAP(dev, qspi_data, K_MEM_CACHE_NONE);
127
128 cad_params->reg_base = DEVICE_MMIO_NAMED_GET(dev, qspi_reg);
129 cad_params->data_base = DEVICE_MMIO_NAMED_GET(dev, qspi_data);
130
131 rc = cad_qspi_init(cad_params, QSPI_CONFIG_CPHA,
132 QSPI_CONFIG_CPOL, QSPI_CONFIG_CSDA,
133 QSPI_CONFIG_CSDADS, QSPI_CONFIG_CSEOT,
134 QSPI_CONFIG_CSSOT, 0);
135
136 if (rc < 0) {
137 LOG_ERR("Cadence QSPI Flash Init Failed");
138 return rc;
139 }
140
141 return 0;
142 }
143
144 #define CREATE_FLASH_CADENCE_QSPI_DEVICE(inst) \
145 static struct flash_cad_priv flash_cad_priv_##inst = { \
146 .params = { \
147 .clk_rate = DT_INST_PROP(inst, clock_frequency),\
148 .data_size = DT_INST_REG_SIZE_BY_IDX(inst, 1), \
149 }, \
150 }; \
151 \
152 static struct flash_cad_config flash_cad_config_##inst = { \
153 DEVICE_MMIO_NAMED_ROM_INIT_BY_NAME( \
154 qspi_reg, DT_DRV_INST(inst)), \
155 DEVICE_MMIO_NAMED_ROM_INIT_BY_NAME( \
156 qspi_data, DT_DRV_INST(inst)), \
157 }; \
158 \
159 DEVICE_DT_INST_DEFINE(inst, \
160 flash_cad_init, \
161 NULL, \
162 &flash_cad_priv_##inst, \
163 &flash_cad_config_##inst, \
164 POST_KERNEL, \
165 CONFIG_KERNEL_INIT_PRIORITY_DEVICE, \
166 &flash_cad_api);
167
168 DT_INST_FOREACH_STATUS_OKAY(CREATE_FLASH_CADENCE_QSPI_DEVICE)
169