From patchwork Tue Sep 20 03:36:41 2016 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Peter Chen X-Patchwork-Id: 9340975 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 225D9601C2 for ; Tue, 20 Sep 2016 03:39:24 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 0F47529B58 for ; Tue, 20 Sep 2016 03:39:24 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id 00CD029BEA; Tue, 20 Sep 2016 03:39:23 +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 D8BCC29B58 for ; Tue, 20 Sep 2016 03:39:22 +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 1bmBsU-0006nw-IQ; Tue, 20 Sep 2016 03:37:50 +0000 Received: from mail-by2nam03on0056.outbound.protection.outlook.com ([104.47.42.56] helo=NAM03-BY2-obe.outbound.protection.outlook.com) by bombadil.infradead.org with esmtps (Exim 4.85_2 #1 (Red Hat Linux)) id 1bmBs5-0006cf-DF for linux-arm-kernel@lists.infradead.org; Tue, 20 Sep 2016 03:37:30 +0000 Received: from BN3PR0301CA0078.namprd03.prod.outlook.com (10.160.152.174) by DM5PR03MB2443.namprd03.prod.outlook.com (10.168.233.13) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384_P384) id 15.1.629.8; Tue, 20 Sep 2016 03:37:15 +0000 Received: from BN1BFFO11FD020.protection.gbl (2a01:111:f400:7c10::1:133) by BN3PR0301CA0078.outlook.office365.com (2a01:111:e400:401e::46) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384_P384) id 15.1.629.8 via Frontend Transport; Tue, 20 Sep 2016 03:37:15 +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 BN1BFFO11FD020.mail.protection.outlook.com (10.58.144.83) with Microsoft SMTP Server (version=TLS1_0, cipher=TLS_RSA_WITH_AES_256_CBC_SHA) id 15.1.629.5 via Frontend Transport; Tue, 20 Sep 2016 03:37:14 +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 u8K3antj006721; Mon, 19 Sep 2016 20:37:06 -0700 From: Peter Chen To: , , , , , , , , Subject: [PATCH v7 2/8] power: add power sequence library Date: Tue, 20 Sep 2016 11:36:41 +0800 Message-ID: <1474342607-27512-3-git-send-email-peter.chen@nxp.com> X-Mailer: git-send-email 2.7.4 In-Reply-To: <1474342607-27512-1-git-send-email-peter.chen@nxp.com> References: <1474342607-27512-1-git-send-email-peter.chen@nxp.com> X-EOPAttributedMessage: 0 X-Matching-Connectors: 131188162352231348; (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)(2980300002)(1110001)(1109001)(339900001)(199003)(189002)(86362001)(189998001)(11100500001)(81166006)(8676002)(586003)(19580405001)(8666005)(97736004)(81156014)(305945005)(2201001)(5001770100001)(50986999)(76176999)(36756003)(356003)(19580395003)(48376002)(15975445007)(77096005)(8936002)(50466002)(2171001)(7416002)(2906002)(47776003)(5003940100001)(92566002)(7846002)(68736007)(104016004)(50226002)(626004)(229853001)(105606002)(87936001)(2950100001)(33646002)(5660300001)(4326007)(106466001)(85426001)(7059030)(2004002)(2101003); DIR:OUT; SFP:1101; SCL:1; SRVR:DM5PR03MB2443; H:tx30smr01.am.freescale.net; FPR:; SPF:Fail; PTR:InfoDomainNonexistent; A:1; MX:1; LANG:en; X-Microsoft-Exchange-Diagnostics: 1; BN1BFFO11FD020; 1:O8tiCayL0Q0kGJCCmZ1i8uFi3PGO23HLUtLiWTnjj2a0ktk2aA5PTMXmGyVA/InDI+0jkZHPtmF8Z/a/XubsrwvpX10HOJQ+vSpjw5aJ1Lpf3jKqyEVqbQCd9xVxJBuoGCFY+SgMYUpoJ5AJ1aTPbEh5mwBwoxezgRBX7ASpecVq3ND0Jgh0SRIJoJDxESZQEe6zdqqnAFXwn9FbIIFMQQcEktu9tmDhWHZcE/YQKpP+ErEwrps5UyTFHG1pxO1VJ6OeZ1NhWGCpWPZHzPcbj5RCtb7ljuX3gyA8q8F4mgRN+JeJHueV9CVZMHsA8n9noo2uLhs+MX4uiqB1XLK6CYmcDTwHqTRLxTEvOOoYbFql6miKGcLi65SE8Sk85MxpiXy0mZlozM7a0bF4rs7Hmc7V0SPhzRU2sYGHNFesuMKyBGtJHzv5EeSRHw2DPG4gK6LcbwQ1diNKI8KExK5boseWaG7kakyxm5e3tFrECitjTEjpjPg/InGLxUtrtgPz6CyeT9WQQn1nHIxRZj5hxzavAtyHQKVleRE5XDJN28qdAF7fJwDdEkYtXrbFAm6ls2v7wagSEuA0jmyw9W8HhR3CIkdSQkkT3lehVfnKv7KNShFtiWTGA+KRhptVAt6cI0yQn3YcAJcjgi4DGs2ikOD/A8Ucg1eEtmcEghzp5tNSoW4T4iQzwDiZkB56WR02N5neD3WhKiomHjzW3FWbp+Hyz3S75RxhRht5bmBiP1f90Quf7sBNZkUVoEwp86Hv MIME-Version: 1.0 X-MS-Office365-Filtering-Correlation-Id: 8a66ecd9-2c14-41a5-5254-08d3e1076a32 X-Microsoft-Exchange-Diagnostics: 1; DM5PR03MB2443; 2:zIa4N6vskVrbKm1PmpjdohoNXieU6DTYk8NKOZhCzLENZfhq2JiNnAjvLL03E6aR9s/6dgWMwWgLpl2S1Dsj6HinVwuCLCAe390uyJ5cPVjguLiJ8B+R/ttdmjcvXWQGhUGqJOLa9PLq0Ruj74l4r/aNCm4gzycd61MjdphSRlO091/xpHd/laz7rj0KcnNT; 3:M39KmrPpRFpIfwrjEh06jXY1A5n9Q6LZI34p2inJOrIQybnM7xmONCdgf9ccTEcELgppFjMhqhVS4d2OLGKhcZqMaA8I1/FIffSpTR0UbHNOPdRj4WI9CcSE4Y3AxrFnML3T/RBNCgNry/pjoO0ODC2VcQS/lgt1JVW+b5Jm1lMXOomJeNXJCku+TG8MbHqM8BrNzWgZZGyLAW2Eu5uV5YbOrGlM9BVch741e9loCmE= X-Microsoft-Antispam: UriScan:;BCL:0;PCL:0;RULEID:;SRVR:DM5PR03MB2443; X-Microsoft-Exchange-Diagnostics: 1; DM5PR03MB2443; 25:YS3r0+T3O5yYrSAwI37uHANqvVNReUA5i0+Gt/be1iXRnOYLFnAX1clCNqS4EspYWsWdBs/Gi9xHnv7ZudIInJsAItsemmUhoUYsLsf3c4r/F9eN4lGK8J+/+pkD6uOIJ/t1vGf555mZqvcq3WKZyecAUE10W3WVxcjWxdZB18m2od4iR3YliqvRa83GfOXS+2K0j9OwXnx+/rcDatSOtEV3w9ow9UEH0OoMskPIFkLzESHIfXvXkAb7NefZNDmCweCJPC1D5D2Ek/S/rQZZ10LezW9ZcEM7hKl8jR5QyfcCgZZrvjfuKVzzWkejgeCC6GXFyGH9fsGJW81NETjs0tqlvRoB7NtSjBam0WNIUOy9iZRZOgETDnoK3e+CKyUx7HMYDM1jDHaSZuadrT4BOgOBus1XRehcmvN6BF+3WkOyelw0ITupY8kTNeTCcp9+Cbvg2m4ksLyX1wVXr0L8raF+vIPROwH6cKf8iXc/liz4eNTneQVPjYnfQNF8BXPPsjdskXKG+C34xiT69+nLhUilb4gPqr6uUsNKE9AVmi4N0mPcpUQEHbnVpCZbhN4jfBbsC/rSm+jyNE6KzuVcqBBQwMXyUiP30mL3ATnkjpds4LqVBW8rnqmQG54R9Aaw8VJ4Qp0rgtZhIWssCZPEHaGRicvPDL3PDGvHZY7UGK73jyJ+IvCCzQ0Zb/W1wN8lMguw60RCvk7RPfbupJjq84KGqY47q2BTpg6/gDDqCzyldOgM7XLMt4GKxCicJxxBG4OX3P+4Rait3pywSAaQ2oInJMs5z654zevfXX2XRdzR53rmuhCGSy4xLTJc/SEG X-Microsoft-Exchange-Diagnostics: 1; DM5PR03MB2443; 31:PYDTAL6vWRrGAuktTjefRjvJSaCyz0Vz39R0LBLNlNf6D9oeTgXjGuotAuH81B1VlIQ9PVKAd/GRyJWxSygdn224UZiMCl9eL3CNB/lFU27h4MZnKoKWNob74bXGTOcPuLeGGhoLbicVet1yo1Qv4PgaNgJKODlgtLATl8uxyr6guHkD9qdQayvSB9yKpRPjUp5xBUnmpJR9A5O6uQw9KXMR58ygdtCVyM3RFbgq+x0=; 4:cSRGrXXfRRWezTFR5nTVwbEeX2/TKCpTcrj2OyHSNPzi3QUWfJcYqil5jxcC6rP08rdvXR4zwP8bh74G6Mdr/asAVaVAwB55xJtpFMIY9sNj7Xo/zag/nM0m8aN1HmDFTQV3b/bGlA5uLynTenGXYTTisbuuHJaAtr2fCLkAtdPbOODnHJzq5ZmcYOydufQrR8LS969fkirbCZ/OfgvMd/bvkzlk9MQCUyO8f1OzIs4cbmLAxNwaCeL6Vx7VIpm2v64zPjLt7ff87RmZLjEgcHuF9vWux0Sicx329bjpIHN55JAXQrUyuIbp9erORp3xupDcWaVovUTByWtbq7g7LtmBNxVwYXGU3JWZ0nJRN67jgKAnBsUF3HLH1wcRFG/6oR9NhYUAhM1PSKemoVf7Fvd2lJnFReGFki9QNhNyVT55bR4Lez6aHDkTiPg3XCeKb25uUGzufqZ7BNMCffwCZw9NZ9/5Im/YkoH2BoOm5E9BHurrW7/AK7Ri0CIDKn7qeuLCQIN4LjmV0vc5BuEWJptx0K/OoFHTwdSpr7Km8PuAX9nRH4HDn4H9VVuEmBUFTMz7sWTOEB/bo7xvC2arkfXqm4I1VROId3VYp3TFXWheTYEoDtC0RtneoznLir2RybMkohsWRc7+kRNoYbLa0ucofUTqgYqiSav+MFa2dto= X-Microsoft-Antispam-PRVS: X-Exchange-Antispam-Report-Test: UriScan:(250305191791016)(22074186197030)(9452136761055)(185117386973197)(84791874153150); X-Exchange-Antispam-Report-CFA-Test: BCL:0; PCL:0; RULEID:(6040176)(601004)(2401047)(13018025)(13024025)(13017025)(13015025)(13023025)(8121501046)(5005006)(3002001)(10201501046)(6055026); SRVR:DM5PR03MB2443; BCL:0; PCL:0; RULEID:(400006); SRVR:DM5PR03MB2443; X-Forefront-PRVS: 0071BFA85B X-Microsoft-Exchange-Diagnostics: =?us-ascii?Q?1; DM5PR03MB2443; 23:t4evndYv/Gx8dOX9Snl11Cpmsshp1CM07T2UaD3hx?= =?us-ascii?Q?TRDEH1TXguQo6wLovFrAVkXAjhhPKbTLJjra3lOmRCEMUKJNXZN8urJVAjay?= =?us-ascii?Q?xVs480245GaSE5FzfdQaW3wVQYRm4iNCnVJr0006MFr56jRsXuwsLT6lsgXr?= =?us-ascii?Q?UK6wXg/TkbcRnUyZSrQ9pyfGF0RxmjFZCgzG7mhl0P5TQMhLJRLdlnrSewwC?= =?us-ascii?Q?C9xiWwg8cdntN/0zbA3DlFOFE6PsP83rMoo7GbhtLSx/NHF1ZkS5kSl2ucio?= =?us-ascii?Q?of1O70sj3ckRd5pHc5JK2Sk0r5L9PSC5UeLyczQGeS9xqfld3+3CgSTjzQ4I?= =?us-ascii?Q?RYfUcrYIYPY8CavfyFd7DsmtOY4IPdyXVL7OSXXGz0jEU1UD+pfvL2F6Y/Kf?= =?us-ascii?Q?QsJsD8ezhdF14cP1IXyv/2fUZEYqfI7QPK72kGgpDSAh3SWHxMXY9EXhYfw8?= =?us-ascii?Q?lu4oe+VuJVXRCDIBt7z9jLbvGqdK3HrLiltSc2EzX4wjQGtIbia/Hfe3aPAI?= =?us-ascii?Q?NPtvQOAUBHKDumh/KDSVhu+ABKkGeXZEyDBO2nVW403GM9mNFOgblJxAPQAn?= =?us-ascii?Q?ZvsHwOx/yuCC/Ve4shUMBgQiSTyLqGcvrlVf0Ti98qV9BI8tyIxSmSnFhYHF?= =?us-ascii?Q?QaHBz5owMUhes5YZWc15tGeKRi0WECphG/kUFP35xeWW8h3deY0BJcOQtMAh?= =?us-ascii?Q?YuhIpYOG4PmR5mJvYsvGXIdD0Laz56S4Tpx2qVXW5mdc+4OBaVRPvjNbP3qb?= =?us-ascii?Q?fRNp2+zgCs1s9C1OytkI5tH951VaSHcY6jhgcDb8tLLSUeBBEBbE4uCdz21+?= =?us-ascii?Q?p3NPwvj62DhZPzkxupFydfSJ/cZBUoDYCr1BwIA4IbdWYUVxnqf0dPq7hrcj?= =?us-ascii?Q?vzMi+DeCx46RMvXN2m/ZFtQqK3qwHw5vqvWYDMpxhNfxEb95VFpWL5ydMpZo?= =?us-ascii?Q?+YEPBaJdHb+BtY77ZE9KCZFnoSX7E1Rks3DZUH1g1YNlHyalBmJy178rzd0H?= =?us-ascii?Q?DEHPrRQTDhkfSjWbiKFJesXK3PwAFibQMHjXgGBBH210Bw400CV73UWhN+8z?= =?us-ascii?Q?msfNOTdfyfI6MEXew65ob6bEcSop2hX2ZWJ4nEG4lrCB7qzhGP3bzxROHkXQ?= =?us-ascii?Q?pbGCrBworA17dLykuvB+WJ9U/sWlt5rmOFOE9mSJ0T2WXUTtS9PUsFqI285D?= =?us-ascii?Q?eigxfRC187a0/jzxvKc+ZGT1dnKWiO027vMYQ+vEzq44371pue4JBCR/Wt7+?= =?us-ascii?Q?or3pg/82fCZLfZbo9SYDVBXzDB9CUO4sL+AKXRso8gsGEvG8/b83i9egbzqk?= =?us-ascii?B?UT09?= X-Microsoft-Exchange-Diagnostics: 1; DM5PR03MB2443; 6:7Ni8/Lb1YWL1yQ0Tqxih516KjeX0OaLfBt61xyYeR/g0paeMMLon7Sisp20EsSLJ0wfIFaV4NzEeG7TCbRs7W102zET278FVhvVj3D7iVwrarhmB6ovOUPl5ecY8CGiaRRGecnCbYl+QCgfrcchpzds7H+2KQj35UQuwj6DE0mFU0CKY0oaS++1BQKPkwefeMNdjAqa9M/a+8seIh0aySPRAev/A1AVc/3RQldnYsAEKQ/6UlSF71lqMrnifdo1WYQuxJ2l3rcs1mHT9Ep/7jx44wU1c5tHL0bXDadEjqLw=; 5:dVkev/JzA9yDHT3eLBu6T9GMRi4iLszUjkU+GG88BpmqZL8HO8e5DUnorQAkGBFitkq2MpHTK2mbrs/7yqE9tvj8hnbHsv9tj5mAFmaFiCiLFWlq2oDCVSiUp2oKCBDpgBFqEDoZAVZH5axbm3EiSLu4Ve51/AYXQLPcYLsJ4X8=; 24:KLLjPK5/9pqQM4vpoklvskUgxCVZEgeMH0aIg/g0b2Eb45srkJPONzc7ZjIqGBo/LzKextrQlJx88A83nt5nTN9ME8k6+/puMI6l96Y0OSw=; 7:BbbEYV/KpiZBWrIpyqKnoOyUWG6wFEvwJ7rOgB+wjbjUSj6lQa9DshTObLyc9ypNbG5X9Gqv/lqG6HY84nkdtW8jsOlVe0+tde2lPjYfTUy/+FghbIcmKnOzZyDwc+bHIZdJDi2gg5Dq8nYpw8C6ZnX1kARjUsFaQVOD4J6L6ZolzYr5QBcV+79jQiwFBfNSKAIb/adrPXaX+e/0Z9HoHkJvQm5C2CuGEjqGdB5uUnhKUzytqzwePlCicrLawKPsuLXa8PfvZMWFexQ+dfRf1N8veZ0BpWoXcAGe7beIOAnKyhMM1Y3PJNsPcy2PTtEg SpamDiagnosticOutput: 1:99 SpamDiagnosticMetadata: NSPM X-MS-Exchange-CrossTenant-OriginalArrivalTime: 20 Sep 2016 03:37:14.7083 (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: DM5PR03MB2443 X-CRM114-Version: 20100106-BlameMichelson ( TRE 0.8.0 (BSD) ) MR-646709E3 X-CRM114-CacheID: sfid-20160919_203725_603722_C70CBD48 X-CRM114-Status: GOOD ( 20.96 ) 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, devicetree@vger.kernel.org, k.kozlowski@samsung.com, stephen.boyd@linaro.org, oscar@naiandei.net, arnd@arndb.de, pawel.moll@arm.com, linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org, s.hauer@pengutronix.de, linux-usb@vger.kernel.org, mail@maciej.szmigiero.name, troy.kisky@boundarydevices.com, vaibhav.hiremath@linaro.org, stillcompiling@gmail.com, Peter Chen , p.zabel@pengutronix.de, festevam@gmail.com, mka@chromium.org, linux-arm-kernel@lists.infradead.org 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. pwrseq_generic is intended for general purpose of power sequence, which handles gpios and clocks currently, and can cover regulator and pinctrl 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). Signed-off-by: Peter Chen 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 | 45 ++++++ drivers/power/pwrseq/Makefile | 3 + drivers/power/pwrseq/core.c | 190 ++++++++++++++++++++++++ drivers/power/pwrseq/pwrseq_compatible_sample.c | 178 ++++++++++++++++++++++ drivers/power/pwrseq/pwrseq_generic.c | 177 ++++++++++++++++++++++ include/linux/power/pwrseq.h | 73 +++++++++ 9 files changed, 677 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_compatible_sample.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 b3e9395..b353769 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -9343,6 +9343,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 M: Dmitry Eremin-Solenikov diff --git a/drivers/power/Kconfig b/drivers/power/Kconfig index acd4a15..f6aa4fd 100644 --- a/drivers/power/Kconfig +++ b/drivers/power/Kconfig @@ -515,3 +515,4 @@ endif # POWER_SUPPLY source "drivers/power/reset/Kconfig" source "drivers/power/avs/Kconfig" +source "drivers/power/pwrseq/Kconfig" diff --git a/drivers/power/Makefile b/drivers/power/Makefile index e46b75d..4ed2e12 100644 --- a/drivers/power/Makefile +++ b/drivers/power/Makefile @@ -74,3 +74,4 @@ obj-$(CONFIG_CHARGER_TPS65217) += tps65217_charger.o obj-$(CONFIG_POWER_RESET) += reset/ obj-$(CONFIG_AXP288_FUEL_GAUGE) += axp288_fuel_gauge.o obj-$(CONFIG_AXP288_CHARGER) += axp288_charger.o +obj-$(CONFIG_POWER_SEQUENCE) += pwrseq/ diff --git a/drivers/power/pwrseq/Kconfig b/drivers/power/pwrseq/Kconfig new file mode 100644 index 0000000..dff5e35 --- /dev/null +++ b/drivers/power/pwrseq/Kconfig @@ -0,0 +1,45 @@ +# +# Power Sequence library +# + +config POWER_SEQUENCE + bool + +menu "Power Sequence Support" + +config PWRSEQ_GENERIC + bool "Generic power sequence control" + depends on OF + select POWER_SEQUENCE + 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. This generic one can be + used for common power sequence control. + +config PWRSEQ_GENERIC_INSTANCE_NUMBER + int "Number of Generic Power Sequence Instance" + depends on PWRSEQ_GENERIC + range 1 10 + default 2 + help + Usually, there are not so many devices needs power sequence, we set two + as default value. + +config PWRSEQ_SAMPLE + bool "sample power sequence control using compatible string" + depends on OF + select POWER_SEQUENCE + help + It is a sample library which implements power sequence for device id, + it is an example purpose. + +config PWRSEQ_SAMPLE_INSTANCE_NUMBER + int "Number of Sample Power Sequence Instance" + depends on PWRSEQ_SAMPLE + range 1 5 + default 1 + help + Usually, this file is special for certain device, so the default for this number + is 1. +endmenu diff --git a/drivers/power/pwrseq/Makefile b/drivers/power/pwrseq/Makefile new file mode 100644 index 0000000..62f3cbf --- /dev/null +++ b/drivers/power/pwrseq/Makefile @@ -0,0 +1,3 @@ +obj-$(CONFIG_POWER_SEQUENCE) += core.o +obj-$(CONFIG_PWRSEQ_GENERIC) += pwrseq_generic.o +obj-$(CONFIG_PWRSEQ_SAMPLE) += pwrseq_compatible_sample.o diff --git a/drivers/power/pwrseq/core.c b/drivers/power/pwrseq/core.c new file mode 100644 index 0000000..6beae20 --- /dev/null +++ b/drivers/power/pwrseq/core.c @@ -0,0 +1,190 @@ +/* + * 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. If not, see . + */ + +#include +#include +#include +#include +#include + +static DEFINE_MUTEX(pwrseq_list_mutex); +static LIST_HEAD(pwrseq_list); + +int pwrseq_get(struct device_node *np, struct pwrseq *p) +{ + if (p && p->get) + return p->get(np, p); + + return -ENOTSUPP; +} +EXPORT_SYMBOL_GPL(pwrseq_get); + +int pwrseq_on(struct pwrseq *p) +{ + if (p && p->on) + return p->on(p); + + return -ENOTSUPP; +} +EXPORT_SYMBOL_GPL(pwrseq_on); + +void pwrseq_off(struct pwrseq *p) +{ + if (p && p->off) + p->off(p); +} +EXPORT_SYMBOL_GPL(pwrseq_off); + +void pwrseq_put(struct pwrseq *p) +{ + if (p && p->put) + p->put(p); +} +EXPORT_SYMBOL_GPL(pwrseq_put); + +void pwrseq_free(struct pwrseq *p) +{ + if (p && p->free) + p->free(p); +} +EXPORT_SYMBOL_GPL(pwrseq_free); + +int pwrseq_suspend(struct pwrseq *p) +{ + if (p && p->suspend) + return p->suspend(p); + + return 0; +} +EXPORT_SYMBOL_GPL(pwrseq_suspend); + +int pwrseq_resume(struct pwrseq *p) +{ + if (p && p->resume) + return p->resume(p); + + return 0; +} +EXPORT_SYMBOL_GPL(pwrseq_resume); + +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); + +static struct pwrseq *pwrseq_find_available_instance(struct device_node *np) +{ + struct pwrseq *pwrseq; + + 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; + 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; + return pwrseq; + } + } + pr_warn("Can't find any pwrseq instances for %s\n", np->full_name); + + return NULL; +} + +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(-ENONET); + + ret = pwrseq_get(np, pwrseq); + if (ret) + goto pwr_free; + + ret = pwrseq_on(pwrseq); + if (ret) + goto pwr_put; + + return pwrseq; + +pwr_put: + pwrseq_put(pwrseq); +pwr_free: + pwrseq_free(pwrseq); + return ERR_PTR(ret); +} +EXPORT_SYMBOL_GPL(of_pwrseq_on); + +void of_pwrseq_off(struct pwrseq *pwrseq) +{ + pwrseq_off(pwrseq); + pwrseq_put(pwrseq); + pwrseq_free(pwrseq); +} +EXPORT_SYMBOL_GPL(of_pwrseq_off); + +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 = of_pwrseq_on(np); + if (IS_ERR(pwrseq)) + return PTR_ERR(pwrseq); + + pwrseq_list_node = kzalloc(sizeof(*pwrseq_list_node), GFP_KERNEL); + if (!pwrseq_list_node) { + of_pwrseq_off(pwrseq); + return -ENOMEM; + } + pwrseq_list_node->pwrseq = pwrseq; + list_add(&pwrseq_list_node->list, head); + + return 0; +} +EXPORT_SYMBOL_GPL(of_pwrseq_on_list); + +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); diff --git a/drivers/power/pwrseq/pwrseq_compatible_sample.c b/drivers/power/pwrseq/pwrseq_compatible_sample.c new file mode 100644 index 0000000..d6bcc6d --- /dev/null +++ b/drivers/power/pwrseq/pwrseq_compatible_sample.c @@ -0,0 +1,178 @@ +/* + * pwrseq_compatible_sample.c Sample power sequence handling for compatible + * + * 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. If not, see . + */ + +#include +#include +#include +#include +#include +#include +#include + +#include + +struct pwrseq_sample { + struct pwrseq pwrseq; + struct gpio_desc *gpiod_reset; + struct clk *clks[PWRSEQ_MAX_CLKS]; + u32 duration_us; +}; + +#define to_generic_pwrseq(p) container_of(p, struct pwrseq_sample, pwrseq) + +static void pwrseq_sample_free(struct pwrseq *pwrseq) +{ + pwrseq->used = false; +} + +static void pwrseq_sample_put(struct pwrseq *pwrseq) +{ + struct pwrseq_sample *pwrseq_sam = to_generic_pwrseq(pwrseq); + int clk; + + if (pwrseq_sam->gpiod_reset) + gpiod_put(pwrseq_sam->gpiod_reset); + + for (clk = 0; clk < PWRSEQ_MAX_CLKS; clk++) + clk_put(pwrseq_sam->clks[clk]); +} + +static void pwrseq_sample_off(struct pwrseq *pwrseq) +{ + struct pwrseq_sample *pwrseq_sam = to_generic_pwrseq(pwrseq); + int clk; + + for (clk = PWRSEQ_MAX_CLKS - 1; clk >= 0; clk--) + clk_disable_unprepare(pwrseq_sam->clks[clk]); +} + +static int pwrseq_sample_on(struct pwrseq *pwrseq) +{ + struct pwrseq_sample *pwrseq_sam = to_generic_pwrseq(pwrseq); + int clk, ret = 0; + struct gpio_desc *gpiod_reset = pwrseq_sam->gpiod_reset; + + for (clk = 0; clk < PWRSEQ_MAX_CLKS && pwrseq_sam->clks[clk]; clk++) { + ret = clk_prepare_enable(pwrseq_sam->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_sam->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_sam->clks[clk]); + + return ret; +} + +static int pwrseq_sample_get(struct device_node *np, struct pwrseq *pwrseq) +{ + struct pwrseq_sample *pwrseq_sam = to_generic_pwrseq(pwrseq); + enum of_gpio_flags flags; + int reset_gpio, clk, ret = 0; + + for (clk = 0; clk < PWRSEQ_MAX_CLKS; clk++) { + pwrseq_sam->clks[clk] = of_clk_get(np, clk); + if (IS_ERR(pwrseq_sam->clks[clk])) { + ret = PTR_ERR(pwrseq_sam->clks[clk]); + if (ret != -ENOENT) + goto err_put_clks; + pwrseq_sam->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_sam->gpiod_reset = gpio_to_desc(reset_gpio); + of_property_read_u32(np, "reset-duration-us", + &pwrseq_sam->duration_us); + } else { + if (reset_gpio == -ENOENT) + return 0; + + ret = reset_gpio; + pr_err("Failed to get reset gpio on %s, err = %d\n", + np->full_name, reset_gpio); + goto err_put_clks; + } + + return ret; + +err_put_clks: + while (--clk >= 0) + clk_put(pwrseq_sam->clks[clk]); + return ret; +} + +static const struct of_device_id sample_id_table[] = { + { .compatible = "usb5e3,608",}, + { .compatible = "usbb95,1708",}, + { /* sentinel */ } +}; + +static int __init pwrseq_compatible_sample_register(void) +{ + struct pwrseq_sample *pwrseq_sam; + int i; + + for (i = 0; i < CONFIG_PWRSEQ_SAMPLE_INSTANCE_NUMBER; i++) { + pwrseq_sam = kzalloc(sizeof(*pwrseq_sam), GFP_KERNEL); + if (!pwrseq_sam) + return -ENOMEM; + + pwrseq_sam->pwrseq.pwrseq_of_match_table = sample_id_table; + pwrseq_sam->pwrseq.get = pwrseq_sample_get; + pwrseq_sam->pwrseq.on = pwrseq_sample_on; + pwrseq_sam->pwrseq.off = pwrseq_sample_off; + pwrseq_sam->pwrseq.put = pwrseq_sample_put; + pwrseq_sam->pwrseq.free = pwrseq_sample_free; + + pwrseq_register(&pwrseq_sam->pwrseq); + } + + return 0; +} +postcore_initcall(pwrseq_compatible_sample_register) diff --git a/drivers/power/pwrseq/pwrseq_generic.c b/drivers/power/pwrseq/pwrseq_generic.c new file mode 100644 index 0000000..bcd16c3 --- /dev/null +++ b/drivers/power/pwrseq/pwrseq_generic.c @@ -0,0 +1,177 @@ +/* + * 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. If not, see . + */ + +#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; +}; + +#define to_generic_pwrseq(p) container_of(p, struct pwrseq_generic, pwrseq) + +static void pwrseq_generic_free(struct pwrseq *pwrseq) +{ + pwrseq->used = false; +} + +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]); +} + +static void pwrseq_generic_off(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]); +} + +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) + return 0; + + ret = reset_gpio; + pr_err("Failed to get reset gpio on %s, err = %d\n", + np->full_name, reset_gpio); + goto err_put_clks; + } + + return ret; + +err_put_clks: + while (--clk >= 0) + clk_put(pwrseq_gen->clks[clk]); + return ret; +} + +static const struct of_device_id generic_id_table[] = { + { .compatible = "generic",}, + { /* sentinel */ } +}; + +static int __init pwrseq_generic_register(void) +{ + struct pwrseq_generic *pwrseq_gen; + int i; + + for (i = 0; i < CONFIG_PWRSEQ_GENERIC_INSTANCE_NUMBER; i++) { + 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.free = pwrseq_generic_free; + + pwrseq_register(&pwrseq_gen->pwrseq); + } + + return 0; +} +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..ae4753f --- /dev/null +++ b/include/linux/power/pwrseq.h @@ -0,0 +1,73 @@ +#ifndef __LINUX_PWRSEQ_H +#define __LINUX_PWRSEQ_H + +#include + +#define PWRSEQ_MAX_CLKS 3 + +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); + void (*free)(struct pwrseq *p); + int (*suspend)(struct pwrseq *p); + int (*resume)(struct pwrseq *p); + bool used; +}; + +/* 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) +int pwrseq_get(struct device_node *np, struct pwrseq *p); +int pwrseq_on(struct pwrseq *p); +void pwrseq_off(struct pwrseq *p); +void pwrseq_put(struct pwrseq *p); +void pwrseq_free(struct pwrseq *p); +int pwrseq_suspend(struct pwrseq *p); +int pwrseq_resume(struct pwrseq *p); +void pwrseq_register(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); +#else +static inline int pwrseq_get(struct device_node *np, struct pwrseq *p) +{ + return 0; +} +static inline int pwrseq_on(struct pwrseq *p) +{ + return 0; +} +static inline void pwrseq_off(struct pwrseq *p) {} +static inline void pwrseq_put(struct pwrseq *p) {} +static inline void pwrseq_free(struct pwrseq *p) {} +static inline int pwrseq_suspend(struct pwrseq *p) +{ + return 0; +} +static inline int pwrseq_resume(struct pwrseq *p) +{ + return 0; +} +static inline void pwrseq_register(struct pwrseq *pwrseq) {} +static inline struct pwrseq *of_pwrseq_on(struct device_node *np) +{ + return NULL; +} +void of_pwrseq_off(struct pwrseq *pwrseq) {} +int of_pwrseq_on_list(struct device_node *np, struct list_head *head) +{ + return 0; +} +void of_pwrseq_off_list(struct list_head *head) {} +#endif /* CONFIG_POWER_SEQUENCE */ + +#endif /* __LINUX_PWRSEQ_H */