[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20200318230938.31573-3-tony@atomide.com>
Date: Wed, 18 Mar 2020 16:09:38 -0700
From: Tony Lindgren <tony@...mide.com>
To: Nick Dyer <nick@...anahar.org>,
Dmitry Torokhov <dmitry.torokhov@...il.com>
Cc: Henrik Rydberg <rydberg@...math.org>,
Arthur Demchenkov <spinal.by@...il.com>,
Ivaylo Dimitrov <ivo.g.dimitrov.75@...il.com>,
Merlijn Wajer <merlijn@...zup.org>,
Pavel Machek <pavel@....cz>,
Sebastian Reichel <sre@...nel.org>,
linux-input@...r.kernel.org, linux-kernel@...r.kernel.org,
linux-omap@...r.kernel.org
Subject: [PATCH 3/3] Input: atmel_mxt_ts - use runtime PM autosuspend for idle config
Let's enable runtime PM autosuspend with a default value of 600 ms and
switch to idle power config for runtime_suspend. The autosuspend timeout
can also be configured also via userspace with value of -1 disabling the
autosuspend.
Let's only allow autosuspend if MXT_SUSPEND_T9_CTRL is not configured for
suspend_mode as MXT_SUSPEND_T9_CTRL mode powers off the controller.
Signed-off-by: Tony Lindgren <tony@...mide.com>
---
drivers/input/touchscreen/atmel_mxt_ts.c | 76 ++++++++++++++++++++----
1 file changed, 66 insertions(+), 10 deletions(-)
diff --git a/drivers/input/touchscreen/atmel_mxt_ts.c b/drivers/input/touchscreen/atmel_mxt_ts.c
--- a/drivers/input/touchscreen/atmel_mxt_ts.c
+++ b/drivers/input/touchscreen/atmel_mxt_ts.c
@@ -1140,21 +1140,26 @@ static irqreturn_t mxt_process_messages(struct mxt_data *data)
static irqreturn_t mxt_interrupt(int irq, void *dev_id)
{
struct mxt_data *data = dev_id;
+ irqreturn_t ret = IRQ_HANDLED;
if (data->in_bootloader) {
/* bootloader state transition completion */
complete(&data->bl_completion);
- return IRQ_HANDLED;
+ goto out_done;
}
if (!data->object_table)
- return IRQ_HANDLED;
+ goto out_done;
- if (data->T44_address) {
- return mxt_process_messages_t44(data);
- } else {
- return mxt_process_messages(data);
- }
+ if (data->T44_address)
+ ret = mxt_process_messages_t44(data);
+ else
+ ret = mxt_process_messages(data);
+
+out_done:
+ pm_runtime_mark_last_busy(&data->client->dev);
+
+ return ret;
}
static int mxt_t6_command(struct mxt_data *data, u16 cmd_offset,
@@ -2957,6 +2962,16 @@ static int mxt_input_open(struct input_dev *input_dev)
return error;
}
+ /*
+ * Prevent autoidle if MXT_SUSPEND_T9_CTRL is the default as it will
+ * power off the controller. Balanced inmxt_input_close().
+ */
+ if (data->suspend_mode == MXT_SUSPEND_T9_CTRL)
+ pm_runtime_get(dev);
+
+ pm_runtime_mark_last_busy(dev);
+ pm_runtime_put_autosuspend(dev);
+
return 0;
}
@@ -2964,8 +2979,21 @@ static void mxt_input_close(struct input_dev *input_dev)
{
struct mxt_data *data = input_get_drvdata(input_dev);
struct device *dev = &data->client->dev;
+ int error;
- pm_runtime_put_sync(dev);
+ /* Release extra usage count for MXT_SUSPEND_T9_CTRL done in open */
+ if (data->suspend_mode == MXT_SUSPEND_T9_CTRL)
+ pm_runtime_put(dev);
+
+ /* Wake the device so autosuspend sees correct input_dev->users */
+ error = pm_runtime_get_sync(dev);
+ if (error < 0) {
+ pm_runtime_put_noidle(dev);
+ return;
+ }
+
+ pm_runtime_mark_last_busy(dev);
+ pm_runtime_put_autosuspend(dev);
}
static int mxt_parse_device_properties(struct mxt_data *data)
@@ -3099,6 +3127,14 @@ static int mxt_probe(struct i2c_client *client, const struct i2c_device_id *id)
}
pm_runtime_enable(dev);
+ pm_runtime_use_autosuspend(dev);
+ pm_runtime_set_autosuspend_delay(dev, 600);
+ error = pm_runtime_get_sync(dev);
+ if (error < 0) {
+ pm_runtime_put_noidle(dev);
+ return error;
+ }
+
error = mxt_initialize(data);
if (error)
@@ -3111,9 +3147,14 @@ static int mxt_probe(struct i2c_client *client, const struct i2c_device_id *id)
goto err_disable;
}
+ pm_runtime_mark_last_busy(dev);
+ pm_runtime_put_autosuspend(dev);
+
return 0;
err_disable:
+ pm_runtime_dont_use_autosuspend(dev);
+ pm_runtime_put_sync(dev);
pm_runtime_disable(dev);
mxt_free_input_device(data);
mxt_free_object_table(data);
@@ -3124,12 +3165,24 @@ static int mxt_remove(struct i2c_client *client)
{
struct mxt_data *data = i2c_get_clientdata(client);
struct device *dev = &data->client->dev;
+ int active;
+
+ /* Attempt to change controller suspend mode to disable on remove */
+ active = pm_runtime_get_sync(dev);
+ if (active < 0)
+ pm_runtime_put_noidle(dev);
+ else
+ data->suspend_mode = MXT_SUSPEND_T9_CTRL;
+
+ pm_runtime_dont_use_autosuspend(dev);
+ if (active >= 0)
+ pm_runtime_put_sync(dev);
+ pm_runtime_disable(dev);
disable_irq(data->irq);
sysfs_remove_group(&client->dev.kobj, &mxt_attr_group);
mxt_free_input_device(data);
mxt_free_object_table(data);
- pm_runtime_disable(dev);
return 0;
}
@@ -3152,7 +3205,10 @@ static int __maybe_unused mxt_runtime_suspend(struct device *dev)
case MXT_SUSPEND_DEEP_SLEEP:
default:
- mxt_set_t7_power_cfg(data, MXT_POWER_CFG_DEEPSLEEP);
+ if (input_dev->users)
+ mxt_set_t7_power_cfg(data, MXT_POWER_CFG_IDLE);
+ else
+ mxt_set_t7_power_cfg(data, MXT_POWER_CFG_DEEPSLEEP);
break;
}
--
2.25.1
Powered by blists - more mailing lists