1 /*
2 * Copyright (c) 2022 Mizuki Agawa <agawa.mizuki@fujitsu.com>
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT invensense_icp10125
8
9 #include <zephyr/drivers/i2c.h>
10 #include <zephyr/drivers/sensor.h>
11 #include <zephyr/logging/log.h>
12 #include <zephyr/sys/byteorder.h>
13
14 #ifdef CONFIG_ICP10125_CHECK_CRC
15 #include <zephyr/sys/crc.h>
16 #endif /* CONFIG_ICP10125_CHECK_CRC */
17
18 LOG_MODULE_REGISTER(ICP10125, CONFIG_SENSOR_LOG_LEVEL);
19
20 #define CRC_POLY 0x31
21 #define SENSOR_DATA_SIZE 2
22
23 #define AMBIENT_TEMP_DATA_NUM 1
24 #define PRESS_DATA_NUM 2
25 #define PRESS_AND_AMBIENT_TEMP_DATA_NUM (AMBIENT_TEMP_DATA_NUM + PRESS_DATA_NUM)
26
27 enum {
28 LOW_POWER,
29 NORMAL,
30 LOW_NOISE,
31 ULTRA_LOW_NOISE,
32 NUM_MEASURE_MODE
33 };
34
35 struct icp10125_data {
36 uint16_t raw_ambient_temp;
37 uint32_t raw_press;
38 float sensor_constants[4];
39 };
40
41 struct icp10125_dev_config {
42 struct i2c_dt_spec i2c;
43 uint8_t ambient_temp_mode;
44 uint8_t press_mode;
45 };
46
47 struct icp10125_cmd {
48 uint8_t data[2];
49 };
50
51 struct icp10125_sensor_data {
52 uint8_t data[2];
53 uint8_t crc;
54 };
55
56 struct icp10125_otp_read_setup {
57 struct icp10125_cmd cmd;
58 uint8_t data[3];
59 } __packed __aligned(1);
60
61 /* ambient temperature measurement command for each mode.
62 * (Section 5.2 MEASUREMENT COMMANDS in the Datasheet)
63 */
64 static const struct icp10125_cmd ambient_temp_measurement_cmds[] = {
65 {{0x60, 0x9C}}, {{0x68, 0x25}}, {{0x70, 0xDF}}, {{0x78, 0x66}}
66 };
67
68 /* pressure measurement command for each mode.
69 * (Section 5.2 MEASUREMENT COMMANDS in the Datasheet)
70 */
71 static const struct icp10125_cmd press_measurement_cmds[] = {
72 {{0x40, 0x1A}}, {{0x48, 0xA3}}, {{0x50, 0x59}}, {{0x59, 0xE0}}
73 };
74
75 /* Request preparation for OTP data read. It should issue before data read request.
76 * (Section 5.2 MEASUREMENT COMMANDS in the Datasheet)
77 */
78 static const struct icp10125_otp_read_setup otp_read_setup = {
79 .cmd = {{0xC5, 0x95}},
80 .data = {0x00, 0x66, 0x9C}
81 };
82
83 /* OTP data read request.
84 * After issue this command 2byte x 4 sensor constant value can readable.
85 */
86 static const struct icp10125_cmd otp_read_request_cmd = {{0xC7, 0xF7}};
87
88 /* The max conversion time for each modes.
89 * (Section 2.2 OPERATION MODES in the Datasheet)
90 */
91 static const uint32_t conv_time_max[] = {1800, 6300, 23800, 94500};
92
93 /* The typical conversion time for each modes.
94 * (Section 2.2 OPERATION MODES in the Datasheet)
95 */
96 static const uint32_t conv_time_typ[] = {1600, 5600, 20800, 83200};
97
98 /* The Datasheet has no mention of the constants and formulas.
99 * Instead, it shows only how to use it in the sample code.
100 * Since there is no detailed description in the ICP10125 product manual,
101 * the calculation of the pressure implements is the same as shown in
102 * the 5.11 SAMPLE CODE: EXAMPLE C SYNTAX
103 */
104
icp10125_calculate_conversion_constants(const float * p_LUT,float * A,float * B,float * C)105 static void icp10125_calculate_conversion_constants(const float *p_LUT, float *A, float *B,
106 float *C)
107 {
108 const float p_Pa[] = {45000.0, 80000.0, 105000.0};
109
110 *C = (p_LUT[0] * p_LUT[1] * (p_Pa[0] - p_Pa[1]) +
111 p_LUT[1] * p_LUT[2] * (p_Pa[1] - p_Pa[2]) +
112 p_LUT[2] * p_LUT[0] * (p_Pa[2] - p_Pa[0])) /
113 (p_LUT[2] * (p_Pa[0] - p_Pa[1]) + p_LUT[0] * (p_Pa[1] - p_Pa[2]) +
114 p_LUT[1] * (p_Pa[2] - p_Pa[0]));
115 *A = (p_Pa[0] * p_LUT[0] - p_Pa[1] * p_LUT[1] - (p_Pa[1] - p_Pa[0]) * (*C)) /
116 (p_LUT[0] - p_LUT[1]);
117 *B = (p_Pa[0] - (*A)) * (p_LUT[0] + (*C));
118 }
119
icp10125_calc_calibrated_ambient_temp(const struct icp10125_data * data)120 static float icp10125_calc_calibrated_ambient_temp(const struct icp10125_data *data)
121 {
122 return -45.f + 175.f / 65536.f * data->raw_ambient_temp;
123 }
124
icp10125_calc_calibrated_press(const struct icp10125_data * data)125 static float icp10125_calc_calibrated_press(const struct icp10125_data *data)
126 {
127 const float quadr_factor = 1 / 16777216.0;
128 const float offst_factor = 2048.0;
129 const float LUT_lower = 3.5 * (1 << 20);
130 const float LUT_upper = 11.5 * (1 << 20);
131 float t;
132 float in[3];
133 float A, B, C;
134
135 t = data->raw_ambient_temp - 32768.f;
136 in[0] = LUT_lower + (data->sensor_constants[0] * t * t) * quadr_factor;
137 in[1] = offst_factor * data->sensor_constants[3] +
138 (data->sensor_constants[1] * t * t) * quadr_factor;
139 in[2] = LUT_upper + (data->sensor_constants[2] * t * t) * quadr_factor;
140 icp10125_calculate_conversion_constants(in, &A, &B, &C);
141
142 return A + B / (C + data->raw_press);
143 }
144
145 /* End of porting the 5.11 SAMPLE CODE: EXAMPLE C SYNTAX */
146
icp10125_read_otp(const struct device * dev)147 static int icp10125_read_otp(const struct device *dev)
148 {
149 struct icp10125_data *data = dev->data;
150 struct icp10125_sensor_data sensor_data;
151
152 const struct icp10125_dev_config *cfg = dev->config;
153 int rc = 0;
154
155 rc = i2c_write_dt(&cfg->i2c, (uint8_t *)&otp_read_setup, sizeof(otp_read_setup));
156 if (rc < 0) {
157 LOG_ERR("Failed to write otp_read_setup.\n");
158 return rc;
159 }
160
161 for (size_t i = 0; i < ARRAY_SIZE(data->sensor_constants); i++) {
162 rc = i2c_write_dt(&cfg->i2c, (uint8_t *)&otp_read_request_cmd,
163 sizeof(otp_read_request_cmd));
164 if (rc < 0) {
165 LOG_ERR("Failed to write otp_read_request.\n");
166 return rc;
167 }
168
169 rc = i2c_read_dt(&cfg->i2c, (uint8_t *)&sensor_data, sizeof(sensor_data));
170 if (rc < 0) {
171 LOG_ERR("Failed to read otp_read_request.\n");
172 return rc;
173 }
174
175 data->sensor_constants[i] = sys_get_be16(sensor_data.data);
176 }
177
178 return 0;
179 }
180
181 #ifdef CONFIG_ICP10125_CHECK_CRC
icp10125_check_crc(const uint8_t * data,const size_t len)182 static int icp10125_check_crc(const uint8_t *data, const size_t len)
183 {
184 /* Details of CRC are described in Chapter 5 Section 8 of the product
185 * specifications.
186 */
187 return crc8(data, len, CRC_POLY, 0xFF, false);
188 }
189 #endif
190
icp10125_measure(const struct i2c_dt_spec * i2c,const struct icp10125_cmd * cmds,const uint8_t mode,struct icp10125_sensor_data * sensor_data,const size_t data_num)191 static int icp10125_measure(const struct i2c_dt_spec *i2c, const struct icp10125_cmd *cmds,
192 const uint8_t mode, struct icp10125_sensor_data *sensor_data,
193 const size_t data_num)
194 {
195 int rc = 0;
196
197 rc = i2c_write_dt(i2c, (uint8_t *)&cmds[mode], sizeof(cmds[mode]));
198 if (rc < 0) {
199 LOG_ERR("Failed to start measurement.\n");
200 return rc;
201 }
202
203 /* Wait for the sensor to become readable.
204 * First wait for the typical time and then read.
205 * If that fails, wait until the time to surely became readable.
206 */
207 k_sleep(K_USEC(conv_time_typ[mode]));
208 if (i2c_read_dt(i2c, (uint8_t *)sensor_data, sizeof(sensor_data[0]) * data_num) < 0) {
209 k_sleep(K_USEC(conv_time_max[mode] - conv_time_typ[mode]));
210 rc = i2c_read_dt(i2c, (uint8_t *)sensor_data, sizeof(sensor_data[0]) * data_num);
211 if (rc < 0) {
212 LOG_ERR("Failed to read measurement.\n");
213 return rc;
214 }
215 }
216
217 #ifdef CONFIG_ICP10125_CHECK_CRC
218 /* Calculate CRC from Chapter 5 Section 8 of ICP10125 Product manuals. */
219 for (size_t i = 0; i < data_num; i++) {
220 if (!icp10125_check_crc(sensor_data[i].data, SENSOR_DATA_SIZE)) {
221 LOG_ERR("Sensor data has invalid CRC.\n");
222 return -EIO;
223 }
224 }
225 #endif /* CONFIG_ICP10125_CHECK_CRC */
226
227 return 0;
228 }
229
icp10125_sample_fetch(const struct device * dev,const enum sensor_channel chan)230 static int icp10125_sample_fetch(const struct device *dev, const enum sensor_channel chan)
231 {
232 struct icp10125_data *data = dev->data;
233 const struct icp10125_dev_config *cfg = dev->config;
234 uint8_t endian_conversion[3];
235 struct icp10125_sensor_data sensor_data[PRESS_AND_AMBIENT_TEMP_DATA_NUM] = {0};
236 int rc = 0;
237
238 if (!(chan == SENSOR_CHAN_AMBIENT_TEMP || chan == SENSOR_CHAN_PRESS ||
239 chan == SENSOR_CHAN_ALL)) {
240 return -ENOTSUP;
241 }
242
243 if (chan == SENSOR_CHAN_AMBIENT_TEMP) {
244 rc = icp10125_measure(&cfg->i2c, ambient_temp_measurement_cmds,
245 cfg->ambient_temp_mode, sensor_data, AMBIENT_TEMP_DATA_NUM);
246 if (rc < 0) {
247 return rc;
248 }
249
250 data->raw_ambient_temp = sys_get_be16(sensor_data[0].data);
251 } else {
252 rc = icp10125_measure(&cfg->i2c, press_measurement_cmds, cfg->press_mode,
253 sensor_data, PRESS_AND_AMBIENT_TEMP_DATA_NUM);
254 if (rc < 0) {
255 return rc;
256 }
257
258 endian_conversion[0] = sensor_data[0].data[0];
259 endian_conversion[1] = sensor_data[0].data[1];
260 endian_conversion[2] = sensor_data[1].data[0];
261 data->raw_press = sys_get_be24(endian_conversion);
262 data->raw_ambient_temp = sys_get_be16(sensor_data[2].data);
263 }
264
265 return 0;
266 }
267
icp10125_convert_press_value(struct icp10125_data * data,struct sensor_value * val)268 static void icp10125_convert_press_value(struct icp10125_data *data, struct sensor_value *val)
269 {
270 sensor_value_from_float(val, icp10125_calc_calibrated_press(data) / 1000.f);
271 }
272
icp10125_convert_ambient_temp_value(struct icp10125_data * data,struct sensor_value * val)273 static void icp10125_convert_ambient_temp_value(struct icp10125_data *data,
274 struct sensor_value *val)
275 {
276 sensor_value_from_float(val, icp10125_calc_calibrated_ambient_temp(data));
277 }
278
icp10125_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)279 static int icp10125_channel_get(const struct device *dev, enum sensor_channel chan,
280 struct sensor_value *val)
281 {
282 struct icp10125_data *data = dev->data;
283
284 if (chan == SENSOR_CHAN_AMBIENT_TEMP) {
285 icp10125_convert_ambient_temp_value(data, val);
286 } else if (chan == SENSOR_CHAN_PRESS) {
287 icp10125_convert_press_value(data, val);
288 } else {
289 return -ENOTSUP;
290 }
291
292 return 0;
293 }
294
icp10125_init(const struct device * dev)295 static int icp10125_init(const struct device *dev)
296 {
297 int rc = icp10125_read_otp(dev);
298
299 if (rc < 0) {
300 return rc;
301 }
302
303 return 0;
304 }
305
306 static DEVICE_API(sensor, icp10125_api_funcs) = {
307 .sample_fetch = icp10125_sample_fetch,
308 .channel_get = icp10125_channel_get,
309 };
310
311 #define ICP10125_DEFINE(inst) \
312 static struct icp10125_data icp10125_drv_##inst; \
313 static const struct icp10125_dev_config icp10125_config_##inst = { \
314 .i2c = I2C_DT_SPEC_INST_GET(inst), \
315 .ambient_temp_mode = DT_INST_ENUM_IDX(inst, temperature_measurement_mode), \
316 .press_mode = DT_INST_ENUM_IDX(inst, pressure_measurement_mode)}; \
317 DEVICE_DT_INST_DEFINE(inst, icp10125_init, NULL, &icp10125_drv_##inst, \
318 &icp10125_config_##inst, POST_KERNEL, CONFIG_SENSOR_INIT_PRIORITY, \
319 &icp10125_api_funcs);
320
321 DT_INST_FOREACH_STATUS_OKAY(ICP10125_DEFINE)
322