Received: by 2002:a25:6193:0:0:0:0:0 with SMTP id v141csp3365211ybb; Tue, 31 Mar 2020 03:52:58 -0700 (PDT) X-Google-Smtp-Source: ADFU+vvTTY/OuBRiFqRji5HpGWNEdV5yFKDVzD8UfeMv437jlxBjR3gvK63dGXFtaRc7q00vgNdY X-Received: by 2002:a05:6830:2147:: with SMTP id r7mr12210664otd.59.1585651978247; Tue, 31 Mar 2020 03:52:58 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1585651978; cv=none; d=google.com; s=arc-20160816; b=mlzLg4HlA3gQX3+OJgPlZLIUwHLDGwVlcXSw0QKx4xS2NMe03B7zDvhFSoLRI9Mqra B6nFxapqbOvW3LN3EdebwBsRUcjFO46/teyHEgyc/+yJwlyOxNKPIFT9QDNYYQKZifjG dZAph0VVOL0+A7mOxF96Su0ZWk4cRf0Es0C6Cu6GcPIHKMVDoNBf1vQSh4x6yngDVRFL X8EhX7oCuUYA8pHCayGdGh3GeJ8Kg4M/2OTm8F5GDcX9ret8ZwwMs5hKm83kbrM1tKPa 6DwYX0dAVuLWj3ciinbM64EepYT3hMSEy8mE50IFPS9JC8Cv21yKwCcxgnSZYCD6Rvor CNNA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:mime-version:references:in-reply-to :message-id:date:subject:cc:to:from:ironport-sdr:ironport-sdr; bh=qPJitLqksE7uPawOGs1K3FjZEP9C86h8KnRYfT6Z6s8=; b=HTzeFD4uV4dH7FJ8r/EbEuxKgcIujpsITrhxlpwhygIgDv6DdyJFxoMDdD4pBDDjGz NhIKLbhn0b30xrCrRAsMdhIepT/poMEKfW5yMh85++IG5+PjKtsfOiCStwuuqZ2rdgrU Potxf8SSCHlDRH/iP64Kue1SvgI2hTPA+FQNdpKETyeOAXfcLGFYi0AyNg/1g/shoY9M uTKg9E+fJbE9/1SeQ6sRvEalmQ3QrbNjnvOkDMoRJIHekV7b87d5Bk9OXZHv3R39ikAc 0TTa4BgY6gctYJaDy//DhnxUUsxlg7ZwpdnsmzTmq9qXtWVBJqkqhMVguDcL94URrmN0 mRHg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id t28si7189025ooc.16.2020.03.31.03.52.37; Tue, 31 Mar 2020 03:52:58 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1730672AbgCaKvx (ORCPT + 99 others); Tue, 31 Mar 2020 06:51:53 -0400 Received: from esa4.mentor.iphmx.com ([68.232.137.252]:43515 "EHLO esa4.mentor.iphmx.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1730433AbgCaKvu (ORCPT ); Tue, 31 Mar 2020 06:51:50 -0400 IronPort-SDR: U3X0+ElcFdZO99sjuo/5i6gpf6OaDJT0iRfT4AdduUerUnpwLFJiXtLm8tyNI/rjOWefrbVByj 7/cfs8rr0Cib/gELZNtkKDlWcAA55hNLVlfDlBa0l66pSbpB8DSv2jT3W8fRWNMLMgV6Gr28nT 3Es8vt5rbHjBaqbu/T1vf0D1pvjam1CZPF8KOl7bA8DwilossogEiXbi3QyLuTPNcKWwhOSR7r 1BwK7N2utjGmdja7upGF6vvLAG0oBG3jKgQs34O2ZaPW9t56NW3l5VaA95QlAvW/cMpew2eOBu VXk= X-IronPort-AV: E=Sophos;i="5.72,327,1580803200"; d="scan'208";a="47330748" Received: from orw-gwy-01-in.mentorg.com ([192.94.38.165]) by esa4.mentor.iphmx.com with ESMTP; 31 Mar 2020 02:51:50 -0800 IronPort-SDR: 7I+4n7Dywgdg7JRZVzjzxmMKCj8CQmEAW8T+UIY8oHDQx+wBbh/b/4c7vi/X/j0cK08eOtHlEv Z8//VjmDYd20xr9vZFtt6HGdGtXXLkCx2ni9rq4xeHyXNVNfe+FoOFj8HkTausaY6VtmiK2YO6 yqZRdmmnQEkZdjNMTW56izh/WKpaqXbESh5F3dPLA/UIW19zQIKq2VNQB5IjxcXTqqiyJw507b 7aPbt/e6vb0bACwF2VVCdD2GgRTaSkq8uDOWfF5dfa9k2KO8Jwx3fBsuHaD3mWbZOFIaZ4AqJa /3o= From: Jiada Wang To: , , , , CC: , , , , Subject: [PATCH v10 14/55] Input: atmel_mxt_ts - add regulator control support Date: Tue, 31 Mar 2020 03:50:10 -0700 Message-ID: <20200331105051.58896-15-jiada_wang@mentor.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20200331105051.58896-1-jiada_wang@mentor.com> References: <20200331105051.58896-1-jiada_wang@mentor.com> MIME-Version: 1.0 Content-Type: text/plain Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Nick Dyer Allow the driver to optionally manage enabling/disable power to the touch controller itself. If the regulators are not present then use the deep sleep power mode instead. For a correct power on sequence, it is required that we have control over the RESET line. Signed-off-by: Nick Dyer Acked-by: Benson Leung Acked-by: Yufeng Shen (cherry picked from ndyer/linux/for-upstream commit 14052b61bb66c2f2283c00e733e131be7a9b8bfc) [gdavis: Resolve forward port conflicts due to v4.14-rc1 commmit f657b00df22e ("Input: atmel_mxt_ts - add support for reset line") and applying upstream commit 96a938aa214e ("Input: atmel_mxt_ts - remove platform data support").] Signed-off-by: George G. Davis [gdavis: Squash fixes from Dirk Behme: - Input: atmel_mxt_ts - in failure case disable the regulator - Input: atmel_mxt_ts - disable only enabled regulators - Input: atmel_mxt_ts - use devm_regulator_get()] Signed-off-by: Dirk Behme [jiada: Replace white-spaces with tab for MXT_CHG_DELAY separate Documentation/ and include/dt-bindings/ portion change to another commit] Signed-off-by: Jiada Wang --- drivers/input/touchscreen/atmel_mxt_ts.c | 130 +++++++++++++++++++++-- 1 file changed, 121 insertions(+), 9 deletions(-) diff --git a/drivers/input/touchscreen/atmel_mxt_ts.c b/drivers/input/touchscreen/atmel_mxt_ts.c index 9aafed92db9c..ef8baf64659e 100644 --- a/drivers/input/touchscreen/atmel_mxt_ts.c +++ b/drivers/input/touchscreen/atmel_mxt_ts.c @@ -26,10 +26,12 @@ #include #include #include +#include #include #include #include #include +#include /* Firmware files */ #define MXT_FW_NAME "maxtouch.fw" @@ -215,6 +217,9 @@ enum t100_type { #define MXT_CRC_TIMEOUT 1000 /* msec */ #define MXT_FW_RESET_TIME 3000 /* msec */ #define MXT_FW_CHG_TIMEOUT 300 /* msec */ +#define MXT_REGULATOR_DELAY 150 /* msec */ +#define MXT_CHG_DELAY 100 /* msec */ +#define MXT_POWERON_DELAY 150 /* msec */ /* Command to unlock bootloader */ #define MXT_UNLOCK_CMD_MSB 0xaa @@ -275,11 +280,6 @@ enum v4l_dbg_inputs { MXT_V4L_INPUT_MAX, }; -enum mxt_suspend_mode { - MXT_SUSPEND_DEEP_SLEEP = 0, - MXT_SUSPEND_T9_CTRL = 1, -}; - /* Config update context */ struct mxt_cfg { u8 *raw; @@ -333,6 +333,8 @@ struct mxt_data { u8 stylus_aux_pressure; u8 stylus_aux_peak; bool use_retrigen_workaround; + struct regulator *reg_vdd; + struct regulator *reg_avdd; /* Cached parameters from object table */ u16 T5_address; @@ -2073,6 +2075,94 @@ static int mxt_read_info_block(struct mxt_data *data) return error; } +static void mxt_regulator_enable(struct mxt_data *data) +{ + int error; + + if (!data->reg_vdd || !data->reg_avdd) + return; + + gpiod_set_value(data->reset_gpio, 0); + + error = regulator_enable(data->reg_vdd); + if (error) + return; + + error = regulator_enable(data->reg_avdd); + if (error) { + regulator_disable(data->reg_vdd); + return; + } + + /* + * According to maXTouch power sequencing specification, RESET line + * must be kept low until some time after regulators come up to + * voltage + */ + msleep(MXT_REGULATOR_DELAY); + gpiod_set_value(data->reset_gpio, 1); + msleep(MXT_CHG_DELAY); + +retry_wait: + reinit_completion(&data->bl_completion); + data->in_bootloader = true; + error = mxt_wait_for_completion(data, &data->bl_completion, + MXT_POWERON_DELAY); + if (error == -EINTR) + goto retry_wait; + + data->in_bootloader = false; +} + +static void mxt_regulator_disable(struct mxt_data *data) +{ + if (!data->reg_vdd || !data->reg_avdd) + return; + + if (regulator_is_enabled(data->reg_vdd)) + regulator_disable(data->reg_vdd); + if (regulator_is_enabled(data->reg_avdd)) + regulator_disable(data->reg_avdd); +} + +static int mxt_probe_regulators(struct mxt_data *data) +{ + struct device *dev = &data->client->dev; + int error; + + /* Must have reset GPIO to use regulator support */ + if (!data->reset_gpio) { + error = -EINVAL; + goto fail; + } + + data->reg_vdd = devm_regulator_get(dev, "vdd"); + if (IS_ERR(data->reg_vdd)) { + error = PTR_ERR(data->reg_vdd); + dev_err(dev, "Error %d getting vdd regulator\n", error); + goto fail; + } + + data->reg_avdd = devm_regulator_get(dev, "avdd"); + if (IS_ERR(data->reg_avdd)) { + error = PTR_ERR(data->reg_avdd); + dev_err(dev, "Error %d getting avdd regulator\n", error); + goto fail_release; + } + + mxt_regulator_enable(data); + + dev_dbg(dev, "Initialised regulators\n"); + return 0; + +fail_release: + regulator_put(data->reg_vdd); +fail: + data->reg_vdd = NULL; + data->reg_avdd = NULL; + return error; +} + static int mxt_read_t9_resolution(struct mxt_data *data) { struct i2c_client *client = data->client; @@ -3139,7 +3229,12 @@ static int mxt_load_fw(struct device *dev, const char *fn) goto release_firmware; if (data->suspended) { - enable_irq(data->irq); + if (data->suspend_mode == MXT_SUSPEND_REGULATOR) + mxt_regulator_enable(data); + + if (data->suspend_mode == MXT_SUSPEND_DEEP_SLEEP) + enable_irq(data->irq); + data->suspended = false; } @@ -3345,6 +3440,11 @@ static void mxt_start(struct mxt_data *data) MXT_TOUCH_MULTI_T9, MXT_T9_CTRL, 0x83); break; + case MXT_SUSPEND_REGULATOR: + enable_irq(data->irq); + mxt_regulator_enable(data); + break; + case MXT_SUSPEND_DEEP_SLEEP: default: /* @@ -3377,6 +3477,12 @@ static void mxt_stop(struct mxt_data *data) MXT_TOUCH_MULTI_T9, MXT_T9_CTRL, 0); break; + case MXT_SUSPEND_REGULATOR: + disable_irq(data->irq); + mxt_regulator_disable(data); + mxt_reset_slots(data); + break; + case MXT_SUSPEND_DEEP_SLEEP: default: disable_irq(data->irq); @@ -3469,6 +3575,8 @@ static int mxt_parse_device_properties(struct mxt_data *data) data->t15_num_keys = n_keys; } + device_property_read_u32(dev, "atmel,suspend-mode", + &data->suspend_mode); return 0; } @@ -3555,14 +3663,18 @@ static int mxt_probe(struct i2c_client *client, const struct i2c_device_id *id) return error; } - disable_irq(client->irq); - - if (data->reset_gpio) { + if (data->suspend_mode == MXT_SUSPEND_REGULATOR) { + error = mxt_probe_regulators(data); + if (error) + return error; + } else if (data->reset_gpio) { msleep(MXT_RESET_GPIO_TIME); gpiod_set_value(data->reset_gpio, 1); msleep(MXT_RESET_INVALID_CHG); } + disable_irq(data->irq); + error = mxt_initialize(data); if (error) return error; -- 2.17.1