1 /*
2 * Copyright (c) 2020, Laird Connectivity
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT honeywell_sm351lt
8
9 #include "sm351lt.h"
10
11 #include <zephyr/init.h>
12 #include <zephyr/sys/byteorder.h>
13 #include <zephyr/logging/log.h>
14 #include <stdio.h>
15 #include <zephyr/sys/util.h>
16 #include <zephyr/drivers/gpio.h>
17 #include <zephyr/drivers/sensor.h>
18 #include <string.h>
19
20 LOG_MODULE_REGISTER(SM351LT, CONFIG_SENSOR_LOG_LEVEL);
21
22 #if CONFIG_SM351LT_TRIGGER
sm351lt_trigger_set(const struct device * dev,const struct sensor_trigger * trig,sensor_trigger_handler_t handler)23 static int sm351lt_trigger_set(const struct device *dev,
24 const struct sensor_trigger *trig,
25 sensor_trigger_handler_t handler)
26 {
27 const struct sm351lt_config *const config = dev->config;
28 struct sm351lt_data *data = dev->data;
29 int ret = -ENOTSUP;
30
31 if (trig->chan == SENSOR_CHAN_PROX) {
32 data->changed_handler = handler;
33 data->changed_trigger = trig;
34 ret = gpio_pin_interrupt_configure_dt(&config->int_gpio,
35 (handler ? data->trigger_type
36 : GPIO_INT_DISABLE));
37
38 if (ret < 0) {
39 return ret;
40 }
41
42 if (handler) {
43 ret = gpio_add_callback(config->int_gpio.port,
44 &data->gpio_cb);
45 } else {
46 ret = gpio_remove_callback(config->int_gpio.port,
47 &data->gpio_cb);
48 }
49 }
50
51 return ret;
52 }
53
sm351lt_gpio_callback(const struct device * dev,struct gpio_callback * cb,uint32_t pins)54 static void sm351lt_gpio_callback(const struct device *dev,
55 struct gpio_callback *cb, uint32_t pins)
56 {
57 struct sm351lt_data *data =
58 CONTAINER_OF(cb, struct sm351lt_data, gpio_cb);
59
60 #if defined(CONFIG_SM351LT_TRIGGER_OWN_THREAD)
61 k_sem_give(&data->gpio_sem);
62 #elif defined(CONFIG_SM351LT_TRIGGER_GLOBAL_THREAD)
63 k_work_submit(&data->work);
64 #endif
65 }
66
sm351lt_thread_cb(const struct device * dev)67 static void sm351lt_thread_cb(const struct device *dev)
68 {
69 struct sm351lt_data *data = dev->data;
70
71 if (likely(data->changed_handler != NULL)) {
72 data->changed_handler(dev, data->changed_trigger);
73 }
74
75 return;
76 }
77
78 #if defined(CONFIG_SM351LT_TRIGGER_OWN_THREAD)
sm351lt_thread(void * arg1,void * unused2,void * unused3)79 static void sm351lt_thread(void *arg1, void *unused2, void *unused3)
80 {
81 struct sm351lt_data *data = arg1;
82
83 ARG_UNUSED(unused2);
84 ARG_UNUSED(unused3);
85
86 while (1) {
87 k_sem_take(&data->gpio_sem, K_FOREVER);
88 sm351lt_thread_cb(data->dev);
89 }
90 }
91 #endif
92
93 #if defined(CONFIG_SM351LT_TRIGGER_GLOBAL_THREAD)
sm351lt_work_cb(struct k_work * work)94 static void sm351lt_work_cb(struct k_work *work)
95 {
96 struct sm351lt_data *data =
97 CONTAINER_OF(work, struct sm351lt_data, work);
98
99 sm351lt_thread_cb(data->dev);
100 }
101 #endif
102 #endif
103
sm351lt_sample_fetch(const struct device * dev,enum sensor_channel chan)104 static int sm351lt_sample_fetch(const struct device *dev,
105 enum sensor_channel chan)
106 {
107 const struct sm351lt_config *config = dev->config;
108 struct sm351lt_data *data = dev->data;
109
110 if (chan != SENSOR_CHAN_ALL && chan != SENSOR_CHAN_PROX) {
111 return -ENOTSUP;
112 }
113
114 data->sample_status = gpio_pin_get_dt(&config->int_gpio);
115
116 return 0;
117 }
118
sm351lt_channel_get(const struct device * dev,enum sensor_channel chan,struct sensor_value * val)119 static int sm351lt_channel_get(const struct device *dev,
120 enum sensor_channel chan,
121 struct sensor_value *val)
122 {
123 struct sm351lt_data *data = dev->data;
124
125 if (chan == SENSOR_CHAN_PROX) {
126 val->val1 = data->sample_status;
127 val->val2 = 0;
128 } else {
129 return -ENOTSUP;
130 }
131
132 return 0;
133 }
134
135 #if CONFIG_SM351LT_TRIGGER
sm351lt_attr_set(const struct device * dev,enum sensor_channel chan,enum sensor_attribute attr,const struct sensor_value * val)136 static int sm351lt_attr_set(const struct device *dev,
137 enum sensor_channel chan,
138 enum sensor_attribute attr,
139 const struct sensor_value *val)
140 {
141 struct sm351lt_data *data = dev->data;
142
143 if (chan == SENSOR_CHAN_PROX) {
144 if (attr == SENSOR_ATTR_SM351LT_TRIGGER_TYPE) {
145 /* Interrupt triggering type */
146 data->trigger_type = val->val1;
147 } else {
148 return -ENOTSUP;
149 }
150 } else {
151 return -ENOTSUP;
152 }
153
154 return 0;
155 }
156
sm351lt_attr_get(const struct device * dev,enum sensor_channel chan,enum sensor_attribute attr,struct sensor_value * val)157 static int sm351lt_attr_get(const struct device *dev,
158 enum sensor_channel chan,
159 enum sensor_attribute attr,
160 struct sensor_value *val)
161 {
162 struct sm351lt_data *data = dev->data;
163
164 if (chan == SENSOR_CHAN_PROX) {
165 if (attr == SENSOR_ATTR_SM351LT_TRIGGER_TYPE) {
166 /* Interrupt triggering type */
167 val->val1 = data->trigger_type;
168 val->val2 = 0;
169 } else {
170 return -ENOTSUP;
171 }
172 } else {
173 return -ENOTSUP;
174 }
175
176 return 0;
177 }
178 #endif
179
180 static DEVICE_API(sensor, sm351lt_api_funcs) = {
181 .sample_fetch = sm351lt_sample_fetch,
182 .channel_get = sm351lt_channel_get,
183 #if CONFIG_SM351LT_TRIGGER
184 .attr_set = sm351lt_attr_set,
185 .attr_get = sm351lt_attr_get,
186 .trigger_set = sm351lt_trigger_set,
187 #endif
188 };
189
sm351lt_init(const struct device * dev)190 static int sm351lt_init(const struct device *dev)
191 {
192 const struct sm351lt_config *const config = dev->config;
193 uint32_t ret;
194
195 if (!gpio_is_ready_dt(&config->int_gpio)) {
196 LOG_ERR("GPIO device not ready");
197 return -ENODEV;
198 }
199
200 ret = gpio_pin_configure_dt(&config->int_gpio, GPIO_INPUT);
201 if (ret) {
202 LOG_ERR("failed to configure gpio: %d", ret);
203 return ret;
204 }
205
206 #if defined(CONFIG_SM351LT_TRIGGER)
207 struct sm351lt_data *data = dev->data;
208 data->dev = dev;
209
210 #if defined(CONFIG_SM351LT_TRIGGER_OWN_THREAD)
211 k_sem_init(&data->gpio_sem, 0, K_SEM_MAX_LIMIT);
212
213 k_thread_create(&data->thread, data->thread_stack,
214 CONFIG_SM351LT_THREAD_STACK_SIZE,
215 sm351lt_thread, data, NULL,
216 NULL, K_PRIO_COOP(CONFIG_SM351LT_THREAD_PRIORITY),
217 0, K_NO_WAIT);
218
219 #if defined(CONFIG_THREAD_NAME) && defined(CONFIG_THREAD_MAX_NAME_LEN)
220 /* Sets up thread name as the device name */
221 k_thread_name_set(&data->thread, dev->name);
222 #endif
223
224 #elif defined(CONFIG_SM351LT_TRIGGER_GLOBAL_THREAD)
225 data->work.handler = sm351lt_work_cb;
226 #endif
227
228 data->trigger_type = GPIO_INT_DISABLE;
229
230 ret = gpio_pin_interrupt_configure_dt(&config->int_gpio,
231 GPIO_INT_DISABLE);
232 if (ret) {
233 LOG_ERR("failed to configure gpio interrupt: %d", ret);
234 return ret;
235 }
236
237 /* Setup callback struct but do not add it yet */
238 gpio_init_callback(&data->gpio_cb, sm351lt_gpio_callback,
239 BIT(config->int_gpio.pin));
240 #endif
241
242 return 0;
243 }
244
245 /* Instantiation macros for each individual device. */
246 #define SM351LT_DEFINE(inst) \
247 static struct sm351lt_data sm351lt_data_##inst; \
248 static const struct sm351lt_config sm351lt_config_##inst = { \
249 .int_gpio = GPIO_DT_SPEC_INST_GET(inst, gpios), \
250 }; \
251 \
252 SENSOR_DEVICE_DT_INST_DEFINE(inst, \
253 sm351lt_init, \
254 NULL, \
255 &sm351lt_data_##inst, \
256 &sm351lt_config_##inst, \
257 POST_KERNEL, \
258 CONFIG_SENSOR_INIT_PRIORITY, \
259 &sm351lt_api_funcs);
260
261
262 /* Main instantiation macro for every configured device in DTS. */
263 DT_INST_FOREACH_STATUS_OKAY(SM351LT_DEFINE)
264