blob: bb6f2fe1466728a8dc41e77b9f633b85a43404e1 [file] [log] [blame]
Sebastian Reichelb98abe52014-05-28 23:51:53 -07001/*
2 * Generic DT helper functions for touchscreen devices
3 *
4 * Copyright (c) 2014 Sebastian Reichel <sre@kernel.org>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 */
11
Dmitry Torokhov4200e832015-07-06 15:18:24 -070012#include <linux/property.h>
Sebastian Reichelb98abe52014-05-28 23:51:53 -070013#include <linux/input.h>
Maxime Ripard0a363a32015-03-21 20:17:57 -070014#include <linux/input/mt.h>
Sebastian Reichelb98abe52014-05-28 23:51:53 -070015#include <linux/input/touchscreen.h>
16
Dmitry Torokhov4200e832015-07-06 15:18:24 -070017static bool touchscreen_get_prop_u32(struct device *dev,
Dmitry Torokhov7c494372015-06-01 10:35:16 -070018 const char *property,
19 unsigned int default_value,
20 unsigned int *value)
Maxime Ripard3eea8b52015-03-21 20:17:48 -070021{
Dmitry Torokhov7c494372015-06-01 10:35:16 -070022 u32 val;
23 int error;
Maxime Ripard3eea8b52015-03-21 20:17:48 -070024
Dmitry Torokhov4200e832015-07-06 15:18:24 -070025 error = device_property_read_u32(dev, property, &val);
Dmitry Torokhov7c494372015-06-01 10:35:16 -070026 if (error) {
27 *value = default_value;
28 return false;
29 }
Maxime Ripard3eea8b52015-03-21 20:17:48 -070030
Dmitry Torokhov7c494372015-06-01 10:35:16 -070031 *value = val;
32 return true;
Maxime Ripard3eea8b52015-03-21 20:17:48 -070033}
34
35static void touchscreen_set_params(struct input_dev *dev,
36 unsigned long axis,
37 int max, int fuzz)
38{
39 struct input_absinfo *absinfo;
40
41 if (!test_bit(axis, dev->absbit)) {
Dmitry Torokhovf61fd212015-07-06 14:45:25 -070042 dev_warn(&dev->dev,
43 "DT specifies parameters but the axis %lu is not set up\n",
44 axis);
Maxime Ripard3eea8b52015-03-21 20:17:48 -070045 return;
46 }
47
48 absinfo = &dev->absinfo[axis];
49 absinfo->maximum = max;
50 absinfo->fuzz = fuzz;
51}
52
Sebastian Reichelb98abe52014-05-28 23:51:53 -070053/**
Dmitry Torokhov4200e832015-07-06 15:18:24 -070054 * touchscreen_parse_properties - parse common touchscreen DT properties
55 * @input: input device that should be parsed
56 * @multitouch: specifies whether parsed properties should be applied to
57 * single-touch or multi-touch axes
Sebastian Reichelb98abe52014-05-28 23:51:53 -070058 *
59 * This function parses common DT properties for touchscreens and setups the
Dmitry Torokhov4200e832015-07-06 15:18:24 -070060 * input device accordingly. The function keeps previously set up default
Sebastian Reichelb98abe52014-05-28 23:51:53 -070061 * values if no value is specified via DT.
62 */
Dmitry Torokhov4200e832015-07-06 15:18:24 -070063void touchscreen_parse_properties(struct input_dev *input, bool multitouch)
Sebastian Reichelb98abe52014-05-28 23:51:53 -070064{
Dmitry Torokhov4200e832015-07-06 15:18:24 -070065 struct device *dev = input->dev.parent;
Dmitry Torokhov7c494372015-06-01 10:35:16 -070066 unsigned int axis;
67 unsigned int maximum, fuzz;
68 bool data_present;
Sebastian Reichelb98abe52014-05-28 23:51:53 -070069
Dmitry Torokhov4200e832015-07-06 15:18:24 -070070 input_alloc_absinfo(input);
71 if (!input->absinfo)
Sebastian Reichelb98abe52014-05-28 23:51:53 -070072 return;
73
Dmitry Torokhov7c494372015-06-01 10:35:16 -070074 axis = multitouch ? ABS_MT_POSITION_X : ABS_X;
Dmitry Torokhov4200e832015-07-06 15:18:24 -070075 data_present = touchscreen_get_prop_u32(dev, "touchscreen-size-x",
76 input_abs_get_max(input,
Dmitry Torokhov51717862015-07-06 14:57:54 -070077 axis) + 1,
Dmitry Torokhov7c494372015-06-01 10:35:16 -070078 &maximum) |
Dmitry Torokhov4200e832015-07-06 15:18:24 -070079 touchscreen_get_prop_u32(dev, "touchscreen-fuzz-x",
80 input_abs_get_fuzz(input, axis),
Dmitry Torokhov7c494372015-06-01 10:35:16 -070081 &fuzz);
82 if (data_present)
Dmitry Torokhov4200e832015-07-06 15:18:24 -070083 touchscreen_set_params(input, axis, maximum - 1, fuzz);
Sebastian Reichelb98abe52014-05-28 23:51:53 -070084
Dmitry Torokhov7c494372015-06-01 10:35:16 -070085 axis = multitouch ? ABS_MT_POSITION_Y : ABS_Y;
Dmitry Torokhov4200e832015-07-06 15:18:24 -070086 data_present = touchscreen_get_prop_u32(dev, "touchscreen-size-y",
87 input_abs_get_max(input,
Dmitry Torokhov51717862015-07-06 14:57:54 -070088 axis) + 1,
Dmitry Torokhov7c494372015-06-01 10:35:16 -070089 &maximum) |
Dmitry Torokhov4200e832015-07-06 15:18:24 -070090 touchscreen_get_prop_u32(dev, "touchscreen-fuzz-y",
91 input_abs_get_fuzz(input, axis),
Dmitry Torokhov7c494372015-06-01 10:35:16 -070092 &fuzz);
93 if (data_present)
Dmitry Torokhov4200e832015-07-06 15:18:24 -070094 touchscreen_set_params(input, axis, maximum - 1, fuzz);
Sebastian Reichelb98abe52014-05-28 23:51:53 -070095
Dmitry Torokhov7c494372015-06-01 10:35:16 -070096 axis = multitouch ? ABS_MT_PRESSURE : ABS_PRESSURE;
Dmitry Torokhov4200e832015-07-06 15:18:24 -070097 data_present = touchscreen_get_prop_u32(dev,
98 "touchscreen-max-pressure",
99 input_abs_get_max(input, axis),
Dmitry Torokhov7c494372015-06-01 10:35:16 -0700100 &maximum) |
Dmitry Torokhov4200e832015-07-06 15:18:24 -0700101 touchscreen_get_prop_u32(dev,
102 "touchscreen-fuzz-pressure",
103 input_abs_get_fuzz(input, axis),
Dmitry Torokhov7c494372015-06-01 10:35:16 -0700104 &fuzz);
105 if (data_present)
Dmitry Torokhov4200e832015-07-06 15:18:24 -0700106 touchscreen_set_params(input, axis, maximum, fuzz);
Sebastian Reichelb98abe52014-05-28 23:51:53 -0700107}
Dmitry Torokhov4200e832015-07-06 15:18:24 -0700108EXPORT_SYMBOL(touchscreen_parse_properties);