1 /*
2  * Copyright (c) 2021 Leica Geosystems AG
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT sbs_sbs_gauge
8 
9 #include <zephyr/drivers/i2c.h>
10 #include <zephyr/drivers/sensor.h>
11 #include <zephyr/sys/byteorder.h>
12 
13 #include "sbs_gauge.h"
14 
15 #include <zephyr/logging/log.h>
16 LOG_MODULE_REGISTER(sbs_gauge, CONFIG_SENSOR_LOG_LEVEL);
17 
sbs_cmd_reg_read(const struct device * dev,uint8_t reg_addr,uint16_t * val)18 static int sbs_cmd_reg_read(const struct device *dev,
19 			    uint8_t reg_addr,
20 			    uint16_t *val)
21 {
22 	const struct sbs_gauge_config *cfg;
23 	uint8_t i2c_data[2];
24 	int status;
25 
26 	cfg = dev->config;
27 	status = i2c_burst_read_dt(&cfg->i2c, reg_addr, i2c_data,
28 				   ARRAY_SIZE(i2c_data));
29 	if (status < 0) {
30 		LOG_ERR("Unable to read register");
31 		return status;
32 	}
33 
34 	*val = sys_get_le16(i2c_data);
35 
36 	return 0;
37 }
38 
39 /**
40  * @brief sensor value get
41  *
42  * @return -ENOTSUP for unsupported channels
43  */
sbs_gauge_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)44 static int sbs_gauge_channel_get(const struct device *dev,
45 				 enum sensor_channel chan,
46 				 struct sensor_value *val)
47 {
48 	struct sbs_gauge_data *data;
49 	int32_t int_temp;
50 
51 	data = dev->data;
52 	val->val2 = 0;
53 
54 	switch (chan) {
55 	case SENSOR_CHAN_GAUGE_VOLTAGE:
56 		val->val1 = data->voltage / 1000;
57 		val->val2 = (data->voltage % 1000) * 1000;
58 		break;
59 
60 	case SENSOR_CHAN_GAUGE_AVG_CURRENT:
61 		val->val1 = data->avg_current / 1000;
62 		val->val2 = (data->avg_current % 1000) * 1000;
63 		break;
64 
65 	case SENSOR_CHAN_GAUGE_TEMP:
66 		int_temp = (data->internal_temperature * 10);
67 		int_temp = int_temp - 27315;
68 		val->val1 = int_temp / 100;
69 		val->val2 = (int_temp % 100) * 1000000;
70 		break;
71 
72 	case SENSOR_CHAN_GAUGE_STATE_OF_CHARGE:
73 		val->val1 = data->state_of_charge;
74 		break;
75 
76 	case SENSOR_CHAN_GAUGE_FULL_CHARGE_CAPACITY:
77 		val->val1 = data->full_charge_capacity;
78 		break;
79 
80 	case SENSOR_CHAN_GAUGE_REMAINING_CHARGE_CAPACITY:
81 		val->val1 = data->remaining_charge_capacity;
82 		break;
83 
84 	case SENSOR_CHAN_GAUGE_NOM_AVAIL_CAPACITY:
85 		val->val1 = data->nom_avail_capacity;
86 		break;
87 
88 	case SENSOR_CHAN_GAUGE_FULL_AVAIL_CAPACITY:
89 		val->val1 = data->full_avail_capacity;
90 		break;
91 
92 	case SENSOR_CHAN_GAUGE_TIME_TO_EMPTY:
93 		val->val1 = data->time_to_empty;
94 		break;
95 
96 	case SENSOR_CHAN_GAUGE_TIME_TO_FULL:
97 		val->val1 = data->time_to_full;
98 		break;
99 
100 	case SENSOR_CHAN_GAUGE_CYCLE_COUNT:
101 		val->val1 = data->cycle_count;
102 		break;
103 
104 	default:
105 		return -ENOTSUP;
106 	}
107 
108 	return 0;
109 }
110 
111 static const uint16_t all_channels[] = {
112 	SENSOR_CHAN_GAUGE_VOLTAGE,
113 	SENSOR_CHAN_GAUGE_AVG_CURRENT,
114 	SENSOR_CHAN_GAUGE_TEMP,
115 	SENSOR_CHAN_GAUGE_STATE_OF_CHARGE,
116 	SENSOR_CHAN_GAUGE_FULL_CHARGE_CAPACITY,
117 	SENSOR_CHAN_GAUGE_REMAINING_CHARGE_CAPACITY,
118 	SENSOR_CHAN_GAUGE_NOM_AVAIL_CAPACITY,
119 	SENSOR_CHAN_GAUGE_FULL_AVAIL_CAPACITY,
120 	SENSOR_CHAN_GAUGE_TIME_TO_EMPTY,
121 	SENSOR_CHAN_GAUGE_TIME_TO_FULL,
122 	SENSOR_CHAN_GAUGE_CYCLE_COUNT
123 };
124 
sbs_gauge_sample_fetch(const struct device * dev,enum sensor_channel chan)125 static int sbs_gauge_sample_fetch(const struct device *dev,
126 			    enum sensor_channel chan)
127 {
128 	struct sbs_gauge_data *data;
129 	int status = 0;
130 
131 	data = dev->data;
132 
133 	switch (chan) {
134 	case SENSOR_CHAN_GAUGE_VOLTAGE:
135 		status = sbs_cmd_reg_read(dev,
136 					  SBS_GAUGE_CMD_VOLTAGE,
137 					  &data->voltage);
138 		if (status < 0) {
139 			LOG_ERR("Failed to read voltage");
140 		}
141 		break;
142 
143 	case SENSOR_CHAN_GAUGE_AVG_CURRENT:
144 		status = sbs_cmd_reg_read(dev,
145 					  SBS_GAUGE_CMD_AVG_CURRENT,
146 					  &data->avg_current);
147 		if (status < 0) {
148 			LOG_ERR("Failed to read average current ");
149 		}
150 		break;
151 
152 	case SENSOR_CHAN_GAUGE_TEMP:
153 		status = sbs_cmd_reg_read(dev,
154 					  SBS_GAUGE_CMD_TEMP,
155 					  &data->internal_temperature);
156 		if (status < 0) {
157 			LOG_ERR("Failed to read internal temperature");
158 		}
159 		break;
160 
161 	case SENSOR_CHAN_GAUGE_STATE_OF_CHARGE:
162 		status = sbs_cmd_reg_read(dev,
163 					  SBS_GAUGE_CMD_ASOC,
164 					  &data->state_of_charge);
165 		if (status < 0) {
166 			LOG_ERR("Failed to read state of charge");
167 		}
168 		break;
169 
170 	case SENSOR_CHAN_GAUGE_FULL_CHARGE_CAPACITY:
171 		status = sbs_cmd_reg_read(dev,
172 					  SBS_GAUGE_CMD_FULL_CAPACITY,
173 					  &data->full_charge_capacity);
174 		if (status < 0) {
175 			LOG_ERR("Failed to read full charge capacity");
176 		}
177 		break;
178 
179 	case SENSOR_CHAN_GAUGE_REMAINING_CHARGE_CAPACITY:
180 		status = sbs_cmd_reg_read(dev,
181 					  SBS_GAUGE_CMD_REM_CAPACITY,
182 					  &data->remaining_charge_capacity);
183 		if (status < 0) {
184 			LOG_ERR("Failed to read remaining charge capacity");
185 		}
186 		break;
187 
188 	case SENSOR_CHAN_GAUGE_NOM_AVAIL_CAPACITY:
189 		status = sbs_cmd_reg_read(dev,
190 					  SBS_GAUGE_CMD_NOM_CAPACITY,
191 					  &data->nom_avail_capacity);
192 		if (status < 0) {
193 			LOG_ERR("Failed to read nominal available capacity");
194 		}
195 		break;
196 
197 	case SENSOR_CHAN_GAUGE_FULL_AVAIL_CAPACITY:
198 		status = sbs_cmd_reg_read(dev,
199 					  SBS_GAUGE_CMD_FULL_CAPACITY,
200 					  &data->full_avail_capacity);
201 		if (status < 0) {
202 			LOG_ERR("Failed to read full available capacity");
203 		}
204 		break;
205 
206 	case SENSOR_CHAN_GAUGE_TIME_TO_EMPTY:
207 		status = sbs_cmd_reg_read(dev,
208 					  SBS_GAUGE_CMD_AVG_TIME2EMPTY,
209 					  &data->time_to_empty);
210 		data->time_to_empty = (data->time_to_empty) & 0x00FF;
211 
212 		if (status < 0) {
213 			LOG_ERR("Failed to read time to empty");
214 		}
215 		break;
216 
217 	case SENSOR_CHAN_GAUGE_TIME_TO_FULL:
218 		status = sbs_cmd_reg_read(dev,
219 					  SBS_GAUGE_CMD_AVG_TIME2FULL,
220 					  &data->time_to_full);
221 		data->time_to_full = (data->time_to_full) & 0x00FF;
222 
223 		if (status < 0) {
224 			LOG_ERR("Failed to read time to full");
225 		}
226 		break;
227 
228 	case SENSOR_CHAN_GAUGE_CYCLE_COUNT:
229 		status = sbs_cmd_reg_read(dev,
230 					  SBS_GAUGE_CMD_CYCLE_COUNT,
231 					  &data->cycle_count);
232 		data->cycle_count = (data->cycle_count) & 0x00FF;
233 
234 		if (status < 0) {
235 			LOG_ERR("Failed to read cycle count");
236 		}
237 		break;
238 
239 	case SENSOR_CHAN_ALL:
240 		for (int i = 0; i < ARRAY_SIZE(all_channels); i++) {
241 			status = sbs_gauge_sample_fetch(dev, all_channels[i]);
242 			if (status != 0) {
243 				break;
244 			}
245 		}
246 
247 		break;
248 
249 	default:
250 		return -ENOTSUP;
251 	}
252 
253 	return status;
254 }
255 
256 /**
257  * @brief initialize the fuel gauge
258  *
259  * @return 0 for success
260  */
sbs_gauge_init(const struct device * dev)261 static int sbs_gauge_init(const struct device *dev)
262 {
263 	const struct sbs_gauge_config *cfg;
264 
265 	cfg = dev->config;
266 
267 	if (!device_is_ready(cfg->i2c.bus)) {
268 		LOG_ERR("Bus device is not ready");
269 		return -ENODEV;
270 	}
271 
272 	return 0;
273 }
274 
275 static DEVICE_API(sensor, sbs_gauge_driver_api) = {
276 	.sample_fetch = sbs_gauge_sample_fetch,
277 	.channel_get = sbs_gauge_channel_get,
278 };
279 
280 #define SBS_GAUGE_INIT(index) \
281 	static struct sbs_gauge_data sbs_gauge_driver_##index; \
282 \
283 	static const struct sbs_gauge_config sbs_gauge_config_##index = { \
284 		.i2c = I2C_DT_SPEC_INST_GET(index), \
285 	}; \
286 \
287 	SENSOR_DEVICE_DT_INST_DEFINE(index, \
288 			    &sbs_gauge_init, \
289 			    NULL, \
290 			    &sbs_gauge_driver_##index, \
291 			    &sbs_gauge_config_##index, POST_KERNEL, \
292 			    CONFIG_SENSOR_INIT_PRIORITY, \
293 			    &sbs_gauge_driver_api);
294 
295 DT_INST_FOREACH_STATUS_OKAY(SBS_GAUGE_INIT)
296