xref: /openbmc/linux/drivers/input/touchscreen/colibri-vf50-ts.c (revision c95baf12f5077419db01313ab61c2aac007d40cd)
1*2874c5fdSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
248ead50cSSanchayan Maity /*
348ead50cSSanchayan Maity  * Toradex Colibri VF50 Touchscreen driver
448ead50cSSanchayan Maity  *
548ead50cSSanchayan Maity  * Copyright 2015 Toradex AG
648ead50cSSanchayan Maity  *
748ead50cSSanchayan Maity  * Originally authored by Stefan Agner for 3.0 kernel
848ead50cSSanchayan Maity  */
948ead50cSSanchayan Maity 
1048ead50cSSanchayan Maity #include <linux/delay.h>
1148ead50cSSanchayan Maity #include <linux/err.h>
1248ead50cSSanchayan Maity #include <linux/gpio/consumer.h>
1348ead50cSSanchayan Maity #include <linux/iio/consumer.h>
1448ead50cSSanchayan Maity #include <linux/iio/types.h>
1548ead50cSSanchayan Maity #include <linux/input.h>
1648ead50cSSanchayan Maity #include <linux/interrupt.h>
1748ead50cSSanchayan Maity #include <linux/kernel.h>
1848ead50cSSanchayan Maity #include <linux/module.h>
19ff84dabeSGeert Uytterhoeven #include <linux/of.h>
2048ead50cSSanchayan Maity #include <linux/pinctrl/consumer.h>
2148ead50cSSanchayan Maity #include <linux/platform_device.h>
2248ead50cSSanchayan Maity #include <linux/slab.h>
2348ead50cSSanchayan Maity #include <linux/types.h>
2448ead50cSSanchayan Maity 
2548ead50cSSanchayan Maity #define DRIVER_NAME			"colibri-vf50-ts"
2648ead50cSSanchayan Maity 
2748ead50cSSanchayan Maity #define VF_ADC_MAX			((1 << 12) - 1)
2848ead50cSSanchayan Maity 
2948ead50cSSanchayan Maity #define COLI_TOUCH_MIN_DELAY_US		1000
3048ead50cSSanchayan Maity #define COLI_TOUCH_MAX_DELAY_US		2000
3148ead50cSSanchayan Maity #define COLI_PULLUP_MIN_DELAY_US	10000
3248ead50cSSanchayan Maity #define COLI_PULLUP_MAX_DELAY_US	11000
3348ead50cSSanchayan Maity #define COLI_TOUCH_NO_OF_AVGS		5
3448ead50cSSanchayan Maity #define COLI_TOUCH_REQ_ADC_CHAN		4
3548ead50cSSanchayan Maity 
3648ead50cSSanchayan Maity struct vf50_touch_device {
3748ead50cSSanchayan Maity 	struct platform_device *pdev;
3848ead50cSSanchayan Maity 	struct input_dev *ts_input;
3948ead50cSSanchayan Maity 	struct iio_channel *channels;
4048ead50cSSanchayan Maity 	struct gpio_desc *gpio_xp;
4148ead50cSSanchayan Maity 	struct gpio_desc *gpio_xm;
4248ead50cSSanchayan Maity 	struct gpio_desc *gpio_yp;
4348ead50cSSanchayan Maity 	struct gpio_desc *gpio_ym;
4448ead50cSSanchayan Maity 	int pen_irq;
4548ead50cSSanchayan Maity 	int min_pressure;
4648ead50cSSanchayan Maity 	bool stop_touchscreen;
4748ead50cSSanchayan Maity };
4848ead50cSSanchayan Maity 
4948ead50cSSanchayan Maity /*
5048ead50cSSanchayan Maity  * Enables given plates and measures touch parameters using ADC
5148ead50cSSanchayan Maity  */
adc_ts_measure(struct iio_channel * channel,struct gpio_desc * plate_p,struct gpio_desc * plate_m)5248ead50cSSanchayan Maity static int adc_ts_measure(struct iio_channel *channel,
5348ead50cSSanchayan Maity 			  struct gpio_desc *plate_p, struct gpio_desc *plate_m)
5448ead50cSSanchayan Maity {
5548ead50cSSanchayan Maity 	int i, value = 0, val = 0;
5648ead50cSSanchayan Maity 	int error;
5748ead50cSSanchayan Maity 
5848ead50cSSanchayan Maity 	gpiod_set_value(plate_p, 1);
5948ead50cSSanchayan Maity 	gpiod_set_value(plate_m, 1);
6048ead50cSSanchayan Maity 
6148ead50cSSanchayan Maity 	usleep_range(COLI_TOUCH_MIN_DELAY_US, COLI_TOUCH_MAX_DELAY_US);
6248ead50cSSanchayan Maity 
6348ead50cSSanchayan Maity 	for (i = 0; i < COLI_TOUCH_NO_OF_AVGS; i++) {
6448ead50cSSanchayan Maity 		error = iio_read_channel_raw(channel, &val);
6548ead50cSSanchayan Maity 		if (error < 0) {
6648ead50cSSanchayan Maity 			value = error;
6748ead50cSSanchayan Maity 			goto error_iio_read;
6848ead50cSSanchayan Maity 		}
6948ead50cSSanchayan Maity 
7048ead50cSSanchayan Maity 		value += val;
7148ead50cSSanchayan Maity 	}
7248ead50cSSanchayan Maity 
7348ead50cSSanchayan Maity 	value /= COLI_TOUCH_NO_OF_AVGS;
7448ead50cSSanchayan Maity 
7548ead50cSSanchayan Maity error_iio_read:
7648ead50cSSanchayan Maity 	gpiod_set_value(plate_p, 0);
7748ead50cSSanchayan Maity 	gpiod_set_value(plate_m, 0);
7848ead50cSSanchayan Maity 
7948ead50cSSanchayan Maity 	return value;
8048ead50cSSanchayan Maity }
8148ead50cSSanchayan Maity 
8248ead50cSSanchayan Maity /*
8348ead50cSSanchayan Maity  * Enable touch detection using falling edge detection on XM
8448ead50cSSanchayan Maity  */
vf50_ts_enable_touch_detection(struct vf50_touch_device * vf50_ts)8548ead50cSSanchayan Maity static void vf50_ts_enable_touch_detection(struct vf50_touch_device *vf50_ts)
8648ead50cSSanchayan Maity {
8748ead50cSSanchayan Maity 	/* Enable plate YM (needs to be strong GND, high active) */
8848ead50cSSanchayan Maity 	gpiod_set_value(vf50_ts->gpio_ym, 1);
8948ead50cSSanchayan Maity 
9048ead50cSSanchayan Maity 	/*
9148ead50cSSanchayan Maity 	 * Let the platform mux to idle state in order to enable
9248ead50cSSanchayan Maity 	 * Pull-Up on GPIO
9348ead50cSSanchayan Maity 	 */
9448ead50cSSanchayan Maity 	pinctrl_pm_select_idle_state(&vf50_ts->pdev->dev);
9548ead50cSSanchayan Maity 
9648ead50cSSanchayan Maity 	/* Wait for the pull-up to be stable on high */
9748ead50cSSanchayan Maity 	usleep_range(COLI_PULLUP_MIN_DELAY_US, COLI_PULLUP_MAX_DELAY_US);
9848ead50cSSanchayan Maity }
9948ead50cSSanchayan Maity 
10048ead50cSSanchayan Maity /*
10148ead50cSSanchayan Maity  * ADC touch screen sampling bottom half irq handler
10248ead50cSSanchayan Maity  */
vf50_ts_irq_bh(int irq,void * private)10348ead50cSSanchayan Maity static irqreturn_t vf50_ts_irq_bh(int irq, void *private)
10448ead50cSSanchayan Maity {
10548ead50cSSanchayan Maity 	struct vf50_touch_device *vf50_ts = private;
10648ead50cSSanchayan Maity 	struct device *dev = &vf50_ts->pdev->dev;
10748ead50cSSanchayan Maity 	int val_x, val_y, val_z1, val_z2, val_p = 0;
10848ead50cSSanchayan Maity 	bool discard_val_on_start = true;
10948ead50cSSanchayan Maity 
11048ead50cSSanchayan Maity 	/* Disable the touch detection plates */
11148ead50cSSanchayan Maity 	gpiod_set_value(vf50_ts->gpio_ym, 0);
11248ead50cSSanchayan Maity 
11348ead50cSSanchayan Maity 	/* Let the platform mux to default state in order to mux as ADC */
11448ead50cSSanchayan Maity 	pinctrl_pm_select_default_state(dev);
11548ead50cSSanchayan Maity 
11648ead50cSSanchayan Maity 	while (!vf50_ts->stop_touchscreen) {
11748ead50cSSanchayan Maity 		/* X-Direction */
11848ead50cSSanchayan Maity 		val_x = adc_ts_measure(&vf50_ts->channels[0],
11948ead50cSSanchayan Maity 				vf50_ts->gpio_xp, vf50_ts->gpio_xm);
12048ead50cSSanchayan Maity 		if (val_x < 0)
12148ead50cSSanchayan Maity 			break;
12248ead50cSSanchayan Maity 
12348ead50cSSanchayan Maity 		/* Y-Direction */
12448ead50cSSanchayan Maity 		val_y = adc_ts_measure(&vf50_ts->channels[1],
12548ead50cSSanchayan Maity 				vf50_ts->gpio_yp, vf50_ts->gpio_ym);
12648ead50cSSanchayan Maity 		if (val_y < 0)
12748ead50cSSanchayan Maity 			break;
12848ead50cSSanchayan Maity 
12948ead50cSSanchayan Maity 		/*
13048ead50cSSanchayan Maity 		 * Touch pressure
13148ead50cSSanchayan Maity 		 * Measure on XP/YM
13248ead50cSSanchayan Maity 		 */
13348ead50cSSanchayan Maity 		val_z1 = adc_ts_measure(&vf50_ts->channels[2],
13448ead50cSSanchayan Maity 				vf50_ts->gpio_yp, vf50_ts->gpio_xm);
13548ead50cSSanchayan Maity 		if (val_z1 < 0)
13648ead50cSSanchayan Maity 			break;
13748ead50cSSanchayan Maity 		val_z2 = adc_ts_measure(&vf50_ts->channels[3],
13848ead50cSSanchayan Maity 				vf50_ts->gpio_yp, vf50_ts->gpio_xm);
13948ead50cSSanchayan Maity 		if (val_z2 < 0)
14048ead50cSSanchayan Maity 			break;
14148ead50cSSanchayan Maity 
14248ead50cSSanchayan Maity 		/* Validate signal (avoid calculation using noise) */
14348ead50cSSanchayan Maity 		if (val_z1 > 64 && val_x > 64) {
14448ead50cSSanchayan Maity 			/*
14548ead50cSSanchayan Maity 			 * Calculate resistance between the plates
14648ead50cSSanchayan Maity 			 * lower resistance means higher pressure
14748ead50cSSanchayan Maity 			 */
14848ead50cSSanchayan Maity 			int r_x = (1000 * val_x) / VF_ADC_MAX;
14948ead50cSSanchayan Maity 
15048ead50cSSanchayan Maity 			val_p = (r_x * val_z2) / val_z1 - r_x;
15148ead50cSSanchayan Maity 
15248ead50cSSanchayan Maity 		} else {
15348ead50cSSanchayan Maity 			val_p = 2000;
15448ead50cSSanchayan Maity 		}
15548ead50cSSanchayan Maity 
15648ead50cSSanchayan Maity 		val_p = 2000 - val_p;
15748ead50cSSanchayan Maity 		dev_dbg(dev,
15848ead50cSSanchayan Maity 			"Measured values: x: %d, y: %d, z1: %d, z2: %d, p: %d\n",
15948ead50cSSanchayan Maity 			val_x, val_y, val_z1, val_z2, val_p);
16048ead50cSSanchayan Maity 
16148ead50cSSanchayan Maity 		/*
16248ead50cSSanchayan Maity 		 * If touch pressure is too low, stop measuring and reenable
16348ead50cSSanchayan Maity 		 * touch detection
16448ead50cSSanchayan Maity 		 */
16548ead50cSSanchayan Maity 		if (val_p < vf50_ts->min_pressure || val_p > 2000)
16648ead50cSSanchayan Maity 			break;
16748ead50cSSanchayan Maity 
16848ead50cSSanchayan Maity 		/*
16948ead50cSSanchayan Maity 		 * The pressure may not be enough for the first x and the
17048ead50cSSanchayan Maity 		 * second y measurement, but, the pressure is ok when the
17148ead50cSSanchayan Maity 		 * driver is doing the third and fourth measurement. To
17248ead50cSSanchayan Maity 		 * take care of this, we drop the first measurement always.
17348ead50cSSanchayan Maity 		 */
17448ead50cSSanchayan Maity 		if (discard_val_on_start) {
17548ead50cSSanchayan Maity 			discard_val_on_start = false;
17648ead50cSSanchayan Maity 		} else {
17748ead50cSSanchayan Maity 			/*
17848ead50cSSanchayan Maity 			 * Report touch position and sleep for
17948ead50cSSanchayan Maity 			 * the next measurement.
18048ead50cSSanchayan Maity 			 */
18148ead50cSSanchayan Maity 			input_report_abs(vf50_ts->ts_input,
18248ead50cSSanchayan Maity 					ABS_X, VF_ADC_MAX - val_x);
18348ead50cSSanchayan Maity 			input_report_abs(vf50_ts->ts_input,
18448ead50cSSanchayan Maity 					ABS_Y, VF_ADC_MAX - val_y);
18548ead50cSSanchayan Maity 			input_report_abs(vf50_ts->ts_input,
18648ead50cSSanchayan Maity 					ABS_PRESSURE, val_p);
18748ead50cSSanchayan Maity 			input_report_key(vf50_ts->ts_input, BTN_TOUCH, 1);
18848ead50cSSanchayan Maity 			input_sync(vf50_ts->ts_input);
18948ead50cSSanchayan Maity 		}
19048ead50cSSanchayan Maity 
19148ead50cSSanchayan Maity 		usleep_range(COLI_PULLUP_MIN_DELAY_US,
19248ead50cSSanchayan Maity 			     COLI_PULLUP_MAX_DELAY_US);
19348ead50cSSanchayan Maity 	}
19448ead50cSSanchayan Maity 
19548ead50cSSanchayan Maity 	/* Report no more touch, re-enable touch detection */
19648ead50cSSanchayan Maity 	input_report_abs(vf50_ts->ts_input, ABS_PRESSURE, 0);
19748ead50cSSanchayan Maity 	input_report_key(vf50_ts->ts_input, BTN_TOUCH, 0);
19848ead50cSSanchayan Maity 	input_sync(vf50_ts->ts_input);
19948ead50cSSanchayan Maity 
20048ead50cSSanchayan Maity 	vf50_ts_enable_touch_detection(vf50_ts);
20148ead50cSSanchayan Maity 
20248ead50cSSanchayan Maity 	return IRQ_HANDLED;
20348ead50cSSanchayan Maity }
20448ead50cSSanchayan Maity 
vf50_ts_open(struct input_dev * dev_input)20548ead50cSSanchayan Maity static int vf50_ts_open(struct input_dev *dev_input)
20648ead50cSSanchayan Maity {
20748ead50cSSanchayan Maity 	struct vf50_touch_device *touchdev = input_get_drvdata(dev_input);
20848ead50cSSanchayan Maity 	struct device *dev = &touchdev->pdev->dev;
20948ead50cSSanchayan Maity 
21048ead50cSSanchayan Maity 	dev_dbg(dev, "Input device %s opened, starting touch detection\n",
21148ead50cSSanchayan Maity 		dev_input->name);
21248ead50cSSanchayan Maity 
21348ead50cSSanchayan Maity 	touchdev->stop_touchscreen = false;
21448ead50cSSanchayan Maity 
21548ead50cSSanchayan Maity 	/* Mux detection before request IRQ, wait for pull-up to settle */
21648ead50cSSanchayan Maity 	vf50_ts_enable_touch_detection(touchdev);
21748ead50cSSanchayan Maity 
21848ead50cSSanchayan Maity 	return 0;
21948ead50cSSanchayan Maity }
22048ead50cSSanchayan Maity 
vf50_ts_close(struct input_dev * dev_input)22148ead50cSSanchayan Maity static void vf50_ts_close(struct input_dev *dev_input)
22248ead50cSSanchayan Maity {
22348ead50cSSanchayan Maity 	struct vf50_touch_device *touchdev = input_get_drvdata(dev_input);
22448ead50cSSanchayan Maity 	struct device *dev = &touchdev->pdev->dev;
22548ead50cSSanchayan Maity 
22648ead50cSSanchayan Maity 	touchdev->stop_touchscreen = true;
22748ead50cSSanchayan Maity 
22848ead50cSSanchayan Maity 	/* Make sure IRQ is not running past close */
22948ead50cSSanchayan Maity 	mb();
23048ead50cSSanchayan Maity 	synchronize_irq(touchdev->pen_irq);
23148ead50cSSanchayan Maity 
23248ead50cSSanchayan Maity 	gpiod_set_value(touchdev->gpio_ym, 0);
23348ead50cSSanchayan Maity 	pinctrl_pm_select_default_state(dev);
23448ead50cSSanchayan Maity 
23548ead50cSSanchayan Maity 	dev_dbg(dev, "Input device %s closed, disable touch detection\n",
23648ead50cSSanchayan Maity 		dev_input->name);
23748ead50cSSanchayan Maity }
23848ead50cSSanchayan Maity 
vf50_ts_get_gpiod(struct device * dev,struct gpio_desc ** gpio_d,const char * con_id,enum gpiod_flags flags)23948ead50cSSanchayan Maity static int vf50_ts_get_gpiod(struct device *dev, struct gpio_desc **gpio_d,
24048ead50cSSanchayan Maity 			     const char *con_id, enum gpiod_flags flags)
24148ead50cSSanchayan Maity {
24248ead50cSSanchayan Maity 	int error;
24348ead50cSSanchayan Maity 
24448ead50cSSanchayan Maity 	*gpio_d = devm_gpiod_get(dev, con_id, flags);
24548ead50cSSanchayan Maity 	if (IS_ERR(*gpio_d)) {
24648ead50cSSanchayan Maity 		error = PTR_ERR(*gpio_d);
24748ead50cSSanchayan Maity 		dev_err(dev, "Could not get gpio_%s %d\n", con_id, error);
24848ead50cSSanchayan Maity 		return error;
24948ead50cSSanchayan Maity 	}
25048ead50cSSanchayan Maity 
25148ead50cSSanchayan Maity 	return 0;
25248ead50cSSanchayan Maity }
25348ead50cSSanchayan Maity 
vf50_ts_channel_release(void * data)25448ead50cSSanchayan Maity static void vf50_ts_channel_release(void *data)
25548ead50cSSanchayan Maity {
25648ead50cSSanchayan Maity 	struct iio_channel *channels = data;
25748ead50cSSanchayan Maity 
25848ead50cSSanchayan Maity 	iio_channel_release_all(channels);
25948ead50cSSanchayan Maity }
26048ead50cSSanchayan Maity 
vf50_ts_probe(struct platform_device * pdev)26148ead50cSSanchayan Maity static int vf50_ts_probe(struct platform_device *pdev)
26248ead50cSSanchayan Maity {
26348ead50cSSanchayan Maity 	struct input_dev *input;
26448ead50cSSanchayan Maity 	struct iio_channel *channels;
26548ead50cSSanchayan Maity 	struct device *dev = &pdev->dev;
26648ead50cSSanchayan Maity 	struct vf50_touch_device *touchdev;
26748ead50cSSanchayan Maity 	int num_adc_channels;
26848ead50cSSanchayan Maity 	int error;
26948ead50cSSanchayan Maity 
27048ead50cSSanchayan Maity 	channels = iio_channel_get_all(dev);
27148ead50cSSanchayan Maity 	if (IS_ERR(channels))
27248ead50cSSanchayan Maity 		return PTR_ERR(channels);
27348ead50cSSanchayan Maity 
27448ead50cSSanchayan Maity 	error = devm_add_action(dev, vf50_ts_channel_release, channels);
27548ead50cSSanchayan Maity 	if (error) {
27648ead50cSSanchayan Maity 		iio_channel_release_all(channels);
27748ead50cSSanchayan Maity 		dev_err(dev, "Failed to register iio channel release action");
27848ead50cSSanchayan Maity 		return error;
27948ead50cSSanchayan Maity 	}
28048ead50cSSanchayan Maity 
28148ead50cSSanchayan Maity 	num_adc_channels = 0;
28248ead50cSSanchayan Maity 	while (channels[num_adc_channels].indio_dev)
28348ead50cSSanchayan Maity 		num_adc_channels++;
28448ead50cSSanchayan Maity 
28548ead50cSSanchayan Maity 	if (num_adc_channels != COLI_TOUCH_REQ_ADC_CHAN) {
28648ead50cSSanchayan Maity 		dev_err(dev, "Inadequate ADC channels specified\n");
28748ead50cSSanchayan Maity 		return -EINVAL;
28848ead50cSSanchayan Maity 	}
28948ead50cSSanchayan Maity 
29048ead50cSSanchayan Maity 	touchdev = devm_kzalloc(dev, sizeof(*touchdev), GFP_KERNEL);
29148ead50cSSanchayan Maity 	if (!touchdev)
29248ead50cSSanchayan Maity 		return -ENOMEM;
29348ead50cSSanchayan Maity 
29448ead50cSSanchayan Maity 	touchdev->pdev = pdev;
29548ead50cSSanchayan Maity 	touchdev->channels = channels;
29648ead50cSSanchayan Maity 
29748ead50cSSanchayan Maity 	error = of_property_read_u32(dev->of_node, "vf50-ts-min-pressure",
29848ead50cSSanchayan Maity 				 &touchdev->min_pressure);
29948ead50cSSanchayan Maity 	if (error)
30048ead50cSSanchayan Maity 		return error;
30148ead50cSSanchayan Maity 
30248ead50cSSanchayan Maity 	input = devm_input_allocate_device(dev);
30348ead50cSSanchayan Maity 	if (!input) {
30448ead50cSSanchayan Maity 		dev_err(dev, "Failed to allocate TS input device\n");
30548ead50cSSanchayan Maity 		return -ENOMEM;
30648ead50cSSanchayan Maity 	}
30748ead50cSSanchayan Maity 
30848ead50cSSanchayan Maity 	input->name = DRIVER_NAME;
30948ead50cSSanchayan Maity 	input->id.bustype = BUS_HOST;
31048ead50cSSanchayan Maity 	input->dev.parent = dev;
31148ead50cSSanchayan Maity 	input->open = vf50_ts_open;
31248ead50cSSanchayan Maity 	input->close = vf50_ts_close;
31348ead50cSSanchayan Maity 
31448ead50cSSanchayan Maity 	input_set_capability(input, EV_KEY, BTN_TOUCH);
31548ead50cSSanchayan Maity 	input_set_abs_params(input, ABS_X, 0, VF_ADC_MAX, 0, 0);
31648ead50cSSanchayan Maity 	input_set_abs_params(input, ABS_Y, 0, VF_ADC_MAX, 0, 0);
31748ead50cSSanchayan Maity 	input_set_abs_params(input, ABS_PRESSURE, 0, VF_ADC_MAX, 0, 0);
31848ead50cSSanchayan Maity 
31948ead50cSSanchayan Maity 	touchdev->ts_input = input;
32048ead50cSSanchayan Maity 	input_set_drvdata(input, touchdev);
32148ead50cSSanchayan Maity 
32248ead50cSSanchayan Maity 	error = input_register_device(input);
32348ead50cSSanchayan Maity 	if (error) {
32448ead50cSSanchayan Maity 		dev_err(dev, "Failed to register input device\n");
32548ead50cSSanchayan Maity 		return error;
32648ead50cSSanchayan Maity 	}
32748ead50cSSanchayan Maity 
32848ead50cSSanchayan Maity 	error = vf50_ts_get_gpiod(dev, &touchdev->gpio_xp, "xp", GPIOD_OUT_LOW);
32948ead50cSSanchayan Maity 	if (error)
33048ead50cSSanchayan Maity 		return error;
33148ead50cSSanchayan Maity 
33248ead50cSSanchayan Maity 	error = vf50_ts_get_gpiod(dev, &touchdev->gpio_xm,
33348ead50cSSanchayan Maity 				"xm", GPIOD_OUT_LOW);
33448ead50cSSanchayan Maity 	if (error)
33548ead50cSSanchayan Maity 		return error;
33648ead50cSSanchayan Maity 
33748ead50cSSanchayan Maity 	error = vf50_ts_get_gpiod(dev, &touchdev->gpio_yp, "yp", GPIOD_OUT_LOW);
33848ead50cSSanchayan Maity 	if (error)
33948ead50cSSanchayan Maity 		return error;
34048ead50cSSanchayan Maity 
34148ead50cSSanchayan Maity 	error = vf50_ts_get_gpiod(dev, &touchdev->gpio_ym, "ym", GPIOD_OUT_LOW);
34248ead50cSSanchayan Maity 	if (error)
34348ead50cSSanchayan Maity 		return error;
34448ead50cSSanchayan Maity 
34548ead50cSSanchayan Maity 	touchdev->pen_irq = platform_get_irq(pdev, 0);
34648ead50cSSanchayan Maity 	if (touchdev->pen_irq < 0)
34748ead50cSSanchayan Maity 		return touchdev->pen_irq;
34848ead50cSSanchayan Maity 
34948ead50cSSanchayan Maity 	error = devm_request_threaded_irq(dev, touchdev->pen_irq,
35048ead50cSSanchayan Maity 					  NULL, vf50_ts_irq_bh, IRQF_ONESHOT,
35148ead50cSSanchayan Maity 					  "vf50 touch", touchdev);
35248ead50cSSanchayan Maity 	if (error) {
35348ead50cSSanchayan Maity 		dev_err(dev, "Failed to request IRQ %d: %d\n",
35448ead50cSSanchayan Maity 			touchdev->pen_irq, error);
35548ead50cSSanchayan Maity 		return error;
35648ead50cSSanchayan Maity 	}
35748ead50cSSanchayan Maity 
35848ead50cSSanchayan Maity 	return 0;
35948ead50cSSanchayan Maity }
36048ead50cSSanchayan Maity 
36148ead50cSSanchayan Maity static const struct of_device_id vf50_touch_of_match[] = {
36248ead50cSSanchayan Maity 	{ .compatible = "toradex,vf50-touchscreen", },
36348ead50cSSanchayan Maity 	{ }
36448ead50cSSanchayan Maity };
36548ead50cSSanchayan Maity MODULE_DEVICE_TABLE(of, vf50_touch_of_match);
36648ead50cSSanchayan Maity 
36748ead50cSSanchayan Maity static struct platform_driver vf50_touch_driver = {
36848ead50cSSanchayan Maity 	.driver = {
36948ead50cSSanchayan Maity 		.name = "toradex,vf50_touchctrl",
37048ead50cSSanchayan Maity 		.of_match_table = vf50_touch_of_match,
37148ead50cSSanchayan Maity 	},
37248ead50cSSanchayan Maity 	.probe = vf50_ts_probe,
37348ead50cSSanchayan Maity };
37448ead50cSSanchayan Maity module_platform_driver(vf50_touch_driver);
37548ead50cSSanchayan Maity 
37648ead50cSSanchayan Maity MODULE_AUTHOR("Sanchayan Maity");
37748ead50cSSanchayan Maity MODULE_DESCRIPTION("Colibri VF50 Touchscreen driver");
37848ead50cSSanchayan Maity MODULE_LICENSE("GPL");
379