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