1 /*
2  * Copyright (c) 2021 Leonard Pollak
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT sensirion_sgp40
8 
9 #include <zephyr/device.h>
10 #include <zephyr/drivers/i2c.h>
11 #include <zephyr/kernel.h>
12 #include <zephyr/drivers/sensor.h>
13 #include <zephyr/logging/log.h>
14 #include <zephyr/pm/device.h>
15 #include <zephyr/sys/byteorder.h>
16 #include <zephyr/sys/crc.h>
17 
18 #include <zephyr/drivers/sensor/sgp40.h>
19 #include "sgp40.h"
20 
21 LOG_MODULE_REGISTER(SGP40, CONFIG_SENSOR_LOG_LEVEL);
22 
sgp40_compute_crc(uint16_t value)23 static uint8_t sgp40_compute_crc(uint16_t value)
24 {
25 	uint8_t buf[2];
26 
27 	sys_put_be16(value, buf);
28 
29 	return crc8(buf, 2, SGP40_CRC_POLY, SGP40_CRC_INIT, false);
30 }
31 
sgp40_write_command(const struct device * dev,uint16_t cmd)32 static int sgp40_write_command(const struct device *dev, uint16_t cmd)
33 {
34 
35 	const struct sgp40_config *cfg = dev->config;
36 	uint8_t tx_buf[2];
37 
38 	sys_put_be16(cmd, tx_buf);
39 
40 	return i2c_write_dt(&cfg->bus, tx_buf, sizeof(tx_buf));
41 }
42 
sgp40_start_measurement(const struct device * dev)43 static int sgp40_start_measurement(const struct device *dev)
44 {
45 	const struct sgp40_config *cfg = dev->config;
46 	struct sgp40_data *data = dev->data;
47 	uint8_t tx_buf[8];
48 
49 	sys_put_be16(SGP40_CMD_MEASURE_RAW, tx_buf);
50 	sys_put_be24(sys_get_be24(data->rh_param), &tx_buf[2]);
51 	sys_put_be24(sys_get_be24(data->t_param), &tx_buf[5]);
52 
53 	return i2c_write_dt(&cfg->bus, tx_buf, sizeof(tx_buf));
54 }
55 
sgp40_attr_set(const struct device * dev,enum sensor_channel chan,enum sensor_attribute attr,const struct sensor_value * val)56 static int sgp40_attr_set(const struct device *dev,
57 				enum sensor_channel chan,
58 				enum sensor_attribute attr,
59 				const struct sensor_value *val)
60 {
61 	struct sgp40_data *data = dev->data;
62 
63 	/*
64 	 * Temperature and RH conversion to ticks as explained in datasheet
65 	 * in section "I2C commands"
66 	 */
67 
68 	switch ((enum sensor_attribute_sgp40)attr) {
69 	case SENSOR_ATTR_SGP40_TEMPERATURE:
70 	{
71 		uint16_t t_ticks;
72 		int16_t tmp;
73 
74 		tmp = (int16_t)CLAMP(val->val1, SGP40_COMP_MIN_T, SGP40_COMP_MAX_T);
75 		/* adding +87 to avoid most rounding errors through truncation */
76 		t_ticks = (uint16_t)((((tmp + 45) * 65535) + 87) / 175);
77 		sys_put_be16(t_ticks, data->t_param);
78 		data->t_param[2] = sgp40_compute_crc(t_ticks);
79 	}
80 		break;
81 	case SENSOR_ATTR_SGP40_HUMIDITY:
82 	{
83 		uint16_t rh_ticks;
84 		uint8_t tmp;
85 
86 		tmp = (uint8_t)CLAMP(val->val1, SGP40_COMP_MIN_RH, SGP40_COMP_MAX_RH);
87 		/* adding +50 to eliminate rounding errors through truncation */
88 		rh_ticks = (uint16_t)(((tmp * 65535U) + 50U) / 100U);
89 		sys_put_be16(rh_ticks, data->rh_param);
90 		data->rh_param[2] = sgp40_compute_crc(rh_ticks);
91 	}
92 		break;
93 	default:
94 		return -ENOTSUP;
95 	}
96 	return 0;
97 }
98 
sgp40_selftest(const struct device * dev)99 static int sgp40_selftest(const struct device *dev)
100 {
101 	const struct sgp40_config *cfg = dev->config;
102 	uint8_t rx_buf[3];
103 	uint16_t raw_sample;
104 	int rc;
105 
106 	rc = sgp40_write_command(dev, SGP40_CMD_MEASURE_TEST);
107 	if (rc < 0) {
108 		LOG_ERR("Failed to start selftest!");
109 		return rc;
110 	}
111 
112 	k_sleep(K_MSEC(SGP40_TEST_WAIT_MS));
113 
114 	rc = i2c_read_dt(&cfg->bus, rx_buf, sizeof(rx_buf));
115 	if (rc < 0) {
116 		LOG_ERR("Failed to read data sample.");
117 		return rc;
118 	}
119 
120 	raw_sample = sys_get_be16(rx_buf);
121 	if (sgp40_compute_crc(raw_sample) != rx_buf[2]) {
122 		LOG_ERR("Received invalid CRC from selftest.");
123 		return -EIO;
124 	}
125 
126 	if (raw_sample != SGP40_TEST_OK) {
127 		LOG_ERR("Selftest failed.");
128 		return -EIO;
129 	}
130 
131 	return 0;
132 }
133 
sgp40_sample_fetch(const struct device * dev,enum sensor_channel chan)134 static int sgp40_sample_fetch(const struct device *dev,
135 			       enum sensor_channel chan)
136 {
137 	struct sgp40_data *data = dev->data;
138 	const struct sgp40_config *cfg = dev->config;
139 	uint8_t rx_buf[3];
140 	uint16_t raw_sample;
141 	int rc;
142 
143 	if (chan != SENSOR_CHAN_GAS_RES && chan != SENSOR_CHAN_ALL) {
144 		return -ENOTSUP;
145 	}
146 
147 	rc = sgp40_start_measurement(dev);
148 	if (rc < 0) {
149 		LOG_ERR("Failed to start measurement.");
150 		return rc;
151 	}
152 
153 	k_sleep(K_MSEC(SGP40_MEASURE_WAIT_MS));
154 
155 	rc = i2c_read_dt(&cfg->bus, rx_buf, sizeof(rx_buf));
156 	if (rc < 0) {
157 		LOG_ERR("Failed to read data sample.");
158 		return rc;
159 	}
160 
161 	raw_sample = sys_get_be16(rx_buf);
162 	if (sgp40_compute_crc(raw_sample) != rx_buf[2]) {
163 		LOG_ERR("Invalid CRC8 for data sample.");
164 		return -EIO;
165 	}
166 
167 	data->raw_sample = raw_sample;
168 
169 	return 0;
170 }
171 
sgp40_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)172 static int sgp40_channel_get(const struct device *dev,
173 			      enum sensor_channel chan,
174 			      struct sensor_value *val)
175 {
176 	const struct sgp40_data *data = dev->data;
177 
178 	if (chan != SENSOR_CHAN_GAS_RES) {
179 		return -ENOTSUP;
180 	}
181 
182 	val->val1 = data->raw_sample;
183 	val->val2 = 0;
184 
185 	return 0;
186 }
187 
188 
189 #ifdef CONFIG_PM_DEVICE
sgp40_pm_action(const struct device * dev,enum pm_device_action action)190 static int sgp40_pm_action(const struct device *dev,
191 			   enum pm_device_action action)
192 {
193 	uint16_t cmd;
194 
195 	switch (action) {
196 	case PM_DEVICE_ACTION_RESUME:
197 		/* activate the hotplate by sending a measure command */
198 		cmd = SGP40_CMD_MEASURE_RAW;
199 		break;
200 	case PM_DEVICE_ACTION_SUSPEND:
201 		cmd = SGP40_CMD_HEATER_OFF;
202 		break;
203 	default:
204 		return -ENOTSUP;
205 	}
206 
207 	return sgp40_write_command(dev, cmd);
208 }
209 #endif /* CONFIG_PM_DEVICE */
210 
sgp40_init(const struct device * dev)211 static int sgp40_init(const struct device *dev)
212 {
213 	const struct sgp40_config *cfg = dev->config;
214 	struct sensor_value comp_data;
215 
216 	if (!device_is_ready(cfg->bus.bus)) {
217 		LOG_ERR("Device not ready.");
218 		return -ENODEV;
219 	}
220 
221 	if (cfg->selftest) {
222 		int rc = sgp40_selftest(dev);
223 
224 		if (rc < 0) {
225 			LOG_ERR("Selftest failed!");
226 			return rc;
227 		}
228 		LOG_DBG("Selftest succeeded!");
229 	}
230 
231 	comp_data.val1 = SGP40_COMP_DEFAULT_T;
232 	sensor_attr_set(dev,
233 			SENSOR_CHAN_GAS_RES,
234 			(enum sensor_attribute) SENSOR_ATTR_SGP40_TEMPERATURE,
235 			&comp_data);
236 	comp_data.val1 = SGP40_COMP_DEFAULT_RH;
237 	sensor_attr_set(dev,
238 			SENSOR_CHAN_GAS_RES,
239 			(enum sensor_attribute) SENSOR_ATTR_SGP40_HUMIDITY,
240 			&comp_data);
241 
242 	return 0;
243 }
244 
245 static DEVICE_API(sensor, sgp40_api) = {
246 	.sample_fetch = sgp40_sample_fetch,
247 	.channel_get = sgp40_channel_get,
248 	.attr_set = sgp40_attr_set,
249 };
250 
251 #define SGP40_INIT(n)						\
252 	static struct sgp40_data sgp40_data_##n;		\
253 								\
254 	static const struct sgp40_config sgp40_config_##n = {	\
255 		.bus = I2C_DT_SPEC_INST_GET(n),			\
256 		.selftest = DT_INST_PROP(n, enable_selftest),	\
257 	};							\
258 								\
259 	PM_DEVICE_DT_INST_DEFINE(n, sgp40_pm_action);		\
260 								\
261 	SENSOR_DEVICE_DT_INST_DEFINE(n,				\
262 			      sgp40_init,			\
263 			      PM_DEVICE_DT_INST_GET(n),	\
264 			      &sgp40_data_##n,			\
265 			      &sgp40_config_##n,		\
266 			      POST_KERNEL,			\
267 			      CONFIG_SENSOR_INIT_PRIORITY,	\
268 			      &sgp40_api);
269 
270 DT_INST_FOREACH_STATUS_OKAY(SGP40_INIT)
271