| .. | .. |
|---|
| 1 | +// SPDX-License-Identifier: GPL-2.0-only |
|---|
| 1 | 2 | /* |
|---|
| 2 | 3 | * Touchkey driver for Freescale MPR121 Controllor |
|---|
| 3 | 4 | * |
|---|
| .. | .. |
|---|
| 5 | 6 | * Author: Zhang Jiejing <jiejing.zhang@freescale.com> |
|---|
| 6 | 7 | * |
|---|
| 7 | 8 | * Based on mcs_touchkey.c |
|---|
| 8 | | - * |
|---|
| 9 | | - * This program is free software; you can redistribute it and/or modify |
|---|
| 10 | | - * it under the terms of the GNU General Public License version 2 as |
|---|
| 11 | | - * published by the Free Software Foundation. |
|---|
| 12 | | - * |
|---|
| 13 | 9 | */ |
|---|
| 14 | 10 | |
|---|
| 15 | 11 | #include <linux/bitops.h> |
|---|
| .. | .. |
|---|
| 57 | 53 | #define TOUCH_STATUS_MASK 0xfff |
|---|
| 58 | 54 | /* MPR121 has 12 keys */ |
|---|
| 59 | 55 | #define MPR121_MAX_KEY_COUNT 12 |
|---|
| 56 | + |
|---|
| 57 | +#define MPR121_MIN_POLL_INTERVAL 10 |
|---|
| 58 | +#define MPR121_MAX_POLL_INTERVAL 200 |
|---|
| 60 | 59 | |
|---|
| 61 | 60 | struct mpr121_touchkey { |
|---|
| 62 | 61 | struct i2c_client *client; |
|---|
| .. | .. |
|---|
| 119 | 118 | return vdd_supply; |
|---|
| 120 | 119 | } |
|---|
| 121 | 120 | |
|---|
| 122 | | -static irqreturn_t mpr_touchkey_interrupt(int irq, void *dev_id) |
|---|
| 121 | +static void mpr_touchkey_report(struct input_dev *dev) |
|---|
| 123 | 122 | { |
|---|
| 124 | | - struct mpr121_touchkey *mpr121 = dev_id; |
|---|
| 125 | | - struct i2c_client *client = mpr121->client; |
|---|
| 123 | + struct mpr121_touchkey *mpr121 = input_get_drvdata(dev); |
|---|
| 126 | 124 | struct input_dev *input = mpr121->input_dev; |
|---|
| 125 | + struct i2c_client *client = mpr121->client; |
|---|
| 127 | 126 | unsigned long bit_changed; |
|---|
| 128 | 127 | unsigned int key_num; |
|---|
| 129 | 128 | int reg; |
|---|
| .. | .. |
|---|
| 131 | 130 | reg = i2c_smbus_read_byte_data(client, ELE_TOUCH_STATUS_1_ADDR); |
|---|
| 132 | 131 | if (reg < 0) { |
|---|
| 133 | 132 | dev_err(&client->dev, "i2c read error [%d]\n", reg); |
|---|
| 134 | | - goto out; |
|---|
| 133 | + return; |
|---|
| 135 | 134 | } |
|---|
| 136 | 135 | |
|---|
| 137 | 136 | reg <<= 8; |
|---|
| 138 | 137 | reg |= i2c_smbus_read_byte_data(client, ELE_TOUCH_STATUS_0_ADDR); |
|---|
| 139 | 138 | if (reg < 0) { |
|---|
| 140 | 139 | dev_err(&client->dev, "i2c read error [%d]\n", reg); |
|---|
| 141 | | - goto out; |
|---|
| 140 | + return; |
|---|
| 142 | 141 | } |
|---|
| 143 | 142 | |
|---|
| 144 | 143 | reg &= TOUCH_STATUS_MASK; |
|---|
| .. | .. |
|---|
| 159 | 158 | |
|---|
| 160 | 159 | } |
|---|
| 161 | 160 | input_sync(input); |
|---|
| 161 | +} |
|---|
| 162 | 162 | |
|---|
| 163 | | -out: |
|---|
| 163 | +static irqreturn_t mpr_touchkey_interrupt(int irq, void *dev_id) |
|---|
| 164 | +{ |
|---|
| 165 | + struct mpr121_touchkey *mpr121 = dev_id; |
|---|
| 166 | + |
|---|
| 167 | + mpr_touchkey_report(mpr121->input_dev); |
|---|
| 168 | + |
|---|
| 164 | 169 | return IRQ_HANDLED; |
|---|
| 165 | 170 | } |
|---|
| 166 | 171 | |
|---|
| .. | .. |
|---|
| 233 | 238 | int vdd_uv; |
|---|
| 234 | 239 | struct mpr121_touchkey *mpr121; |
|---|
| 235 | 240 | struct input_dev *input_dev; |
|---|
| 241 | + u32 poll_interval = 0; |
|---|
| 236 | 242 | int error; |
|---|
| 237 | 243 | int i; |
|---|
| 238 | | - |
|---|
| 239 | | - if (!client->irq) { |
|---|
| 240 | | - dev_err(dev, "irq number should not be zero\n"); |
|---|
| 241 | | - return -EINVAL; |
|---|
| 242 | | - } |
|---|
| 243 | 244 | |
|---|
| 244 | 245 | vdd_supply = mpr121_vdd_supply_init(dev); |
|---|
| 245 | 246 | if (IS_ERR(vdd_supply)) |
|---|
| .. | .. |
|---|
| 257 | 258 | |
|---|
| 258 | 259 | mpr121->client = client; |
|---|
| 259 | 260 | mpr121->input_dev = input_dev; |
|---|
| 260 | | - mpr121->keycount = device_property_read_u32_array(dev, "linux,keycodes", |
|---|
| 261 | | - NULL, 0); |
|---|
| 261 | + mpr121->keycount = device_property_count_u32(dev, "linux,keycodes"); |
|---|
| 262 | 262 | if (mpr121->keycount > MPR121_MAX_KEY_COUNT) { |
|---|
| 263 | 263 | dev_err(dev, "too many keys defined (%d)\n", mpr121->keycount); |
|---|
| 264 | 264 | return -EINVAL; |
|---|
| .. | .. |
|---|
| 279 | 279 | if (device_property_read_bool(dev, "autorepeat")) |
|---|
| 280 | 280 | __set_bit(EV_REP, input_dev->evbit); |
|---|
| 281 | 281 | input_set_capability(input_dev, EV_MSC, MSC_SCAN); |
|---|
| 282 | + input_set_drvdata(input_dev, mpr121); |
|---|
| 282 | 283 | |
|---|
| 283 | 284 | input_dev->keycode = mpr121->keycodes; |
|---|
| 284 | 285 | input_dev->keycodesize = sizeof(mpr121->keycodes[0]); |
|---|
| .. | .. |
|---|
| 293 | 294 | return error; |
|---|
| 294 | 295 | } |
|---|
| 295 | 296 | |
|---|
| 296 | | - error = devm_request_threaded_irq(dev, client->irq, NULL, |
|---|
| 297 | | - mpr_touchkey_interrupt, |
|---|
| 298 | | - IRQF_TRIGGER_FALLING | IRQF_ONESHOT, |
|---|
| 299 | | - dev->driver->name, mpr121); |
|---|
| 300 | | - if (error) { |
|---|
| 301 | | - dev_err(dev, "Failed to register interrupt\n"); |
|---|
| 302 | | - return error; |
|---|
| 297 | + device_property_read_u32(dev, "poll-interval", &poll_interval); |
|---|
| 298 | + |
|---|
| 299 | + if (client->irq) { |
|---|
| 300 | + error = devm_request_threaded_irq(dev, client->irq, NULL, |
|---|
| 301 | + mpr_touchkey_interrupt, |
|---|
| 302 | + IRQF_TRIGGER_FALLING | |
|---|
| 303 | + IRQF_ONESHOT, |
|---|
| 304 | + dev->driver->name, mpr121); |
|---|
| 305 | + if (error) { |
|---|
| 306 | + dev_err(dev, "Failed to register interrupt\n"); |
|---|
| 307 | + return error; |
|---|
| 308 | + } |
|---|
| 309 | + } else if (poll_interval) { |
|---|
| 310 | + if (poll_interval < MPR121_MIN_POLL_INTERVAL) |
|---|
| 311 | + return -EINVAL; |
|---|
| 312 | + |
|---|
| 313 | + if (poll_interval > MPR121_MAX_POLL_INTERVAL) |
|---|
| 314 | + return -EINVAL; |
|---|
| 315 | + |
|---|
| 316 | + error = input_setup_polling(input_dev, mpr_touchkey_report); |
|---|
| 317 | + if (error) { |
|---|
| 318 | + dev_err(dev, "Failed to setup polling\n"); |
|---|
| 319 | + return error; |
|---|
| 320 | + } |
|---|
| 321 | + |
|---|
| 322 | + input_set_poll_interval(input_dev, poll_interval); |
|---|
| 323 | + input_set_min_poll_interval(input_dev, |
|---|
| 324 | + MPR121_MIN_POLL_INTERVAL); |
|---|
| 325 | + input_set_max_poll_interval(input_dev, |
|---|
| 326 | + MPR121_MAX_POLL_INTERVAL); |
|---|
| 327 | + } else { |
|---|
| 328 | + dev_err(dev, |
|---|
| 329 | + "invalid IRQ number and polling not configured\n"); |
|---|
| 330 | + return -EINVAL; |
|---|
| 303 | 331 | } |
|---|
| 304 | 332 | |
|---|
| 305 | 333 | error = input_register_device(input_dev); |
|---|