From patchwork Tue Jan 3 06:33:25 2017 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Peter Chen X-Patchwork-Id: 9494449 Return-Path: Received: from mail.wl.linuxfoundation.org (pdx-wl-mail.web.codeaurora.org [172.30.200.125]) by pdx-korg-patchwork.web.codeaurora.org (Postfix) with ESMTP id 1C15260414 for ; Tue, 3 Jan 2017 06:35:30 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 0D7272015F for ; Tue, 3 Jan 2017 06:35:30 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id 0218A2679B; Tue, 3 Jan 2017 06:35:29 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on pdx-wl-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-4.2 required=2.0 tests=BAD_ENC_HEADER,BAYES_00, RCVD_IN_DNSWL_MED autolearn=unavailable version=3.3.1 Received: from bombadil.infradead.org (bombadil.infradead.org [198.137.202.9]) (using TLSv1.2 with cipher AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by mail.wl.linuxfoundation.org (Postfix) with ESMTPS id E6855266F3 for ; Tue, 3 Jan 2017 06:35:28 +0000 (UTC) Received: from localhost ([127.0.0.1] helo=bombadil.infradead.org) by bombadil.infradead.org with esmtp (Exim 4.85_2 #1 (Red Hat Linux)) id 1cOIfL-0004Ye-Ph; Tue, 03 Jan 2017 06:33:47 +0000 Received: from mail-bn3nam01on0050.outbound.protection.outlook.com ([104.47.33.50] helo=NAM01-BN3-obe.outbound.protection.outlook.com) by bombadil.infradead.org with esmtps (Exim 4.85_2 #1 (Red Hat Linux)) id 1cOIfB-0004R5-S5 for linux-arm-kernel@lists.infradead.org; Tue, 03 Jan 2017 06:33:43 +0000 Received: from CY4PR03CA0007.namprd03.prod.outlook.com (10.168.162.17) by BN1PR0301MB0738.namprd03.prod.outlook.com (10.160.78.145) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384_P384) id 15.1.817.10; Tue, 3 Jan 2017 06:33:15 +0000 Received: from BL2FFO11OLC009.protection.gbl (2a01:111:f400:7c09::128) by CY4PR03CA0007.outlook.office365.com (2603:10b6:903:33::17) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384_P384) id 15.1.817.10 via Frontend Transport; Tue, 3 Jan 2017 06:33:14 +0000 Authentication-Results: spf=fail (sender IP is 192.88.168.50) smtp.mailfrom=nxp.com; nxp.com; dkim=none (message not signed) header.d=none; nxp.com; dmarc=fail action=none header.from=nxp.com; nxp.com; dkim=none (message not signed) header.d=none; Received-SPF: Fail (protection.outlook.com: domain of nxp.com does not designate 192.88.168.50 as permitted sender) receiver=protection.outlook.com; client-ip=192.88.168.50; helo=tx30smr01.am.freescale.net; Received: from tx30smr01.am.freescale.net (192.88.168.50) by BL2FFO11OLC009.mail.protection.outlook.com (10.173.160.145) with Microsoft SMTP Server (version=TLS1_0, cipher=TLS_RSA_WITH_AES_256_CBC_SHA) id 15.1.803.8 via Frontend Transport; Tue, 3 Jan 2017 06:33:13 +0000 Received: from b29397-desktop.ap.freescale.net (b29397-desktop.ap.freescale.net [10.192.242.114]) by tx30smr01.am.freescale.net (8.14.3/8.14.0) with ESMTP id v036WhRC011369; Mon, 2 Jan 2017 23:33:04 -0700 From: Peter Chen To: , , , , , , , , Subject: [PATCH 2/8] power: add power sequence library Date: Tue, 3 Jan 2017 14:33:25 +0800 Message-ID: <1483425211-14473-3-git-send-email-peter.chen@nxp.com> X-Mailer: git-send-email 2.7.4 In-Reply-To: <1483425211-14473-1-git-send-email-peter.chen@nxp.com> References: <1483425211-14473-1-git-send-email-peter.chen@nxp.com> X-EOPAttributedMessage: 0 X-Matching-Connectors: 131278987938958611; (91ab9b29-cfa4-454e-5278-08d120cd25b8); () X-Forefront-Antispam-Report: CIP:192.88.168.50; IPV:NLI; CTRY:US; EFV:NLI; SFV:NSPM; SFS:(10009020)(6009001)(7916002)(336005)(39840400002)(39860400002)(39380400002)(39850400002)(39400400002)(39410400002)(39450400003)(2980300002)(1109001)(1110001)(339900001)(199003)(189002)(105606002)(54906002)(8676002)(104016004)(2171001)(81166006)(5660300001)(5003940100001)(50986999)(68736007)(36756003)(85426001)(47776003)(305945005)(626004)(6666003)(2906002)(76176999)(33646002)(92566002)(81156014)(8936002)(2950100002)(8656002)(356003)(86362001)(7416002)(50226002)(4326007)(106466001)(39060400001)(38730400001)(2201001)(7406005)(77096006)(97736004)(48376002)(5001770100001)(50466002)(189998001)(2004002); DIR:OUT; SFP:1101; SCL:1; SRVR:BN1PR0301MB0738; H:tx30smr01.am.freescale.net; FPR:; SPF:Fail; PTR:InfoDomainNonexistent; A:1; MX:1; LANG:en; X-Microsoft-Exchange-Diagnostics: 1; BL2FFO11OLC009; 1:bt1Mda/JQSbZuXs1Pxb7W5gGchrvXtHZvPZJ6Hugd0mJpO8QpruJHr8aqAAIEjhIM7Ei90/9Nl2125Ils+17WorEWFleEgs7MKiW3TOuDPDLQ5HpeHYHQeSxp7ztgMZAGTVWn1LFlQfM9Az3PTjGixh/iPTN088K6rcONj/wi3lUisUeRCt1GmvGiDbgh5oJ+QXe9gAk/01VBCU2MidP0zgN4ehq3TCEHRnE/OWtKlMhWbTDxuiPXtZf5B80+6ov38DOO037wIq9t/QGZmqfZD0pjPOC4rA/D60cbrYnmnW+XipGf/cPojZL2TBevL7gOa27/KYPrEXe8oWXZxBFm9ycebLuz2t4B1AqeICgd/iIdYSiqIDq59ZP6XmVBw9/3lsZhWBdS+bUuQ771GZ6MI2sZHDgTmsZGq2SLIHG+z5nzFS6YuAeyaLh63py/2yijFaQC6N6R+A9FyCy/bczrvoZg+gXWAF8PZU3VMPzhuUmuAsxX+gD95+vBNFLl1jjH3lTzhw+9CkwIYS6mSpAPF19na3LJis2iZvd6XTBByOQe5WVqrCj/vUBF1T417I6FnERa5/UYv1SKzI1jQtYMbGC7yZrjw9KkQM6CMKIKSrT3G9+W+ZgzeJz65MDB18gocbK2TqS17X5u6q0Mi8Dj377ULZLzYCSxZpDImOva3N835iEe9pYCLvX0TFLcd29k2KRtodAruR47//bB5veLmV2rTa58SKGz4unQqWP6eDPS4++VFl3a45g5yjUktZ4qNePxLLqfmUXf582AxRQjw== MIME-Version: 1.0 X-MS-Office365-Filtering-Correlation-Id: 8bc7a18c-39a4-46ae-abb8-08d433a26507 X-Microsoft-Antispam: UriScan:; BCL:0; PCL:0; RULEID:(22001); SRVR:BN1PR0301MB0738; X-Microsoft-Exchange-Diagnostics: 1; BN1PR0301MB0738; 3:l/zJ/nMMY2TVdGCzCgrmHpYm6Er0Dh21y6en+SOzUsWyj74494eY9CEG6A99br4eZifYJ/cY2Ioa39H37oRxqk2/Rftg6nwVXMitK6hUaOKbLAsY85kzPonaSIVmf5FpXUSq4qWsgXOogjJXklrEUZd963+Qb6IpwxQcCZvlKUBP/FnWxuWy0AnaSiaxBElnnH+zCFDOKCBTqlXBHKmpNok/0v/LanLr9DvY8AFdfn8FhTKQsCtCWyeu5Inia2LnMQ3ztngMoCZHhguehCME62biuq1oF4cZhO2UtmQJB/jdnaQ+Rx+alXoUjC9ACeXdGgQdyB6j5RnHX91c57ZD2u6793rBdUDNiWJVCkjdWNVaGtOly5Ti4kgswSkaNBF9; 25:iSkXklypV/gNw9w426e4JYgZLDQfyFK81wK7rZfX5MjoIamBYxbit8QODoZuQ6U5JThGiCr8dyTSxcaGGmFxRXtnUSJDUcSYdoc+BUClmmtKsusgWDoe8HuKdsV1AQPFARbJHU4gy3bBA9+t0flW88EN5z1xgE/1c4+kad/4U/rv75OUy/nBIBveSHNZD4aECC5Kp4EeOiyjzhHmxYdJ9GhoQd9A3lxJYVvEl6b3W+u4DcqvRt26Cf+EsSOCXSSyY497rayCo2KQfQF9y5UGyyquJ3FxD+X9JBPUj+hnvSwvFDOYnNYpuvNXNUtv3fjPThDCAWydplsIMX0hkJxGwG6DSR8CmskLJpxv/5mlvTVTi7/Y6OEVsgkjAOVpqylGBWayiM0sVa8KVpdXmVSBqCD9cm6yjhrqeFlvaHw7VJ8DglXZx+00lRrqmUJqShh/Q3uYypawxmYmiuDRxtMgMg== X-Microsoft-Exchange-Diagnostics: 1; BN1PR0301MB0738; 31:8Hy9iN/Q/Q4LUFFrnvGLq5vCLlVNmboFGZm+VSRXK7gizskcWeOZxFNgbbn2kSRt1aqRgV4Xw6CZL7jBtulV5gUWP07r0NOv32aB/Gh0UtfIe/a5xyZ4vhpFK7nQk7axQloRx0Js43Rf4XZ1TWv0BJUvhfz6WMF6bmWuCP4JVonDTdU+I1OYYxIjQuRj60ojxzyo8LrAMUus5v63al7NLNJuu2G0QHXwG4OGJuGsimntomee+UqSuuQeCS4jpWKcyBwGlcSuGpVmzT3oo/f7qw== X-Microsoft-Antispam-PRVS: X-Exchange-Antispam-Report-Test: UriScan:(9452136761055)(185117386973197)(84791874153150); X-Exchange-Antispam-Report-CFA-Test: BCL:0; PCL:0; RULEID:(6095060)(601004)(2401047)(13024025)(13023025)(13015025)(13017025)(13018025)(8121501046)(5005006)(3002001)(10201501046)(6055026)(6096035)(20161123561025)(20161123559025)(20161123556025)(20161123565025)(20161123563025); SRVR:BN1PR0301MB0738; BCL:0; PCL:0; RULEID:(400006); SRVR:BN1PR0301MB0738; X-Microsoft-Exchange-Diagnostics: 1; BN1PR0301MB0738; 4:fyHMnllKjQwk+JkRyPQZZe2nMDKDM+prjQIhEUT8yqup3Jni6KHGmPvJ6hb8GMpDS0ZBzqJxm3VQLw8zIMOVY/x6C/+ojv7lcMbBZ7JbauvR9yZPF7NzRRidu4UcrUZfM/wj/bpYtgNFQDHSnDHnt8wvLsbBbcElAnLljQeNCcjn5Fqadx/D6W92VYK30kUbJz6YkuCGQHggwK1mnHyy7hRKLbf2hpeBQd3Fo8NVds8GZ2ftz100dHG5RPN2lG/dErDY0/EdhQiGocIvOQ4aBq96nA27hOc0YtDGW0sOEPaSpjnKHzbatJTPmu3K7j+KaSjx+II1WMJIiA1pDSPfsQh2QcSENLhxFmuP1i30T9JxgRRsTTsM3aThSXitwxPY3MntQGk4u8o8Gh/ThlA3qPlFjTSUvNwIluyXlxK5yo/NKyHJNp7AwpomTD2T4p65Tvjgn6e7EO+01L5BDy7W7jAXxn7LlClBbaImSmi7FXYhmXdxWVeO+dYjeKiqpAxI+/yCeloeoxoT6DwNawqsR4jmH8cUYgwqngxueO22SSUB7D2esI3hOjcztkAkTj76XuZ57iXZHCSS1qi6ZhKUYCGl3afeLZ+gzypKau8z72zVRQG5wAxJEf070SvPb1Oyb3q0txS0ZtoJLQHBgGgaKLvLKWpbIhdI8VOx9/gUAHDWqndj/b/xY9e+Ewi4UU1HdthbZd+x58KKR6mgN95+EK1IJjitRhsVT+6fFxHccLOY4v0IQky0CFdgzB8DWeXh93yLC/013UxauM2zQUrOe/2tGb4IqjCeQd0liKdLqJlYXQgc8s1vFHEDobTXr9Rs X-Forefront-PRVS: 01762B0D64 X-Microsoft-Exchange-Diagnostics: =?us-ascii?Q?1; BN1PR0301MB0738; 23:3JW4KtWCqB6bzUBj1Ln3AUDhTpjYJ3Mz/HpYlgc?= =?us-ascii?Q?Cu2N69UgZ3f2KAzQxkUY/doorX7L0Z7C1QWphUusZs8E6jki00u1jfcE+mPF?= =?us-ascii?Q?O2zdZuwLCu0DaeuRdUjI7z/P6M7MD0jAH2eElaXPXcmBPGvAFeeYmwyIFB84?= =?us-ascii?Q?IriSPazDPiNtt7P5ltbhzYvzcF5OAiC4bmL8QfB/ENKNiQrLKtRSXYz8Bo3j?= =?us-ascii?Q?tHiTrBrCuoGtFv5XTYU+GqkDP5a54IeJIOXUZGm8+A1fF5FY4fBsHPC5nota?= =?us-ascii?Q?ys/riDSmCp+puMf4RsccR7KgFShUgHYnaobZv+sbbTPN2u/QO1LuZo0EIoTK?= =?us-ascii?Q?+rNeAqYliNHeNvDSdLpYJLhUg81PyKlMzxlIWxGBJ8qNbE5YcY71GqifsARd?= =?us-ascii?Q?zznuSiEaYUn9zJrf42idF2KQYkJ1sqI8eevcOdcKXwKFeIZpUgfR4lSvauKg?= =?us-ascii?Q?6XHbK0O3QZTwNdkbqy5nb/9iuQamLa8K0Z7l5vxwvQZ0bDS7tQcfiuemG3Mg?= =?us-ascii?Q?No9IAvf3dzyoHwEPI63kysXwR7NaOleram+oNRtjEpL/ZvXpu4yYvIhZTzKb?= =?us-ascii?Q?BUSUxlxWrRuYyn022VJo4gjnjgSkpFolLaLRyuAsTm31q1qT39OcMVlCo3hM?= =?us-ascii?Q?2TtgHBfl+baKt1v4+05NLP+mqIWMLdIQmvTqXmC/u+AAz756ifTkakVKXUAH?= =?us-ascii?Q?P1L+Vc4S8yh0iSC0O0eAeiREEKL8vkhultx5PefGmYWVYeD+GZyk3SJGhMt0?= =?us-ascii?Q?BuRvPwWJhthSfxQ02QnM2fOactD+5VGnh/ZeUyO8aaUb51PXp6hsjSIywk70?= =?us-ascii?Q?+T1j6iDVhUy3VsNFopb9/5770D51+9YXpkDfLXq19k6VPr0/wiG7F4SF0ljO?= =?us-ascii?Q?rHelIIhlpLfN81kF9yEm24iyDwNNWeimCBUHkokaA+HsQEXb3/N/tUma8awY?= =?us-ascii?Q?sahUbmXhZq7uUFl10XPwiXCMjxfKg8dg3FcO65o28XBVNUTB+epdQGUPvJAP?= =?us-ascii?Q?ObdrJFHKPkooUgBv/755ecxaDBFvKEinMN1Pe+Z9UzfGhLA28G6E+/3JliYY?= =?us-ascii?Q?CWA9lQTAsRN7n/0H44QnAKFIaTJK5XfbcoXuUzMSRvoKoouWqKOTyFaGUoMr?= =?us-ascii?Q?UFD8UxpigZ9+qaoA0ZlUskxwBixVrAbbaflpUhG2Huue1LloXwY9NUGEv1Ej?= =?us-ascii?Q?c+7OBy1Nn7gqJbE7JtCTTy0YKYUX4YCIRDESbIEEp+1eBbeKmlDRozSMASF5?= =?us-ascii?Q?m2N2cS13qAIYZO5lU/8KYnB2OMmuyfZxuON/lqVlUPzhH1nbzqoq7RQTqUJA?= =?us-ascii?Q?kbybm3xQREyOFq171oZsA5aE2JDUeloU+5lnMqzU9iqxRnUoBxYy1mWVBmtG?= =?us-ascii?Q?esDnnpWeSdLxnmIE7ab+qDI/wz2k=3D?= X-Microsoft-Exchange-Diagnostics: 1; BN1PR0301MB0738; 6:WrRwwPkp34nSmQJCJTc4eg4mWZeTrWmDTx+fD8Uj9zwO74EHCn+4E5OnnE0GDvq/qS2Wd9LD0nvuZkyuFB9+e6AQjgr+6wbXM464GoGd10GHyzDfJKjN6jaeZ5GJKQUkDDVvRNNuD0xxxeGFbbXavggOpPPV8vbeIuiKr2K7Bp7xS6+cZUBIDhRey+ZWi1K7as387AdV5Mc9WXJlHMNjE0lQBcDWeU6QOj2p3BUdz4I9xcgIlIKHzxrvVNfwoyDRPh0UDb9I59qJ19G/n4MZuPfW++Ml6XAeCOYgVkvdMgDytP/PxRaHg8Z6b7GxmcOG7aD+9IhRw4zyWMqOC2gULYtnCXLXRz6mrqsj2uprSvtKTdESbIU8bnE82LVlDYMhMD/0L0xk4/oxf1CN4UmJu8oZV5B4NEf9MxpY6bO8k3jKRKCyIQA+H1OIy5thnCDu; 5:tyykPPe49VexZWIDehAV2aXcTKrCAM0snDRoHVm5c61pgDdspGBesddJ3cApcYOCYBc/VnjP4eagR3EKdWrZ7BdW+AeAZa90tFCEMOHOXiQOoSWVFqoGU+SgNI0pUupoYRE76v154ceeIdhr/9llONtA9lorcPRmKesL3JINnY5mDZ3y22YlZWnZfXL7HZel; 24:lwnKTpPuKf7iBbPqwP+omBsvBiYy2og2Ji6v78IuDidmWoGWdgFsGIOaIV1EEXWU+a7+B1hJcwJYryhjAY985j9YI8f76Krkfgx3m023nmU= SpamDiagnosticOutput: 1:99 SpamDiagnosticMetadata: NSPM X-Microsoft-Exchange-Diagnostics: 1; BN1PR0301MB0738; 7:PnKDyj+ggR+czlTRNKrK1KaPyN4GdsyMz+oplVIg0+2BAa5LR4NCYD/DUeDJfPw/8Fn9poqVNkIqfIIbUmKg9u+EjsKhrraZLg1r3eNnWlTuYZ+Sww7Sfm01HgVr8gL/Uw9X1oWgArllDL3zTZH9r71UBGrJACPQTrUq7id4pdSHKRAelxqaRuvytVo0Q9WVMtkKCaDm4wSqjnR6z0Yh5dhdakouxSfcX6LlGH5zPX0oCseIbcb5/IvIl9mIiK29QWz3Hk7AfaS3dgk3dYpca6jdfxEA6O2FtymByA1LbOC2+WJ9MkG3GojnJujSRtQzCcSFr8ty+g6KnWHynEyh3lanymLqJ8p7S7ARJ/IlZNb5PZVMJauyeZIB7kh1q90Hl0NlZ5d6tPVb0Doqmyw09iN0LdzTNxlLaXtO8BGGvW5j6IWC20MQHmXaODh3yao+r9Eh2luh9D2oO4LzJe6CCA== X-MS-Exchange-CrossTenant-OriginalArrivalTime: 03 Jan 2017 06:33:13.4590 (UTC) X-MS-Exchange-CrossTenant-Id: 5afe0b00-7697-4969-b663-5eab37d5f47e X-MS-Exchange-CrossTenant-OriginalAttributedTenantConnectingIp: TenantId=5afe0b00-7697-4969-b663-5eab37d5f47e; Ip=[192.88.168.50]; Helo=[tx30smr01.am.freescale.net] X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: BN1PR0301MB0738 X-CRM114-Version: 20100106-BlameMichelson ( TRE 0.8.0 (BSD) ) MR-646709E3 X-CRM114-CacheID: sfid-20170102_223338_299334_7788BA0F X-CRM114-Status: GOOD ( 19.40 ) X-BeenThere: linux-arm-kernel@lists.infradead.org X-Mailman-Version: 2.1.20 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Cc: mark.rutland@arm.com, Peter Chen , heiko@sntech.de, stephen.boyd@linaro.org, gary.bisson@boundarydevices.com, festevam@gmail.com, stillcompiling@gmail.com, arnd@arndb.de, vaibhav.hiremath@linaro.org, krzk@kernel.org, mka@chromium.org, devicetree@vger.kernel.org, mail@maciej.szmigiero.name, pawel.moll@arm.com, linux-pm@vger.kernel.org, s.hauer@pengutronix.de, troy.kisky@boundarydevices.com, linux-arm-kernel@lists.infradead.org, hverkuil@xs4all.nl, oscar@naiandei.net, linux-usb@vger.kernel.org, linux-kernel@vger.kernel.org, p.zabel@pengutronix.de Sender: "linux-arm-kernel" Errors-To: linux-arm-kernel-bounces+patchwork-linux-arm=patchwork.kernel.org@lists.infradead.org X-Virus-Scanned: ClamAV using ClamSMTP We have an well-known problem that the device needs to do some power sequence before it can be recognized by related host, the typical example like hard-wired mmc devices and usb devices. This power sequence is hard to be described at device tree and handled by related host driver, so we have created a common power sequence library to cover this requirement. The core code has supplied some common helpers for host driver, and individual power sequence libraries handle kinds of power sequence for devices. The pwrseq librares always need to allocate extra instance for compatible string match. pwrseq_generic is intended for general purpose of power sequence, which handles gpios and clocks currently, and can cover other controls in future. The host driver just needs to call of_pwrseq_on/of_pwrseq_off if only one power sequence is needed, else call of_pwrseq_on_list /of_pwrseq_off_list instead (eg, USB hub driver). For new power sequence library, it can add its compatible string to pwrseq_of_match_table, then the pwrseq core will match it with DT's, and choose this library at runtime. Signed-off-by: Peter Chen Tested-by: Maciej S. Szmigiero Tested-by Joshua Clayton Reviewed-by: Matthias Kaehlcke Tested-by: Matthias Kaehlcke --- MAINTAINERS | 9 + drivers/power/Kconfig | 1 + drivers/power/Makefile | 1 + drivers/power/pwrseq/Kconfig | 20 ++ drivers/power/pwrseq/Makefile | 2 + drivers/power/pwrseq/core.c | 335 ++++++++++++++++++++++++++++++++++ drivers/power/pwrseq/pwrseq_generic.c | 224 +++++++++++++++++++++++ include/linux/power/pwrseq.h | 81 ++++++++ 8 files changed, 673 insertions(+) create mode 100644 drivers/power/pwrseq/Kconfig create mode 100644 drivers/power/pwrseq/Makefile create mode 100644 drivers/power/pwrseq/core.c create mode 100644 drivers/power/pwrseq/pwrseq_generic.c create mode 100644 include/linux/power/pwrseq.h diff --git a/MAINTAINERS b/MAINTAINERS index cfff2c9..ae2aa25 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -9828,6 +9828,15 @@ F: include/linux/pm_* F: include/linux/powercap.h F: drivers/powercap/ +POWER SEQUENCE LIBRARY +M: Peter Chen +T: git git://git.kernel.org/pub/scm/linux/kernel/git/peter.chen/usb.git +L: linux-pm@vger.kernel.org +S: Maintained +F: Documentation/devicetree/bindings/power/pwrseq/ +F: drivers/power/pwrseq/ +F: include/linux/power/pwrseq.h + POWER SUPPLY CLASS/SUBSYSTEM and DRIVERS M: Sebastian Reichel L: linux-pm@vger.kernel.org diff --git a/drivers/power/Kconfig b/drivers/power/Kconfig index 63454b5..c1bb046 100644 --- a/drivers/power/Kconfig +++ b/drivers/power/Kconfig @@ -1,3 +1,4 @@ source "drivers/power/avs/Kconfig" source "drivers/power/reset/Kconfig" source "drivers/power/supply/Kconfig" +source "drivers/power/pwrseq/Kconfig" diff --git a/drivers/power/Makefile b/drivers/power/Makefile index ff35c71..7db8035 100644 --- a/drivers/power/Makefile +++ b/drivers/power/Makefile @@ -1,3 +1,4 @@ obj-$(CONFIG_POWER_AVS) += avs/ obj-$(CONFIG_POWER_RESET) += reset/ obj-$(CONFIG_POWER_SUPPLY) += supply/ +obj-$(CONFIG_POWER_SEQUENCE) += pwrseq/ diff --git a/drivers/power/pwrseq/Kconfig b/drivers/power/pwrseq/Kconfig new file mode 100644 index 0000000..c6b3569 --- /dev/null +++ b/drivers/power/pwrseq/Kconfig @@ -0,0 +1,20 @@ +# +# Power Sequence library +# + +menuconfig POWER_SEQUENCE + bool "Power sequence control" + help + It is used for drivers which needs to do power sequence + (eg, turn on clock, toggle reset gpio) before the related + devices can be found by hardware, eg, USB bus. + +if POWER_SEQUENCE + +config PWRSEQ_GENERIC + bool "Generic power sequence control" + depends on OF + help + This is the generic power sequence control library, and is + supposed to support common power sequence usage. +endif diff --git a/drivers/power/pwrseq/Makefile b/drivers/power/pwrseq/Makefile new file mode 100644 index 0000000..ad82389 --- /dev/null +++ b/drivers/power/pwrseq/Makefile @@ -0,0 +1,2 @@ +obj-$(CONFIG_POWER_SEQUENCE) += core.o +obj-$(CONFIG_PWRSEQ_GENERIC) += pwrseq_generic.o diff --git a/drivers/power/pwrseq/core.c b/drivers/power/pwrseq/core.c new file mode 100644 index 0000000..3d19e62 --- /dev/null +++ b/drivers/power/pwrseq/core.c @@ -0,0 +1,335 @@ +/* + * core.c power sequence core file + * + * Copyright (C) 2016 Freescale Semiconductor, Inc. + * Author: Peter Chen + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 of + * the License as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. + */ + +#include +#include +#include +#include +#include + +static DEFINE_MUTEX(pwrseq_list_mutex); +static LIST_HEAD(pwrseq_list); + +static int pwrseq_get(struct device_node *np, struct pwrseq *p) +{ + if (p && p->get) + return p->get(np, p); + + return -ENOTSUPP; +} + +static int pwrseq_on(struct pwrseq *p) +{ + if (p && p->on) + return p->on(p); + + return -ENOTSUPP; +} + +static void pwrseq_off(struct pwrseq *p) +{ + if (p && p->off) + p->off(p); +} + +static void pwrseq_put(struct pwrseq *p) +{ + if (p && p->put) + p->put(p); +} + +/** + * pwrseq_register - Add pwrseq instance to global pwrseq list + * + * @pwrseq: the pwrseq instance + */ +void pwrseq_register(struct pwrseq *pwrseq) +{ + mutex_lock(&pwrseq_list_mutex); + list_add(&pwrseq->node, &pwrseq_list); + mutex_unlock(&pwrseq_list_mutex); +} +EXPORT_SYMBOL_GPL(pwrseq_register); + +/** + * pwrseq_unregister - Remove pwrseq instance from global pwrseq list + * + * @pwrseq: the pwrseq instance + */ +void pwrseq_unregister(struct pwrseq *pwrseq) +{ + mutex_lock(&pwrseq_list_mutex); + list_del(&pwrseq->node); + mutex_unlock(&pwrseq_list_mutex); +} +EXPORT_SYMBOL_GPL(pwrseq_unregister); + +static struct pwrseq *pwrseq_find_available_instance(struct device_node *np) +{ + struct pwrseq *pwrseq; + + mutex_lock(&pwrseq_list_mutex); + list_for_each_entry(pwrseq, &pwrseq_list, node) { + if (pwrseq->used) + continue; + + /* compare compatible string for pwrseq node */ + if (of_match_node(pwrseq->pwrseq_of_match_table, np)) { + pwrseq->used = true; + mutex_unlock(&pwrseq_list_mutex); + return pwrseq; + } + + /* return generic pwrseq instance */ + if (!strcmp(pwrseq->pwrseq_of_match_table->compatible, + "generic")) { + pr_debug("using generic pwrseq instance for %s\n", + np->full_name); + pwrseq->used = true; + mutex_unlock(&pwrseq_list_mutex); + return pwrseq; + } + } + mutex_unlock(&pwrseq_list_mutex); + pr_debug("Can't find any pwrseq instances for %s\n", np->full_name); + + return NULL; +} + +/** + * of_pwrseq_on - Carry out power sequence on for device node + * + * @np: the device node would like to power on + * + * Carry out a single device power on. If multiple devices + * need to be handled, use of_pwrseq_on_list() instead. + * + * Return a pointer to the power sequence instance on success, + * or an error code otherwise. + */ +struct pwrseq *of_pwrseq_on(struct device_node *np) +{ + struct pwrseq *pwrseq; + int ret; + + pwrseq = pwrseq_find_available_instance(np); + if (!pwrseq) + return ERR_PTR(-ENOENT); + + ret = pwrseq_get(np, pwrseq); + if (ret) { + /* Mark current pwrseq as unused */ + pwrseq->used = false; + return ERR_PTR(ret); + } + + ret = pwrseq_on(pwrseq); + if (ret) + goto pwr_put; + + return pwrseq; + +pwr_put: + pwrseq_put(pwrseq); + return ERR_PTR(ret); +} +EXPORT_SYMBOL_GPL(of_pwrseq_on); + +/** + * of_pwrseq_off - Carry out power sequence off for this pwrseq instance + * + * @pwrseq: the pwrseq instance which related device would like to be off + * + * This API is used to power off single device, it is the opposite + * operation for of_pwrseq_on. + */ +void of_pwrseq_off(struct pwrseq *pwrseq) +{ + pwrseq_off(pwrseq); + pwrseq_put(pwrseq); +} +EXPORT_SYMBOL_GPL(of_pwrseq_off); + +/** + * of_pwrseq_on_list - Carry out power sequence on for list + * + * @np: the device node would like to power on + * @head: the list head for pwrseq list on this bus + * + * This API is used to power on multiple devices at single bus. + * If there are several devices on bus (eg, USB bus), uses this + * this API. Otherwise, use of_pwrseq_on instead. After the device + * is powered on successfully, it will be added to pwrseq list for + * this bus. The caller needs to use mutex_lock for concurrent. + * + * Return 0 on success, or an error value otherwise. + */ +int of_pwrseq_on_list(struct device_node *np, struct list_head *head) +{ + struct pwrseq *pwrseq; + struct pwrseq_list_per_dev *pwrseq_list_node; + + pwrseq_list_node = kzalloc(sizeof(*pwrseq_list_node), GFP_KERNEL); + if (!pwrseq_list_node) + return -ENOMEM; + + pwrseq = of_pwrseq_on(np); + if (IS_ERR(pwrseq)) { + kfree(pwrseq_list_node); + return PTR_ERR(pwrseq); + } + + pwrseq_list_node->pwrseq = pwrseq; + list_add(&pwrseq_list_node->list, head); + + return 0; +} +EXPORT_SYMBOL_GPL(of_pwrseq_on_list); + +/** + * of_pwrseq_off_list - Carry out power sequence off for the list + * + * @head: the list head for pwrseq instance list on this bus + * + * This API is used to power off all devices on this bus, it is + * the opposite operation for of_pwrseq_on_list. + * The caller needs to use mutex_lock for concurrent. + */ +void of_pwrseq_off_list(struct list_head *head) +{ + struct pwrseq *pwrseq; + struct pwrseq_list_per_dev *pwrseq_list_node, *tmp_node; + + list_for_each_entry_safe(pwrseq_list_node, tmp_node, head, list) { + pwrseq = pwrseq_list_node->pwrseq; + of_pwrseq_off(pwrseq); + list_del(&pwrseq_list_node->list); + kfree(pwrseq_list_node); + } +} +EXPORT_SYMBOL_GPL(of_pwrseq_off_list); + +/** + * pwrseq_suspend - Carry out power sequence suspend for this pwrseq instance + * + * @pwrseq: the pwrseq instance + * + * This API is used to do suspend operation on pwrseq instance. + * + * Return 0 on success, or an error value otherwise. + */ +int pwrseq_suspend(struct pwrseq *p) +{ + int ret = 0; + + if (p && p->suspend) + ret = p->suspend(p); + else + return ret; + + if (!ret) + p->suspended = true; + else + pr_err("%s failed\n", __func__); + + return ret; +} +EXPORT_SYMBOL_GPL(pwrseq_suspend); + +/** + * pwrseq_resume - Carry out power sequence resume for this pwrseq instance + * + * @pwrseq: the pwrseq instance + * + * This API is used to do resume operation on pwrseq instance. + * + * Return 0 on success, or an error value otherwise. + */ +int pwrseq_resume(struct pwrseq *p) +{ + int ret = 0; + + if (p && p->resume) + ret = p->resume(p); + else + return ret; + + if (!ret) + p->suspended = false; + else + pr_err("%s failed\n", __func__); + + return ret; +} +EXPORT_SYMBOL_GPL(pwrseq_resume); + +/** + * pwrseq_suspend_list - Carry out power sequence suspend for list + * + * @head: the list head for pwrseq instance list on this bus + * + * This API is used to do suspend on all power sequence instances on this bus. + * The caller needs to use mutex_lock for concurrent. + */ +int pwrseq_suspend_list(struct list_head *head) +{ + struct pwrseq *pwrseq; + struct pwrseq_list_per_dev *pwrseq_list_node; + int ret = 0; + + list_for_each_entry(pwrseq_list_node, head, list) { + ret = pwrseq_suspend(pwrseq_list_node->pwrseq); + if (ret) + break; + } + + if (ret) { + list_for_each_entry(pwrseq_list_node, head, list) { + pwrseq = pwrseq_list_node->pwrseq; + if (pwrseq->suspended) + pwrseq_resume(pwrseq); + } + } + + return ret; +} +EXPORT_SYMBOL_GPL(pwrseq_suspend_list); + +/** + * pwrseq_resume_list - Carry out power sequence resume for the list + * + * @head: the list head for pwrseq instance list on this bus + * + * This API is used to do resume on all power sequence instances on this bus. + * The caller needs to use mutex_lock for concurrent. + */ +int pwrseq_resume_list(struct list_head *head) +{ + struct pwrseq_list_per_dev *pwrseq_list_node; + int ret = 0; + + list_for_each_entry(pwrseq_list_node, head, list) { + ret = pwrseq_resume(pwrseq_list_node->pwrseq); + if (ret) + break; + } + + return ret; +} +EXPORT_SYMBOL_GPL(pwrseq_resume_list); diff --git a/drivers/power/pwrseq/pwrseq_generic.c b/drivers/power/pwrseq/pwrseq_generic.c new file mode 100644 index 0000000..0e70a38 --- /dev/null +++ b/drivers/power/pwrseq/pwrseq_generic.c @@ -0,0 +1,224 @@ +/* + * pwrseq_generic.c Generic power sequence handling + * + * Copyright (C) 2016 Freescale Semiconductor, Inc. + * Author: Peter Chen + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 of + * the License as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. + */ + +#include +#include +#include +#include +#include +#include +#include + +#include + +struct pwrseq_generic { + struct pwrseq pwrseq; + struct gpio_desc *gpiod_reset; + struct clk *clks[PWRSEQ_MAX_CLKS]; + u32 duration_us; + bool suspended; +}; + +#define to_generic_pwrseq(p) container_of(p, struct pwrseq_generic, pwrseq) + +static int pwrseq_generic_alloc_instance(void); +static const struct of_device_id generic_id_table[] = { + { .compatible = "generic",}, + { /* sentinel */ } +}; + +static int pwrseq_generic_suspend(struct pwrseq *pwrseq) +{ + struct pwrseq_generic *pwrseq_gen = to_generic_pwrseq(pwrseq); + int clk; + + for (clk = PWRSEQ_MAX_CLKS - 1; clk >= 0; clk--) + clk_disable_unprepare(pwrseq_gen->clks[clk]); + + pwrseq_gen->suspended = true; + return 0; +} + +static int pwrseq_generic_resume(struct pwrseq *pwrseq) +{ + struct pwrseq_generic *pwrseq_gen = to_generic_pwrseq(pwrseq); + int clk, ret = 0; + + for (clk = 0; clk < PWRSEQ_MAX_CLKS && pwrseq_gen->clks[clk]; clk++) { + ret = clk_prepare_enable(pwrseq_gen->clks[clk]); + if (ret) { + pr_err("Can't enable clock, ret=%d\n", ret); + goto err_disable_clks; + } + } + + pwrseq_gen->suspended = false; + return ret; + +err_disable_clks: + while (--clk >= 0) + clk_disable_unprepare(pwrseq_gen->clks[clk]); + + return ret; +} + +static void pwrseq_generic_put(struct pwrseq *pwrseq) +{ + struct pwrseq_generic *pwrseq_gen = to_generic_pwrseq(pwrseq); + int clk; + + if (pwrseq_gen->gpiod_reset) + gpiod_put(pwrseq_gen->gpiod_reset); + + for (clk = 0; clk < PWRSEQ_MAX_CLKS; clk++) + clk_put(pwrseq_gen->clks[clk]); + + pwrseq_unregister(&pwrseq_gen->pwrseq); + kfree(pwrseq_gen); +} + +static void pwrseq_generic_off(struct pwrseq *pwrseq) +{ + struct pwrseq_generic *pwrseq_gen = to_generic_pwrseq(pwrseq); + int clk; + + if (pwrseq_gen->suspended) + return; + + for (clk = PWRSEQ_MAX_CLKS - 1; clk >= 0; clk--) + clk_disable_unprepare(pwrseq_gen->clks[clk]); +} + +static int pwrseq_generic_on(struct pwrseq *pwrseq) +{ + struct pwrseq_generic *pwrseq_gen = to_generic_pwrseq(pwrseq); + int clk, ret = 0; + struct gpio_desc *gpiod_reset = pwrseq_gen->gpiod_reset; + + for (clk = 0; clk < PWRSEQ_MAX_CLKS && pwrseq_gen->clks[clk]; clk++) { + ret = clk_prepare_enable(pwrseq_gen->clks[clk]); + if (ret) { + pr_err("Can't enable clock, ret=%d\n", ret); + goto err_disable_clks; + } + } + + if (gpiod_reset) { + u32 duration_us = pwrseq_gen->duration_us; + + if (duration_us <= 10) + udelay(10); + else + usleep_range(duration_us, duration_us + 100); + gpiod_set_value(gpiod_reset, 0); + } + + return ret; + +err_disable_clks: + while (--clk >= 0) + clk_disable_unprepare(pwrseq_gen->clks[clk]); + + return ret; +} + +static int pwrseq_generic_get(struct device_node *np, struct pwrseq *pwrseq) +{ + struct pwrseq_generic *pwrseq_gen = to_generic_pwrseq(pwrseq); + enum of_gpio_flags flags; + int reset_gpio, clk, ret = 0; + + for (clk = 0; clk < PWRSEQ_MAX_CLKS; clk++) { + pwrseq_gen->clks[clk] = of_clk_get(np, clk); + if (IS_ERR(pwrseq_gen->clks[clk])) { + ret = PTR_ERR(pwrseq_gen->clks[clk]); + if (ret != -ENOENT) + goto err_put_clks; + pwrseq_gen->clks[clk] = NULL; + break; + } + } + + reset_gpio = of_get_named_gpio_flags(np, "reset-gpios", 0, &flags); + if (gpio_is_valid(reset_gpio)) { + unsigned long gpio_flags; + + if (flags & OF_GPIO_ACTIVE_LOW) + gpio_flags = GPIOF_ACTIVE_LOW | GPIOF_OUT_INIT_LOW; + else + gpio_flags = GPIOF_OUT_INIT_HIGH; + + ret = gpio_request_one(reset_gpio, gpio_flags, + "pwrseq-reset-gpios"); + if (ret) + goto err_put_clks; + + pwrseq_gen->gpiod_reset = gpio_to_desc(reset_gpio); + of_property_read_u32(np, "reset-duration-us", + &pwrseq_gen->duration_us); + } else if (reset_gpio == -ENOENT) { + ; /* no such gpio */ + } else { + ret = reset_gpio; + pr_err("Failed to get reset gpio on %s, err = %d\n", + np->full_name, reset_gpio); + goto err_put_clks; + } + + /* allocate new one for later pwrseq instance request */ + ret = pwrseq_generic_alloc_instance(); + if (ret) + goto err_put_gpio; + + return 0; + +err_put_gpio: + if (pwrseq_gen->gpiod_reset) + gpiod_put(pwrseq_gen->gpiod_reset); +err_put_clks: + while (--clk >= 0) + clk_put(pwrseq_gen->clks[clk]); + return ret; +} + +static int pwrseq_generic_alloc_instance(void) +{ + struct pwrseq_generic *pwrseq_gen; + + pwrseq_gen = kzalloc(sizeof(*pwrseq_gen), GFP_KERNEL); + if (!pwrseq_gen) + return -ENOMEM; + + pwrseq_gen->pwrseq.pwrseq_of_match_table = generic_id_table; + pwrseq_gen->pwrseq.get = pwrseq_generic_get; + pwrseq_gen->pwrseq.on = pwrseq_generic_on; + pwrseq_gen->pwrseq.off = pwrseq_generic_off; + pwrseq_gen->pwrseq.put = pwrseq_generic_put; + pwrseq_gen->pwrseq.suspend = pwrseq_generic_suspend; + pwrseq_gen->pwrseq.resume = pwrseq_generic_resume; + + pwrseq_register(&pwrseq_gen->pwrseq); + return 0; +} + +static int __init pwrseq_generic_register(void) +{ + return pwrseq_generic_alloc_instance(); +} +postcore_initcall(pwrseq_generic_register) diff --git a/include/linux/power/pwrseq.h b/include/linux/power/pwrseq.h new file mode 100644 index 0000000..cbc344c --- /dev/null +++ b/include/linux/power/pwrseq.h @@ -0,0 +1,81 @@ +#ifndef __LINUX_PWRSEQ_H +#define __LINUX_PWRSEQ_H + +#include + +#define PWRSEQ_MAX_CLKS 3 + +/** + * struct pwrseq - the power sequence structure + * @pwrseq_of_match_table: the OF device id table this pwrseq library supports + * @node: the list pointer to be added to pwrseq list + * @get: the API is used to get pwrseq instance from the device node + * @on: do power on for this pwrseq instance + * @off: do power off for this pwrseq instance + * @put: release the resources on this pwrseq instance + * @suspend: do suspend operation on this pwrseq instance + * @resume: do resume operation on this pwrseq instance + * @used: this pwrseq instance is used by device + */ +struct pwrseq { + const struct of_device_id *pwrseq_of_match_table; + struct list_head node; + int (*get)(struct device_node *np, struct pwrseq *p); + int (*on)(struct pwrseq *p); + void (*off)(struct pwrseq *p); + void (*put)(struct pwrseq *p); + int (*suspend)(struct pwrseq *p); + int (*resume)(struct pwrseq *p); + bool used; + bool suspended; +}; + +/* used for power sequence instance list in one driver */ +struct pwrseq_list_per_dev { + struct pwrseq *pwrseq; + struct list_head list; +}; + +#if IS_ENABLED(CONFIG_POWER_SEQUENCE) +void pwrseq_register(struct pwrseq *pwrseq); +void pwrseq_unregister(struct pwrseq *pwrseq); +struct pwrseq *of_pwrseq_on(struct device_node *np); +void of_pwrseq_off(struct pwrseq *pwrseq); +int of_pwrseq_on_list(struct device_node *np, struct list_head *head); +void of_pwrseq_off_list(struct list_head *head); +int pwrseq_suspend(struct pwrseq *p); +int pwrseq_resume(struct pwrseq *p); +int pwrseq_suspend_list(struct list_head *head); +int pwrseq_resume_list(struct list_head *head); +#else +static inline void pwrseq_register(struct pwrseq *pwrseq) {} +static inline void pwrseq_unregister(struct pwrseq *pwrseq) {} +static inline struct pwrseq *of_pwrseq_on(struct device_node *np) +{ + return NULL; +} +static void of_pwrseq_off(struct pwrseq *pwrseq) {} +static int of_pwrseq_on_list(struct device_node *np, struct list_head *head) +{ + return 0; +} +static void of_pwrseq_off_list(struct list_head *head) {} +static int pwrseq_suspend(struct pwrseq *p) +{ + return 0; +} +static int pwrseq_resume(struct pwrseq *p) +{ + return 0; +} +static int pwrseq_suspend_list(struct list_head *head) +{ + return 0; +} +static int pwrseq_resume_list(struct list_head *head) +{ + return 0; +} +#endif /* CONFIG_POWER_SEQUENCE */ + +#endif /* __LINUX_PWRSEQ_H */