From patchwork Sat Mar 5 05:23:01 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Lizhi Hou X-Patchwork-Id: 548781 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id A6ECEC433F5 for ; Sat, 5 Mar 2022 05:23:43 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231238AbiCEFYa (ORCPT ); Sat, 5 Mar 2022 00:24:30 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:41862 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230077AbiCEFY3 (ORCPT ); Sat, 5 Mar 2022 00:24:29 -0500 Received: from NAM12-DM6-obe.outbound.protection.outlook.com (mail-dm6nam12on2072.outbound.protection.outlook.com [40.107.243.72]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 23983177D37; Fri, 4 Mar 2022 21:23:39 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=bttdaXIcjL65w1ROwszhINJT0QV8VW4pf79YwGTbOI3FIxcSskIrhMICiXSRviY7HA3bdXGsVRRO+lVpDbz54eW3Lm+xajCEDc5LxYdkWYYB5sVSa7hS+UJbDHigkR2LAwalSPOD/ZBcjER5zH07WqkMALBhftgW5oR5wk+P101Oy3TFdQEYVCxGW9s2l++w1Oy7z/7Jkbmw7uqREhx0uAcV2Zwnue2yzJYfngnrUGFQH2jk65nQIVemqCHYSn/R8pC3SeB0mbcuTBQjkClQZrVAsn2UIlQ7eUTnyxVErMfzTp1ztfvf4Y+ijevlEUGkQkvUuvDB4hkAoh9LKLzmyA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=9YaIMcnkqkA7VrLiz5PDiYFwijwL2DpD7MCfhu9ZTGc=; b=Ttxpd7ul+F4owJFMvEy9Mu49xrS0W4dVfwxPsN+ksV/GI6EeCcD1rAp13ukbHblQpUWtzeae5vHiIyYzpWYBN4xwjCE54j7cN+XR3KDxRV+Bem7ahuLy33Qcu+6ZpA2Gxg322flc2wk1Ls8surphmCvLJXE89aEQ38Q8G1IxkLaKGSmHyTqr8oLaMID79OBRBuH4ISfrbjkwfJnOpUBsYtO9as4Hwq5wR71D+BqPwsC6RQg09WKGEpOT4eJtkdQNhNjYY96bHSioUFRE87afbZkhSriYBENLRJXuC2iEA8SWnjXf3ssNH8h+z+4c+nN1DP8OgQTlvY8QLZ6BNs2esw== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=pass (sender ip is 149.199.62.198) smtp.rcpttodomain=infradead.org smtp.mailfrom=xilinx.com; dmarc=pass (p=none sp=none pct=100) action=none header.from=xilinx.com; dkim=none (message not signed); arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=xilinx.onmicrosoft.com; s=selector2-xilinx-onmicrosoft-com; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=9YaIMcnkqkA7VrLiz5PDiYFwijwL2DpD7MCfhu9ZTGc=; b=L9j39z0eD6ko2SlYmcGhq9xF9wrBG9KkZGceeD6jZQPhEpw7xihG9SMFJ3MiihJsULRAzuGOuicj4/9Ohpf2e4Y0kiUeMVjKOFmih+PL6YFcJIzmYEd1cjTj2jFMjTv5sRPhl0hlTYV7LOkIYLX3wK2u+pMLuKsR1ouHYbwtmIg= Received: from SA9PR13CA0006.namprd13.prod.outlook.com (2603:10b6:806:21::11) by DM6PR02MB4731.namprd02.prod.outlook.com (2603:10b6:5:fa::14) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.5017.26; Sat, 5 Mar 2022 05:23:35 +0000 Received: from SN1NAM02FT0011.eop-nam02.prod.protection.outlook.com (2603:10b6:806:21:cafe::b) by SA9PR13CA0006.outlook.office365.com (2603:10b6:806:21::11) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.5061.8 via Frontend Transport; Sat, 5 Mar 2022 05:23:35 +0000 X-MS-Exchange-Authentication-Results: spf=pass (sender IP is 149.199.62.198) smtp.mailfrom=xilinx.com; dkim=none (message not signed) header.d=none;dmarc=pass action=none header.from=xilinx.com; Received-SPF: Pass (protection.outlook.com: domain of xilinx.com designates 149.199.62.198 as permitted sender) receiver=protection.outlook.com; client-ip=149.199.62.198; helo=xsj-pvapexch02.xlnx.xilinx.com; Received: from xsj-pvapexch02.xlnx.xilinx.com (149.199.62.198) by SN1NAM02FT0011.mail.protection.outlook.com (10.97.5.171) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.20.5038.14 via Frontend Transport; Sat, 5 Mar 2022 05:23:35 +0000 Received: from xsj-pvapexch02.xlnx.xilinx.com (172.19.86.41) by xsj-pvapexch02.xlnx.xilinx.com (172.19.86.41) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.2176.14; Fri, 4 Mar 2022 21:23:34 -0800 Received: from smtp.xilinx.com (172.19.127.95) by xsj-pvapexch02.xlnx.xilinx.com (172.19.86.41) with Microsoft SMTP Server id 15.1.2176.14 via Frontend Transport; Fri, 4 Mar 2022 21:23:34 -0800 Envelope-to: dwmw2@infradead.org, yilun.xu@intel.com, mdf@kernel.org, robh@kernel.org, trix@redhat.com, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, linux-pci@vger.kernel.org Received: from [172.19.72.93] (port=44430 helo=xsj-xw9400.xilinx.com) by smtp.xilinx.com with esmtp (Exim 4.90) (envelope-from ) id 1nQMtC-000BYk-KM; Fri, 04 Mar 2022 21:23:34 -0800 Received: by xsj-xw9400.xilinx.com (Postfix, from userid 21952) id 23376600197; Fri, 4 Mar 2022 21:23:09 -0800 (PST) From: Lizhi Hou To: , , CC: Lizhi Hou , , , , , , , , , , , Max Zhen Subject: [PATCH V1 RESEND 1/4] pci: add interface to create pci-ep device tree node Date: Fri, 4 Mar 2022 21:23:01 -0800 Message-ID: <20220305052304.726050-2-lizhi.hou@xilinx.com> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20220305052304.726050-1-lizhi.hou@xilinx.com> References: <20220305052304.726050-1-lizhi.hou@xilinx.com> MIME-Version: 1.0 X-EOPAttributedMessage: 0 X-MS-PublicTrafficType: Email X-MS-Office365-Filtering-Correlation-Id: a0816dcd-ad84-4245-edd5-08d9fe684be3 X-MS-TrafficTypeDiagnostic: DM6PR02MB4731:EE_ X-Microsoft-Antispam-PRVS: X-MS-Exchange-SenderADCheck: 1 X-MS-Exchange-AntiSpam-Relay: 0 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: oqwIPOQcFcXoakUoM86cdwmj0Gg3ACsGMyB1GutBAdAN9OF5rUt8dXFem1NsWtlzbol4gE+UuQviEzW+oWpsGP9Fi8EBiHNIontZp1zQz+0K34FnnUxe1ZrImdCSRZlfD6ljX8vWErAuTjqYoSnB+PCpQWHYP1mpDXwMJZjeU/2j4MHmsv0HqDddA/u42ExyKSdJKeWloDHWEnL9nNkpj7yhpQpiW1cBNzjYSBgz7cjbnxj/LIMloUK8jyN2UynIEbouWj5dInnUqG45aaa2s+aa97YpKUgLoLC2eeAzz3GopsQa8cDw1X7fkcqpb1ejcXcCVwapl0WEP2LI81Ejd96+YXcBbDl4zPqorCPcotUjiv14RUww4yTKiTJMV5OGaAdKl/JUpqOE3KY5WhkKFPTzZGEISScCq9I27mu5L62Vr7BhCktHhmI+TsPRhFAgmTxCdYzQMK8Bhrlhtf1kxmmGjrvGB8TNwJQRp/kbHqn3xFXjtj+VPUNb4rdqTrBUnxx87q0hF92HdvVa9mKHF+d/IhTHElspMC5exhAGXimVaVy2wDSsx4mdTW9Bf4zO4SKCYoH1lPOffmdkDtXBTy/bxTzi1S2ZA9/tY2iGFFt0e3YfPJhP3obs8CGeCd/UgiBGTx4+/+pynf7GRc3R5g== X-Forefront-Antispam-Report: CIP:149.199.62.198; CTRY:US; LANG:en; SCL:1; SRV:; IPV:NLI; SFV:NSPM; H:xsj-pvapexch02.xlnx.xilinx.com; PTR:unknown-62-198.xilinx.com; CAT:NONE; SFS:(13230001)(4636009)(46966006)(36840700001)(26005)(54906003)(508600001)(1076003)(186003)(316002)(8676002)(107886003)(5660300002)(42186006)(110136005)(8936002)(70586007)(4326008)(6666004)(82310400004)(2616005)(36756003)(83380400001)(6266002)(426003)(2906002)(44832011)(356005)(336012)(7636003)(36860700001)(47076005)(70206006); DIR:OUT; SFP:1101; X-OriginatorOrg: xilinx.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 05 Mar 2022 05:23:35.2137 (UTC) X-MS-Exchange-CrossTenant-Network-Message-Id: a0816dcd-ad84-4245-edd5-08d9fe684be3 X-MS-Exchange-CrossTenant-Id: 657af505-d5df-48d0-8300-c31994686c5c X-MS-Exchange-CrossTenant-OriginalAttributedTenantConnectingIp: TenantId=657af505-d5df-48d0-8300-c31994686c5c; Ip=[149.199.62.198]; Helo=[xsj-pvapexch02.xlnx.xilinx.com] X-MS-Exchange-CrossTenant-AuthSource: SN1NAM02FT0011.eop-nam02.prod.protection.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Anonymous X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: DM6PR02MB4731 Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org This patch enables PCIe device to uses flattened device tree to describe apertures in its PCIe BARs. The aperture address consists of PCIe BAR index and offset. For this kind of device, the driver probe routine calls the new added interface to create a device tree node. This device tree node is attached under system device tree root. Then the driver may load the flatten device tree overlay and attach it under this node. And the node also contains 'ranges' property which is used to translate aperture address(BAR index and offset) to CPU address. Signed-off-by: Sonal Santan Signed-off-by: Max Zhen Signed-off-by: Lizhi Hou --- drivers/pci/of.c | 180 +++++++++++++++++++++++++++++++++++++++++ include/linux/of_pci.h | 15 ++++ 2 files changed, 195 insertions(+) diff --git a/drivers/pci/of.c b/drivers/pci/of.c index cb2e8351c2cc..198f08351070 100644 --- a/drivers/pci/of.c +++ b/drivers/pci/of.c @@ -605,6 +605,186 @@ int devm_of_pci_bridge_init(struct device *dev, struct pci_host_bridge *bridge) return pci_parse_request_of_pci_ranges(dev, bridge); } +#if IS_ENABLED(CONFIG_OF_DYNAMIC) + +static void devm_of_pci_destroy_bus_endpoint(struct device *dev, void *res) +{ + struct device_node *node = res; + + of_detach_node(node); +} + +static int of_ep_add_property(struct device *dev, struct property **proplist, const char *name, + const int length, void *value) +{ + struct property *new; + + new = devm_kzalloc(dev, sizeof(*new), GFP_KERNEL); + if (!new) + return -ENOMEM; + + new->name = devm_kstrdup(dev, name, GFP_KERNEL); + if (!new->name) + return -ENOMEM; + + new->value = devm_kmalloc(dev, length, GFP_KERNEL); + if (!new->value) + return -ENOMEM; + + memcpy(new->value, value, length); + new->length = length; + new->next = *proplist; + *proplist = new; + + return 0; +} + +static struct device_node *of_ep_alloc_node(struct pci_dev *pdev, const char *name) +{ + struct device_node *node; + char *full_name; + + node = devres_alloc(devm_of_pci_destroy_bus_endpoint, sizeof(*node), GFP_KERNEL); + if (!node) + return NULL; + + full_name = devm_kasprintf(&pdev->dev, GFP_KERNEL, "/%s@%llx", name, + (u64)pci_resource_start(pdev, 0)); + if (!full_name) + return NULL; + + node->parent = of_root; + node->full_name = full_name; + of_node_set_flag(node, OF_DYNAMIC); + of_node_init(node); + + return node; +} + +/** + * devm_of_pci_create_bus_endpoint - Create a device node for the given pci device. + * @pdev: PCI device pointer. + * + * For PCI device which uses flattened device tree to describe apertures in its BARs, + * a device node for the given pci device is required. Then the flattened device tree + * overlay from the device can be applied to the base tree. + * The device node is under root node and act like bus node. It contains a "ranges" + * property which is used for address translation of its children. Each child node + * corresponds an aperture and use BAR index and offset as its address. + + * Returns 0 on success or a negative error-code on failure. + */ +int devm_of_pci_create_bus_endpoint(struct pci_dev *pdev) +{ + struct property *proplist = NULL; + struct device *dev = &pdev->dev; + int range_ncells, addr_ncells; + struct device_node *node; + void *prop = NULL; + u32 *range_cell; + __be32 val; + int i, ret; + + node = of_ep_alloc_node(pdev, "pci-ep-bus"); + if (!node) + return -ENOMEM; + + /* the endpoint node works as 'simple-bus' to translate aperture addresses. */ + prop = "simple-bus"; + ret = of_ep_add_property(dev, &proplist, "compatible", strlen(prop) + 1, prop); + if (ret) + goto cleanup; + + /* The address and size cells of nodes underneath are 2 */ + val = cpu_to_be32(2); + ret = of_ep_add_property(dev, &proplist, "#address-cells", sizeof(u32), &val); + if (ret) + goto cleanup; + + ret = of_ep_add_property(dev, &proplist, "#size-cells", sizeof(u32), &val); + if (ret) + goto cleanup; + + /* child address format: 0xIooooooo oooooooo, I = bar index, o = offset on bar */ + addr_ncells = of_n_addr_cells(node); + if (addr_ncells > 2) { + /* does not support number of address cells greater than 2 */ + ret = -EINVAL; + goto cleanup; + } + + /* range cells include */ + range_ncells = addr_ncells + 4; + prop = kzalloc(range_ncells * sizeof(u32) * PCI_STD_NUM_BARS, GFP_KERNEL); + if (!prop) { + ret = -ENOMEM; + goto cleanup; + } + + range_cell = prop; + for (i = 0; i < PCI_STD_NUM_BARS; i++) { + if (!pci_resource_len(pdev, i)) + continue; + /* highest 4 bits of address are bar index */ + *(__be64 *)range_cell = cpu_to_be64((u64)i << 60); + range_cell += 2; + if (addr_ncells == 2) + *(__be64 *)range_cell = cpu_to_be64((u64)pci_resource_start(pdev, i)); + else + *(__be32 *)range_cell = cpu_to_be32((u32)pci_resource_start(pdev, i)); + + range_cell += addr_ncells; + *(__be64 *)range_cell = cpu_to_be64((u64)pci_resource_len(pdev, i)); + range_cell += 2; + } + + /* error out if there is not PCI BAR been found */ + if ((void *)range_cell == prop) { + ret = -EINVAL; + goto cleanup; + } + + ret = of_ep_add_property(dev, &proplist, "ranges", (void *)range_cell - prop, prop); + kfree(prop); + if (ret) + goto cleanup; + + node->properties = proplist; + ret = of_attach_node(node); + if (ret) + goto cleanup; + + devres_add(dev, node); + + return 0; + +cleanup: + kfree(prop); + if (node) + devres_free(node); + + return ret; +} +EXPORT_SYMBOL_GPL(devm_of_pci_create_bus_endpoint); + +struct device_node *of_pci_find_bus_endpoint(struct pci_dev *pdev) +{ + struct device_node *dn; + char *path; + + path = kasprintf(GFP_KERNEL, "/pci-ep-bus@%llx", + (u64)pci_resource_start(pdev, 0)); + if (!path) + return NULL; + + dn = of_find_node_by_path(path); + kfree(path); + + return dn; +} +EXPORT_SYMBOL_GPL(of_pci_find_bus_endpoint); +#endif /* CONFIG_OF_DYNAMIC */ + #endif /* CONFIG_PCI */ /** diff --git a/include/linux/of_pci.h b/include/linux/of_pci.h index 29658c0ee71f..c1d86be321b2 100644 --- a/include/linux/of_pci.h +++ b/include/linux/of_pci.h @@ -38,4 +38,19 @@ of_irq_parse_and_map_pci(const struct pci_dev *dev, u8 slot, u8 pin) } #endif +#if IS_ENABLED(CONFIG_OF_DYNAMIC) && IS_ENABLED(CONFIG_PCI) +int devm_of_pci_create_bus_endpoint(struct pci_dev *pdev); +struct device_node *of_pci_find_bus_endpoint(struct pci_dev *pdev); +#else +static inline int devm_of_pci_create_bus_endpoint(struct pci_dev *pdev) +{ + return -EINVAL; +} + +static inline struct device_node *of_pci_find_bus_endpoint(struct pci_dev *pdev) +{ + return NULL; +} +#endif + #endif