1 /*
2  * Copyright (c) 2019 Kwon Tae-young <tykwon@m2i.co.kr>
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT st_stm32_eeprom
8 
9 #include <zephyr/drivers/eeprom.h>
10 #include <soc.h>
11 #include <zephyr/kernel.h>
12 
13 #define LOG_LEVEL CONFIG_EEPROM_LOG_LEVEL
14 #include <zephyr/logging/log.h>
15 LOG_MODULE_REGISTER(eeprom_stm32);
16 
17 K_MUTEX_DEFINE(lock);
18 
19 struct eeprom_stm32_config {
20 	uint32_t addr;
21 	size_t size;
22 };
23 
eeprom_stm32_read(const struct device * dev,off_t offset,void * buf,size_t len)24 static int eeprom_stm32_read(const struct device *dev, off_t offset,
25 				void *buf,
26 				size_t len)
27 {
28 	const struct eeprom_stm32_config *config = dev->config;
29 	uint8_t *pbuf = buf;
30 
31 	if (!len) {
32 		return 0;
33 	}
34 
35 	if ((offset + len) > config->size) {
36 		LOG_WRN("attempt to read past device boundary");
37 		return -EINVAL;
38 	}
39 
40 	k_mutex_lock(&lock, K_FOREVER);
41 
42 	while (len) {
43 		*pbuf = *(__IO uint8_t*)(config->addr + offset);
44 
45 		pbuf++;
46 		offset++;
47 		len--;
48 	}
49 
50 	k_mutex_unlock(&lock);
51 
52 	return 0;
53 }
54 
eeprom_stm32_write(const struct device * dev,off_t offset,const void * buf,size_t len)55 static int eeprom_stm32_write(const struct device *dev, off_t offset,
56 				const void *buf, size_t len)
57 {
58 	const struct eeprom_stm32_config *config = dev->config;
59 	const uint8_t *pbuf = buf;
60 	HAL_StatusTypeDef ret = HAL_OK;
61 
62 	if (!len) {
63 		return 0;
64 	}
65 
66 	if ((offset + len) > config->size) {
67 		LOG_WRN("attempt to write past device boundary");
68 		return -EINVAL;
69 	}
70 
71 	k_mutex_lock(&lock, K_FOREVER);
72 
73 	HAL_FLASHEx_DATAEEPROM_Unlock();
74 
75 	while (len) {
76 		ret = HAL_FLASHEx_DATAEEPROM_Program(
77 						FLASH_TYPEPROGRAMDATA_BYTE,
78 						config->addr + offset, *pbuf);
79 		if (ret) {
80 			LOG_ERR("failed to write to EEPROM (err %d)", ret);
81 			HAL_FLASHEx_DATAEEPROM_Lock();
82 			k_mutex_unlock(&lock);
83 			return ret;
84 		}
85 
86 		pbuf++;
87 		offset++;
88 		len--;
89 	}
90 
91 	ret = HAL_FLASHEx_DATAEEPROM_Lock();
92 	if (ret) {
93 		LOG_ERR("failed to lock EEPROM (err %d)", ret);
94 		k_mutex_unlock(&lock);
95 		return ret;
96 	}
97 
98 	k_mutex_unlock(&lock);
99 
100 	return ret;
101 }
102 
eeprom_stm32_size(const struct device * dev)103 static size_t eeprom_stm32_size(const struct device *dev)
104 {
105 	const struct eeprom_stm32_config *config = dev->config;
106 
107 	return config->size;
108 }
109 
110 static DEVICE_API(eeprom, eeprom_stm32_api) = {
111 	.read = eeprom_stm32_read,
112 	.write = eeprom_stm32_write,
113 	.size = eeprom_stm32_size,
114 };
115 
116 static const struct eeprom_stm32_config eeprom_config = {
117 	.addr = DT_INST_REG_ADDR(0),
118 	.size = DT_INST_REG_SIZE(0),
119 };
120 
121 DEVICE_DT_INST_DEFINE(0, NULL, NULL, NULL, &eeprom_config, POST_KERNEL,
122 		      CONFIG_EEPROM_INIT_PRIORITY, &eeprom_stm32_api);
123