1 /* 2 * Touchscreen driver for the TS-4800 board 3 * 4 * Copyright (c) 2015 - Savoir-faire Linux 5 * 6 * This file is licensed under the terms of the GNU General Public 7 * License version 2. This program is licensed "as is" without any 8 * warranty of any kind, whether express or implied. 9 */ 10 11 #include <linux/bitops.h> 12 #include <linux/input.h> 13 #include <linux/io.h> 14 #include <linux/kernel.h> 15 #include <linux/mfd/syscon.h> 16 #include <linux/module.h> 17 #include <linux/of.h> 18 #include <linux/platform_device.h> 19 #include <linux/regmap.h> 20 21 /* polling interval in ms */ 22 #define POLL_INTERVAL 3 23 24 #define DEBOUNCE_COUNT 1 25 26 /* sensor values are 12-bit wide */ 27 #define MAX_12BIT ((1 << 12) - 1) 28 29 #define PENDOWN_MASK 0x1 30 31 #define X_OFFSET 0x0 32 #define Y_OFFSET 0x2 33 34 struct ts4800_ts { 35 struct input_dev *input; 36 struct device *dev; 37 char phys[32]; 38 39 void __iomem *base; 40 struct regmap *regmap; 41 unsigned int reg; 42 unsigned int bit; 43 44 bool pendown; 45 int debounce; 46 }; 47 48 static int ts4800_ts_open(struct input_dev *input_dev) 49 { 50 struct ts4800_ts *ts = input_get_drvdata(input_dev); 51 int error; 52 53 ts->pendown = false; 54 ts->debounce = DEBOUNCE_COUNT; 55 56 error = regmap_update_bits(ts->regmap, ts->reg, ts->bit, ts->bit); 57 if (error) { 58 dev_warn(ts->dev, "Failed to enable touchscreen: %d\n", error); 59 return error; 60 } 61 62 return 0; 63 } 64 65 static void ts4800_ts_close(struct input_dev *input_dev) 66 { 67 struct ts4800_ts *ts = input_get_drvdata(input_dev); 68 int ret; 69 70 ret = regmap_update_bits(ts->regmap, ts->reg, ts->bit, 0); 71 if (ret) 72 dev_warn(ts->dev, "Failed to disable touchscreen\n"); 73 74 } 75 76 static void ts4800_ts_poll(struct input_dev *input_dev) 77 { 78 struct ts4800_ts *ts = input_get_drvdata(input_dev); 79 u16 last_x = readw(ts->base + X_OFFSET); 80 u16 last_y = readw(ts->base + Y_OFFSET); 81 bool pendown = last_x & PENDOWN_MASK; 82 83 if (pendown) { 84 if (ts->debounce) { 85 ts->debounce--; 86 return; 87 } 88 89 if (!ts->pendown) { 90 input_report_key(input_dev, BTN_TOUCH, 1); 91 ts->pendown = true; 92 } 93 94 last_x = ((~last_x) >> 4) & MAX_12BIT; 95 last_y = ((~last_y) >> 4) & MAX_12BIT; 96 97 input_report_abs(input_dev, ABS_X, last_x); 98 input_report_abs(input_dev, ABS_Y, last_y); 99 input_sync(input_dev); 100 } else if (ts->pendown) { 101 ts->pendown = false; 102 ts->debounce = DEBOUNCE_COUNT; 103 input_report_key(input_dev, BTN_TOUCH, 0); 104 input_sync(input_dev); 105 } 106 } 107 108 static int ts4800_parse_dt(struct platform_device *pdev, 109 struct ts4800_ts *ts) 110 { 111 struct device *dev = &pdev->dev; 112 struct device_node *np = dev->of_node; 113 u32 reg, bit; 114 int error; 115 116 struct device_node *syscon_np __free(device_node) = 117 of_parse_phandle(np, "syscon", 0); 118 if (!syscon_np) { 119 dev_err(dev, "no syscon property\n"); 120 return -ENODEV; 121 } 122 123 ts->regmap = syscon_node_to_regmap(syscon_np); 124 if (IS_ERR(ts->regmap)) { 125 dev_err(dev, "cannot get parent's regmap\n"); 126 return PTR_ERR(ts->regmap); 127 } 128 129 error = of_property_read_u32_index(np, "syscon", 1, ®); 130 if (error < 0) { 131 dev_err(dev, "no offset in syscon\n"); 132 return error; 133 } 134 135 ts->reg = reg; 136 137 error = of_property_read_u32_index(np, "syscon", 2, &bit); 138 if (error < 0) { 139 dev_err(dev, "no bit in syscon\n"); 140 return error; 141 } 142 143 ts->bit = BIT(bit); 144 145 return 0; 146 } 147 148 static int ts4800_ts_probe(struct platform_device *pdev) 149 { 150 struct input_dev *input_dev; 151 struct ts4800_ts *ts; 152 int error; 153 154 ts = devm_kzalloc(&pdev->dev, sizeof(*ts), GFP_KERNEL); 155 if (!ts) 156 return -ENOMEM; 157 158 error = ts4800_parse_dt(pdev, ts); 159 if (error) 160 return error; 161 162 ts->base = devm_platform_ioremap_resource(pdev, 0); 163 if (IS_ERR(ts->base)) 164 return PTR_ERR(ts->base); 165 166 input_dev = devm_input_allocate_device(&pdev->dev); 167 if (!input_dev) 168 return -ENOMEM; 169 170 snprintf(ts->phys, sizeof(ts->phys), "%s/input0", dev_name(&pdev->dev)); 171 ts->input = input_dev; 172 ts->dev = &pdev->dev; 173 174 input_set_drvdata(input_dev, ts); 175 176 input_dev->name = "TS-4800 Touchscreen"; 177 input_dev->phys = ts->phys; 178 179 input_dev->open = ts4800_ts_open; 180 input_dev->close = ts4800_ts_close; 181 182 input_set_capability(input_dev, EV_KEY, BTN_TOUCH); 183 input_set_abs_params(input_dev, ABS_X, 0, MAX_12BIT, 0, 0); 184 input_set_abs_params(input_dev, ABS_Y, 0, MAX_12BIT, 0, 0); 185 186 error = input_setup_polling(input_dev, ts4800_ts_poll); 187 if (error) { 188 dev_err(&pdev->dev, "Unable to set up polling: %d\n", error); 189 return error; 190 } 191 192 input_set_poll_interval(input_dev, POLL_INTERVAL); 193 194 error = input_register_device(input_dev); 195 if (error) { 196 dev_err(&pdev->dev, 197 "Unable to register input device: %d\n", error); 198 return error; 199 } 200 201 return 0; 202 } 203 204 static const struct of_device_id ts4800_ts_of_match[] = { 205 { .compatible = "technologic,ts4800-ts", }, 206 { }, 207 }; 208 MODULE_DEVICE_TABLE(of, ts4800_ts_of_match); 209 210 static struct platform_driver ts4800_ts_driver = { 211 .driver = { 212 .name = "ts4800-ts", 213 .of_match_table = ts4800_ts_of_match, 214 }, 215 .probe = ts4800_ts_probe, 216 }; 217 module_platform_driver(ts4800_ts_driver); 218 219 MODULE_AUTHOR("Damien Riegel <damien.riegel@savoirfairelinux.com>"); 220 MODULE_DESCRIPTION("TS-4800 Touchscreen Driver"); 221 MODULE_LICENSE("GPL v2"); 222 MODULE_ALIAS("platform:ts4800_ts"); 223