Received: by 10.223.185.116 with SMTP id b49csp2664566wrg; Mon, 5 Mar 2018 06:51:22 -0800 (PST) X-Google-Smtp-Source: AG47ELvl0dwWLHBihmLwHXY0IiEi5vsXtMt9Xza5IMIXKPFNb3suyzl1abkdFKV58NErH53AeYZ9 X-Received: by 10.98.218.7 with SMTP id c7mr10712139pfh.162.1520261482522; Mon, 05 Mar 2018 06:51:22 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1520261482; cv=none; d=google.com; s=arc-20160816; b=KIVfPVWNONxwhMyRUdZdzlizXbGmEk/b7vYPGG5zVreVsn/spQwehvx6VhsDtcOEKM /upUHbW8O5Zt5tzU7r3g+qRLzNscneGWSCqA6RZpXgfrBVBWTnW7fQFMZCLgMD3KCqCD eV1dKgBjsLrZHuFYFv+sENoyED0VEYA8NgsjKUqWpt6FQximq2Y0l//QdQS1mwB1SofE tT1HRIjgi/cDljhvgjn55owfBwbDkfs6XLDmC/aLjIsEdbK+HgBI4MNvl+vdKkxoaU8K 3nvo4YLRh0ajYbgRUoinfBYRTpxrblKfrKogO7+HdV+9qxF3FVZ/4HgnBkq1gub/miy7 rzxQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject:dkim-signature :arc-authentication-results; bh=SN2lVyguLYXCsYRiL9/lvw92lWmyeDnOydcd+kL8qD8=; b=T9ozTL2zWjgt38WN+8/JTdpbnNaMO9vTQpTJCH1pNJKLcynbAF27CewyreUOKb2Lbq j1iWvrsa4KmpAV9+cBY/MOSM/Icq5b1uRgCuERc3q28dsoy6jKtI2YmII2j/eiEMkiGi 2xir8vaxjUCCvtCyJ4lAxpVmfnw8UOFVqnQinTpyj2tUST9fXNBNp8T0RgWi+ZoSsWfX smjjm/LRw3PvdMBCIPKWvJjTObTPXMbsbVE19/b3CjzCidCdpD4ONNFBsXikeTmLdGqr FWTnyZl2uqhzwxxrPZdI4pNuzfuYq4SeiTQA/L0ncR1hf9eqOItFca4VqZ3Vm6TZAu6F iIkQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=fail header.i=@gmail.com header.s=20161025 header.b=tgA1swVD; 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 k30si8451126pgn.319.2018.03.05.06.51.07; Mon, 05 Mar 2018 06:51:22 -0800 (PST) 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; dkim=fail header.i=@gmail.com header.s=20161025 header.b=tgA1swVD; 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 S1752407AbeCEOtZ (ORCPT + 99 others); Mon, 5 Mar 2018 09:49:25 -0500 Received: from mail-pl0-f65.google.com ([209.85.160.65]:46044 "EHLO mail-pl0-f65.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1752063AbeCEOtV (ORCPT ); Mon, 5 Mar 2018 09:49:21 -0500 Received: by mail-pl0-f65.google.com with SMTP id v9-v6so9740793plp.12; Mon, 05 Mar 2018 06:49:20 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=sender:subject:to:cc:references:from:message-id:date:user-agent :mime-version:in-reply-to:content-language:content-transfer-encoding; bh=SN2lVyguLYXCsYRiL9/lvw92lWmyeDnOydcd+kL8qD8=; b=tgA1swVDOOzdWmRZRNpcaIixQHOshVEgq2i6L0oV+5xFacYSLe4Aq8VEynyErS4DgB i8N7DXcaX2tKGHCIFK7tXI/3ewKAAHeT8hJn27X0LX0UFTDE+rPNTKswRyCb05PO4gFn wmYgyeBidVWuLIsT2bAfPkyal84DtJX0Ix/ztxvOfVqDelBVaub84G3VkKjI4VGyYiWW LDfNa0Vk9UAsoZTgHEThoTK1EzU6737cCm6Hv8g7ZcWkpoaNkEZInopxLEIQnEgUuBbb kA3HrYB4pYAfsRPnE8pP7JSyiTQHXpkJUVKjjlQDhK0aYSepTb6h4ylYO2wUD0hcHPEr K/ng== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:sender:subject:to:cc:references:from:message-id :date:user-agent:mime-version:in-reply-to:content-language :content-transfer-encoding; bh=SN2lVyguLYXCsYRiL9/lvw92lWmyeDnOydcd+kL8qD8=; b=g+SkbIGJ9Z8K3I3zkuz6jPaNWUJdk6bQbUY3lvJzfdSYkBWKsO+0Vs7DnERjbZhxQ3 /YDZrWaJj41dCDWf9bBuNe1dqpcY/74hqUMuY+QQd7lkI+d1K5h7UIBOBugeyImSBu2a HdUTAtmgtJoy3NzLPHHxLXGW4a8ZpOyzVQmYfZycp92Ovsx5OKtBwrBBrN3E4OUhKqmQ ciKiHhSaeyFyFrNCAC0P5hlDeQyyyWS+tcKsHRRp2UO+CN2jOMBLxgihsEnRPi+ZaoNX 9HeNxgOqrqs+Y8k2xJnRiJyFoX2qOJJwaYMououMshf8D3GrXD5tnte6w5BQxtqml5M7 ZWhg== X-Gm-Message-State: APf1xPBYuWJnHUuFhQge8FK8XvodbW5o0Wll0SDIUJbyaj47Q2SjVRjv 8fOolCoLLqUrZFBF1AFZRqkKH4Is X-Received: by 2002:a17:902:b704:: with SMTP id d4-v6mr13661145pls.406.1520261360185; Mon, 05 Mar 2018 06:49:20 -0800 (PST) Received: from server.roeck-us.net (108-223-40-66.lightspeed.sntcca.sbcglobal.net. [108.223.40.66]) by smtp.gmail.com with ESMTPSA id 83sm27007111pfj.151.2018.03.05.06.49.18 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Mon, 05 Mar 2018 06:49:19 -0800 (PST) Subject: Re: [PATCH v6] staging: typec: handle vendor defined part and modify drp toggling flow To: ShuFan Lee , heikki.krogerus@linux.intel.com, greg@kroah.com Cc: shufan_lee@richtek.com, cy_huang@richtek.com, jun.li@nxp.com, linux-kernel@vger.kernel.org, linux-usb@vger.kernel.org References: <1520223402-19906-1-git-send-email-leechu729@gmail.com> From: Guenter Roeck Message-ID: <646aee73-935f-0cbc-5d61-8697f8bba4ac@roeck-us.net> Date: Mon, 5 Mar 2018 06:49:17 -0800 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:52.0) Gecko/20100101 Thunderbird/52.6.0 MIME-Version: 1.0 In-Reply-To: <1520223402-19906-1-git-send-email-leechu729@gmail.com> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 03/04/2018 08:16 PM, ShuFan Lee wrote: > From: ShuFan Lee > > Handle vendor defined behavior in tcpci_init, tcpci_set_vconn, tcpci_start_drp_toggling > and export tcpci_irq. More operations can be extended in tcpci_data if needed. > According to TCPCI specification, 4.4.5.2 ROLE_CONTROL, > TCPC shall not start DRP toggling until subsequently the TCPM > writes to the COMMAND register to start DRP toggling. > DRP toggling flow is chagned as following: s/chagned/changed/ > - Write DRP = 1, Rp level and RC.CCx to Rd/Rd or Rp/Rp > - Set LOOK4CONNECTION command > > Signed-off-by: ShuFan Lee > --- > drivers/staging/typec/tcpci.c | 134 ++++++++++++++++++++++++++++++++++-------- > drivers/staging/typec/tcpci.h | 15 +++++ > 2 files changed, 123 insertions(+), 26 deletions(-) > > patch changelogs between v1 & v2 > - Remove unnecessary i2c_client in the structure of tcpci > - Rename structure of tcpci_vendor_data to tcpci_data > - Not exporting tcpci read/write wrappers but register i2c regmap in glue driver > - Add set_vconn ops in tcpci_data > (It is necessary for RT1711H to enable/disable idle mode before disabling/enabling vconn) > - Export tcpci_irq so that vendor can call it in their own IRQ handler > > patch changelogs between v2 & v3 > - Change the return type of tcpci_irq from int to irqreturn_t > > patch changelogs between v3 & v4 > - Directly return regmap_write at the end of drp_toggling function > - Because tcpci_irq is called in _tcpci_irq, move _tcpci_irq to the place after tcpci_irq > > patch changelogs between v4 & v5 > - Add a space between my first & last name, from ShuFanLee to ShuFan Lee. > > patch changelogs between v5 & v6 > - Add start_drp_toggling in tcpci_data and call it at the beginning of tcpci_start_drp_toggling > - Modify the flow of tcpci_start_drp_toggling as following > - Set Rp level, RC.CCx to Rd/Rd or Rp/Rp and DRP = 1 > - Set LOOK4CONNECTION command > > diff --git a/drivers/staging/typec/tcpci.c b/drivers/staging/typec/tcpci.c > index 9bd4412..9e600f7 100644 > --- a/drivers/staging/typec/tcpci.c > +++ b/drivers/staging/typec/tcpci.c > @@ -21,7 +21,6 @@ > > struct tcpci { > struct device *dev; > - struct i2c_client *client; > > struct tcpm_port *port; > > @@ -30,6 +29,12 @@ struct tcpci { > bool controls_vbus; > > struct tcpc_dev tcpc; > + struct tcpci_data *data; > +}; > + > +struct tcpci_chip { > + struct tcpci *tcpci; > + struct tcpci_data data; > }; > > static inline struct tcpci *tcpc_to_tcpci(struct tcpc_dev *tcpc) > @@ -37,8 +42,7 @@ static inline struct tcpci *tcpc_to_tcpci(struct tcpc_dev *tcpc) > return container_of(tcpc, struct tcpci, tcpc); > } > > -static int tcpci_read16(struct tcpci *tcpci, unsigned int reg, > - u16 *val) > +static int tcpci_read16(struct tcpci *tcpci, unsigned int reg, u16 *val) > { > return regmap_raw_read(tcpci->regmap, reg, val, sizeof(u16)); > } > @@ -98,9 +102,19 @@ static int tcpci_set_cc(struct tcpc_dev *tcpc, enum typec_cc_status cc) > static int tcpci_start_drp_toggling(struct tcpc_dev *tcpc, > enum typec_cc_status cc) > { > + int ret; > struct tcpci *tcpci = tcpc_to_tcpci(tcpc); > unsigned int reg = TCPC_ROLE_CTRL_DRP; > > + if (tcpci->data) { > + if (tcpci->data->start_drp_toggling) { From the code flow it is guaranteed that ->data is set. It should therefore be unnecessary to check for it (we don't check if ->reg is set either). > + ret = tcpci->data->start_drp_toggling(tcpci, > + tcpci->data, cc); > + if (ret < 0) > + return ret; > + } > + } > + > switch (cc) { > default: > case TYPEC_CC_RP_DEF: > @@ -117,7 +131,17 @@ static int tcpci_start_drp_toggling(struct tcpc_dev *tcpc, > break; > } > > - return regmap_write(tcpci->regmap, TCPC_ROLE_CTRL, reg); > + if (cc == TYPEC_CC_RD) > + reg |= (TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC1_SHIFT) | > + (TCPC_ROLE_CTRL_CC_RD << TCPC_ROLE_CTRL_CC2_SHIFT); > + else > + reg |= (TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC1_SHIFT) | > + (TCPC_ROLE_CTRL_CC_RP << TCPC_ROLE_CTRL_CC2_SHIFT); > + ret = regmap_write(tcpci->regmap, TCPC_ROLE_CTRL, reg); > + if (ret < 0) > + return ret; > + return regmap_write(tcpci->regmap, TCPC_COMMAND, > + TCPC_CMD_LOOK4CONNECTION); > } > > static enum typec_cc_status tcpci_to_typec_cc(unsigned int cc, bool sink) > @@ -178,6 +202,16 @@ static int tcpci_set_vconn(struct tcpc_dev *tcpc, bool enable) > struct tcpci *tcpci = tcpc_to_tcpci(tcpc); > int ret; > > + /* Handle vendor set vconn */ > + if (tcpci->data) { > + if (tcpci->data->set_vconn) { > + ret = tcpci->data->set_vconn(tcpci, tcpci->data, > + enable); > + if (ret < 0) > + return ret; > + } > + } > + > ret = regmap_write(tcpci->regmap, TCPC_POWER_CTRL, > enable ? TCPC_POWER_CTRL_VCONN_ENABLE : 0); > if (ret < 0) > @@ -323,6 +357,15 @@ static int tcpci_init(struct tcpc_dev *tcpc) > if (time_after(jiffies, timeout)) > return -ETIMEDOUT; > > + /* Handle vendor init */ > + if (tcpci->data) { > + if (tcpci->data->init) { > + ret = tcpci->data->init(tcpci, tcpci->data); > + if (ret < 0) > + return ret; > + } > + } > + > /* Clear all events */ > ret = tcpci_write16(tcpci, TCPC_ALERT, 0xffff); > if (ret < 0) > @@ -344,9 +387,8 @@ static int tcpci_init(struct tcpc_dev *tcpc) > return tcpci_write16(tcpci, TCPC_ALERT_MASK, reg); > } > > -static irqreturn_t tcpci_irq(int irq, void *dev_id) > +irqreturn_t tcpci_irq(struct tcpci *tcpci) > { > - struct tcpci *tcpci = dev_id; > u16 status; > > tcpci_read16(tcpci, TCPC_ALERT, &status); > @@ -412,6 +454,14 @@ static irqreturn_t tcpci_irq(int irq, void *dev_id) > > return IRQ_HANDLED; > } > +EXPORT_SYMBOL_GPL(tcpci_irq); > + > +static irqreturn_t _tcpci_irq(int irq, void *dev_id) > +{ > + struct tcpci *tcpci = dev_id; > + > + return tcpci_irq(tcpci); > +} > > static const struct regmap_config tcpci_regmap_config = { > .reg_bits = 8, > @@ -435,22 +485,18 @@ static int tcpci_parse_config(struct tcpci *tcpci) > return 0; > } > > -static int tcpci_probe(struct i2c_client *client, > - const struct i2c_device_id *i2c_id) > +struct tcpci *tcpci_register_port(struct device *dev, struct tcpci_data *data) > { > struct tcpci *tcpci; > int err; > > - tcpci = devm_kzalloc(&client->dev, sizeof(*tcpci), GFP_KERNEL); > + tcpci = devm_kzalloc(dev, sizeof(*tcpci), GFP_KERNEL); > if (!tcpci) > - return -ENOMEM; > + return ERR_PTR(-ENOMEM); > > - tcpci->client = client; > - tcpci->dev = &client->dev; > - i2c_set_clientdata(client, tcpci); > - tcpci->regmap = devm_regmap_init_i2c(client, &tcpci_regmap_config); > - if (IS_ERR(tcpci->regmap)) > - return PTR_ERR(tcpci->regmap); > + tcpci->dev = dev; > + tcpci->data = data; > + tcpci->regmap = data->regmap; > > tcpci->tcpc.init = tcpci_init; > tcpci->tcpc.get_vbus = tcpci_get_vbus; > @@ -467,27 +513,63 @@ static int tcpci_probe(struct i2c_client *client, > > err = tcpci_parse_config(tcpci); > if (err < 0) > - return err; > + return ERR_PTR(err); > + > + tcpci->port = tcpm_register_port(tcpci->dev, &tcpci->tcpc); > + if (PTR_ERR_OR_ZERO(tcpci->port)) > + return ERR_CAST(tcpci->port); > > - /* Disable chip interrupts */ > - tcpci_write16(tcpci, TCPC_ALERT_MASK, 0); > + return tcpci; > +} > +EXPORT_SYMBOL_GPL(tcpci_register_port); > + > +void tcpci_unregister_port(struct tcpci *tcpci) > +{ > + tcpm_unregister_port(tcpci->port); > +} > +EXPORT_SYMBOL_GPL(tcpci_unregister_port); > > - err = devm_request_threaded_irq(tcpci->dev, client->irq, NULL, > - tcpci_irq, > +static int tcpci_probe(struct i2c_client *client, > + const struct i2c_device_id *i2c_id) > +{ > + struct tcpci_chip *chip; > + int err; > + u16 val = 0; > + > + chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL); > + if (!chip) > + return -ENOMEM; > + > + chip->data.regmap = devm_regmap_init_i2c(client, &tcpci_regmap_config); > + if (IS_ERR(chip->data.regmap)) > + return PTR_ERR(chip->data.regmap); > + > + /* Disable chip interrupts before requesting irq */ > + err = regmap_raw_write(chip->data.regmap, TCPC_ALERT_MASK, &val, > + sizeof(u16)); > + if (err < 0) > + return err; > + > + err = devm_request_threaded_irq(&client->dev, client->irq, NULL, > + _tcpci_irq, > IRQF_ONESHOT | IRQF_TRIGGER_LOW, > - dev_name(tcpci->dev), tcpci); > + dev_name(&client->dev), chip); > if (err < 0) > return err; > > - tcpci->port = tcpm_register_port(tcpci->dev, &tcpci->tcpc); > - return PTR_ERR_OR_ZERO(tcpci->port); > + chip->tcpci = tcpci_register_port(&client->dev, &chip->data); > + if (PTR_ERR_OR_ZERO(chip->tcpci)) > + return PTR_ERR(chip->tcpci); > + > + i2c_set_clientdata(client, chip); > + return 0; > } > > static int tcpci_remove(struct i2c_client *client) > { > - struct tcpci *tcpci = i2c_get_clientdata(client); > + struct tcpci_chip *chip = i2c_get_clientdata(client); > > - tcpm_unregister_port(tcpci->port); > + tcpci_unregister_port(chip->tcpci); > > return 0; > } > diff --git a/drivers/staging/typec/tcpci.h b/drivers/staging/typec/tcpci.h > index fdfb06c..a2c1754 100644 > --- a/drivers/staging/typec/tcpci.h > +++ b/drivers/staging/typec/tcpci.h > @@ -59,6 +59,7 @@ > #define TCPC_POWER_CTRL_VCONN_ENABLE BIT(0) > > #define TCPC_CC_STATUS 0x1d > +#define TCPC_CC_STATUS_DRPRST BIT(5) > #define TCPC_CC_STATUS_TERM BIT(4) > #define TCPC_CC_STATUS_CC2_SHIFT 2 > #define TCPC_CC_STATUS_CC2_MASK 0x3 > @@ -121,4 +122,18 @@ > #define TCPC_VBUS_VOLTAGE_ALARM_HI_CFG 0x76 > #define TCPC_VBUS_VOLTAGE_ALARM_LO_CFG 0x78 > > +struct tcpci; > +struct tcpci_data { > + struct regmap *regmap; > + int (*init)(struct tcpci *tcpci, struct tcpci_data *data); > + int (*set_vconn)(struct tcpci *tcpci, struct tcpci_data *data, > + bool enable); > + int (*start_drp_toggling)(struct tcpci *tcpci, struct tcpci_data *data, > + enum typec_cc_status cc); > +}; > + > +struct tcpci *tcpci_register_port(struct device *dev, struct tcpci_data *data); > +void tcpci_unregister_port(struct tcpci *tcpci); > +irqreturn_t tcpci_irq(struct tcpci *tcpci); > + > #endif /* __LINUX_USB_TCPCI_H */ >