1 /*
2 * Copyright (c) 2022 BrainCo Inc.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT gd_gd32_flash_controller
8
9 #include "flash_gd32.h"
10
11 #include <zephyr/kernel.h>
12 #include <zephyr/devicetree.h>
13 #include <zephyr/drivers/flash.h>
14 #include <zephyr/logging/log.h>
15
16 #include <gd32_fmc.h>
17
18 LOG_MODULE_REGISTER(flash_gd32, CONFIG_FLASH_LOG_LEVEL);
19
20 struct flash_gd32_data {
21 struct k_sem mutex;
22 };
23
24 static struct flash_gd32_data flash_data;
25
26 static const struct flash_parameters flash_gd32_parameters = {
27 .write_block_size = SOC_NV_FLASH_PRG_SIZE,
28 .erase_value = 0xff,
29 };
30
flash_gd32_read(const struct device * dev,off_t offset,void * data,size_t len)31 static int flash_gd32_read(const struct device *dev, off_t offset,
32 void *data, size_t len)
33 {
34 if ((offset > SOC_NV_FLASH_SIZE) ||
35 ((offset + len) > SOC_NV_FLASH_SIZE)) {
36 return -EINVAL;
37 }
38
39 if (len == 0U) {
40 return 0;
41 }
42
43 memcpy(data, (uint8_t *)SOC_NV_FLASH_ADDR + offset, len);
44
45 return 0;
46 }
47
flash_gd32_write(const struct device * dev,off_t offset,const void * data,size_t len)48 static int flash_gd32_write(const struct device *dev, off_t offset,
49 const void *data, size_t len)
50 {
51 struct flash_gd32_data *dev_data = dev->data;
52 int ret = 0;
53
54 if (!flash_gd32_valid_range(offset, len, true)) {
55 return -EINVAL;
56 }
57
58 if (len == 0U) {
59 return 0;
60 }
61
62 k_sem_take(&dev_data->mutex, K_FOREVER);
63
64 ret = flash_gd32_write_range(offset, data, len);
65
66 k_sem_give(&dev_data->mutex);
67
68 return ret;
69 }
70
flash_gd32_erase(const struct device * dev,off_t offset,size_t size)71 static int flash_gd32_erase(const struct device *dev, off_t offset, size_t size)
72 {
73 struct flash_gd32_data *data = dev->data;
74 int ret = 0;
75
76 if (size == 0U) {
77 return 0;
78 }
79
80 if (!flash_gd32_valid_range(offset, size, false)) {
81 return -EINVAL;
82 }
83
84 k_sem_take(&data->mutex, K_FOREVER);
85
86 ret = flash_gd32_erase_block(offset, size);
87
88 k_sem_give(&data->mutex);
89
90 return ret;
91 }
92
93 static const struct flash_parameters*
flash_gd32_get_parameters(const struct device * dev)94 flash_gd32_get_parameters(const struct device *dev)
95 {
96 ARG_UNUSED(dev);
97
98 return &flash_gd32_parameters;
99 }
100
101 static DEVICE_API(flash, flash_gd32_driver_api) = {
102 .read = flash_gd32_read,
103 .write = flash_gd32_write,
104 .erase = flash_gd32_erase,
105 .get_parameters = flash_gd32_get_parameters,
106 #ifdef CONFIG_FLASH_PAGE_LAYOUT
107 .page_layout = flash_gd32_pages_layout,
108 #endif
109 };
110
flash_gd32_init(const struct device * dev)111 static int flash_gd32_init(const struct device *dev)
112 {
113 struct flash_gd32_data *data = dev->data;
114
115 k_sem_init(&data->mutex, 1, 1);
116
117 return 0;
118 }
119
120 DEVICE_DT_INST_DEFINE(0, flash_gd32_init, NULL,
121 &flash_data, NULL, POST_KERNEL,
122 CONFIG_FLASH_INIT_PRIORITY, &flash_gd32_driver_api);
123