Message ID | 20230214164135.17039-3-quic_devipriy@quicinc.com |
---|---|
State | New |
Headers |
Return-Path: <linux-kernel-owner@vger.kernel.org> Delivered-To: ouuuleilei@gmail.com Received: by 2002:adf:eb09:0:0:0:0:0 with SMTP id s9csp3085500wrn; Tue, 14 Feb 2023 08:48:48 -0800 (PST) X-Google-Smtp-Source: AK7set+WfmuUPPVNTI6WDKqxnh2K4V2BHeDDfqspLQLRO/bRccVkk2c9ghPN7SnU/ZAu3SkLqH9n X-Received: by 2002:a17:902:e887:b0:198:a49b:9f3b with SMTP id w7-20020a170902e88700b00198a49b9f3bmr4425367plg.22.1676393327850; Tue, 14 Feb 2023 08:48:47 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1676393327; cv=none; d=google.com; s=arc-20160816; b=s56OQzUCDHObfbTHJgJTO6UX5aBmAKAEP1c86q93dcN6QIWkowYuoEyJr+++89OKQ5 UTgppWofmEOPp2U3FPoJOBAzLZS6HvBpsWDcph2oj5L4jOTYBtVJralRptnyrFr35pBc EvHsyH31BZpZSqO9iRiS+hk9tNGZmLtB9Uc0HclQtw7FhqZZUdYnC/7oumqweSuzEnFx IG8os+DS6VQJmlYRDGCaERcBs2X+gb7vY+DrwyTRVm6C59Bkx1ZaMyyut3b0zNje0Dnw LoNAAn8OfUGIIVzur3lckiyM1394rizLrTkbgI4bFtTZY61qPdjwn3F1+oUOFx0jqxXY rmiQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:mime-version:references:in-reply-to:message-id :date:subject:cc:to:from:dkim-signature; bh=Eej72nqvlgNpGnu0Iij8jCmTl+SLlvH3sjmLIhu8SZk=; b=uHg9MhOtp7XQpIwHSxk25zOC+KWtEPLxr9KFnN0P5PwuKNir3KBou3u1jRd7D6rN3M 3FfhccESiYOjxshlSrGLD/B7zC7R1F7ZFFmYi2toF7Pnu962G6mdNN9k6kvrW9GO1j8U 4DmuEkEJgQ3aarRMPjhK7Za7mFbQ87tdItsNpkijkCWIaFlbcTsWymaH2nUjWWi/1TeN NN7P4JqW3L4/hPZA2WnmWqYCyYka01vzo4UR+sYLXIP+1+RvqGH/PTarrizWlP6WRJ5V Vfd42W6AjWHjlX6eOJG8U47yT3z4RhQWutO5d7CHImViG26j3ITqvX/ojBcSrk3rmX9R clcA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@quicinc.com header.s=qcppdkim1 header.b="CNinT/zI"; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=quicinc.com Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id d2-20020a170902728200b001992f8b147bsi14868614pll.372.2023.02.14.08.48.34; Tue, 14 Feb 2023 08:48:47 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@quicinc.com header.s=qcppdkim1 header.b="CNinT/zI"; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=quicinc.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232429AbjBNQmy (ORCPT <rfc822;tebrre53rla2o@gmail.com> + 99 others); Tue, 14 Feb 2023 11:42:54 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60792 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229514AbjBNQmt (ORCPT <rfc822;linux-kernel@vger.kernel.org>); Tue, 14 Feb 2023 11:42:49 -0500 Received: from mx0b-0031df01.pphosted.com (mx0b-0031df01.pphosted.com [205.220.180.131]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id AD2C929E03; Tue, 14 Feb 2023 08:42:38 -0800 (PST) Received: from pps.filterd (m0279873.ppops.net [127.0.0.1]) by mx0a-0031df01.pphosted.com (8.17.1.19/8.17.1.19) with ESMTP id 31ECITvB002701; Tue, 14 Feb 2023 16:42:24 GMT DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=quicinc.com; h=from : to : cc : subject : date : message-id : in-reply-to : references : mime-version : content-type; s=qcppdkim1; bh=Eej72nqvlgNpGnu0Iij8jCmTl+SLlvH3sjmLIhu8SZk=; b=CNinT/zI0OtaJpn+t5oB+u21AADpsUUP4T11hEBoruMztq1afPrSV9JmxXPlpqzKSGR+ HScJNsnU9H1i3vsZIESwltonG8F2SCjgEv6UUdW7jpquVaf4u6oNiDAVmbTfIBc3Az/J qwrzy1OPfjHj2p+0rdWOP1Ri0MbWZ9uUG1IsPNG4Ftvrcak8Ugtp7l5FHWhhsaXEicos o+cs25MRZtMlcvOWJcpdClbjZmxKNX7Ba2mqtOABw754MRrVVaDppT3O+fJ3/gWWMp1D hgjDweoDb1W60UxFI+eONyx3I3EiUtw+XAdSPdgf21q2eqvvJk7MXKZkof8oY4qFWppm xg== Received: from nalasppmta05.qualcomm.com (Global_NAT1.qualcomm.com [129.46.96.20]) by mx0a-0031df01.pphosted.com (PPS) with ESMTPS id 3nr4kp9had-1 (version=TLSv1.2 cipher=ECDHE-RSA-AES256-GCM-SHA384 bits=256 verify=NOT); Tue, 14 Feb 2023 16:42:24 +0000 Received: from nalasex01a.na.qualcomm.com (nalasex01a.na.qualcomm.com [10.47.209.196]) by NALASPPMTA05.qualcomm.com (8.17.1.5/8.17.1.5) with ESMTPS id 31EGgN1M017963 (version=TLSv1.2 cipher=ECDHE-RSA-AES256-GCM-SHA384 bits=256 verify=NOT); Tue, 14 Feb 2023 16:42:23 GMT Received: from devipriy-linux.qualcomm.com (10.80.80.8) by nalasex01a.na.qualcomm.com (10.47.209.196) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.2.986.36; Tue, 14 Feb 2023 08:42:15 -0800 From: Devi Priya <quic_devipriy@quicinc.com> To: <agross@kernel.org>, <andersson@kernel.org>, <konrad.dybcio@linaro.org>, <lpieralisi@kernel.org>, <kw@linux.com>, <robh@kernel.org>, <bhelgaas@google.com>, <krzysztof.kozlowski+dt@linaro.org>, <vkoul@kernel.org>, <kishon@kernel.org>, <mturquette@baylibre.com>, <sboyd@kernel.org>, <mani@kernel.org>, <p.zabel@pengutronix.de>, <svarbanov@mm-sol.com>, <linux-arm-msm@vger.kernel.org>, <linux-pci@vger.kernel.org>, <devicetree@vger.kernel.org>, <linux-kernel@vger.kernel.org>, <linux-phy@lists.infradead.org>, <linux-clk@vger.kernel.org> CC: <quic_srichara@quicinc.com>, <quic_gokulsri@quicinc.com>, <quic_sjaganat@quicinc.com>, <quic_kathirav@quicinc.com>, <quic_arajkuma@quicinc.com>, <quic_anusha@quicinc.com> Subject: [PATCH 2/7] PCI: qcom: Add IPQ9574 PCIe support Date: Tue, 14 Feb 2023 22:11:30 +0530 Message-ID: <20230214164135.17039-3-quic_devipriy@quicinc.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20230214164135.17039-1-quic_devipriy@quicinc.com> References: <20230214164135.17039-1-quic_devipriy@quicinc.com> MIME-Version: 1.0 Content-Type: text/plain X-Originating-IP: [10.80.80.8] X-ClientProxiedBy: nasanex01b.na.qualcomm.com (10.46.141.250) To nalasex01a.na.qualcomm.com (10.47.209.196) X-QCInternal: smtphost X-Proofpoint-Virus-Version: vendor=nai engine=6200 definitions=5800 signatures=585085 X-Proofpoint-GUID: rKiOfBVDdt6F_a9ZkVUVRyfMoGbgYFDX X-Proofpoint-ORIG-GUID: rKiOfBVDdt6F_a9ZkVUVRyfMoGbgYFDX X-Proofpoint-Virus-Version: vendor=baseguard engine=ICAP:2.0.219,Aquarius:18.0.930,Hydra:6.0.562,FMLib:17.11.170.22 definitions=2023-02-14_11,2023-02-14_01,2023-02-09_01 X-Proofpoint-Spam-Details: rule=outbound_notspam policy=outbound score=0 mlxscore=0 suspectscore=0 spamscore=0 priorityscore=1501 phishscore=0 bulkscore=0 clxscore=1015 impostorscore=0 lowpriorityscore=0 adultscore=0 malwarescore=0 mlxlogscore=999 classifier=spam adjust=0 reason=mlx scancount=1 engine=8.12.0-2212070000 definitions=main-2302140142 X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,SPF_HELO_NONE,SPF_PASS autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: <linux-kernel.vger.kernel.org> X-Mailing-List: linux-kernel@vger.kernel.org X-getmail-retrieved-from-mailbox: =?utf-8?q?INBOX?= X-GMAIL-THRID: =?utf-8?q?1757825810165688860?= X-GMAIL-MSGID: =?utf-8?q?1757825810165688860?= |
Series |
Add PCIe support for IPQ9574
|
|
Commit Message
Devi Priya
Feb. 14, 2023, 4:41 p.m. UTC
Adding PCIe support for IPQ9574 SoC Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> --- drivers/pci/controller/dwc/pcie-qcom.c | 119 +++++++++++++++++++++++++ 1 file changed, 119 insertions(+)
Comments
Hi Devi, On 2/14/2023 10:11 PM, Devi Priya wrote: > Adding PCIe support for IPQ9574 SoC > > Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> > Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> > Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> > --- > drivers/pci/controller/dwc/pcie-qcom.c | 119 +++++++++++++++++++++++++ > 1 file changed, 119 insertions(+) > > diff --git a/drivers/pci/controller/dwc/pcie-qcom.c b/drivers/pci/controller/dwc/pcie-qcom.c > index a232b04af048..57606c113d45 100644 > --- a/drivers/pci/controller/dwc/pcie-qcom.c > +++ b/drivers/pci/controller/dwc/pcie-qcom.c > @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { > struct reset_control *rst; > }; > > +struct qcom_pcie_resources_1_27_0 { > + struct clk_bulk_data *clks; > + struct reset_control *rst; > + int num_clks; > +}; > + > union qcom_pcie_resources { > struct qcom_pcie_resources_1_0_0 v1_0_0; > struct qcom_pcie_resources_2_1_0 v2_1_0; > @@ -201,6 +207,7 @@ union qcom_pcie_resources { > struct qcom_pcie_resources_2_4_0 v2_4_0; > struct qcom_pcie_resources_2_7_0 v2_7_0; > struct qcom_pcie_resources_2_9_0 v2_9_0; > + struct qcom_pcie_resources_1_27_0 v1_27_0; > }; > > struct qcom_pcie; > @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct qcom_pcie *pcie) > return 0; > } > > +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) > +{ > + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; > + struct dw_pcie *pci = pcie->pci; > + struct device *dev = pci->dev; > + > + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); > + if (res->clks < 0) > + return res->num_clks; > + > + res->rst = devm_reset_control_array_get_exclusive(dev); > + if (IS_ERR(res->rst)) > + return PTR_ERR(res->rst); > + > + return 0; > +} > + > +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) > +{ > + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; > + > + clk_bulk_disable_unprepare(res->num_clks, res->clks); > +} > + > +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) > +{ > + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; > + struct device *dev = pcie->pci->dev; > + int ret; > + > + ret = reset_control_assert(res->rst); > + if (ret) { > + dev_err(dev, "reset assert failed (%d)\n", ret); > + return ret; > + } > + > + /* > + * Delay periods before and after reset deassert are working values > + * from downstream Codeaurora kernel > + */ > + usleep_range(2000, 2500); > + > + ret = reset_control_deassert(res->rst); > + if (ret) { > + dev_err(dev, "reset deassert failed (%d)\n", ret); > + return ret; > + } > + > + usleep_range(2000, 2500); > + > + return clk_bulk_prepare_enable(res->num_clks, res->clks); > +} > + > +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) > +{ > + struct dw_pcie *pci = pcie->pci; > + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); > + u32 val; > + int i; > + > + writel(0x8000000, pcie->parf + PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); > + > + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); > + val &= ~BIT(0); > + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); > + > + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); > + > + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); > + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, > + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); > + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | > + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, > + pci->dbi_base + GEN3_RELATED_OFF); > + > + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | > + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | > + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, > + pcie->parf + PCIE20_PARF_SYS_CTRL); > + > + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); > + > + dw_pcie_dbi_ro_wr_en(pci); > + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + PCI_EXP_SLTCAP); > + > + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); > + val &= ~PCI_EXP_LNKCAP_ASPMS; > + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); > + > + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + > + PCI_EXP_DEVCTL2); > + > + for (i = 0; i < 256; i++) > + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 * i)); > + > + return 0; > +} > + > static int qcom_pcie_link_up(struct dw_pcie *pci) > { > u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); > @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { > .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, > }; > > +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ > +static const struct qcom_pcie_ops ops_1_27_0 = { > + .get_resources = qcom_pcie_get_resources_1_27_0, > + .init = qcom_pcie_init_1_27_0, > + .post_init = qcom_pcie_post_init_1_27_0, > + .deinit = qcom_pcie_deinit_1_27_0, > + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, > +}; > + > static const struct qcom_pcie_cfg cfg_1_0_0 = { > .ops = &ops_1_0_0, > }; > @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { > .ops = &ops_2_9_0, > }; > > +static const struct qcom_pcie_cfg cfg_1_27_0 = { > + .ops = &ops_1_27_0, > +}; > + > static const struct dw_pcie_ops dw_pcie_ops = { > .link_up = qcom_pcie_link_up, > .start_link = qcom_pcie_start_link, > @@ -1829,6 +1947,7 @@ static const struct of_device_id qcom_pcie_match[] = { > { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, > { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, > { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, > + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, I do not see much difference between 2_9_0 and 1_27_0. Is this patch really required. Can you check if it works with 2_9_0 itself ? Regards, Sricharan
Hi Sri, Thanks for taking time to review the patch! On 2/16/2023 5:08 PM, Sricharan Ramabadhran wrote: > Hi Devi, > > On 2/14/2023 10:11 PM, Devi Priya wrote: >> Adding PCIe support for IPQ9574 SoC >> >> Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> >> Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> >> Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> >> --- >> drivers/pci/controller/dwc/pcie-qcom.c | 119 +++++++++++++++++++++++++ >> 1 file changed, 119 insertions(+) >> >> diff --git a/drivers/pci/controller/dwc/pcie-qcom.c >> b/drivers/pci/controller/dwc/pcie-qcom.c >> index a232b04af048..57606c113d45 100644 >> --- a/drivers/pci/controller/dwc/pcie-qcom.c >> +++ b/drivers/pci/controller/dwc/pcie-qcom.c >> @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { >> struct reset_control *rst; >> }; >> +struct qcom_pcie_resources_1_27_0 { >> + struct clk_bulk_data *clks; >> + struct reset_control *rst; >> + int num_clks; >> +}; >> + >> union qcom_pcie_resources { >> struct qcom_pcie_resources_1_0_0 v1_0_0; >> struct qcom_pcie_resources_2_1_0 v2_1_0; >> @@ -201,6 +207,7 @@ union qcom_pcie_resources { >> struct qcom_pcie_resources_2_4_0 v2_4_0; >> struct qcom_pcie_resources_2_7_0 v2_7_0; >> struct qcom_pcie_resources_2_9_0 v2_9_0; >> + struct qcom_pcie_resources_1_27_0 v1_27_0; >> }; >> struct qcom_pcie; >> @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct >> qcom_pcie *pcie) >> return 0; >> } >> +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >> + struct dw_pcie *pci = pcie->pci; >> + struct device *dev = pci->dev; >> + >> + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); >> + if (res->clks < 0) >> + return res->num_clks; >> + >> + res->rst = devm_reset_control_array_get_exclusive(dev); >> + if (IS_ERR(res->rst)) >> + return PTR_ERR(res->rst); >> + >> + return 0; >> +} >> + >> +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >> + >> + clk_bulk_disable_unprepare(res->num_clks, res->clks); >> +} >> + >> +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >> + struct device *dev = pcie->pci->dev; >> + int ret; >> + >> + ret = reset_control_assert(res->rst); >> + if (ret) { >> + dev_err(dev, "reset assert failed (%d)\n", ret); >> + return ret; >> + } >> + >> + /* >> + * Delay periods before and after reset deassert are working values >> + * from downstream Codeaurora kernel >> + */ >> + usleep_range(2000, 2500); >> + >> + ret = reset_control_deassert(res->rst); >> + if (ret) { >> + dev_err(dev, "reset deassert failed (%d)\n", ret); >> + return ret; >> + } >> + >> + usleep_range(2000, 2500); >> + >> + return clk_bulk_prepare_enable(res->num_clks, res->clks); >> +} >> + >> +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct dw_pcie *pci = pcie->pci; >> + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >> + u32 val; >> + int i; >> + >> + writel(0x8000000, pcie->parf + PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); >> + >> + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); >> + val &= ~BIT(0); >> + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); >> + >> + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); >> + >> + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); >> + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, >> + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); >> + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | >> + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, >> + pci->dbi_base + GEN3_RELATED_OFF); >> + >> + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | >> + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | >> + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, >> + pcie->parf + PCIE20_PARF_SYS_CTRL); >> + >> + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); >> + >> + dw_pcie_dbi_ro_wr_en(pci); >> + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + PCI_EXP_SLTCAP); >> + >> + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); >> + val &= ~PCI_EXP_LNKCAP_ASPMS; >> + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); >> + >> + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + >> + PCI_EXP_DEVCTL2); >> + >> + for (i = 0; i < 256; i++) >> + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 * >> i)); >> + >> + return 0; >> +} >> + >> static int qcom_pcie_link_up(struct dw_pcie *pci) >> { >> u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >> @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { >> .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >> }; >> +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ >> +static const struct qcom_pcie_ops ops_1_27_0 = { >> + .get_resources = qcom_pcie_get_resources_1_27_0, >> + .init = qcom_pcie_init_1_27_0, >> + .post_init = qcom_pcie_post_init_1_27_0, >> + .deinit = qcom_pcie_deinit_1_27_0, >> + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >> +}; >> + >> static const struct qcom_pcie_cfg cfg_1_0_0 = { >> .ops = &ops_1_0_0, >> }; >> @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { >> .ops = &ops_2_9_0, >> }; >> +static const struct qcom_pcie_cfg cfg_1_27_0 = { >> + .ops = &ops_1_27_0, >> +}; >> + >> static const struct dw_pcie_ops dw_pcie_ops = { >> .link_up = qcom_pcie_link_up, >> .start_link = qcom_pcie_start_link, >> @@ -1829,6 +1947,7 @@ static const struct of_device_id >> qcom_pcie_match[] = { >> { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, >> { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, >> { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, >> + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, > > I do not see much difference between 2_9_0 and 1_27_0. Is this patch > really required. Can you check if it works with 2_9_0 itself ? Yes right Sri, Only the clocks seem to differ between 2_9_0 and 1_27_0. Will update 2_9_0 ops to get the clocks from the DT and use the same for ipq9574 in the next spin. Best Regards, Devi Priya > > Regards, > Sricharan
On 2/20/2023 7:11 PM, Devi Priya wrote: > Hi Sri, > Thanks for taking time to review the patch! > > On 2/16/2023 5:08 PM, Sricharan Ramabadhran wrote: >> Hi Devi, >> >> On 2/14/2023 10:11 PM, Devi Priya wrote: >>> Adding PCIe support for IPQ9574 SoC >>> >>> Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> >>> Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> >>> Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> >>> --- >>> drivers/pci/controller/dwc/pcie-qcom.c | 119 >>> +++++++++++++++++++++++++ >>> 1 file changed, 119 insertions(+) >>> >>> diff --git a/drivers/pci/controller/dwc/pcie-qcom.c >>> b/drivers/pci/controller/dwc/pcie-qcom.c >>> index a232b04af048..57606c113d45 100644 >>> --- a/drivers/pci/controller/dwc/pcie-qcom.c >>> +++ b/drivers/pci/controller/dwc/pcie-qcom.c >>> @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { >>> struct reset_control *rst; >>> }; >>> +struct qcom_pcie_resources_1_27_0 { >>> + struct clk_bulk_data *clks; >>> + struct reset_control *rst; >>> + int num_clks; >>> +}; >>> + >>> union qcom_pcie_resources { >>> struct qcom_pcie_resources_1_0_0 v1_0_0; >>> struct qcom_pcie_resources_2_1_0 v2_1_0; >>> @@ -201,6 +207,7 @@ union qcom_pcie_resources { >>> struct qcom_pcie_resources_2_4_0 v2_4_0; >>> struct qcom_pcie_resources_2_7_0 v2_7_0; >>> struct qcom_pcie_resources_2_9_0 v2_9_0; >>> + struct qcom_pcie_resources_1_27_0 v1_27_0; >>> }; >>> struct qcom_pcie; >>> @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct >>> qcom_pcie *pcie) >>> return 0; >>> } >>> +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) >>> +{ >>> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >>> + struct dw_pcie *pci = pcie->pci; >>> + struct device *dev = pci->dev; >>> + >>> + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); >>> + if (res->clks < 0) >>> + return res->num_clks; >>> + >>> + res->rst = devm_reset_control_array_get_exclusive(dev); >>> + if (IS_ERR(res->rst)) >>> + return PTR_ERR(res->rst); >>> + >>> + return 0; >>> +} >>> + >>> +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) >>> +{ >>> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >>> + >>> + clk_bulk_disable_unprepare(res->num_clks, res->clks); >>> +} >>> + >>> +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) >>> +{ >>> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >>> + struct device *dev = pcie->pci->dev; >>> + int ret; >>> + >>> + ret = reset_control_assert(res->rst); >>> + if (ret) { >>> + dev_err(dev, "reset assert failed (%d)\n", ret); >>> + return ret; >>> + } >>> + >>> + /* >>> + * Delay periods before and after reset deassert are working >>> values >>> + * from downstream Codeaurora kernel >>> + */ >>> + usleep_range(2000, 2500); >>> + >>> + ret = reset_control_deassert(res->rst); >>> + if (ret) { >>> + dev_err(dev, "reset deassert failed (%d)\n", ret); >>> + return ret; >>> + } >>> + >>> + usleep_range(2000, 2500); >>> + >>> + return clk_bulk_prepare_enable(res->num_clks, res->clks); >>> +} >>> + >>> +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) >>> +{ >>> + struct dw_pcie *pci = pcie->pci; >>> + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >>> + u32 val; >>> + int i; >>> + >>> + writel(0x8000000, pcie->parf + >>> PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); Devi, Above statement also differs. You need to consider this also when you use the 2_9_0 ops. Thanks, >>> + >>> + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); >>> + val &= ~BIT(0); >>> + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); >>> + >>> + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); >>> + >>> + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); >>> + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, >>> + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); >>> + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | >>> + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, >>> + pci->dbi_base + GEN3_RELATED_OFF); >>> + >>> + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | >>> + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | >>> + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, >>> + pcie->parf + PCIE20_PARF_SYS_CTRL); >>> + >>> + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); >>> + >>> + dw_pcie_dbi_ro_wr_en(pci); >>> + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + >>> PCI_EXP_SLTCAP); >>> + >>> + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); >>> + val &= ~PCI_EXP_LNKCAP_ASPMS; >>> + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); >>> + >>> + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + >>> + PCI_EXP_DEVCTL2); >>> + >>> + for (i = 0; i < 256; i++) >>> + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 >>> * i)); >>> + >>> + return 0; >>> +} >>> + >>> static int qcom_pcie_link_up(struct dw_pcie *pci) >>> { >>> u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >>> @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { >>> .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >>> }; >>> +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ >>> +static const struct qcom_pcie_ops ops_1_27_0 = { >>> + .get_resources = qcom_pcie_get_resources_1_27_0, >>> + .init = qcom_pcie_init_1_27_0, >>> + .post_init = qcom_pcie_post_init_1_27_0, >>> + .deinit = qcom_pcie_deinit_1_27_0, >>> + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >>> +}; >>> + >>> static const struct qcom_pcie_cfg cfg_1_0_0 = { >>> .ops = &ops_1_0_0, >>> }; >>> @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { >>> .ops = &ops_2_9_0, >>> }; >>> +static const struct qcom_pcie_cfg cfg_1_27_0 = { >>> + .ops = &ops_1_27_0, >>> +}; >>> + >>> static const struct dw_pcie_ops dw_pcie_ops = { >>> .link_up = qcom_pcie_link_up, >>> .start_link = qcom_pcie_start_link, >>> @@ -1829,6 +1947,7 @@ static const struct of_device_id >>> qcom_pcie_match[] = { >>> { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, >>> { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, >>> { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, >>> + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, >> >> I do not see much difference between 2_9_0 and 1_27_0. Is this patch >> really required. Can you check if it works with 2_9_0 itself ? > Yes right Sri, Only the clocks seem to differ between 2_9_0 and 1_27_0. > Will update 2_9_0 ops to get the clocks from the DT and use the same > for ipq9574 in the next spin. > > Best Regards, > Devi Priya >> >> Regards, >> Sricharan
On 2/20/2023 8:21 PM, Kathiravan T wrote: > > On 2/20/2023 7:11 PM, Devi Priya wrote: >> Hi Sri, >> Thanks for taking time to review the patch! >> >> On 2/16/2023 5:08 PM, Sricharan Ramabadhran wrote: >>> Hi Devi, >>> >>> On 2/14/2023 10:11 PM, Devi Priya wrote: >>>> Adding PCIe support for IPQ9574 SoC >>>> >>>> Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> >>>> Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> >>>> Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> >>>> --- >>>> drivers/pci/controller/dwc/pcie-qcom.c | 119 >>>> +++++++++++++++++++++++++ >>>> 1 file changed, 119 insertions(+) >>>> >>>> diff --git a/drivers/pci/controller/dwc/pcie-qcom.c >>>> b/drivers/pci/controller/dwc/pcie-qcom.c >>>> index a232b04af048..57606c113d45 100644 >>>> --- a/drivers/pci/controller/dwc/pcie-qcom.c >>>> +++ b/drivers/pci/controller/dwc/pcie-qcom.c >>>> @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { >>>> struct reset_control *rst; >>>> }; >>>> +struct qcom_pcie_resources_1_27_0 { >>>> + struct clk_bulk_data *clks; >>>> + struct reset_control *rst; >>>> + int num_clks; >>>> +}; >>>> + >>>> union qcom_pcie_resources { >>>> struct qcom_pcie_resources_1_0_0 v1_0_0; >>>> struct qcom_pcie_resources_2_1_0 v2_1_0; >>>> @@ -201,6 +207,7 @@ union qcom_pcie_resources { >>>> struct qcom_pcie_resources_2_4_0 v2_4_0; >>>> struct qcom_pcie_resources_2_7_0 v2_7_0; >>>> struct qcom_pcie_resources_2_9_0 v2_9_0; >>>> + struct qcom_pcie_resources_1_27_0 v1_27_0; >>>> }; >>>> struct qcom_pcie; >>>> @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct >>>> qcom_pcie *pcie) >>>> return 0; >>>> } >>>> +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) >>>> +{ >>>> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >>>> + struct dw_pcie *pci = pcie->pci; >>>> + struct device *dev = pci->dev; >>>> + >>>> + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); >>>> + if (res->clks < 0) >>>> + return res->num_clks; >>>> + >>>> + res->rst = devm_reset_control_array_get_exclusive(dev); >>>> + if (IS_ERR(res->rst)) >>>> + return PTR_ERR(res->rst); >>>> + >>>> + return 0; >>>> +} >>>> + >>>> +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) >>>> +{ >>>> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >>>> + >>>> + clk_bulk_disable_unprepare(res->num_clks, res->clks); >>>> +} >>>> + >>>> +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) >>>> +{ >>>> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >>>> + struct device *dev = pcie->pci->dev; >>>> + int ret; >>>> + >>>> + ret = reset_control_assert(res->rst); >>>> + if (ret) { >>>> + dev_err(dev, "reset assert failed (%d)\n", ret); >>>> + return ret; >>>> + } >>>> + >>>> + /* >>>> + * Delay periods before and after reset deassert are working >>>> values >>>> + * from downstream Codeaurora kernel >>>> + */ >>>> + usleep_range(2000, 2500); >>>> + >>>> + ret = reset_control_deassert(res->rst); >>>> + if (ret) { >>>> + dev_err(dev, "reset deassert failed (%d)\n", ret); >>>> + return ret; >>>> + } >>>> + >>>> + usleep_range(2000, 2500); >>>> + >>>> + return clk_bulk_prepare_enable(res->num_clks, res->clks); >>>> +} >>>> + >>>> +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) >>>> +{ >>>> + struct dw_pcie *pci = pcie->pci; >>>> + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >>>> + u32 val; >>>> + int i; >>>> + >>>> + writel(0x8000000, pcie->parf + >>>> PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); > > > Devi, > > > Above statement also differs. You need to consider this also when you > use the 2_9_0 ops. > Sure, thanks kathir. Will take care of this as well > Best Regards, Devi Priya > Thanks, > > >>>> + >>>> + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); >>>> + val &= ~BIT(0); >>>> + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); >>>> + >>>> + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); >>>> + >>>> + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); >>>> + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, >>>> + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); >>>> + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | >>>> + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, >>>> + pci->dbi_base + GEN3_RELATED_OFF); >>>> + >>>> + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | >>>> + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | >>>> + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, >>>> + pcie->parf + PCIE20_PARF_SYS_CTRL); >>>> + >>>> + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); >>>> + >>>> + dw_pcie_dbi_ro_wr_en(pci); >>>> + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + >>>> PCI_EXP_SLTCAP); >>>> + >>>> + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); >>>> + val &= ~PCI_EXP_LNKCAP_ASPMS; >>>> + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); >>>> + >>>> + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + >>>> + PCI_EXP_DEVCTL2); >>>> + >>>> + for (i = 0; i < 256; i++) >>>> + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 >>>> * i)); >>>> + >>>> + return 0; >>>> +} >>>> + >>>> static int qcom_pcie_link_up(struct dw_pcie *pci) >>>> { >>>> u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >>>> @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { >>>> .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >>>> }; >>>> +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ >>>> +static const struct qcom_pcie_ops ops_1_27_0 = { >>>> + .get_resources = qcom_pcie_get_resources_1_27_0, >>>> + .init = qcom_pcie_init_1_27_0, >>>> + .post_init = qcom_pcie_post_init_1_27_0, >>>> + .deinit = qcom_pcie_deinit_1_27_0, >>>> + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >>>> +}; >>>> + >>>> static const struct qcom_pcie_cfg cfg_1_0_0 = { >>>> .ops = &ops_1_0_0, >>>> }; >>>> @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { >>>> .ops = &ops_2_9_0, >>>> }; >>>> +static const struct qcom_pcie_cfg cfg_1_27_0 = { >>>> + .ops = &ops_1_27_0, >>>> +}; >>>> + >>>> static const struct dw_pcie_ops dw_pcie_ops = { >>>> .link_up = qcom_pcie_link_up, >>>> .start_link = qcom_pcie_start_link, >>>> @@ -1829,6 +1947,7 @@ static const struct of_device_id >>>> qcom_pcie_match[] = { >>>> { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, >>>> { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, >>>> { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, >>>> + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, >>> >>> I do not see much difference between 2_9_0 and 1_27_0. Is this patch >>> really required. Can you check if it works with 2_9_0 itself ? >> Yes right Sri, Only the clocks seem to differ between 2_9_0 and 1_27_0. >> Will update 2_9_0 ops to get the clocks from the DT and use the same >> for ipq9574 in the next spin. >> >> Best Regards, >> Devi Priya >>> >>> Regards, >>> Sricharan
On Tue, Feb 14, 2023 at 10:11:30PM +0530, Devi Priya wrote: > Adding PCIe support for IPQ9574 SoC > Please add some information about the PCIe IP in the commit message. Like, the Synopsys IP version, Gen speed etc... and how the support is added (using existing config or a new one). As others have mentioned, you could reuse the existing config. In that case, add the info in commit message as I suggested above. And the PCI patch should come last in the series after bindings and dts patches. Thanks, Mani > Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> > Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> > Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> > --- > drivers/pci/controller/dwc/pcie-qcom.c | 119 +++++++++++++++++++++++++ > 1 file changed, 119 insertions(+) > > diff --git a/drivers/pci/controller/dwc/pcie-qcom.c b/drivers/pci/controller/dwc/pcie-qcom.c > index a232b04af048..57606c113d45 100644 > --- a/drivers/pci/controller/dwc/pcie-qcom.c > +++ b/drivers/pci/controller/dwc/pcie-qcom.c > @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { > struct reset_control *rst; > }; > > +struct qcom_pcie_resources_1_27_0 { > + struct clk_bulk_data *clks; > + struct reset_control *rst; > + int num_clks; > +}; > + > union qcom_pcie_resources { > struct qcom_pcie_resources_1_0_0 v1_0_0; > struct qcom_pcie_resources_2_1_0 v2_1_0; > @@ -201,6 +207,7 @@ union qcom_pcie_resources { > struct qcom_pcie_resources_2_4_0 v2_4_0; > struct qcom_pcie_resources_2_7_0 v2_7_0; > struct qcom_pcie_resources_2_9_0 v2_9_0; > + struct qcom_pcie_resources_1_27_0 v1_27_0; > }; > > struct qcom_pcie; > @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct qcom_pcie *pcie) > return 0; > } > > +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) > +{ > + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; > + struct dw_pcie *pci = pcie->pci; > + struct device *dev = pci->dev; > + > + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); > + if (res->clks < 0) > + return res->num_clks; > + > + res->rst = devm_reset_control_array_get_exclusive(dev); > + if (IS_ERR(res->rst)) > + return PTR_ERR(res->rst); > + > + return 0; > +} > + > +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) > +{ > + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; > + > + clk_bulk_disable_unprepare(res->num_clks, res->clks); > +} > + > +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) > +{ > + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; > + struct device *dev = pcie->pci->dev; > + int ret; > + > + ret = reset_control_assert(res->rst); > + if (ret) { > + dev_err(dev, "reset assert failed (%d)\n", ret); > + return ret; > + } > + > + /* > + * Delay periods before and after reset deassert are working values > + * from downstream Codeaurora kernel > + */ > + usleep_range(2000, 2500); > + > + ret = reset_control_deassert(res->rst); > + if (ret) { > + dev_err(dev, "reset deassert failed (%d)\n", ret); > + return ret; > + } > + > + usleep_range(2000, 2500); > + > + return clk_bulk_prepare_enable(res->num_clks, res->clks); > +} > + > +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) > +{ > + struct dw_pcie *pci = pcie->pci; > + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); > + u32 val; > + int i; > + > + writel(0x8000000, pcie->parf + PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); > + > + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); > + val &= ~BIT(0); > + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); > + > + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); > + > + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); > + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, > + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); > + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | > + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, > + pci->dbi_base + GEN3_RELATED_OFF); > + > + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | > + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | > + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, > + pcie->parf + PCIE20_PARF_SYS_CTRL); > + > + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); > + > + dw_pcie_dbi_ro_wr_en(pci); > + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + PCI_EXP_SLTCAP); > + > + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); > + val &= ~PCI_EXP_LNKCAP_ASPMS; > + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); > + > + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + > + PCI_EXP_DEVCTL2); > + > + for (i = 0; i < 256; i++) > + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 * i)); > + > + return 0; > +} > + > static int qcom_pcie_link_up(struct dw_pcie *pci) > { > u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); > @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { > .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, > }; > > +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ > +static const struct qcom_pcie_ops ops_1_27_0 = { > + .get_resources = qcom_pcie_get_resources_1_27_0, > + .init = qcom_pcie_init_1_27_0, > + .post_init = qcom_pcie_post_init_1_27_0, > + .deinit = qcom_pcie_deinit_1_27_0, > + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, > +}; > + > static const struct qcom_pcie_cfg cfg_1_0_0 = { > .ops = &ops_1_0_0, > }; > @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { > .ops = &ops_2_9_0, > }; > > +static const struct qcom_pcie_cfg cfg_1_27_0 = { > + .ops = &ops_1_27_0, > +}; > + > static const struct dw_pcie_ops dw_pcie_ops = { > .link_up = qcom_pcie_link_up, > .start_link = qcom_pcie_start_link, > @@ -1829,6 +1947,7 @@ static const struct of_device_id qcom_pcie_match[] = { > { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, > { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, > { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, > + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, > { .compatible = "qcom,pcie-msm8996", .data = &cfg_2_3_2 }, > { .compatible = "qcom,pcie-qcs404", .data = &cfg_2_4_0 }, > { .compatible = "qcom,pcie-sa8540p", .data = &cfg_1_9_0 }, > -- > 2.17.1 >
On 2/24/2023 1:59 PM, Manivannan Sadhasivam wrote: > On Tue, Feb 14, 2023 at 10:11:30PM +0530, Devi Priya wrote: >> Adding PCIe support for IPQ9574 SoC >> > > Please add some information about the PCIe IP in the commit message. > Like, the Synopsys IP version, Gen speed etc... and how the support > is added (using existing config or a new one). > > As others have mentioned, you could reuse the existing config. In that > case, add the info in commit message as I suggested above. > > And the PCI patch should come last in the series after bindings and dts > patches. > Sure, will update this accordingly in the next spin > Thanks, > Mani > >> Co-developed-by: Anusha Rao <quic_anusha@quicinc.com> >> Signed-off-by: Anusha Rao <quic_anusha@quicinc.com> >> Signed-off-by: Devi Priya <quic_devipriy@quicinc.com> >> --- >> drivers/pci/controller/dwc/pcie-qcom.c | 119 +++++++++++++++++++++++++ >> 1 file changed, 119 insertions(+) >> >> diff --git a/drivers/pci/controller/dwc/pcie-qcom.c b/drivers/pci/controller/dwc/pcie-qcom.c >> index a232b04af048..57606c113d45 100644 >> --- a/drivers/pci/controller/dwc/pcie-qcom.c >> +++ b/drivers/pci/controller/dwc/pcie-qcom.c >> @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { >> struct reset_control *rst; >> }; >> >> +struct qcom_pcie_resources_1_27_0 { >> + struct clk_bulk_data *clks; >> + struct reset_control *rst; >> + int num_clks; >> +}; >> + >> union qcom_pcie_resources { >> struct qcom_pcie_resources_1_0_0 v1_0_0; >> struct qcom_pcie_resources_2_1_0 v2_1_0; >> @@ -201,6 +207,7 @@ union qcom_pcie_resources { >> struct qcom_pcie_resources_2_4_0 v2_4_0; >> struct qcom_pcie_resources_2_7_0 v2_7_0; >> struct qcom_pcie_resources_2_9_0 v2_9_0; >> + struct qcom_pcie_resources_1_27_0 v1_27_0; >> }; >> >> struct qcom_pcie; >> @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct qcom_pcie *pcie) >> return 0; >> } >> >> +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >> + struct dw_pcie *pci = pcie->pci; >> + struct device *dev = pci->dev; >> + >> + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); >> + if (res->clks < 0) >> + return res->num_clks; >> + >> + res->rst = devm_reset_control_array_get_exclusive(dev); >> + if (IS_ERR(res->rst)) >> + return PTR_ERR(res->rst); >> + >> + return 0; >> +} >> + >> +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >> + >> + clk_bulk_disable_unprepare(res->num_clks, res->clks); >> +} >> + >> +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; >> + struct device *dev = pcie->pci->dev; >> + int ret; >> + >> + ret = reset_control_assert(res->rst); >> + if (ret) { >> + dev_err(dev, "reset assert failed (%d)\n", ret); >> + return ret; >> + } >> + >> + /* >> + * Delay periods before and after reset deassert are working values >> + * from downstream Codeaurora kernel >> + */ >> + usleep_range(2000, 2500); >> + >> + ret = reset_control_deassert(res->rst); >> + if (ret) { >> + dev_err(dev, "reset deassert failed (%d)\n", ret); >> + return ret; >> + } >> + >> + usleep_range(2000, 2500); >> + >> + return clk_bulk_prepare_enable(res->num_clks, res->clks); >> +} >> + >> +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) >> +{ >> + struct dw_pcie *pci = pcie->pci; >> + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >> + u32 val; >> + int i; >> + >> + writel(0x8000000, pcie->parf + PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); >> + >> + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); >> + val &= ~BIT(0); >> + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); >> + >> + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); >> + >> + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); >> + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, >> + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); >> + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | >> + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, >> + pci->dbi_base + GEN3_RELATED_OFF); >> + >> + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | >> + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | >> + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, >> + pcie->parf + PCIE20_PARF_SYS_CTRL); >> + >> + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); >> + >> + dw_pcie_dbi_ro_wr_en(pci); >> + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + PCI_EXP_SLTCAP); >> + >> + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); >> + val &= ~PCI_EXP_LNKCAP_ASPMS; >> + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); >> + >> + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + >> + PCI_EXP_DEVCTL2); >> + >> + for (i = 0; i < 256; i++) >> + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 * i)); >> + >> + return 0; >> +} >> + >> static int qcom_pcie_link_up(struct dw_pcie *pci) >> { >> u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); >> @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { >> .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >> }; >> >> +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ >> +static const struct qcom_pcie_ops ops_1_27_0 = { >> + .get_resources = qcom_pcie_get_resources_1_27_0, >> + .init = qcom_pcie_init_1_27_0, >> + .post_init = qcom_pcie_post_init_1_27_0, >> + .deinit = qcom_pcie_deinit_1_27_0, >> + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, >> +}; >> + >> static const struct qcom_pcie_cfg cfg_1_0_0 = { >> .ops = &ops_1_0_0, >> }; >> @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { >> .ops = &ops_2_9_0, >> }; >> >> +static const struct qcom_pcie_cfg cfg_1_27_0 = { >> + .ops = &ops_1_27_0, >> +}; >> + >> static const struct dw_pcie_ops dw_pcie_ops = { >> .link_up = qcom_pcie_link_up, >> .start_link = qcom_pcie_start_link, >> @@ -1829,6 +1947,7 @@ static const struct of_device_id qcom_pcie_match[] = { >> { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, >> { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, >> { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, >> + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, >> { .compatible = "qcom,pcie-msm8996", .data = &cfg_2_3_2 }, >> { .compatible = "qcom,pcie-qcs404", .data = &cfg_2_4_0 }, >> { .compatible = "qcom,pcie-sa8540p", .data = &cfg_1_9_0 }, >> -- >> 2.17.1 >> > Thanks, Devi Priya
diff --git a/drivers/pci/controller/dwc/pcie-qcom.c b/drivers/pci/controller/dwc/pcie-qcom.c index a232b04af048..57606c113d45 100644 --- a/drivers/pci/controller/dwc/pcie-qcom.c +++ b/drivers/pci/controller/dwc/pcie-qcom.c @@ -193,6 +193,12 @@ struct qcom_pcie_resources_2_9_0 { struct reset_control *rst; }; +struct qcom_pcie_resources_1_27_0 { + struct clk_bulk_data *clks; + struct reset_control *rst; + int num_clks; +}; + union qcom_pcie_resources { struct qcom_pcie_resources_1_0_0 v1_0_0; struct qcom_pcie_resources_2_1_0 v2_1_0; @@ -201,6 +207,7 @@ union qcom_pcie_resources { struct qcom_pcie_resources_2_4_0 v2_4_0; struct qcom_pcie_resources_2_7_0 v2_7_0; struct qcom_pcie_resources_2_9_0 v2_9_0; + struct qcom_pcie_resources_1_27_0 v1_27_0; }; struct qcom_pcie; @@ -1409,6 +1416,104 @@ static int qcom_pcie_post_init_2_9_0(struct qcom_pcie *pcie) return 0; } +static int qcom_pcie_get_resources_1_27_0(struct qcom_pcie *pcie) +{ + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; + struct dw_pcie *pci = pcie->pci; + struct device *dev = pci->dev; + + res->num_clks = devm_clk_bulk_get_all(dev, &res->clks); + if (res->clks < 0) + return res->num_clks; + + res->rst = devm_reset_control_array_get_exclusive(dev); + if (IS_ERR(res->rst)) + return PTR_ERR(res->rst); + + return 0; +} + +static void qcom_pcie_deinit_1_27_0(struct qcom_pcie *pcie) +{ + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; + + clk_bulk_disable_unprepare(res->num_clks, res->clks); +} + +static int qcom_pcie_init_1_27_0(struct qcom_pcie *pcie) +{ + struct qcom_pcie_resources_1_27_0 *res = &pcie->res.v1_27_0; + struct device *dev = pcie->pci->dev; + int ret; + + ret = reset_control_assert(res->rst); + if (ret) { + dev_err(dev, "reset assert failed (%d)\n", ret); + return ret; + } + + /* + * Delay periods before and after reset deassert are working values + * from downstream Codeaurora kernel + */ + usleep_range(2000, 2500); + + ret = reset_control_deassert(res->rst); + if (ret) { + dev_err(dev, "reset deassert failed (%d)\n", ret); + return ret; + } + + usleep_range(2000, 2500); + + return clk_bulk_prepare_enable(res->num_clks, res->clks); +} + +static int qcom_pcie_post_init_1_27_0(struct qcom_pcie *pcie) +{ + struct dw_pcie *pci = pcie->pci; + u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); + u32 val; + int i; + + writel(0x8000000, pcie->parf + PCIE20_v3_PARF_SLV_ADDR_SPACE_SIZE); + + val = readl(pcie->parf + PCIE20_PARF_PHY_CTRL); + val &= ~BIT(0); + writel(val, pcie->parf + PCIE20_PARF_PHY_CTRL); + + writel(0, pcie->parf + PCIE20_PARF_DBI_BASE_ADDR); + + writel(DEVICE_TYPE_RC, pcie->parf + PCIE20_PARF_DEVICE_TYPE); + writel(BYPASS | MSTR_AXI_CLK_EN | AHB_CLK_EN, + pcie->parf + PCIE20_PARF_MHI_CLOCK_RESET_CTRL); + writel(GEN3_RELATED_OFF_RXEQ_RGRDLESS_RXTS | + GEN3_RELATED_OFF_GEN3_ZRXDC_NONCOMPL, + pci->dbi_base + GEN3_RELATED_OFF); + + writel(MST_WAKEUP_EN | SLV_WAKEUP_EN | MSTR_ACLK_CGC_DIS | + SLV_ACLK_CGC_DIS | CORE_CLK_CGC_DIS | + AUX_PWR_DET | L23_CLK_RMV_DIS | L1_CLK_RMV_DIS, + pcie->parf + PCIE20_PARF_SYS_CTRL); + + writel(0, pcie->parf + PCIE20_PARF_Q2A_FLUSH); + + dw_pcie_dbi_ro_wr_en(pci); + writel(PCIE_CAP_SLOT_VAL, pci->dbi_base + offset + PCI_EXP_SLTCAP); + + val = readl(pci->dbi_base + offset + PCI_EXP_LNKCAP); + val &= ~PCI_EXP_LNKCAP_ASPMS; + writel(val, pci->dbi_base + offset + PCI_EXP_LNKCAP); + + writel(PCI_EXP_DEVCTL2_COMP_TMOUT_DIS, pci->dbi_base + offset + + PCI_EXP_DEVCTL2); + + for (i = 0; i < 256; i++) + writel(0, pcie->parf + PCIE20_PARF_BDF_TO_SID_TABLE_N + (4 * i)); + + return 0; +} + static int qcom_pcie_link_up(struct dw_pcie *pci) { u16 offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); @@ -1620,6 +1725,15 @@ static const struct qcom_pcie_ops ops_2_9_0 = { .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, }; +/* Qcom IP rev.: 1.27.0 Synopsys IP rev.: 5.80a */ +static const struct qcom_pcie_ops ops_1_27_0 = { + .get_resources = qcom_pcie_get_resources_1_27_0, + .init = qcom_pcie_init_1_27_0, + .post_init = qcom_pcie_post_init_1_27_0, + .deinit = qcom_pcie_deinit_1_27_0, + .ltssm_enable = qcom_pcie_2_3_2_ltssm_enable, +}; + static const struct qcom_pcie_cfg cfg_1_0_0 = { .ops = &ops_1_0_0, }; @@ -1652,6 +1766,10 @@ static const struct qcom_pcie_cfg cfg_2_9_0 = { .ops = &ops_2_9_0, }; +static const struct qcom_pcie_cfg cfg_1_27_0 = { + .ops = &ops_1_27_0, +}; + static const struct dw_pcie_ops dw_pcie_ops = { .link_up = qcom_pcie_link_up, .start_link = qcom_pcie_start_link, @@ -1829,6 +1947,7 @@ static const struct of_device_id qcom_pcie_match[] = { { .compatible = "qcom,pcie-ipq8064-v2", .data = &cfg_2_1_0 }, { .compatible = "qcom,pcie-ipq8074", .data = &cfg_2_3_3 }, { .compatible = "qcom,pcie-ipq8074-gen3", .data = &cfg_2_9_0 }, + { .compatible = "qcom,pcie-ipq9574", .data = &cfg_1_27_0 }, { .compatible = "qcom,pcie-msm8996", .data = &cfg_2_3_2 }, { .compatible = "qcom,pcie-qcs404", .data = &cfg_2_4_0 }, { .compatible = "qcom,pcie-sa8540p", .data = &cfg_1_9_0 },