summaryrefslogtreecommitdiff
path: root/app/drivers/zephyr/ec11_trigger.c
blob: a04a5610e5d6697c128cb35301dc4780984da1e4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
/*
 * Copyright (c) 2020 The ZMK Contributors
 *
 * SPDX-License-Identifier: MIT
 */

#define DT_DRV_COMPAT alps_ec11

#include <device.h>
#include <drivers/gpio.h>
#include <sys/util.h>
#include <kernel.h>
#include <drivers/sensor.h>

#include "ec11.h"

extern struct ec11_data ec11_driver;

#include <logging/log.h>
LOG_MODULE_DECLARE(EC11, CONFIG_SENSOR_LOG_LEVEL);

static inline void setup_int(struct device *dev,
			     bool enable)
{
	struct ec11_data *data = dev->driver_data;
	const struct ec11_config *cfg = dev->config_info;

	LOG_DBG("enabled %s", (enable ? "true" : "false"));

	if (gpio_pin_interrupt_configure(data->a,
				     cfg->a_pin,
				     enable
				     ? GPIO_INT_EDGE_BOTH
				     : GPIO_INT_DISABLE)) {
						 LOG_WRN("Unable to set A pin GPIO interrupt");
					 }

	if (gpio_pin_interrupt_configure(data->b,
				     cfg->b_pin,
				     enable
				     ? GPIO_INT_EDGE_BOTH
				     : GPIO_INT_DISABLE)) {
						 LOG_WRN("Unable to set A pin GPIO interrupt");
					 }
}

static void ec11_a_gpio_callback(struct device *dev,
				 struct gpio_callback *cb, u32_t pins)
{
	struct ec11_data *drv_data =
		CONTAINER_OF(cb, struct ec11_data, a_gpio_cb);

	LOG_DBG("");

	setup_int(drv_data->dev, false);

#if defined(CONFIG_EC11_TRIGGER_OWN_THREAD)
	k_sem_give(&drv_data->gpio_sem);
#elif defined(CONFIG_EC11_TRIGGER_GLOBAL_THREAD)
	k_work_submit(&drv_data->work);
#endif
}

static void ec11_b_gpio_callback(struct device *dev,
				 struct gpio_callback *cb, u32_t pins)
{
	struct ec11_data *drv_data =
		CONTAINER_OF(cb, struct ec11_data, b_gpio_cb);

	LOG_DBG("");

	setup_int(drv_data->dev, false);

#if defined(CONFIG_EC11_TRIGGER_OWN_THREAD)
	k_sem_give(&drv_data->gpio_sem);
#elif defined(CONFIG_EC11_TRIGGER_GLOBAL_THREAD)
	k_work_submit(&drv_data->work);
#endif
}

static void ec11_thread_cb(void *arg)
{
	struct device *dev = arg;
	struct ec11_data *drv_data = dev->driver_data;

	drv_data->handler(dev, drv_data->trigger);

	setup_int(dev, true);
}

#ifdef CONFIG_EC11_TRIGGER_OWN_THREAD
static void ec11_thread(int dev_ptr, int unused)
{
	struct device *dev = INT_TO_POINTER(dev_ptr);
	struct ec11_data *drv_data = dev->driver_data;

	ARG_UNUSED(unused);

	while (1) {
		k_sem_take(&drv_data->gpio_sem, K_FOREVER);
		ec11_thread_cb(dev);
	}
}
#endif

#ifdef CONFIG_EC11_TRIGGER_GLOBAL_THREAD
static void ec11_work_cb(struct k_work *work)
{
	struct ec11_data *drv_data =
		CONTAINER_OF(work, struct ec11_data, work);

	LOG_DBG("");

	ec11_thread_cb(drv_data->dev);
}
#endif

int ec11_trigger_set(struct device *dev,
		       const struct sensor_trigger *trig,
		       sensor_trigger_handler_t handler)
{
	struct ec11_data *drv_data = dev->driver_data;

	setup_int(dev, false);

	k_msleep(5);

	drv_data->trigger = trig;
	drv_data->handler = handler;

	setup_int(dev, true);

	return 0;
}

int ec11_init_interrupt(struct device *dev)
{
	struct ec11_data *drv_data = dev->driver_data;
	const struct ec11_config *drv_cfg = dev->config_info;

	drv_data->dev = dev;
	/* setup gpio interrupt */


	gpio_init_callback(&drv_data->a_gpio_cb,
			   ec11_a_gpio_callback,
			   BIT(drv_cfg->a_pin));

	if (gpio_add_callback(drv_data->a, &drv_data->a_gpio_cb) < 0) {
		LOG_DBG("Failed to set A callback!");
		return -EIO;
	}

	gpio_init_callback(&drv_data->b_gpio_cb,
			   ec11_b_gpio_callback,
			   BIT(drv_cfg->b_pin));

	if (gpio_add_callback(drv_data->b, &drv_data->b_gpio_cb) < 0) {
		LOG_DBG("Failed to set B callback!");
		return -EIO;
	}

#if defined(CONFIG_EC11_TRIGGER_OWN_THREAD)
	k_sem_init(&drv_data->gpio_sem, 0, UINT_MAX);

	k_thread_create(&drv_data->thread, drv_data->thread_stack,
			CONFIG_EC11_THREAD_STACK_SIZE,
			(k_thread_entry_t)ec11_thread, dev,
			0, NULL, K_PRIO_COOP(CONFIG_EC11_THREAD_PRIORITY),
			0, K_NO_WAIT);
#elif defined(CONFIG_EC11_TRIGGER_GLOBAL_THREAD)
	k_work_init(&drv_data->work, ec11_work_cb);
#endif

	return 0;
}