From patchwork Tue Feb 23 15:44:43 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Ben Levinsky X-Patchwork-Id: 386380 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-16.7 required=3.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID, HEADER_FROM_DIFFERENT_DOMAINS, INCLUDES_CR_TRAILER, INCLUDES_PATCH, MAILING_LIST_MULTI, SPF_HELO_NONE, SPF_PASS, URIBL_BLOCKED, USER_AGENT_GIT autolearn=unavailable autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id 14A61C433E9 for ; Tue, 23 Feb 2021 15:46:24 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by mail.kernel.org (Postfix) with ESMTP id DF74E64E20 for ; Tue, 23 Feb 2021 15:46:23 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233339AbhBWPp6 (ORCPT ); Tue, 23 Feb 2021 10:45:58 -0500 Received: from mail-dm6nam10on2072.outbound.protection.outlook.com ([40.107.93.72]:54542 "EHLO NAM10-DM6-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S233313AbhBWPpt (ORCPT ); Tue, 23 Feb 2021 10:45:49 -0500 ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=D9AAw1uaGBAIHVudafQcgKQTV2O2rU3aZSxFKEIebxmZ71b5oYYAJMDZAb80W8OXk9o75oErxKTqrKQi6wdn/vbtxnI9OQRaLQQLQeppGS4T/WuN9MVxOgf6Whpyv45Ttd+xjE5mLxYwP0wPWJqp3XoSOg4xiSW43hd1bpoGP2endLQHWhBKWatH4QuA7UAXT4SL98T0t2YzYvR9uXAuVT2Cuy1g6Z3OMwQ2UFPV8tLu8m48i8Uwjk2S+CBJCPPy9dx6lK89LMbxJv1Xs4MmozgN4pFBACmjGNezIUixADutbY9JE7O+WHdVIicUX+zPPElVkUY+kom8lt7qmRwdaw== 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-SenderADCheck; bh=jYIeQpzyVpt1bQOQiAd2rj0avg9N9KkuqLNxbzqangA=; b=bkhP9dbuKsYTPVbQ5bsfQjaSrJfjPrvGibfMSqjGxQHKYCzglEhaAw1yP2YsiXz+G4B/TasuiYps6lyyWfdTO+EXUNtEigYEVnDf+9CPQEghfM20NYVrD52tqjOeH3LORlnKEJZDCUCXIdMjPrtRp0ZnT4+lrb0ilkecUOOjp/5h441/MMEViFeNkO0qmKaSGExduX5jtv7vopx1Ciy+/r1wttdFwY5r4nyNanLCeQw47f5Guko9AMXeL5eViBUnVRvXqAyzZcFQYQuhgd000i+o/38vdeTCLAQeqtuLCw3pWZ3fXQNS6A4hrz42lrX5V4f1WZOuC3nGin3X8PSuwQ== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=pass (sender ip is 149.199.62.198) smtp.rcpttodomain=linaro.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=jYIeQpzyVpt1bQOQiAd2rj0avg9N9KkuqLNxbzqangA=; b=fdmNhWzXBBSMPfHNkSuRDFoWWdwvpgrWNgD6txppAKd/Sa2xrNB2zOdpKy37nxRhGy61l4g3Obs6DyT7LsCs8tLJhccYBtdFiaxjDiHPe4CCJB78inBcIb3VaxS9AWUxhQNFVMDoANPc6Xr0/JH96fmO98G2GuLxdegrjFeS158= Received: from BL1PR13CA0066.namprd13.prod.outlook.com (2603:10b6:208:2b8::11) by BYAPR02MB4455.namprd02.prod.outlook.com (2603:10b6:a03:10::19) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.3868.27; Tue, 23 Feb 2021 15:44:53 +0000 Received: from BL2NAM02FT057.eop-nam02.prod.protection.outlook.com (2603:10b6:208:2b8:cafe::a3) by BL1PR13CA0066.outlook.office365.com (2603:10b6:208:2b8::11) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.3890.9 via Frontend Transport; Tue, 23 Feb 2021 15:44:53 +0000 X-MS-Exchange-Authentication-Results: spf=pass (sender IP is 149.199.62.198) smtp.mailfrom=xilinx.com; linaro.org; dkim=none (message not signed) header.d=none; linaro.org; 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-pvapexch01.xlnx.xilinx.com; Received: from xsj-pvapexch01.xlnx.xilinx.com (149.199.62.198) by BL2NAM02FT057.mail.protection.outlook.com (10.152.77.36) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.20.3868.27 via Frontend Transport; Tue, 23 Feb 2021 15:44:52 +0000 Received: from xsj-pvapexch02.xlnx.xilinx.com (172.19.86.41) by xsj-pvapexch01.xlnx.xilinx.com (172.19.86.40) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.1913.5; Tue, 23 Feb 2021 07:44:47 -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.1913.5 via Frontend Transport; Tue, 23 Feb 2021 07:44:47 -0800 Envelope-to: michal.simek@xilinx.com, mathieu.poirier@linaro.org, devicetree@vger.kernel.org, linux-remoteproc@vger.kernel.org, linux-kernel@vger.kernel.org, linux-arm-kernel@lists.infradead.org Received: from [172.19.2.206] (port=40282 helo=xsjblevinsk50.xilinx.com) by smtp.xilinx.com with esmtp (Exim 4.90) (envelope-from ) id 1lEZrj-0006uk-8d; Tue, 23 Feb 2021 07:44:47 -0800 From: Ben Levinsky To: CC: , , , , Subject: [PATCH v26 1/5] firmware: xilinx: Add ZynqMP firmware ioctl enums for RPU configuration. Date: Tue, 23 Feb 2021 07:44:43 -0800 Message-ID: <20210223154447.13247-2-ben.levinsky@xilinx.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20210223154447.13247-1-ben.levinsky@xilinx.com> References: <20210223154447.13247-1-ben.levinsky@xilinx.com> MIME-Version: 1.0 X-EOPAttributedMessage: 0 X-MS-Office365-Filtering-HT: Tenant X-MS-PublicTrafficType: Email X-MS-Office365-Filtering-Correlation-Id: a1f1468e-cb52-4061-56e8-08d8d811f646 X-MS-TrafficTypeDiagnostic: BYAPR02MB4455: X-Microsoft-Antispam-PRVS: X-Auto-Response-Suppress: DR, RN, NRN, OOF, AutoReply X-MS-Oob-TLC-OOBClassifiers: OLM:254; X-MS-Exchange-SenderADCheck: 1 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: 0blNoj4i821hHPjJizIhvV+Y8CxT1Q3TyTd2nDkqpCPc9OL4z3RWZbwMu9/rt85UBfjUGrNCubkIcV5KfPRk0MlsVmTQCFtcenXBQ0vLPksE1pejZmi5VGkqjXQUc4vUZ4txsN5WYNt/rERjJzGg3dh9QfuFYtp8vZYiyHdkrcn7G/joO2RHgJ5wLDjnb//uFHCyuuxmA1aqxqdhaWTAL2yV/u/UuGqGCwL81Ef+XqFW9Mkk39WBKCPEQCZ1YYFu4Fn9bWnGpNMgKlYPUJk6EZqHdN+BqxgV+ujkyvv+mN7yPa60LjardjUujTOG8+td4WgLkrJg7BZSQBR/DpfOTj+CpJAtVMIX6+QtKrhtJlyeVvEjLRiDnsLJu03+dFTWb38DToMXIhjlyyvgVPXxuBj//LOHT9qmtY6hyX/XIYom1CBSS7q1qh+RhEvRJ9vJ6jvglodrGhedhw2PXsFJZnuU0c4QiZNhLmZ6rFzg+ddefOc20yuJr4dNDkGamvzSpXuxrUJfE3PMTd3kwGuxhOotBrMvedJCGCLpkg7vRu+O8AlS7Fs1kJktMOSzpSAU6EF00MixKxrI1yaLd3gQTQDTFTK8a5vOhqVqjv0IjmYpM7QBZcSswPxKNlvO4YUXjxEHgTum/7uuaNtdLAX+pMpKopzrJP3qMfhTh9se2B6GPLua9AOBsV9W7Fq9wEuj X-Forefront-Antispam-Report: CIP:149.199.62.198; CTRY:US; LANG:en; SCL:1; SRV:; IPV:NLI; SFV:NSPM; H:xsj-pvapexch01.xlnx.xilinx.com; PTR:unknown-62-198.xilinx.com; CAT:NONE; SFS:(4636009)(346002)(136003)(376002)(39860400002)(396003)(36840700001)(46966006)(8936002)(82740400003)(8676002)(82310400003)(36860700001)(26005)(4326008)(6666004)(5660300002)(2906002)(1076003)(54906003)(316002)(36756003)(70206006)(36906005)(186003)(356005)(6916009)(2616005)(83380400001)(47076005)(7636003)(426003)(107886003)(7696005)(70586007)(478600001)(9786002)(44832011)(336012)(102446001); DIR:OUT; SFP:1101; X-OriginatorOrg: xilinx.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 23 Feb 2021 15:44:52.9364 (UTC) X-MS-Exchange-CrossTenant-Network-Message-Id: a1f1468e-cb52-4061-56e8-08d8d811f646 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-pvapexch01.xlnx.xilinx.com] X-MS-Exchange-CrossTenant-AuthSource: BL2NAM02FT057.eop-nam02.prod.protection.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Anonymous X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: BYAPR02MB4455 Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org Add ZynqMP firmware ioctl enums for RPU configuration and TCM Nodes for later use via request_node and release_node Signed-off-by: Ben Levinsky --- include/linux/firmware/xlnx-zynqmp.h | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/include/linux/firmware/xlnx-zynqmp.h b/include/linux/firmware/xlnx-zynqmp.h index 5968df82b991..e4044cae0713 100644 --- a/include/linux/firmware/xlnx-zynqmp.h +++ b/include/linux/firmware/xlnx-zynqmp.h @@ -104,6 +104,10 @@ enum pm_ret_status { }; enum pm_ioctl_id { + IOCTL_GET_RPU_OPER_MODE = 0, + IOCTL_SET_RPU_OPER_MODE = 1, + IOCTL_RPU_BOOT_ADDR_CONFIG = 2, + IOCTL_TCM_COMB_CONFIG = 3, IOCTL_SD_DLL_RESET = 6, IOCTL_SET_SD_TAPDELAY, IOCTL_SET_PLL_FRAC_MODE, @@ -129,6 +133,21 @@ enum pm_query_id { PM_QID_CLOCK_GET_MAX_DIVISOR, }; +enum rpu_oper_mode { + PM_RPU_MODE_LOCKSTEP = 0, + PM_RPU_MODE_SPLIT = 1, +}; + +enum rpu_boot_mem { + PM_RPU_BOOTMEM_LOVEC = 0, + PM_RPU_BOOTMEM_HIVEC = 1, +}; + +enum rpu_tcm_comb { + PM_RPU_TCM_SPLIT = 0, + PM_RPU_TCM_COMB = 1, +}; + enum zynqmp_pm_reset_action { PM_RESET_ACTION_RELEASE, PM_RESET_ACTION_ASSERT, @@ -273,6 +292,10 @@ enum zynqmp_pm_request_ack { }; enum pm_node_id { + NODE_TCM_0_A = 15, + NODE_TCM_0_B, + NODE_TCM_1_A, + NODE_TCM_1_B, NODE_SD_0 = 39, NODE_SD_1, }; From patchwork Tue Feb 23 15:44:46 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Ben Levinsky X-Patchwork-Id: 386381 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-16.7 required=3.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID, HEADER_FROM_DIFFERENT_DOMAINS, INCLUDES_CR_TRAILER, INCLUDES_PATCH, MAILING_LIST_MULTI, SPF_HELO_NONE, SPF_PASS, URIBL_BLOCKED, USER_AGENT_GIT autolearn=unavailable autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id 757E6C433E6 for ; Tue, 23 Feb 2021 15:45:54 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by mail.kernel.org (Postfix) with ESMTP id 4FBBF64E77 for ; Tue, 23 Feb 2021 15:45:54 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233243AbhBWPpy (ORCPT ); Tue, 23 Feb 2021 10:45:54 -0500 Received: from mail-dm6nam08on2072.outbound.protection.outlook.com ([40.107.102.72]:24065 "EHLO NAM04-DM6-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S233274AbhBWPpq (ORCPT ); Tue, 23 Feb 2021 10:45:46 -0500 ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=jd0gPYr5SPP72w96NcvXnkKk+b573MUc54m6xbPWWz+UziHrqWiIhBKwTIvkQ2NOi3nGXcE8LdSOWtq4zOpJpg7uN+P87zz8O1czMxq1be2l1jICIowhyUZ+QUUwiSL1QAhwSKT65qq41v6KyRKTp8XfieLt6XWDXmok2gUWfynEEPeLN9kuJIJGqchpwlZIcnHWz77vBagZIIlfV/Xs16SfQJze0Lf58LDFiRXgog6zNu/ZJ4e1OLuv1LIK0XYtI9ZKaMRibHkalEdQlTM7g52Ex2HPguqTtq9c+hEoAM9P2+C3UiC9qrmiE4/Cbxzzwg6pQxos0EPqOxc7pj4d3w== 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-SenderADCheck; bh=pjjQkbWISU7bk9PEYhNmG4HT88EK20drwrAsoqLg5jA=; b=IM3JmoiTnoD2iY6lChw1X2gde8PBd5B+YxNrWvBDk6YxXnfHH7HPj05eoWUVHydXA8FRx6eoA8jN6/GIzMh6t/mloBLdTzVGhnn3n5ENjuJgGm5OirP2Xt7c2mrCl1PEbt1hRdUDNGIIYzddnsjYTDL7EUuD30VhSRBR4y2AZq9wB1McQQD4X7GeNYkcFlcMOkCGDrIPoq682GxkN8K9IUVrd0fKfQoQ3Q2hsmy0mhNq7tYUKoCbF6Xv5SjFHk1iA10N/D64dMOqmHBP8VOP9hZJ4MJHGb+xdIBv8ozVwFIWN9z4FnYTQAtZiKuOUsGaa52xfsyW8eL3gOLWv4UtVA== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=pass (sender ip is 149.199.62.198) smtp.rcpttodomain=linaro.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=pjjQkbWISU7bk9PEYhNmG4HT88EK20drwrAsoqLg5jA=; b=is4Km+6yb7xDnCs5lwhHcAOlAu9FvjUUrWbflUZzmprombW4BADGhd86MqS2Fg1KJfAxq6BErf87hJm9maCjkXwEDYzh6q6LbUiDI+34p7w6sNJf1oek4OzPBVft+bxQ9BhY5ATO6UOhCDGyEYD5yDaMRXlLCz6ePwFR90+hFiE= Received: from SN4PR0501CA0088.namprd05.prod.outlook.com (2603:10b6:803:22::26) by BYAPR02MB4280.namprd02.prod.outlook.com (2603:10b6:a03:5a::29) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.3868.32; Tue, 23 Feb 2021 15:44:48 +0000 Received: from SN1NAM02FT035.eop-nam02.prod.protection.outlook.com (2603:10b6:803:22:cafe::6) by SN4PR0501CA0088.outlook.office365.com (2603:10b6:803:22::26) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.3890.10 via Frontend Transport; Tue, 23 Feb 2021 15:44:47 +0000 X-MS-Exchange-Authentication-Results: spf=pass (sender IP is 149.199.62.198) smtp.mailfrom=xilinx.com; linaro.org; dkim=none (message not signed) header.d=none; linaro.org; 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 SN1NAM02FT035.mail.protection.outlook.com (10.152.72.145) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.20.3868.27 via Frontend Transport; Tue, 23 Feb 2021 15:44:47 +0000 Received: from xsj-pvapexch01.xlnx.xilinx.com (172.19.86.40) 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.1913.5; Tue, 23 Feb 2021 07:44:47 -0800 Received: from smtp.xilinx.com (172.19.127.95) by xsj-pvapexch01.xlnx.xilinx.com (172.19.86.40) with Microsoft SMTP Server id 15.1.1913.5 via Frontend Transport; Tue, 23 Feb 2021 07:44:47 -0800 Envelope-to: michal.simek@xilinx.com, mathieu.poirier@linaro.org, devicetree@vger.kernel.org, linux-remoteproc@vger.kernel.org, linux-kernel@vger.kernel.org, linux-arm-kernel@lists.infradead.org Received: from [172.19.2.206] (port=40282 helo=xsjblevinsk50.xilinx.com) by smtp.xilinx.com with esmtp (Exim 4.90) (envelope-from ) id 1lEZrj-0006uk-9z; Tue, 23 Feb 2021 07:44:47 -0800 From: Ben Levinsky To: CC: , , , , Subject: [PATCH v26 4/5] dt-bindings: remoteproc: Add documentation for ZynqMP R5 rproc bindings Date: Tue, 23 Feb 2021 07:44:46 -0800 Message-ID: <20210223154447.13247-5-ben.levinsky@xilinx.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20210223154447.13247-1-ben.levinsky@xilinx.com> References: <20210223154447.13247-1-ben.levinsky@xilinx.com> MIME-Version: 1.0 X-EOPAttributedMessage: 0 X-MS-Office365-Filtering-HT: Tenant X-MS-PublicTrafficType: Email X-MS-Office365-Filtering-Correlation-Id: 91e0c5ac-184d-40e2-df4d-08d8d811f32f X-MS-TrafficTypeDiagnostic: BYAPR02MB4280: X-Microsoft-Antispam-PRVS: X-Auto-Response-Suppress: DR, RN, NRN, OOF, AutoReply X-MS-Oob-TLC-OOBClassifiers: OLM:8882; X-MS-Exchange-SenderADCheck: 1 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: +obes9re7umEJDR7rJNpeICUHUAoOpX9IMMralgb7DDfAiIkrbH2EIRfiGh5YlKFqsa9GdGtb4Nhg4Ny1cqSXAsI8Jd14waRL/TM6ggOl8iOqDZyOVETiUyh5T7ygGM6l+WyY8yp7oK+AJeSg3QG3SpkH2L9d6lVjnjgQHM5ustnKaVtXoWbamR996zBXoo+iwRw2uMgW9lW09PoMAbhl20YQxk0t028S2Y+2dZG9r408EncOSznur91Vr8u09s0r83YhJ3ggolWdzneFMneU1d6YFJV3hsOisqFHoOKRRVdl9wLB6fH1YxQqk5DIVHuH4tpZ1yBzGjQ1+g8/Wa0HtQK+qEMw+OF0g8721xbA726ITSDxK2DNEoIuzrQbtTLK3446zsCjJMyxq37d9V0VhTMjsdLm6+3pjyr9JFx4dPFauQfq1jjeNNn32UYwM+1D1yoXPTqGel/8RlGRqbum/uKDwlojPdzIXXLBHa+kScFqqedgYSe1DySWAnGz0xju8/I9w1gtGcFPP/ZdOxLGWW/Fv+FIMugcjVCj7vGusHrQQV01Ey6Oc/VtKR7J4Ad8iocoBIypqKnGvvGFQ3ZcUqKIv39/XiCB1TYONSP0UeEMaLQ54bMEIkFt/QXPacQ41zV3+0V+iKBgnVn2CTeuNjNhThz3Vd5ctG8Bhwc6c1COc7CLo+DX7LPGtUPf1hLjHKPEH6d83k0aN3Ox2rzm6kVMMoM5QodXvWd1/A2sG2A+MB5L6VxJlvJzP52RplN 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:(4636009)(346002)(396003)(136003)(376002)(39860400002)(36840700001)(46966006)(107886003)(47076005)(36756003)(356005)(6916009)(8676002)(2906002)(478600001)(4326008)(70586007)(83380400001)(36860700001)(70206006)(5660300002)(9786002)(82740400003)(7636003)(36906005)(7696005)(82310400003)(44832011)(316002)(54906003)(426003)(186003)(8936002)(2616005)(336012)(1076003)(26005)(102446001); DIR:OUT; SFP:1101; X-OriginatorOrg: xilinx.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 23 Feb 2021 15:44:47.8130 (UTC) X-MS-Exchange-CrossTenant-Network-Message-Id: 91e0c5ac-184d-40e2-df4d-08d8d811f32f 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: SN1NAM02FT035.eop-nam02.prod.protection.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Anonymous X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: BYAPR02MB4280 Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org Add binding for ZynqMP R5 OpenAMP. Represent the RPU domain resources in one device node. Each RPU processor is a subnode of the top RPU domain node. Signed-off-by: Jason Wu Signed-off-by: Wendy Liang Signed-off-by: Michal Simek Signed-off-by: Ben Levinsky --- .../xilinx,zynqmp-r5-remoteproc.yaml | 223 ++++++++++++++++++ 1 file changed, 223 insertions(+) create mode 100644 Documentation/devicetree/bindings/remoteproc/xilinx,zynqmp-r5-remoteproc.yaml diff --git a/Documentation/devicetree/bindings/remoteproc/xilinx,zynqmp-r5-remoteproc.yaml b/Documentation/devicetree/bindings/remoteproc/xilinx,zynqmp-r5-remoteproc.yaml new file mode 100644 index 000000000000..b9412c2def7b --- /dev/null +++ b/Documentation/devicetree/bindings/remoteproc/xilinx,zynqmp-r5-remoteproc.yaml @@ -0,0 +1,223 @@ +# SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause) +%YAML 1.2 +--- +$id: "http://devicetree.org/schemas/remoteproc/xilinx,zynqmp-r5-remoteproc.yaml#" +$schema: "http://devicetree.org/meta-schemas/core.yaml#" + +title: Xilinx R5 remote processor controller bindings + +description: + This document defines the binding for the remoteproc component that loads and + boots firmwares on the Xilinx Zynqmp and Versal family chipsets. + + Note that the Linux has global addressing view of the R5-related memory (TCM) + so the absolute address ranges are provided in TCM reg's. + +maintainers: + - Ed Mooring + - Ben Levinsky + +properties: + $nodename: + pattern: "^r5fss(@.*)?" + + compatible: + enum: + - xlnx,zynqmp-r5-remoteproc + + reg: + items: + - description: Address and Size of Xilinx RPU Configuration register + + "#address-cells": + const: 2 + + "#size-cells": + const: 2 + + ranges: true + +# Optional properties: +# -------------------- + xlnx,cluster-mode: + $ref: /schemas/types.yaml#/definitions/uint32 + enum: [0, 1] + description: | + Configuration Mode for the Dual R5F cores within the R5F cluster. + Should be either a value of 1 (LockStep mode) or 0 (Split mode), + default is LockStep mode if omitted. + + +#R5F Processor Child Nodes: +# ========================== + +patternProperties: + "^r5f_[a-f0-9]+$": + type: object + description: | + The R5F Sub-System device node should define one or two R5F child nodes, + each node representing a Xilinx instantiation of the Arm Cortex R5F core. + There should be one or two child nodes if the R5F is in Split mode and + one child node if the R5F is in Lockstep mode. + + In Split mode each R5F core has two associated TCM Banks. R5_0 has + TCM Banks 0A and 0B and R5_1 has TCM Banks 1A and 1B. + + In Lockstep mode only one R5F child node should be defined. This one + child has access to TCM Banks 0A, 0B, 1A and 1B and any of the four can + be included in the child R5F's sram property. + + The example below shows Split mode with two child nodes. + + properties: + compatible: + enum: + - xilinx,r5f + +# The following properties are mandatory for R5F Core0 in both LockStep and Split +# modes, and are mandatory for R5F Core1 _only_ in Split mode. + + memory-region: + description: | + Phandles to the memory nodes to be associated with the + The reserved memory nodes should be carveout nodes, and + should be defined with a "no-map" property as per the bindings in + Documentation/devicetree/bindings/reserved-memory/reserved-memory.txt + minItems: 1 + maxItems: 6 + items: + - description: Region used for dynamic DMA allocations like vrings and + vring buffers + - description: Region reserved for firmware image sections + additionalItems: true + + power-domain: + description: | + Power node ID that is used to uniquely identify the RPU for Xilinx + Power Management. + maxItems: 1 + +# Optional properties: +# -------------------- +# The following properties are optional properties for each of the R5F cores: + + mboxes: + description: | + Standard property to specify a mailbox + This property is required only if the rpmsg/virtio functionality + is used + + Refer to the zynqmp-ipi-mailbox documentation for client usage of this + property + maxItems: 1 + + mbox-names: + description: | + Refer to the zynqmp-ipi-mailbox documentation for client usage of this + property + items: + - const: tx + - const: rx + + sram: + $ref: /schemas/types.yaml#/definitions/phandle-array + minItems: 1 + maxItems: 4 + description: | + Phandles to one or more reserved on-chip SRAM regions. The regions + should be defined as child nodes of the respective SRAM node, and + should be defined as per the generic bindings in + Documentation/devicetree/bindings/sram/sram.yaml + + required: + - compatible + - power-domain + + unevaluatedProperties: false + +required: + - reg + - compatible + - "#address-cells" + - "#size-cells" + - ranges + +additionalProperties: false + +examples: + - | + / { + compatible = "xlnx,zynqmp-zcu102-rev1.0", "xlnx,zynqmp-zcu102", "xlnx,zynqmp"; + #address-cells = <2>; + #size-cells = <2>; + model = "ZynqMP ZCU102 "; + + zynqmp_ipi1 { + compatible = "xlnx,zynqmp-ipi-mailbox"; + interrupt-parent = <&gic>; + interrupts = <0 33 4>; + xlnx,ipi-id = <5>; + #address-cells = <1>; + #size-cells = <0>; + + ipi_mailbox_rpu0: mailbox@ff990600 { + reg = <0xff990600 0x20>, + <0xff990620 0x20>, + <0xff9900c0 0x20>, + <0xff9900e0 0x20>; + reg-names = "local_request_region", + "local_response_region", + "remote_request_region", + "remote_response_region"; + #mbox-cells = <1>; + xlnx,ipi-id = <3>; + }; + ipi_mailbox_rpu1: mailbox@ff990780 { + reg = <0xff990780 0x20>, + <0xff9907a0 0x20>, + <0xff9907c0 0x20>, + <0xff9905a0 0x20>; + reg-names = "local_request_region", + "local_response_region", + "remote_request_region", + "remote_response_region"; + #mbox-cells = <1>; + xlnx,ipi-id = <3>; + }; + }; + + r5fss@ff9a0000 { + compatible = "xlnx,zynqmp-r5-remoteproc"; + #address-cells = <2>; + #size-cells = <2>; + ranges; + reg = <0x0 0xff9a0000 0x0 0x10000>; + xlnx,cluster-mode = <0>; + + r5f_0 { + compatible = "xilinx,r5f"; + memory-region = <&elf_load0>, + <&rpu0vdev0vring0>, + <&rpu0vdev0vring1>, + <&rpu0vdev0buffer>; + sram = <&tcm_0a>, <&tcm_0b>; + mboxes = <&ipi_mailbox_rpu0 0x0 &ipi_mailbox_rpu0 0x1>; + mbox-names = "tx", "rx"; + power-domain = <0x7>; + }; + r5f_1 { + compatible = "xilinx,r5f"; + memory-region = <&elf_load1>, + <&rpu1vdev0vring0>, + <&rpu1vdev0vring1>, + <&rpu1vdev0buffer>; + sram = <&tcm_1a>, <&tcm_1b>; + mboxes = <&ipi_mailbox_rpu1 0x0 &ipi_mailbox_rpu1 0x1>; + mbox-names = "tx", "rx"; + power-domain = <0x8>; + }; + + }; + }; + +... From patchwork Tue Feb 23 15:44:47 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Ben Levinsky X-Patchwork-Id: 386379 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-16.7 required=3.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID, HEADER_FROM_DIFFERENT_DOMAINS, INCLUDES_CR_TRAILER, INCLUDES_PATCH, MAILING_LIST_MULTI, SPF_HELO_NONE, SPF_PASS, URIBL_BLOCKED, USER_AGENT_GIT autolearn=unavailable autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id 2C72FC433E9 for ; Tue, 23 Feb 2021 15:46:43 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by mail.kernel.org (Postfix) with ESMTP id E98B964E20 for ; Tue, 23 Feb 2021 15:46:42 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233373AbhBWPqa (ORCPT ); Tue, 23 Feb 2021 10:46:30 -0500 Received: from mail-dm6nam11on2078.outbound.protection.outlook.com ([40.107.223.78]:60576 "EHLO NAM11-DM6-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S233314AbhBWPpy (ORCPT ); Tue, 23 Feb 2021 10:45:54 -0500 ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=DXVBUmVt7E1SSDZCwAj6yTS+ZB+JSJP6V7PAjq/dFO6PINq+SyjZHtStuhGSAql88LlRRmfOrOyi3VbAnl9sIZDVGiGVN5e+hbo4dP6E+hCBvjjyUN6LX2J7AmQt7fBx9Ht1ESBkJtFfzarix27shYKOcpAYWBJkgEAOl9FP4WwqT1tQEc8uC9MH345ipA0VYoTa/0hB8Tyx0sG+jtvmEbHF2L+AYYjV7XYGPal8Tc56TtfBPepQG7N395gSBci9u5Ub3j6cRkLhoSAHTi9ZaRQu6zjoTCVkztoR4dF1LQSBlbc/Us02JgPZgVlYJCeaT27c8C+5XRYbgpKAvAl2Zw== 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-SenderADCheck; bh=FlbRUKp5jgpT0bfskoLhAHcYQDPzjvcJjZkzbEsPAW4=; b=lDA6tYLCoyfDZlb1y04T19m2cAnQIBwYbDFMYMHsXCGv5ZC4mzIthx3j34MiH/VCH8qnT2rAAVEQhh/BomVc8QmZdBAdTgDAQkN3DpMUH2F4ONTtU5e86wGagXLcoOsld2t3FnfJqFF25sp7x0k4APjXCHbDjDA/O1rgWQCImliKeCxfam+5ZgD1ZeuPqMKU1DhEjg2yJH4AY7SoIvaFeb3vrYeUYMhx4kx89DkiUqS0+4xYR2PgUPaWjmevCF6ajw5Eq0qUqxxq0E/RDiYy28f2trFKzoGlGd0/B2I4FKI/Aj7jb42cH0TzPB0+yZ2qjjS9gswZarO3eYWgsSYIGw== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=pass (sender ip is 149.199.62.198) smtp.rcpttodomain=linaro.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=FlbRUKp5jgpT0bfskoLhAHcYQDPzjvcJjZkzbEsPAW4=; b=ZhNru5edt1MDBnX4gXkEOFKyj92KPyXQdlGIn1zbFd8QsLS3Is1Y1iun1YIqfDDXtnkWhKvavwVcJ19HRbdNGMBenKncrY/g5Q2SBltgR8HeDXBXj/F5pGnsfishhZvcRpF0ehaGINutqZhroMRmJZWmoj4w3kmOxlOF9hTkRq0= Received: from BL1PR13CA0069.namprd13.prod.outlook.com (2603:10b6:208:2b8::14) by BYAPR02MB5432.namprd02.prod.outlook.com (2603:10b6:a03:9a::10) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.3846.27; Tue, 23 Feb 2021 15:44:54 +0000 Received: from BL2NAM02FT057.eop-nam02.prod.protection.outlook.com (2603:10b6:208:2b8:cafe::76) by BL1PR13CA0069.outlook.office365.com (2603:10b6:208:2b8::14) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.3890.10 via Frontend Transport; Tue, 23 Feb 2021 15:44:53 +0000 X-MS-Exchange-Authentication-Results: spf=pass (sender IP is 149.199.62.198) smtp.mailfrom=xilinx.com; linaro.org; dkim=none (message not signed) header.d=none; linaro.org; 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-pvapexch01.xlnx.xilinx.com; Received: from xsj-pvapexch01.xlnx.xilinx.com (149.199.62.198) by BL2NAM02FT057.mail.protection.outlook.com (10.152.77.36) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.20.3868.27 via Frontend Transport; Tue, 23 Feb 2021 15:44:53 +0000 Received: from xsj-pvapexch01.xlnx.xilinx.com (172.19.86.40) by xsj-pvapexch01.xlnx.xilinx.com (172.19.86.40) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.1913.5; Tue, 23 Feb 2021 07:44:47 -0800 Received: from smtp.xilinx.com (172.19.127.95) by xsj-pvapexch01.xlnx.xilinx.com (172.19.86.40) with Microsoft SMTP Server id 15.1.1913.5 via Frontend Transport; Tue, 23 Feb 2021 07:44:47 -0800 Envelope-to: michal.simek@xilinx.com, mathieu.poirier@linaro.org, devicetree@vger.kernel.org, linux-remoteproc@vger.kernel.org, linux-kernel@vger.kernel.org, linux-arm-kernel@lists.infradead.org Received: from [172.19.2.206] (port=40282 helo=xsjblevinsk50.xilinx.com) by smtp.xilinx.com with esmtp (Exim 4.90) (envelope-from ) id 1lEZrj-0006uk-Af; Tue, 23 Feb 2021 07:44:47 -0800 From: Ben Levinsky To: CC: , , , , Subject: [PATCH v26 5/5] remoteproc: Add initial zynqmp R5 remoteproc driver Date: Tue, 23 Feb 2021 07:44:47 -0800 Message-ID: <20210223154447.13247-6-ben.levinsky@xilinx.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20210223154447.13247-1-ben.levinsky@xilinx.com> References: <20210223154447.13247-1-ben.levinsky@xilinx.com> MIME-Version: 1.0 X-EOPAttributedMessage: 0 X-MS-Office365-Filtering-HT: Tenant X-MS-PublicTrafficType: Email X-MS-Office365-Filtering-Correlation-Id: 416e5243-0dae-4db7-dc4b-08d8d811f6ba X-MS-TrafficTypeDiagnostic: BYAPR02MB5432: X-Microsoft-Antispam-PRVS: X-Auto-Response-Suppress: DR, RN, NRN, OOF, AutoReply X-MS-Oob-TLC-OOBClassifiers: OLM:541; X-MS-Exchange-SenderADCheck: 1 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: Ct1yACSP5SIGGOzsw+6pTQTsm5JhlpTdueTRurIvNzvwV97E0P2z7jWlGWhFqr1dxljkVxh+4/5jvG31JH2QF1T5V5lbwixvK6chO4VFOGnR7z2lcHF/t41Z0atPBV0rsb+yg3tzXBP11kcqQ+gqJ0bPybLqz1G3FvM0p397KCGV4CtaRD5ZjS3+7qGegXyqp+M/918nEdUOrCm851M5H+fQrL2eQwMOBtnXTmlRSkTQoN5TDevvTkXnaww6kfg7LnD3UypK/oxO+JH4/oF/M/Pdo+Ab8Bp1cUtBANI3b/vyPCP24H3NiOHppqZs8o5K84oxlEaa4IpTVvPEJHEtvJNRY4d3DVjDHhHWtzg2AH5gbSdp77+jGn9h6UQNEhkKxm5aj7G2/+6acqW5nAnKkHtgXWPOoKakFvUGAs4GEEmqr+gKczp34jUjsx1cWGjV9yp5a/woy4iLcAGIBeiNOcHLO2uEoxQe9WSsjAyuY3EDYnWYKILBCjRN2Cl9TJr7Dm4oO8ZyU5Brh4QtxEjjVv8llBhJVOWg3eBbYVCGnYlWMrHiGtoDh1mbKD1H/IMUCrgMUxNyMKKJlGloCWVnVr73fQE7XpE0FUaFue9JF1iaiVjQIlrfXRL1bGtyEatQd0LiJFpdy0RKhsytyqR+iv05ArscPGjssMRM3kHTOSbqSPfYqmv0oOY8uXX5n/hP X-Forefront-Antispam-Report: CIP:149.199.62.198; CTRY:US; LANG:en; SCL:1; SRV:; IPV:NLI; SFV:NSPM; H:xsj-pvapexch01.xlnx.xilinx.com; PTR:unknown-62-198.xilinx.com; CAT:NONE; SFS:(4636009)(396003)(346002)(136003)(39860400002)(376002)(36840700001)(46966006)(70586007)(70206006)(8676002)(6916009)(83380400001)(107886003)(30864003)(426003)(1076003)(2616005)(4326008)(9786002)(186003)(44832011)(2906002)(5660300002)(8936002)(336012)(82310400003)(82740400003)(26005)(47076005)(36860700001)(478600001)(7636003)(36756003)(316002)(356005)(7696005)(54906003)(36906005)(102446001); DIR:OUT; SFP:1101; X-OriginatorOrg: xilinx.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 23 Feb 2021 15:44:53.6980 (UTC) X-MS-Exchange-CrossTenant-Network-Message-Id: 416e5243-0dae-4db7-dc4b-08d8d811f6ba 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-pvapexch01.xlnx.xilinx.com] X-MS-Exchange-CrossTenant-AuthSource: BL2NAM02FT057.eop-nam02.prod.protection.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Anonymous X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: BYAPR02MB5432 Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org R5 is included in Xilinx Zynq UltraScale MPSoC so by adding this remoteproc driver, we can boot the R5 sub-system in two different configurations - * Split * Lockstep The Xilinx R5 Remoteproc Driver boots the R5's via calls to the Xilinx Platform Management Unit that handles the R5 configuration, memory access and R5 lifecycle management. The interface to this manager is done in this driver via zynqmp_pm_* function calls. Signed-off-by: Wendy Liang Signed-off-by: Michal Simek Signed-off-by: Ed Mooring Signed-off-by: Jason Wu Signed-off-by: Ben Levinsky --- drivers/remoteproc/Kconfig | 8 + drivers/remoteproc/Makefile | 1 + drivers/remoteproc/zynqmp_r5_remoteproc.c | 954 ++++++++++++++++++++++ 3 files changed, 963 insertions(+) create mode 100644 drivers/remoteproc/zynqmp_r5_remoteproc.c diff --git a/drivers/remoteproc/Kconfig b/drivers/remoteproc/Kconfig index c6659dfea7c7..c2fe54b1d94f 100644 --- a/drivers/remoteproc/Kconfig +++ b/drivers/remoteproc/Kconfig @@ -275,6 +275,14 @@ config TI_K3_DSP_REMOTEPROC It's safe to say N here if you're not interested in utilizing the DSP slave processors. +config ZYNQMP_R5_REMOTEPROC + tristate "ZynqMP R5 remoteproc support" + depends on PM && ARCH_ZYNQMP + select RPMSG_VIRTIO + select ZYNQMP_IPI_MBOX + help + Say y or m here to support ZynqMP R5 remote processors via the remote + processor framework. endif # REMOTEPROC endmenu diff --git a/drivers/remoteproc/Makefile b/drivers/remoteproc/Makefile index 3dfa28e6c701..ef1abff654c2 100644 --- a/drivers/remoteproc/Makefile +++ b/drivers/remoteproc/Makefile @@ -33,3 +33,4 @@ obj-$(CONFIG_ST_REMOTEPROC) += st_remoteproc.o obj-$(CONFIG_ST_SLIM_REMOTEPROC) += st_slim_rproc.o obj-$(CONFIG_STM32_RPROC) += stm32_rproc.o obj-$(CONFIG_TI_K3_DSP_REMOTEPROC) += ti_k3_dsp_remoteproc.o +obj-$(CONFIG_ZYNQMP_R5_REMOTEPROC) += zynqmp_r5_remoteproc.o diff --git a/drivers/remoteproc/zynqmp_r5_remoteproc.c b/drivers/remoteproc/zynqmp_r5_remoteproc.c new file mode 100644 index 000000000000..4bcff2daceaf --- /dev/null +++ b/drivers/remoteproc/zynqmp_r5_remoteproc.c @@ -0,0 +1,954 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Zynq R5 Remote Processor driver + * + * Based on origin OMAP and Zynq Remote Processor driver + * + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "remoteproc_internal.h" + +#define MAX_RPROCS 2 /* Support up to 2 RPU */ +#define MAX_MEM_PNODES 4 /* Max power nodes for one RPU memory instance */ + +#define BANK_LIST_PROP "sram" +#define DDR_LIST_PROP "memory-region" + +/* IPI buffer MAX length */ +#define IPI_BUF_LEN_MAX 32U +/* RX mailbox client buffer max length */ +#define RX_MBOX_CLIENT_BUF_MAX (IPI_BUF_LEN_MAX + \ + sizeof(struct zynqmp_ipi_message)) + +/* + * Map each Xilinx on-chip SRAM Bank address to their own respective + * pm_node_id. + * + * size can differ based on R5 cluster configuration so record from + * device tree in zynqmp_r5_probe. + */ +struct sram_addr_data { + phys_addr_t addr; + enum pm_node_id id; + resource_size_t size; +}; + +#define NUM_SRAMS 4U +static const struct sram_addr_data zynqmp_banks[NUM_SRAMS] = { + {0xffe00000UL, NODE_TCM_0_A, 0}, + {0xffe20000UL, NODE_TCM_0_B, 0}, + {0xffe90000UL, NODE_TCM_1_A, 0}, + {0xffeb0000UL, NODE_TCM_1_B, 0}, +}; + +/** + * struct zynqmp_r5_rproc - ZynqMP R5 core structure + * + * @rx_mc_buf: rx mailbox client buffer to save the rx message + * @srams : srams Xilinx mgmt IDs for srams that will be used by R5 core. + * @tx_mc: tx mailbox client + * @rx_mc: rx mailbox client + * @mbox_work: mbox_work for the RPU remoteproc + * @tx_mc_skbs: socket buffers for tx mailbox client + * @rproc: rproc handle + * @tx_chan: tx mailbox channel + * @rx_chan: rx mailbox channel + * @pnode_id: RPU CPU power domain id + * @elem: linked list item + */ +struct zynqmp_r5_rproc { + unsigned char rx_mc_buf[RX_MBOX_CLIENT_BUF_MAX]; + struct sram_addr_data srams[NUM_SRAMS]; + struct mbox_client tx_mc; + struct mbox_client rx_mc; + struct work_struct mbox_work; + struct sk_buff_head tx_mc_skbs; + struct rproc *rproc; + struct mbox_chan *tx_chan; + struct mbox_chan *rx_chan; + u32 pnode_id; + struct list_head elem; +}; + +/* + * r5_set_mode + * @z_rproc: Remote processor private data + * @rpu_mode: mode specified by device tree to configure the RPU to + * + * set RPU operation mode + * + * Return: 0 for success, negative value for failure + */ +static int r5_set_mode(struct zynqmp_r5_rproc *z_rproc, + enum rpu_oper_mode rpu_mode) +{ + enum rpu_tcm_comb tcm_mode; + enum rpu_oper_mode cur_rpu_mode; + int ret; + + ret = zynqmp_pm_get_rpu_mode(z_rproc->pnode_id, &cur_rpu_mode); + if (ret < 0) + return ret; + + if (rpu_mode != cur_rpu_mode) { + ret = zynqmp_pm_set_rpu_mode(z_rproc->pnode_id, rpu_mode); + if (ret < 0) + return ret; + } + + tcm_mode = (rpu_mode == PM_RPU_MODE_LOCKSTEP) ? + PM_RPU_TCM_COMB : PM_RPU_TCM_SPLIT; + return zynqmp_pm_set_tcm_config(z_rproc->pnode_id, tcm_mode); +} + +/* + * r5_rproc_prepare + * @rproc: single R5 core's corresponding rproc instance + * + * The Xilinx Versal and ZU+ SoC's require devices (in this case srams) to be + * requested for a given device to be used. As the R5's TCM banks may be used + * by the cores, each of these banks has to be requested before they can be + * used. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int r5_rproc_prepare(struct rproc *rproc) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + int ret, i; + + for (i = 0; i < NUM_SRAMS; i++) { + if (!z_rproc->srams[i].id) + continue; + + ret = zynqmp_pm_request_node(z_rproc->srams[i].id, + ZYNQMP_PM_CAPABILITY_ACCESS, 0, + ZYNQMP_PM_REQUEST_ACK_BLOCKING); + if (ret < 0) + return ret; + } + + return ret; +} + +/* + * r5_rproc_unprepare + * @rproc: single R5 core's corresponding rproc instance + * + * This function implements the .unprepare() ops and performs the complementary + * operations to that of the .prepare() ops. This instructs the power + * management controller that the node is no longer needed by that cluster, + * allowing the node to be placed into an inactive state. The cores + * themselves are only halted in the .stop() callback. The .unprepare() ops are + * invoked by the remoteproc core after the remoteproc is stopped. + * + * Return on first error seen as whatever causes the first error + * may cause subsequent iterations to fail with a different error. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int r5_rproc_unprepare(struct rproc *rproc) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + int ret, i; + + for (i = 0; i < NUM_SRAMS; i++) { + if (!z_rproc->srams[i].id) + continue; + + ret = zynqmp_pm_release_node(z_rproc->srams[i].id); + if (ret < 0) + return ret; + } + + return ret; +} + +/* + * zynqmp_r5_rproc_start + * @rproc: single R5 core's corresponding rproc instance + * + * Start R5 Core from designated boot address. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int zynqmp_r5_rproc_start(struct rproc *rproc) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + enum rpu_boot_mem bootmem; + + bootmem = (rproc->bootaddr & 0xF0000000) == 0xF0000000 ? + PM_RPU_BOOTMEM_HIVEC : PM_RPU_BOOTMEM_LOVEC; + + dev_dbg(rproc->dev.parent, "RPU boot from %s.", + bootmem == PM_RPU_BOOTMEM_HIVEC ? "OCM" : "TCM"); + + return zynqmp_pm_request_wake(z_rproc->pnode_id, 1, + bootmem, ZYNQMP_PM_REQUEST_ACK_NO); +} + +/* + * zynqmp_r5_rproc_stop + * @rproc: single R5 core's corresponding rproc instance + * + * Power down R5 Core. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int zynqmp_r5_rproc_stop(struct rproc *rproc) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + + return zynqmp_pm_force_pwrdwn(z_rproc->pnode_id, + ZYNQMP_PM_REQUEST_ACK_BLOCKING); +} + +/* + * zynqmp_r5_rproc_mem_alloc + * @rproc: single R5 core's corresponding rproc instance + * @mem: mem entry to map + * + * Callback to map va for memory-region's carveout. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int zynqmp_r5_rproc_mem_alloc(struct rproc *rproc, + struct rproc_mem_entry *mem) +{ + void *va; + + va = ioremap_wc(mem->dma, mem->len); + if (IS_ERR_OR_NULL(va)) + return -ENOMEM; + + mem->va = va; + + return 0; +} + +/* + * zynqmp_r5_rproc_mem_release + * @rproc: single R5 core's corresponding rproc instance + * @mem: mem entry to unmap + * + * Unmap memory-region carveout + * + * return 0 on success, otherwise non-zero value on failure + */ +static int zynqmp_r5_rproc_mem_release(struct rproc *rproc, + struct rproc_mem_entry *mem) +{ + iounmap(mem->va); + return 0; +} + +/* + * parse_mem_regions + * @rproc: single R5 core's corresponding rproc instance + * + * Construct rproc mem carveouts from carveout provided in + * memory-region property + * + * return 0 on success, otherwise non-zero value on failure + */ +static int parse_mem_regions(struct rproc *rproc) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + struct device_node *np = z_rproc->rproc->dev.parent->of_node; + struct device *dev = &rproc->dev; + struct of_phandle_iterator it; + struct rproc_mem_entry *mem; + struct reserved_mem *rmem; + int index = 0; + + /* Register associated reserved memory regions */ + of_phandle_iterator_init(&it, np, DDR_LIST_PROP, NULL, 0); + while (of_phandle_iterator_next(&it) == 0) { + rmem = of_reserved_mem_lookup(it.node); + if (!rmem) { + dev_err(dev, "unable to acquire %s\n", DDR_LIST_PROP); + return -EINVAL; + } + + if (strstr(it.node->name, "vdev0buffer")) { + mem = rproc_of_resm_mem_entry_init(dev, index, + rmem->size, + rmem->base, + "vdev0buffer"); + } else { + /* + * The ensuing carveout is either for vring, which has + * device tree node name of length 15 characters, or + * firmware mem, where name is not used by remoteproc + * core later on. So default to vring length of 15. + * + * Extra char for null-terminated string. + * + * only allocate in negative case as strstr returns + * pointer + offset to string in positive case. + */ + char *name; + + name = strstr(it.node->name, "vdev0vring"); + if (name) { + /* + * Expect "rpuXvdev0vringX" as documented + * in xilinx remoteproc device tree binding + */ + if (strlen(it.node->name) < 15) { + dev_err(dev, "%pOF is less than 15 chars", + it.node); + return -EINVAL; + } + } else { + name = devm_kzalloc(dev, 16 * sizeof(char), + GFP_KERNEL); + if (!name) + return -ENOMEM; + strncpy(name, it.node->name, 16); + } + + mem = rproc_mem_entry_init(dev, NULL, + (dma_addr_t)rmem->base, + rmem->size, rmem->base, + zynqmp_r5_rproc_mem_alloc, + zynqmp_r5_rproc_mem_release, + name); + } + + if (!mem) + return -ENOMEM; + + rproc_add_carveout(rproc, mem); + index++; + } + + return 0; +} + +/* + * tcm_mem_alloc + * @rproc: single R5 core's corresponding rproc instance + * @mem: mem entry to initialize the va and da fields of + * + * Given TCM bank entry, this callback will set device address for R5 + * running on TCM and also setup virtual address for TCM bank + * remoteproc carveout. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int tcm_mem_alloc(struct rproc *rproc, + struct rproc_mem_entry *mem) +{ + void *va; + struct device *dev = rproc->dev.parent; + + va = ioremap_wc(mem->dma, mem->len); + if (IS_ERR_OR_NULL(va)) + return -ENOMEM; + + /* Update memory entry va */ + mem->va = va; + + /* + * The R5s expect their TCM banks to be at address 0x0 and 0x2000, + * while on the Linux side they are at 0xffexxxxx. + * + * Zero out the high 12 bits of the address. This will give + * expected values for TCM Banks 0A and 0B (0x0 and 0x20000). + */ + mem->da &= 0x000fffff; + + /* + * TCM Banks 1A and 1B still have to be translated. + * + * Below handle these two banks' absolute addresses (0xffe90000 and + * 0xffeb0000) and convert to the expected relative addresses + * (0x0 and 0x20000). + */ + if (mem->da == 0x90000 || mem->da == 0xB0000) + mem->da -= 0x90000; + + /* if translated TCM bank address is not valid report error */ + if (mem->da != 0x0 && mem->da != 0x20000) { + dev_err(dev, "invalid TCM bank address: %x\n", mem->da); + return -EINVAL; + } + + return 0; +} + +/* + * tcm_mem_release + * @rproc: single R5 core's corresponding rproc instance + * @mem: mem entry to unmap + * + * Unmap TCM banks when powering down R5 core. + * + * return 0 on success, otherwise non-zero value on failure + */ +static int tcm_mem_release(struct rproc *rproc, struct rproc_mem_entry *mem) +{ + iounmap(mem->va); + return 0; +} + +/* + * parse_tcm_banks + * @rproc: single R5 core's corresponding rproc instance + * + * Given R5 node in remoteproc instance + * allocate remoteproc carveout for TCM memory + * needed for firmware to be loaded + * + * return 0 on success, otherwise non-zero value on failure + */ +static int parse_tcm_banks(struct rproc *rproc) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + struct rproc_mem_entry *mem; + int i; + + for (i = 0; i < NUM_SRAMS; i++) { + if (!z_rproc->srams[i].id) + continue; + + mem = rproc_mem_entry_init(&rproc->dev, NULL, + z_rproc->srams[i].addr, + (size_t)z_rproc->srams[i].size, + z_rproc->srams[i].addr, + tcm_mem_alloc, + tcm_mem_release, "sram"); + if (!mem) + return -ENOMEM; + + rproc_add_carveout(rproc, mem); + } + + return 0; +} + +/* + * zynqmp_r5_parse_fw + * @rproc: single R5 core's corresponding rproc instance + * @fw: ptr to firmware to be loaded onto r5 core + * + * When loading firmware, ensure the necessary carveouts are in remoteproc + * + * return 0 on success, otherwise non-zero value on failure + */ +static int zynqmp_r5_parse_fw(struct rproc *rproc, const struct firmware *fw) +{ + int ret; + + ret = parse_tcm_banks(rproc); + if (ret) + return ret; + + ret = parse_mem_regions(rproc); + if (ret) + return ret; + + ret = rproc_elf_load_rsc_table(rproc, fw); + if (ret == -EINVAL) { + /* + * resource table only required for IPC. + * if not present, this is not necessarily an error; + * for example, loading r5 hello world application + * so simply inform user and keep going. + */ + dev_info(&rproc->dev, "no resource table found.\n"); + ret = 0; + } + return ret; +} + +/* + * zynqmp_r5_rproc_kick - kick a firmware if mbox is provided + * @rproc: r5 core's corresponding rproc structure + * @vqid: virtqueue ID + */ +static void zynqmp_r5_rproc_kick(struct rproc *rproc, int vqid) +{ + struct zynqmp_r5_rproc *z_rproc = rproc->priv; + struct device *dev = rproc->dev.parent; + struct zynqmp_ipi_message *mb_msg = NULL; + unsigned int skb_len; + struct sk_buff *skb; + int ret; + + /* If vqid is negative, do not pass the vqid to + * mailbox as vqid is supposed to be 0 or positive. + * It also gives a way to just kick instead but + * not use the IPI buffer. + * + * For now use negative vqid to assume no message will be + * passed with IPI buffer, but just raise interrupt. + * This will be faster as it doesn't need to copy the + * message to the ZynqMP/Versal IPI message buffer. + */ + if (vqid >= 0) { + skb_len = (unsigned int)(sizeof(vqid) + sizeof(mb_msg)); + skb = alloc_skb(skb_len, GFP_KERNEL); + if (!skb) + return; + + mb_msg = (struct zynqmp_ipi_message *)skb_put(skb, skb_len); + mb_msg->len = sizeof(vqid); + memcpy(mb_msg->data, &vqid, sizeof(vqid)); + + skb_queue_tail(&z_rproc->tx_mc_skbs, skb); + } + + ret = mbox_send_message(z_rproc->tx_chan, mb_msg); + if (ret < 0) { + dev_warn(dev, "Failed to kick remote.\n"); + if (mb_msg) { + skb_dequeue_tail(&z_rproc->tx_mc_skbs); + kfree_skb(skb); + } + } +} + +static struct rproc_ops zynqmp_r5_rproc_ops = { + .start = zynqmp_r5_rproc_start, + .stop = zynqmp_r5_rproc_stop, + .load = rproc_elf_load_segments, + .parse_fw = zynqmp_r5_parse_fw, + .find_loaded_rsc_table = rproc_elf_find_loaded_rsc_table, + .sanity_check = rproc_elf_sanity_check, + .get_boot_addr = rproc_elf_get_boot_addr, + .kick = zynqmp_r5_rproc_kick, + .prepare = r5_rproc_prepare, + .unprepare = r5_rproc_unprepare, +}; + +/** + * event_notified_idr_cb - event notified idr callback + * @id: idr id + * @ptr: pointer to idr private data + * @data: data passed to idr_for_each callback + * + * Pass notification to remoteproc virtio + * + * Return: 0. having return is to satisfy the idr_for_each() function + * pointer input argument requirement. + **/ +static int event_notified_idr_cb(int id, void *ptr, void *data) +{ + struct rproc *rproc = data; + + if (rproc_vq_interrupt(rproc, id) == IRQ_NONE) + dev_dbg(&rproc->dev, "no message was found in idr with id: %d\n", id); + return 0; +} + +/** + * handle_event_notified - remoteproc notification work function + * @work: pointer to the work structure + * + * It checks each registered remoteproc notify IDs. + */ +static void handle_event_notified(struct work_struct *work) +{ + struct zynqmp_r5_rproc *z_rproc; + struct rproc *rproc; + + z_rproc = container_of(work, struct zynqmp_r5_rproc, mbox_work); + rproc = z_rproc->rproc; + + /* + * This is to simply raise interrupt as ACK for remote. + * + * Xilinx mailbox, ATF and SoC specific IPI mapping will handle + * mapping to remote processor. + */ + if (mbox_send_message(z_rproc->rx_chan, NULL) < 0) + dev_warn(rproc->dev.parent, "Failed to kick remote.\n"); + + /* + * We only use IPI for interrupt. The firmware side may or may + * not write the notifyid when it trigger IPI. + * And thus, we scan through all the registered notifyids. + */ + idr_for_each(&rproc->notifyids, event_notified_idr_cb, rproc); +} + +/** + * zynqmp_r5_mb_rx_cb - Receive channel mailbox callback + * @cl: mailbox client + * @msg: message pointer + * + * It will schedule the R5 notification work. + */ +static void zynqmp_r5_mb_rx_cb(struct mbox_client *cl, void *msg) +{ + struct zynqmp_r5_rproc *z_rproc; + + z_rproc = container_of(cl, struct zynqmp_r5_rproc, rx_mc); + /* + * Notification can be received with message payload. + * To handle this, check for message contents. + * If there are contents, copy to ipi message buffer payload location. + * + * If message is empty, then that means that was simple raising + * of interrupt. no payload to process. + * + * Note: enqueue work regardless of msg as kick may imply that remote + * is waiting for ack. + */ + if (msg) { + struct zynqmp_ipi_message *ipi_msg, *buf_msg; + size_t len; + + ipi_msg = (struct zynqmp_ipi_message *)msg; + buf_msg = (struct zynqmp_ipi_message *)z_rproc->rx_mc_buf; + len = (ipi_msg->len >= IPI_BUF_LEN_MAX) ? + IPI_BUF_LEN_MAX : ipi_msg->len; + buf_msg->len = len; + memcpy(buf_msg->data, ipi_msg->data, len); + } + schedule_work(&z_rproc->mbox_work); +} + +/** + * zynqmp_r5_mb_tx_done - Request has been sent to the remote + * @cl: mailbox client + * @msg: pointer to the message which has been sent + * @r: status of last TX - OK or error + * + * It will be called by the mailbox framework when the last TX has done. + */ +static void zynqmp_r5_mb_tx_done(struct mbox_client *cl, void *msg, int r) +{ + struct zynqmp_r5_rproc *z_rproc; + struct sk_buff *skb; + + if (!msg) + return; + z_rproc = container_of(cl, struct zynqmp_r5_rproc, tx_mc); + skb = skb_dequeue(&z_rproc->tx_mc_skbs); + kfree_skb(skb); +} + +/** + * zynqmp_r5_setup_mbox - Setup mailboxes + * this is used for each individual R5 core + * + * @z_rproc: pointer to the ZynqMP R5 processor platform data + * @node: pointer of the device node + * + * Function to setup mailboxes to talk to RPU. + * + * Return: 0 for success, negative value for failure. + */ +static int zynqmp_r5_setup_mbox(struct zynqmp_r5_rproc *z_rproc, + struct device_node *node) +{ + struct mbox_client *mclient; + + /* Setup TX mailbox channel client */ + mclient = &z_rproc->tx_mc; + mclient->dev = z_rproc->rproc->dev.parent; + mclient->tx_done = zynqmp_r5_mb_tx_done; + mclient->tx_block = false; + mclient->knows_txdone = false; + mclient->rx_callback = NULL; + + /* Setup RX mailbox channel client */ + mclient = &z_rproc->rx_mc; + mclient->dev = z_rproc->rproc->dev.parent; + mclient->tx_done = NULL; + mclient->tx_block = false; + mclient->knows_txdone = false; + mclient->rx_callback = zynqmp_r5_mb_rx_cb; + + INIT_WORK(&z_rproc->mbox_work, handle_event_notified); + + /* Request TX and RX channels */ + z_rproc->tx_chan = mbox_request_channel_byname(&z_rproc->tx_mc, "tx"); + if (IS_ERR(z_rproc->tx_chan)) { + dev_err(z_rproc->rproc->dev.parent, + "failed to request mbox tx channel.\n"); + return PTR_ERR(z_rproc->tx_chan); + } + + z_rproc->rx_chan = mbox_request_channel_byname(&z_rproc->rx_mc, "rx"); + if (IS_ERR(z_rproc->rx_chan)) { + dev_err(z_rproc->rproc->dev.parent, + "failed to request mbox rx channel.\n"); + return PTR_ERR(z_rproc->rx_chan); + } + skb_queue_head_init(&z_rproc->tx_mc_skbs); + + return 0; +} + +static void zynqmp_r5_cleanup_mbox(struct zynqmp_r5_rproc *z_rproc) +{ + mbox_free_channel(z_rproc->tx_chan); + mbox_free_channel(z_rproc->rx_chan); +} + +/** + * zynqmp_r5_probe - Probes ZynqMP R5 processor device node + * this is called for each individual R5 core to + * set up mailbox, Xilinx platform manager unique ID, + * add to rproc core + * + * @pdev: domain platform device for current R5 core + * @node: pointer of the device node for current R5 core + * @rpu_mode: mode to configure RPU, split or lockstep + * + * Return: 0 for success, negative value for failure. + */ +static struct zynqmp_r5_rproc *zynqmp_r5_probe(struct platform_device *pdev, + struct device_node *node, + enum rpu_oper_mode rpu_mode) +{ + int ret, num_banks; + struct device *dev = &pdev->dev; + struct rproc *rproc_ptr; + struct zynqmp_r5_rproc *z_rproc; + struct device_node *r5_node; + + /* Allocate remoteproc instance */ + rproc_ptr = devm_rproc_alloc(dev, dev_name(dev), &zynqmp_r5_rproc_ops, + NULL, sizeof(struct zynqmp_r5_rproc)); + if (!rproc_ptr) { + ret = -ENOMEM; + goto error; + } + + rproc_ptr->auto_boot = false; + z_rproc = rproc_ptr->priv; + z_rproc->rproc = rproc_ptr; + r5_node = z_rproc->rproc->dev.parent->of_node; + + /* Set up DMA mask */ + ret = dma_set_coherent_mask(dev, DMA_BIT_MASK(32)); + if (ret) + goto error; + + /* Get R5 power domain node */ + ret = of_property_read_u32(node, "power-domain", &z_rproc->pnode_id); + if (ret) + goto error; + + ret = r5_set_mode(z_rproc, rpu_mode); + if (ret) + goto error; + + if (of_property_read_bool(node, "mboxes")) { + ret = zynqmp_r5_setup_mbox(z_rproc, node); + if (ret) + goto error; + } + + /* go through TCM banks for r5 node */ + num_banks = of_count_phandle_with_args(r5_node, BANK_LIST_PROP, NULL); + if (num_banks <= 0) { + dev_err(dev, "need to specify TCM banks\n"); + ret = -EINVAL; + goto error; + } + + if (num_banks > NUM_SRAMS) { + dev_err(dev, "max number of srams is %d. given: %d \r\n", + NUM_SRAMS, num_banks); + ret = -EINVAL; + goto error; + } + + /* construct collection of srams used by the current R5 core */ + for (; num_banks; num_banks--) { + struct resource rsc; + struct device_node *dt_node; + resource_size_t size; + int i; + + dt_node = of_parse_phandle(r5_node, BANK_LIST_PROP, i); + if (!dt_node) { + ret = -EINVAL; + goto error; + } + + ret = of_address_to_resource(dt_node, 0, &rsc); + if (ret < 0) { + of_node_put(dt_node); + goto error; + } + + of_node_put(dt_node); + size = resource_size(&rsc); + + /* + * Find corresponding Xilinx platform management ID. + * The bank information is used in prepare/unprepare and + * parse_fw. + */ + for (i = 0; i < NUM_SRAMS; i++) { + if (rsc.start == zynqmp_banks[i].addr) { + z_rproc->srams[i].addr = rsc.start; + z_rproc->srams[i].size = size; + z_rproc->srams[i].id = zynqmp_banks[i].id; + break; + } + } + + if (i == NUM_SRAMS) { + dev_err(dev, "sram %llx is not valid.\n", rsc.start); + ret = -EINVAL; + goto error; + } + } + + /* Add R5 remoteproc */ + ret = devm_rproc_add(dev, rproc_ptr); + if (ret) { + zynqmp_r5_cleanup_mbox(z_rproc); + goto error; + } + + return z_rproc; +error: + return ERR_PTR(ret); +} + +/* + * zynqmp_r5_remoteproc_probe + * + * @pdev: domain platform device for R5 cluster + * + * called when driver is probed, for each R5 core specified in DT, + * setup as needed to do remoteproc-related operations + * + * Return: 0 for success, negative value for failure. + */ +static int zynqmp_r5_remoteproc_probe(struct platform_device *pdev) +{ + int ret, core_count; + struct device *dev = &pdev->dev; + struct device_node *nc; + enum rpu_oper_mode rpu_mode = PM_RPU_MODE_LOCKSTEP; + struct list_head *cluster; /* list to track each core's rproc */ + struct zynqmp_r5_rproc *z_rproc; + struct platform_device *child_pdev; + struct list_head *pos; + + ret = of_property_read_u32(dev->of_node, "xlnx,cluster-mode", &rpu_mode); + if (ret < 0 || (rpu_mode != PM_RPU_MODE_LOCKSTEP && + rpu_mode != PM_RPU_MODE_SPLIT)) { + dev_err(dev, "invalid cluster mode: ret %d mode %x\n", + ret, rpu_mode); + return ret; + } + + dev_dbg(dev, "RPU configuration: %s\n", + rpu_mode == PM_RPU_MODE_LOCKSTEP ? "lockstep" : "split"); + + /* + * if 2 RPUs provided but one is lockstep, then we have an + * invalid configuration. + */ + + core_count = of_get_available_child_count(dev->of_node); + if ((rpu_mode == PM_RPU_MODE_LOCKSTEP && core_count != 1) || + core_count > MAX_RPROCS) + return -EINVAL; + + cluster = devm_kzalloc(dev, sizeof(*cluster), GFP_KERNEL); + if (!cluster) + return -ENOMEM; + INIT_LIST_HEAD(cluster); + + ret = devm_of_platform_populate(dev); + if (ret) { + dev_err(dev, "devm_of_platform_populate failed, ret = %d\n", ret); + return ret; + } + + /* probe each individual r5 core's remoteproc-related info */ + for_each_available_child_of_node(dev->of_node, nc) { + child_pdev = of_find_device_by_node(nc); + if (!child_pdev) { + dev_err(dev, "could not get R5 core platform device\n"); + ret = -ENODEV; + goto out; + } + + z_rproc = zynqmp_r5_probe(child_pdev, nc, rpu_mode); + dev_dbg(dev, "%s to probe rpu %pOF\n", + ret ? "Failed" : "Able", nc); + if (IS_ERR(z_rproc)) { + ret = PTR_ERR(z_rproc); + goto out; + } + list_add_tail(&z_rproc->elem, cluster); + } + /* wire in so each core can be cleaned up at driver remove */ + platform_set_drvdata(pdev, cluster); + return 0; +out: + list_for_each(pos, cluster) { + z_rproc = list_entry(pos, struct zynqmp_r5_rproc, elem); + zynqmp_r5_cleanup_mbox(z_rproc); + } + return ret; +} + +/* + * zynqmp_r5_remoteproc_remove + * + * @pdev: domain platform device for R5 cluster + * + * When the driver is unloaded, clean up the mailboxes for each + * remoteproc that was initially probed. + */ +static int zynqmp_r5_remoteproc_remove(struct platform_device *pdev) +{ + struct list_head *pos, *temp, *cluster = (struct list_head *) + platform_get_drvdata(pdev); + struct zynqmp_r5_rproc *z_rproc = NULL; + + list_for_each_safe(pos, temp, cluster) { + z_rproc = list_entry(pos, struct zynqmp_r5_rproc, elem); + zynqmp_r5_cleanup_mbox(z_rproc); + } + return 0; +} + +/* Match table for OF platform binding */ +static const struct of_device_id zynqmp_r5_remoteproc_match[] = { + { .compatible = "xlnx,zynqmp-r5-remoteproc", }, + { /* end of list */ }, +}; +MODULE_DEVICE_TABLE(of, zynqmp_r5_remoteproc_match); + +static struct platform_driver zynqmp_r5_remoteproc_driver = { + .probe = zynqmp_r5_remoteproc_probe, + .remove = zynqmp_r5_remoteproc_remove, + .driver = { + .name = "zynqmp_r5_remoteproc", + .of_match_table = zynqmp_r5_remoteproc_match, + }, +}; +module_platform_driver(zynqmp_r5_remoteproc_driver); + +MODULE_AUTHOR("Ben Levinsky "); +MODULE_LICENSE("GPL v2");