Received: by 2002:a25:d7c1:0:0:0:0:0 with SMTP id o184csp5385696ybg; Tue, 22 Oct 2019 02:24:15 -0700 (PDT) X-Google-Smtp-Source: APXvYqy0AgU5TzyaQO3R6P0t0xk/zu8RPmPkpFbp36EgxoaozK3XL11p3sEGPFvvrxU3UnJIMYs0 X-Received: by 2002:a05:6402:13c5:: with SMTP id a5mr11173455edx.106.1571736255442; Tue, 22 Oct 2019 02:24:15 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1571736255; cv=none; d=google.com; s=arc-20160816; b=jvN5wsScOb2wuQU6FqAWwnRnm/RE15YrxiHNtxvj+qlujJILKdelvddnkOM9OV4BBq HKrphb20CE63QBcx7T+OugZ3G/URqoMjCJBO3/PfwhddSWoJeDC0jdAJxSPPGQpNo0oq Bzsj6beh4Xp+zPgi6xDXd9FXq4nXUc29BHt2zsqQKwUYsKQ9E2HqEmGUMHgsGx3gquZs nS1L8k1jshCVRA1YlAyzBIX5LNrS45ulxr5GOFXZCsQvDig6fTydpFjh9skUb+PHwYoe kFkUtIfsua19PP1Y8N8oJBuOsDDspfeVcU0VpXMdTxoe9KqbEA6cBhi9sXVqtC8hITgI vDZA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-language :content-transfer-encoding:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject; bh=S0n9YUpj/P2nM4lvJ65Y1ExPNZN9Tx6sSWD0VUqK/+U=; b=lgnWBNBzTcbSNQ6YY3Oc2smqElnm2hC8hx8S0GhI4n0Sqd4l0Hw3Kbxoz2PvCoDPmi KblrxMi25XGwXBAADw7NXixY+dprr8LNIJ+zfT8+RuBMZfYW66daGrNgKvQJSkum6ik0 mlZVluJk6qXnD/+n+WVNVfdErIB6zPErve1ZJXkf+nzNLlqddq7JAk4JVRBdD4rQGAZQ csbEh/Qr+T3F7BO0M70CLOjwboV5Dz8BTvVM0yI0S1RQUjbirzGTF6yuzqdmIiGTMwZA qzK2exjsTR8WmMMCK7kmjVlPqxRi7vQaxgFiwx9trdD7Zyjwg/cir0L2jdG5rMl8izh1 YshQ== 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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id l24si11400606edq.90.2019.10.22.02.23.50; Tue, 22 Oct 2019 02:24:15 -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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2388566AbfJVJUN (ORCPT + 99 others); Tue, 22 Oct 2019 05:20:13 -0400 Received: from mga06.intel.com ([134.134.136.31]:34158 "EHLO mga06.intel.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2388423AbfJVJUM (ORCPT ); Tue, 22 Oct 2019 05:20:12 -0400 X-Amp-Result: SKIPPED(no attachment in message) X-Amp-File-Uploaded: False Received: from orsmga002.jf.intel.com ([10.7.209.21]) by orsmga104.jf.intel.com with ESMTP/TLS/DHE-RSA-AES256-GCM-SHA384; 22 Oct 2019 02:20:07 -0700 X-ExtLoop1: 1 X-IronPort-AV: E=Sophos;i="5.67,326,1566889200"; d="scan'208";a="209620833" Received: from linux.intel.com ([10.54.29.200]) by orsmga002.jf.intel.com with ESMTP; 22 Oct 2019 02:20:04 -0700 Received: from [10.226.39.21] (unknown [10.226.39.21]) by linux.intel.com (Postfix) with ESMTP id 6786A580127; Tue, 22 Oct 2019 02:20:01 -0700 (PDT) Subject: Re: [PATCH v4 3/3] pci: intel: Add sysfs attributes to configure pcie link To: Andrew Murray Cc: jingoohan1@gmail.com, gustavo.pimentel@synopsys.com, lorenzo.pieralisi@arm.com, robh@kernel.org, martin.blumenstingl@googlemail.com, linux-pci@vger.kernel.org, hch@infradead.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, andriy.shevchenko@intel.com, cheol.yong.kim@intel.com, chuanhua.lei@linux.intel.com, qi-ming.wu@intel.com References: <20191021133849.GQ47056@e119886-lin.cambridge.arm.com> From: Dilip Kota Message-ID: <6a209452-f569-4f6a-8aea-5c9f84167f5a@linux.intel.com> Date: Tue, 22 Oct 2019 17:20:00 +0800 User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64; rv:60.0) Gecko/20100101 Thunderbird/60.9.0 MIME-Version: 1.0 In-Reply-To: <20191021133849.GQ47056@e119886-lin.cambridge.arm.com> Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: 7bit Content-Language: en-US Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Andrew Murray, On 10/21/2019 9:38 PM, Andrew Murray wrote: > On Mon, Oct 21, 2019 at 02:39:20PM +0800, Dilip Kota wrote: >> PCIe RC driver on Intel Gateway SoCs have a requirement >> of changing link width and speed on the fly. >> So add the sysfs attributes to show and store the link >> properties. >> Add the respective link resize function in pcie DesignWare >> framework so that Intel PCIe driver can use during link >> width configuration on the fly. >> >> Signed-off-by: Dilip Kota >> --- >> drivers/pci/controller/dwc/pcie-designware.c | 9 +++ >> drivers/pci/controller/dwc/pcie-designware.h | 3 + >> drivers/pci/controller/dwc/pcie-intel-gw.c | 112 ++++++++++++++++++++++++++- >> 3 files changed, 123 insertions(+), 1 deletion(-) >> >> diff --git a/drivers/pci/controller/dwc/pcie-designware.c b/drivers/pci/controller/dwc/pcie-designware.c >> index 4c391bfd681a..662fdcb4f2d6 100644 >> --- a/drivers/pci/controller/dwc/pcie-designware.c >> +++ b/drivers/pci/controller/dwc/pcie-designware.c >> @@ -474,6 +474,15 @@ int dw_pcie_link_up(struct dw_pcie *pci) >> (!(val & PCIE_PORT_DEBUG1_LINK_IN_TRAINING))); >> } >> >> +void dw_pcie_link_width_resize(struct dw_pcie *pci, u32 lane_width) >> +{ >> + u32 val; >> + >> + val = dw_pcie_readl_dbi(pci, PCIE_PORT_MULTI_LANE_CTRL); >> + val &= ~(PORT_MLTI_LNK_WDTH_CHNG | PORT_MLTI_LNK_WDTH); >> + val |= PORT_MLTI_LNK_WDTH_CHNG | lane_width; >> + dw_pcie_writel_dbi(pci, PCIE_PORT_MULTI_LANE_CTRL, val); >> +} >> >> void dw_pcie_upconfig_setup(struct dw_pcie *pci) >> { >> diff --git a/drivers/pci/controller/dwc/pcie-designware.h b/drivers/pci/controller/dwc/pcie-designware.h >> index 3beac10e4a4c..fcf0442341fd 100644 >> --- a/drivers/pci/controller/dwc/pcie-designware.h >> +++ b/drivers/pci/controller/dwc/pcie-designware.h >> @@ -67,6 +67,8 @@ >> #define PCIE_MSI_INTR0_STATUS 0x830 >> >> #define PCIE_PORT_MULTI_LANE_CTRL 0x8C0 >> +#define PORT_MLTI_LNK_WDTH GENMASK(5, 0) >> +#define PORT_MLTI_LNK_WDTH_CHNG BIT(6) >> #define PORT_MLTI_UPCFG_SUPPORT BIT(7) >> >> #define PCIE_ATU_VIEWPORT 0x900 >> @@ -282,6 +284,7 @@ void dw_pcie_write_dbi2(struct dw_pcie *pci, u32 reg, size_t size, u32 val); >> u32 dw_pcie_read_atu(struct dw_pcie *pci, u32 reg, size_t size); >> void dw_pcie_write_atu(struct dw_pcie *pci, u32 reg, size_t size, u32 val); >> int dw_pcie_link_up(struct dw_pcie *pci); >> +void dw_pcie_link_width_resize(struct dw_pcie *pci, u32 lane_width); >> void dw_pcie_upconfig_setup(struct dw_pcie *pci); >> void dw_pcie_link_speed_change(struct dw_pcie *pci, bool enable); >> void dw_pcie_link_set_n_fts(struct dw_pcie *pci, u32 n_fts); >> diff --git a/drivers/pci/controller/dwc/pcie-intel-gw.c b/drivers/pci/controller/dwc/pcie-intel-gw.c >> index 9142c70db808..b9be0921671d 100644 >> --- a/drivers/pci/controller/dwc/pcie-intel-gw.c >> +++ b/drivers/pci/controller/dwc/pcie-intel-gw.c >> @@ -146,6 +146,22 @@ static void intel_pcie_ltssm_disable(struct intel_pcie_port *lpp) >> pcie_app_wr_mask(lpp, PCIE_APP_CCR_LTSSM_ENABLE, 0, PCIE_APP_CCR); >> } >> >> +static const char *pcie_link_gen_to_str(int gen) >> +{ >> + switch (gen) { >> + case PCIE_LINK_SPEED_GEN1: >> + return "2.5"; >> + case PCIE_LINK_SPEED_GEN2: >> + return "5.0"; >> + case PCIE_LINK_SPEED_GEN3: >> + return "8.0"; >> + case PCIE_LINK_SPEED_GEN4: >> + return "16.0"; >> + default: >> + return "???"; >> + } >> +} >> + >> static void intel_pcie_link_setup(struct intel_pcie_port *lpp) >> { >> u32 val; >> @@ -444,6 +460,91 @@ static int intel_pcie_host_setup(struct intel_pcie_port *lpp) >> return ret; >> } >> >> +static ssize_t pcie_link_status_show(struct device *dev, >> + struct device_attribute *attr, char *buf) >> +{ >> + struct intel_pcie_port *lpp = dev_get_drvdata(dev); >> + u32 reg, width, gen; >> + >> + reg = pcie_rc_cfg_rd(lpp, PCIE_CAP_OFST + PCI_EXP_LNKCTL); >> + width = FIELD_GET(PCI_EXP_LNKSTA_NLW, reg >> 16); >> + gen = FIELD_GET(PCI_EXP_LNKSTA_CLS, reg >> 16); >> + >> + if (gen > lpp->max_speed) >> + return -EINVAL; >> + >> + return sprintf(buf, "Port %2u Width x%u Speed %s GT/s\n", lpp->id, >> + width, pcie_link_gen_to_str(gen)); >> +} >> +static DEVICE_ATTR_RO(pcie_link_status); >> + >> +static ssize_t pcie_speed_store(struct device *dev, >> + struct device_attribute *attr, >> + const char *buf, size_t len) >> +{ >> + struct intel_pcie_port *lpp = dev_get_drvdata(dev); >> + unsigned long val; >> + int ret; >> + >> + ret = kstrtoul(buf, 10, &val); >> + if (ret) >> + return ret; >> + >> + if (val > lpp->max_speed) >> + return -EINVAL; >> + >> + lpp->link_gen = val; >> + intel_pcie_max_speed_setup(lpp); >> + dw_pcie_link_speed_change(&lpp->pci, false); >> + dw_pcie_link_speed_change(&lpp->pci, true); >> + >> + return len; >> +} >> +static DEVICE_ATTR_WO(pcie_speed); >> + >> +/* >> + * Link width change on the fly is not always successful. >> + * It also depends on the partner. >> + */ >> +static ssize_t pcie_width_store(struct device *dev, >> + struct device_attribute *attr, >> + const char *buf, size_t len) >> +{ >> + struct intel_pcie_port *lpp = dev_get_drvdata(dev); >> + unsigned long val; >> + int ret; >> + >> + lpp = dev_get_drvdata(dev); >> + >> + ret = kstrtoul(buf, 10, &val); >> + if (ret) >> + return ret; >> + >> + if (val > lpp->max_width) >> + return -EINVAL; >> + >> + /* HW auto bandwidth negotiation must be enabled */ >> + pcie_rc_cfg_wr_mask(lpp, PCI_EXP_LNKCTL_HAWD, 0, >> + PCIE_CAP_OFST + PCI_EXP_LNKCTL); >> + dw_pcie_link_width_resize(&lpp->pci, val); >> + >> + return len; >> +} >> +static DEVICE_ATTR_WO(pcie_width); >> + >> +static struct attribute *pcie_cfg_attrs[] = { >> + &dev_attr_pcie_link_status.attr, >> + &dev_attr_pcie_speed.attr, >> + &dev_attr_pcie_width.attr, >> + NULL, >> +}; > Is there a reason that these are limited only to the Intel driver and > not the wider set of DWC drivers? > > Is there anything specific here about the Intel GW driver? Yes, they need intel_pcie_max_speed_setup() and pcie_link_gen_to_str(). Once intel_pcie_max_speed_setup() moved to DesignWare framework (as per Bjorn Helgaas inputs) and use pcie_link_speed[] array instead of pcie_link_gen_to_str() (as per gustavo pimentel inputs) we can move this to PCIe DesignWare framework or to pci sysfs file. Regards, Dilip > > Thanks, > > Andrew Murray > >> +ATTRIBUTE_GROUPS(pcie_cfg); >> + >> +static int intel_pcie_sysfs_init(struct intel_pcie_port *lpp) >> +{ >> + return devm_device_add_groups(lpp->pci.dev, pcie_cfg_groups); >> +} >> + >> static void __intel_pcie_remove(struct intel_pcie_port *lpp) >> { >> intel_pcie_core_irq_disable(lpp); >> @@ -490,8 +591,17 @@ static int intel_pcie_rc_init(struct pcie_port *pp) >> { >> struct dw_pcie *pci = to_dw_pcie_from_pp(pp); >> struct intel_pcie_port *lpp = dev_get_drvdata(pci->dev); >> + int ret; >> >> - return intel_pcie_host_setup(lpp); >> + ret = intel_pcie_host_setup(lpp); >> + if (ret) >> + return ret; >> + >> + ret = intel_pcie_sysfs_init(lpp); >> + if (ret) >> + __intel_pcie_remove(lpp); >> + >> + return ret; >> } >> >> int intel_pcie_msi_init(struct pcie_port *pp) >> -- >> 2.11.0 >>