2019-12-10 03:44:44 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2019 Kwon Tae-young <tykwon@m2i.co.kr>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2020-03-24 20:28:48 +01:00
|
|
|
#define DT_DRV_COMPAT st_stm32_eeprom
|
|
|
|
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/drivers/eeprom.h>
|
2019-12-10 03:44:44 +01:00
|
|
|
#include <soc.h>
|
2022-10-13 11:08:32 +02:00
|
|
|
#include <zephyr/kernel.h>
|
2019-12-10 03:44:44 +01:00
|
|
|
|
|
|
|
#define LOG_LEVEL CONFIG_EEPROM_LOG_LEVEL
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/logging/log.h>
|
2019-12-10 03:44:44 +01:00
|
|
|
LOG_MODULE_REGISTER(eeprom_stm32);
|
|
|
|
|
|
|
|
K_MUTEX_DEFINE(lock);
|
|
|
|
|
|
|
|
struct eeprom_stm32_config {
|
2020-05-27 18:26:57 +02:00
|
|
|
uint32_t addr;
|
2019-12-10 03:44:44 +01:00
|
|
|
size_t size;
|
|
|
|
};
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int eeprom_stm32_read(const struct device *dev, off_t offset,
|
|
|
|
void *buf,
|
2019-12-10 03:44:44 +01:00
|
|
|
size_t len)
|
|
|
|
{
|
2020-05-28 20:44:16 +02:00
|
|
|
const struct eeprom_stm32_config *config = dev->config;
|
2020-05-27 18:26:57 +02:00
|
|
|
uint8_t *pbuf = buf;
|
2019-12-10 03:44:44 +01:00
|
|
|
|
|
|
|
if (!len) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((offset + len) > config->size) {
|
|
|
|
LOG_WRN("attempt to read past device boundary");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_mutex_lock(&lock, K_FOREVER);
|
|
|
|
|
|
|
|
while (len) {
|
2020-05-27 18:26:57 +02:00
|
|
|
*pbuf = *(__IO uint8_t*)(config->addr + offset);
|
2019-12-10 03:44:44 +01:00
|
|
|
|
|
|
|
pbuf++;
|
|
|
|
offset++;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_mutex_unlock(&lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int eeprom_stm32_write(const struct device *dev, off_t offset,
|
2019-12-10 03:44:44 +01:00
|
|
|
const void *buf, size_t len)
|
|
|
|
{
|
2020-05-28 20:44:16 +02:00
|
|
|
const struct eeprom_stm32_config *config = dev->config;
|
2020-05-27 18:26:57 +02:00
|
|
|
const uint8_t *pbuf = buf;
|
2019-12-10 03:44:44 +01:00
|
|
|
HAL_StatusTypeDef ret = HAL_OK;
|
|
|
|
|
|
|
|
if (!len) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((offset + len) > config->size) {
|
|
|
|
LOG_WRN("attempt to write past device boundary");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_mutex_lock(&lock, K_FOREVER);
|
|
|
|
|
|
|
|
HAL_FLASHEx_DATAEEPROM_Unlock();
|
|
|
|
|
|
|
|
while (len) {
|
|
|
|
ret = HAL_FLASHEx_DATAEEPROM_Program(
|
|
|
|
FLASH_TYPEPROGRAMDATA_BYTE,
|
|
|
|
config->addr + offset, *pbuf);
|
|
|
|
if (ret) {
|
|
|
|
LOG_ERR("failed to write to EEPROM (err %d)", ret);
|
|
|
|
HAL_FLASHEx_DATAEEPROM_Lock();
|
|
|
|
k_mutex_unlock(&lock);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
pbuf++;
|
|
|
|
offset++;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = HAL_FLASHEx_DATAEEPROM_Lock();
|
|
|
|
if (ret) {
|
|
|
|
LOG_ERR("failed to lock EEPROM (err %d)", ret);
|
|
|
|
k_mutex_unlock(&lock);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_mutex_unlock(&lock);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static size_t eeprom_stm32_size(const struct device *dev)
|
2019-12-10 03:44:44 +01:00
|
|
|
{
|
2020-05-28 20:44:16 +02:00
|
|
|
const struct eeprom_stm32_config *config = dev->config;
|
2019-12-10 03:44:44 +01:00
|
|
|
|
|
|
|
return config->size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct eeprom_driver_api eeprom_stm32_api = {
|
|
|
|
.read = eeprom_stm32_read,
|
|
|
|
.write = eeprom_stm32_write,
|
|
|
|
.size = eeprom_stm32_size,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct eeprom_stm32_config eeprom_config = {
|
2020-03-24 20:28:48 +01:00
|
|
|
.addr = DT_INST_REG_ADDR(0),
|
|
|
|
.size = DT_INST_REG_SIZE(0),
|
2019-12-10 03:44:44 +01:00
|
|
|
};
|
|
|
|
|
2023-04-18 12:25:15 +02:00
|
|
|
DEVICE_DT_INST_DEFINE(0, NULL, NULL, NULL, &eeprom_config, POST_KERNEL,
|
|
|
|
CONFIG_EEPROM_INIT_PRIORITY, &eeprom_stm32_api);
|