aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorHeungjun Kim <riverful.kim@samsung.com>2011-02-24 00:40:11 -0500
committerDmitry Torokhov <dmitry.torokhov@gmail.com>2011-02-24 00:43:31 -0500
commitadf779c1ee1d5556ebd83e39a7189022d4ebce3a (patch)
treeadd454348d5a2eca573fa6751fb20d576a5bce8e
parent5ad567ffbaf208bd35bff4341906346c1a1d4574 (diff)
Input: mcs_touchkey - add support for suspend/resume
This adds support for system-level suspend/resume to the driver. Signed-off-by: Heungjun Kim <riverful.kim@samsung.com> Signed-off-by: Kyungmin Park <kyungmin.park@samsung.com> Signed-off-by: Dmitry Torokhov <dtor@mail.ru>
-rw-r--r--drivers/input/keyboard/mcs_touchkey.c48
-rw-r--r--include/linux/i2c/mcs.h1
2 files changed, 48 insertions, 1 deletions
diff --git a/drivers/input/keyboard/mcs_touchkey.c b/drivers/input/keyboard/mcs_touchkey.c
index 63b849d7e90b..03fa59a0b220 100644
--- a/drivers/input/keyboard/mcs_touchkey.c
+++ b/drivers/input/keyboard/mcs_touchkey.c
@@ -1,5 +1,5 @@
1/* 1/*
2 * mcs_touchkey.c - Touchkey driver for MELFAS MCS5000/5080 controller 2 * Touchkey driver for MELFAS MCS5000/5080 controller
3 * 3 *
4 * Copyright (C) 2010 Samsung Electronics Co.Ltd 4 * Copyright (C) 2010 Samsung Electronics Co.Ltd
5 * Author: HeungJun Kim <riverful.kim@samsung.com> 5 * Author: HeungJun Kim <riverful.kim@samsung.com>
@@ -19,6 +19,7 @@
19#include <linux/input.h> 19#include <linux/input.h>
20#include <linux/irq.h> 20#include <linux/irq.h>
21#include <linux/slab.h> 21#include <linux/slab.h>
22#include <linux/pm.h>
22 23
23/* MCS5000 Touchkey */ 24/* MCS5000 Touchkey */
24#define MCS5000_TOUCHKEY_STATUS 0x04 25#define MCS5000_TOUCHKEY_STATUS 0x04
@@ -45,6 +46,8 @@ struct mcs_touchkey_chip {
45}; 46};
46 47
47struct mcs_touchkey_data { 48struct mcs_touchkey_data {
49 void (*poweron)(bool);
50
48 struct i2c_client *client; 51 struct i2c_client *client;
49 struct input_dev *input_dev; 52 struct input_dev *input_dev;
50 struct mcs_touchkey_chip chip; 53 struct mcs_touchkey_chip chip;
@@ -169,6 +172,11 @@ static int __devinit mcs_touchkey_probe(struct i2c_client *client,
169 if (pdata->cfg_pin) 172 if (pdata->cfg_pin)
170 pdata->cfg_pin(); 173 pdata->cfg_pin();
171 174
175 if (pdata->poweron) {
176 data->poweron = pdata->poweron;
177 data->poweron(true);
178 }
179
172 error = request_threaded_irq(client->irq, NULL, mcs_touchkey_interrupt, 180 error = request_threaded_irq(client->irq, NULL, mcs_touchkey_interrupt,
173 IRQF_TRIGGER_FALLING, client->dev.driver->name, data); 181 IRQF_TRIGGER_FALLING, client->dev.driver->name, data);
174 if (error) { 182 if (error) {
@@ -196,12 +204,49 @@ static int __devexit mcs_touchkey_remove(struct i2c_client *client)
196 struct mcs_touchkey_data *data = i2c_get_clientdata(client); 204 struct mcs_touchkey_data *data = i2c_get_clientdata(client);
197 205
198 free_irq(client->irq, data); 206 free_irq(client->irq, data);
207 if (data->poweron)
208 data->poweron(false);
199 input_unregister_device(data->input_dev); 209 input_unregister_device(data->input_dev);
200 kfree(data); 210 kfree(data);
201 211
202 return 0; 212 return 0;
203} 213}
204 214
215#ifdef CONFIG_PM_SLEEP
216static int mcs_touchkey_suspend(struct device *dev)
217{
218 struct mcs_touchkey_data *data = dev_get_drvdata(dev);
219 struct i2c_client *client = data->client;
220
221 /* Disable the work */
222 disable_irq(client->irq);
223
224 /* Finally turn off the power */
225 if (data->poweron)
226 data->poweron(false);
227
228 return 0;
229}
230
231static int mcs_touchkey_resume(struct device *dev)
232{
233 struct mcs_touchkey_data *data = dev_get_drvdata(dev);
234 struct i2c_client *client = data->client;
235
236 /* Enable the device first */
237 if (data->poweron)
238 data->poweron(true);
239
240 /* Enable irq again */
241 enable_irq(client->irq);
242
243 return 0;
244}
245#endif
246
247static SIMPLE_DEV_PM_OPS(mcs_touchkey_pm_ops,
248 mcs_touchkey_suspend, mcs_touchkey_resume);
249
205static const struct i2c_device_id mcs_touchkey_id[] = { 250static const struct i2c_device_id mcs_touchkey_id[] = {
206 { "mcs5000_touchkey", MCS5000_TOUCHKEY }, 251 { "mcs5000_touchkey", MCS5000_TOUCHKEY },
207 { "mcs5080_touchkey", MCS5080_TOUCHKEY }, 252 { "mcs5080_touchkey", MCS5080_TOUCHKEY },
@@ -213,6 +258,7 @@ static struct i2c_driver mcs_touchkey_driver = {
213 .driver = { 258 .driver = {
214 .name = "mcs_touchkey", 259 .name = "mcs_touchkey",
215 .owner = THIS_MODULE, 260 .owner = THIS_MODULE,
261 .pm = &mcs_touchkey_pm_ops,
216 }, 262 },
217 .probe = mcs_touchkey_probe, 263 .probe = mcs_touchkey_probe,
218 .remove = __devexit_p(mcs_touchkey_remove), 264 .remove = __devexit_p(mcs_touchkey_remove),
diff --git a/include/linux/i2c/mcs.h b/include/linux/i2c/mcs.h
index 725ae7c313ff..61bb18a4fd3c 100644
--- a/include/linux/i2c/mcs.h
+++ b/include/linux/i2c/mcs.h
@@ -18,6 +18,7 @@
18#define MCS_KEY_CODE(v) ((v) & 0xffff) 18#define MCS_KEY_CODE(v) ((v) & 0xffff)
19 19
20struct mcs_platform_data { 20struct mcs_platform_data {
21 void (*poweron)(bool);
21 void (*cfg_pin)(void); 22 void (*cfg_pin)(void);
22 23
23 /* touchscreen */ 24 /* touchscreen */