1 /*
2 * Copyright 2021 Google LLC
3 * SPDX-License-Identifier: Apache-2.0
4 */
5
6 #define DT_DRV_COMPAT zephyr_bbram_emul
7
8 #include <zephyr/drivers/bbram.h>
9 #include <string.h>
10
11 #include <zephyr/logging/log.h>
12 LOG_MODULE_REGISTER(bbram, CONFIG_BBRAM_LOG_LEVEL);
13
14 /** Device config */
15 struct bbram_emul_config {
16 /** BBRAM size (Unit:bytes) */
17 int size;
18 };
19
20 /** Device data */
21 struct bbram_emul_data {
22 /** Memory */
23 uint8_t *data;
24 /** Status register */
25 struct {
26 /** True if BBRAM is in an invalid state */
27 uint8_t is_invalid : 1;
28 /** True if BBRAM incurred a standby power failure */
29 uint8_t standby_failure : 1;
30 /** True if BBRAM incurred a power failure */
31 uint8_t power_failure : 1;
32 } status;
33 };
34
bbram_emul_set_invalid(const struct device * dev,bool is_invalid)35 int bbram_emul_set_invalid(const struct device *dev, bool is_invalid)
36 {
37 struct bbram_emul_data *data = dev->data;
38
39 data->status.is_invalid = is_invalid;
40 return 0;
41 }
42
bbram_emul_set_standby_power_state(const struct device * dev,bool failure)43 int bbram_emul_set_standby_power_state(const struct device *dev, bool failure)
44 {
45 struct bbram_emul_data *data = dev->data;
46
47 data->status.standby_failure = failure;
48 return 0;
49 }
50
bbram_emul_set_power_state(const struct device * dev,bool failure)51 int bbram_emul_set_power_state(const struct device *dev, bool failure)
52 {
53 struct bbram_emul_data *data = dev->data;
54
55 data->status.power_failure = failure;
56 return 0;
57 }
58
bbram_emul_check_invalid(const struct device * dev)59 static int bbram_emul_check_invalid(const struct device *dev)
60 {
61 struct bbram_emul_data *data = dev->data;
62 bool is_invalid = data->status.is_invalid;
63
64 data->status.is_invalid = false;
65 return is_invalid;
66 }
67
bbram_emul_check_standby_power(const struct device * dev)68 static int bbram_emul_check_standby_power(const struct device *dev)
69 {
70 struct bbram_emul_data *data = dev->data;
71 bool failure = data->status.standby_failure;
72
73 data->status.standby_failure = false;
74 return failure;
75 }
76
bbram_emul_check_power(const struct device * dev)77 static int bbram_emul_check_power(const struct device *dev)
78 {
79 struct bbram_emul_data *data = dev->data;
80 bool failure = data->status.power_failure;
81
82 data->status.power_failure = false;
83 return failure;
84 }
85
bbram_emul_get_size(const struct device * dev,size_t * size)86 static int bbram_emul_get_size(const struct device *dev, size_t *size)
87 {
88 const struct bbram_emul_config *config = dev->config;
89
90 *size = config->size;
91 return 0;
92 }
93
bbram_emul_read(const struct device * dev,size_t offset,size_t size,uint8_t * data)94 static int bbram_emul_read(const struct device *dev, size_t offset, size_t size,
95 uint8_t *data)
96 {
97 const struct bbram_emul_config *config = dev->config;
98 struct bbram_emul_data *dev_data = dev->data;
99
100 if (size < 1 || offset + size > config->size || bbram_emul_check_invalid(dev)) {
101 return -EFAULT;
102 }
103
104 memcpy(data, dev_data->data + offset, size);
105 return 0;
106 }
107
bbram_emul_write(const struct device * dev,size_t offset,size_t size,const uint8_t * data)108 static int bbram_emul_write(const struct device *dev, size_t offset, size_t size,
109 const uint8_t *data)
110 {
111 const struct bbram_emul_config *config = dev->config;
112 struct bbram_emul_data *dev_data = dev->data;
113
114 if (size < 1 || offset + size > config->size || bbram_emul_check_invalid(dev)) {
115 return -EFAULT;
116 }
117
118 memcpy(dev_data->data + offset, data, size);
119 return 0;
120 }
121
122 static DEVICE_API(bbram, bbram_emul_driver_api) = {
123 .check_invalid = bbram_emul_check_invalid,
124 .check_standby_power = bbram_emul_check_standby_power,
125 .check_power = bbram_emul_check_power,
126 .get_size = bbram_emul_get_size,
127 .read = bbram_emul_read,
128 .write = bbram_emul_write,
129 };
130
131 #define BBRAM_INIT(inst) \
132 static uint8_t bbram_emul_mem_##inst[DT_INST_PROP(inst, size)]; \
133 static struct bbram_emul_data bbram_emul_data_##inst = { \
134 .data = bbram_emul_mem_##inst, \
135 }; \
136 static struct bbram_emul_config bbram_emul_config_##inst = { \
137 .size = DT_INST_PROP(inst, size), \
138 }; \
139 DEVICE_DT_INST_DEFINE(inst, NULL, NULL, &bbram_emul_data_##inst, \
140 &bbram_emul_config_##inst, PRE_KERNEL_1, CONFIG_BBRAM_INIT_PRIORITY, \
141 &bbram_emul_driver_api);
142
143 DT_INST_FOREACH_STATUS_OKAY(BBRAM_INIT);
144