Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1752134AbbFZHtA (ORCPT ); Fri, 26 Jun 2015 03:49:00 -0400 Received: from mail-bl2on0105.outbound.protection.outlook.com ([65.55.169.105]:4861 "EHLO na01-bl2-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1751655AbbFZHsu (ORCPT ); Fri, 26 Jun 2015 03:48:50 -0400 Authentication-Results: spf=fail (sender IP is 192.88.168.50) smtp.mailfrom=freescale.com; freescale.mail.onmicrosoft.com; dkim=none (message not signed) header.d=none; From: Yuantian.Tang@freescale.com To: b07421@freescale.com CC: linuxppc-dev@lists.ozlabs.org, linux-kernel@vger.kernel.org, devicetree@vger.kernel.org, Tang Yuantian , Chenhui Zhao Subject: [PATCH v5] powerpc/rcpm: add RCPM driver Date: Fri, 26 Jun 2015 15:44:16 +0800 Message-ID: <1435304656-34505-1-git-send-email-Yuantian.Tang@freescale.com> X-Mailer: git-send-email 2.1.0.27.g96db324 X-EOPAttributedMessage: 0 X-Microsoft-Exchange-Diagnostics: 1;BY2FFO11OLC012;1:fcmsfML+ar4c4jzGBfSNlcOH+3uIKRR8c0lDW1FoU3QvlCknA+f4QvCZHGMudrNePsixteas6wqcscyL6pRbdBcOmx1O1G3+9VX/sUWFIO+WqA0aXqlMih7TL4Fqp2r/4CyaU0om8+VLkmS7DXvQcBpPUZ3zmlPXI8NHSlfghU4C5pQ+d5E3IngnmLMgZY2Nvny66siWwaEI9cJJPRLklMGXYLM67n55/lbQMUmIr0+HQn3UxRBFcA8MVUectOc/id2GVSk5vS70zvQo5E/KUy/4GYYd4erfJiw0knjUGaha2WLFByYARNPW8dZdtkjyHrARFtvnqB4Sd0MGkY+3bUKmBBuTe+uhrsM8kQ0rhJgGZtpbuP8kjwl0c8YF61Y2 X-Forefront-Antispam-Report: CIP:192.88.168.50;CTRY:US;IPV:NLI;EFV:NLI;SFV:NSPM;SFS:(10019020)(6009001)(2980300002)(1060300003)(339900001)(199003)(189002)(50986999)(4290100001)(48376002)(50226001)(19580395003)(19580405001)(85782001)(87936001)(85426001)(47776003)(50466002)(6806004)(86362001)(2371004)(36756003)(77096005)(229853001)(2361001)(106466001)(105606002)(110136002)(62966003)(77156002)(92566002)(5001960100002)(107886002)(104016003)(110436001)(2351001)(46102003)(2004002)(4001450100001)(4001430100001)(85772001);DIR:OUT;SFP:1102;SCL:1;SRVR:BLUPR03MB390;H:tx30smr01.am.freescale.net;FPR:;SPF:Fail;MLV:sfv;A:1;MX:1;LANG:en; MIME-Version: 1.0 Content-Type: text/plain X-Microsoft-Exchange-Diagnostics: 1;BLUPR03MB390;2:/Q/8byyTbU8tWe6+KATKOydykEVtqZDIlo3Vn/n22nEMPXcA3LhUtuaNxQHaU+G5;3:4tmH3KDUrprl1vK7JpactBF2O/ltUbdUloq8cmApZTiIPLJqTqcZmQy1SP8Nu99K0VsFEULTQXaBzrQnBPaxYSHySIqcJknfKc6LA+n+1WklZa9gHXaEmjDNw2OCvTWx05BOhBhj6mySauKqG3b56EEES0zeHB1gw7eZwIT3w0lGp9Rn5sFr7JucGICadQfddP+l1hRUMqfrpJjJUOKCXiA3FmIxvdEgzaSVrKKQvUI=;25:2E4TGxXQRZCBxZd52Jvx7EBqZtuqS9L9di6zIm4NJ356HWuifwiOrD4MqPGe2ZZfx0DMhOzgfBOc08zous2i7JrhcHznoMl6Drre7DKfElHR0bsQiOFn0AtEsL2y4reO5WGbz4Bmpq83DYaiPNIBA7j/J3e3UuzafwBa0EQ48wIvWwvmdM3pBE+t1kj/ZkxeY2Pk3VS2GHsbxKlVtW0dUI5wCXswXlwH1VG7g4ejtPRuN7s7dtxx1cOp3vty3wwg;20:A9e2Qgch5tTkm9i1ueNlnvdJx/8QTQIx78DAJ9ehSgKGQBC4AbGMjI3l6NtMW9YrhndgWKNzw0L7mkRzpNe7S5iXtDHwe0eFto6au0hYXGXIVgHaVTVRpa0Agzie8IzTlWnooFeqBs17tvOKid5D3zQa6mKUSzG+I9wSI20Ms/YooJLw1bY/n/md3t27GR2tpf8zE9ygqkqkUTBzOhMLJKrbWwK+cCecxD3NSSp70/zNCXCqmaXxG7UPsvRsHOyWTrpV0B/ZnM9uRYl3iTx1dBbDZFY2Ie2aEe0IKM5UEtbblU/C3xt1FPkwqIlK/UDcsTuX6JQpVZWTx/Zf28GwdgbtbzncDiv6kImH+sPZB9Q= X-Microsoft-Antispam: UriScan:;BCL:1;PCL:0;RULEID:;SRVR:BLUPR03MB390; X-Microsoft-Antispam-PRVS: X-Exchange-Antispam-Report-Test: UriScan:; X-Exchange-Antispam-Report-CFA-Test: BCL:1;PCL:0;RULEID:(601004)(1201001)(5005006)(3002001);SRVR:BLUPR03MB390;BCL:1;PCL:0;RULEID:;SRVR:BLUPR03MB390; X-Microsoft-Exchange-Diagnostics: 1;BLUPR03MB390;4:FUppvsgpLE1uRkSvWvGOIsI1Q/iqEQHZls7rJl4F4u9KA9O9LSOD+KU3vIjZNqxB3SoagDn7rf4e/GAKCiO45booI5OkgIShZkTY/Q6Rb8l4b456UEMLhgZcUSNk8qgQsT7Ecu9yXElFwyEesk65v/mG905uogQdSfzrxCkyppJ1Rk+j5GnwZWB/2/tU7+aUGPdfyEk3JgssBNK4X86P+UOlRUKRB2EWcy+qSVAD+CyNtFXGB8dQEqFJ1ctlX3MwYCiK2VBN89WQDzMjBlbIbWkQSJ447+0pkHpV7itm6XRHF32j4JoUeOQbG+PIauT9 X-Forefront-PRVS: 0619D53754 X-Microsoft-Exchange-Diagnostics: =?us-ascii?Q?1;BLUPR03MB390;23:dNB1KHtnvi1kNfcGDwOwSlazNyZa8dB8eiyHFZu3pn?= =?us-ascii?Q?gnqUTZfSJraBb+XNjFyT5qv96DkCjeiIxzHNsQ8Q69QFg3lvTzFLIYKbgQaj?= =?us-ascii?Q?Fmzxf6UCQ+qW/ekyVGq3fcz7XqPdjXRPBzOjUFS6cpa/XOs0nbWYv2JiAqez?= =?us-ascii?Q?y6EkiI3C5iqjzQBz7Ph4N13tItQLPZfoRHuaMb9FWhtQGqX+3KJhb/FeBx5W?= =?us-ascii?Q?tdxOUbp/IkqjqUQWjcwgx5y/dUhfUBUvZhLg4SVkryyIl0AEabIJriLG3eJe?= =?us-ascii?Q?3LDCP8R1BHrx6yUzGZpSgO/zvJw/hZXajs+4NWTMRG5TzWyNlFsgN21y6AIJ?= =?us-ascii?Q?kKoVnp9ll1Qnae5Wsp/s/4hdUjPzp+IjsA8+bDHp6rzyRmJC8BFY2FVsXp+u?= =?us-ascii?Q?IvRIkZs2/VcvZmjcLkM214OHTpkBveDxOyPOS32adBsb98GmhcR0pIYMhuzR?= =?us-ascii?Q?QO0ndF8MUmErlcEYoVhhXMmxYLZXCo6/TobSRNudt3jvZMUQ3iQLNfdjBRzD?= =?us-ascii?Q?cu1yA3rYR+83BjQWz1bu7ftK1SvXpIPuY1QfKDLMSsTAWg0rme7d81u4R+0m?= =?us-ascii?Q?uozrFVoROmGc/sUZjeXs4MvEp4RxJ6YzO19vqGaEih9zrnocNJyIfipf5Qp5?= =?us-ascii?Q?uJvx1YFWszElC5W1eD6Jfl3GbrETj8sw0TQjuUJC6BPzZOEUrnDrwBVxXFZI?= =?us-ascii?Q?IoKVA3W4drdNrL1am/5l7fZKYqSiIR+W9doJES52/k5o025BCkMlFOE4N7Nu?= =?us-ascii?Q?SQUkdMUQ3vlJcU62RDH//L/WBCJHDO85d2fnfg9nxZDx/I5vKRQCpTEiKxa2?= =?us-ascii?Q?xaxLCwBVaLhwf9r2YAfhLWYA5EzN/LCPt7wGAVS9S4wyY6Dy4Kvqu7GwhSRv?= =?us-ascii?Q?akw/vuQVaXIvMpkSvjYQIwhzmjKPSaf5BoX0pnKPzElZmEEO+JP+YoqZ2Rxg?= =?us-ascii?Q?LTBblfp4uP+RwY7LxqV+mq7CTBRmgKibFFB4SVaUGJob59GQRurQQCw38pD9?= =?us-ascii?Q?umFvbuMlgBDbO0Bgqi1NyrSTH2GGwiOeUpdqHO+C4ja514i3o1AYBN9cr7jv?= =?us-ascii?Q?CzGDY=3D?= X-Microsoft-Exchange-Diagnostics: 1;BLUPR03MB390;5:9Bbrjv24KSF9v0DMWRbbjNUczvFLmJa/QV9rz6Gj7vSiER/92KoYtINcfJzL4NkA2SECLpJUMzfgpypVlD0HVUd/xRvRW1OFSoKVfXV0No2XW68yBL1Ss8Lbly14kE6gbBTory/2vrSR3SxBdSOf+w==;24:z8kXkDzDAs9lMQXCqi49IDMJ7tcbT6v1DKTVNmGErWD/NwHvVNpuz3q/tM8eAB6atCoRfmaxv57IHHdrz31R9DUT7hVg4hYQrhXdt5eZMO0=;20:ZlS/WfKhw5Vfx1AMX86Y0tGWG1L11ujQAIcZVReRlhJwUX2+nfVEvQ9VxYlIvhdt82vXKwZV1cTWKVCmra+7Sg== X-OriginatorOrg: freescale.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 26 Jun 2015 07:48:47.5207 (UTC) X-MS-Exchange-CrossTenant-Id: 710a03f5-10f6-4d38-9ff4-a80b81da590d X-MS-Exchange-CrossTenant-OriginalAttributedTenantConnectingIp: TenantId=710a03f5-10f6-4d38-9ff4-a80b81da590d;Ip=[192.88.168.50];Helo=[tx30smr01.am.freescale.net] X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: BLUPR03MB390 Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 19663 Lines: 642 From: Tang Yuantian There is a RCPM (Run Control/Power Management) in Freescale QorIQ series processors. The device performs tasks associated with device run control and power management. The driver implements some features: mask/unmask irq, enter/exit low power states, freeze time base, etc. Signed-off-by: Chenhui Zhao Signed-off-by: Tang Yuantian --- v5: - update chassis version for t1040 in bindings v4: - refine bindings document v3: - added static and __init modifier to fsl_rcpm_init v2: - fix code style issues - refine compatible string match part Documentation/devicetree/bindings/soc/fsl/rcpm.txt | 44 +++ arch/powerpc/include/asm/fsl_guts.h | 105 +++++++ arch/powerpc/include/asm/fsl_pm.h | 48 +++ arch/powerpc/platforms/85xx/Kconfig | 1 + arch/powerpc/sysdev/Kconfig | 5 + arch/powerpc/sysdev/Makefile | 1 + arch/powerpc/sysdev/fsl_rcpm.c | 338 +++++++++++++++++++++ 7 files changed, 542 insertions(+) create mode 100644 Documentation/devicetree/bindings/soc/fsl/rcpm.txt create mode 100644 arch/powerpc/include/asm/fsl_pm.h create mode 100644 arch/powerpc/sysdev/fsl_rcpm.c diff --git a/Documentation/devicetree/bindings/soc/fsl/rcpm.txt b/Documentation/devicetree/bindings/soc/fsl/rcpm.txt new file mode 100644 index 0000000..dc52f70 --- /dev/null +++ b/Documentation/devicetree/bindings/soc/fsl/rcpm.txt @@ -0,0 +1,44 @@ +* Run Control and Power Management + +The RCPM performs all device-level tasks associated with device run control +and power management. + +Required properites: + - reg : Offset and length of the register set of RCPM block. + - compatible : Sould contain a chip-specific RCPM block compatible string + and (if applicable) may contain a chassis-version RCPM compatible string. + Chip-specific strings are of the form "fsl,-rcpm", such as: + * "fsl,p2041-rcpm" + * "fsl,p3041-rcpm" + * "fsl,p4080-rcpm" + * "fsl,p5020-rcpm" + * "fsl,p5040-rcpm" + * "fsl,t4240-rcpm" + * "fsl,b4420-rcpm" + * "fsl,b4860-rcpm" + + Chassis-version RCPM strings include: + * "fsl,qoriq-rcpm-1.0": for chassis 1.0 rcpm + * "fsl,qoriq-rcpm-2.0": for chassis 2.0 rcpm + * "fsl,qoriq-rcpm-2.1": for chassis 2.1 rcpm + +All references to "1.0" and "2.0" refer to the QorIQ chassis version to +which the chip complies. +Chassis Version Example Chips +--------------- ------------------------------- +1.0 p4080, p5020, p5040, p2041, p3041 +2.0 t4240, b4860, b4420 +2.1 t1040 + +Example: +The RCPM node for T4240: + rcpm: global-utilities@e2000 { + compatible = "fsl,t4240-rcpm", "fsl,qoriq-rcpm-2.0"; + reg = <0xe2000 0x1000>; + }; + +The RCPM node for P4080: + rcpm: global-utilities@e2000 { + compatible = "fsl,qoriq-rcpm-1.0"; + reg = <0xe2000 0x1000>; + }; diff --git a/arch/powerpc/include/asm/fsl_guts.h b/arch/powerpc/include/asm/fsl_guts.h index 43b6bb1..a67413c 100644 --- a/arch/powerpc/include/asm/fsl_guts.h +++ b/arch/powerpc/include/asm/fsl_guts.h @@ -188,5 +188,110 @@ static inline void guts_set_pmuxcr_dma(struct ccsr_guts __iomem *guts, #endif +struct ccsr_rcpm_v1 { + u8 res0000[4]; + __be32 cdozsr; /* 0x0004 Core Doze Status Register */ + u8 res0008[4]; + __be32 cdozcr; /* 0x000c Core Doze Control Register */ + u8 res0010[4]; + __be32 cnapsr; /* 0x0014 Core Nap Status Register */ + u8 res0018[4]; + __be32 cnapcr; /* 0x001c Core Nap Control Register */ + u8 res0020[4]; + __be32 cdozpsr; /* 0x0024 Core Doze Previous Status Register */ + u8 res0028[4]; + __be32 cnappsr; /* 0x002c Core Nap Previous Status Register */ + u8 res0030[4]; + __be32 cwaitsr; /* 0x0034 Core Wait Status Register */ + u8 res0038[4]; + __be32 cwdtdsr; /* 0x003c Core Watchdog Detect Status Register */ + __be32 powmgtcsr; /* 0x0040 PM Control&Status Register */ +#define RCPM_POWMGTCSR_SLP 0x00020000 + u8 res0044[12]; + __be32 ippdexpcr; /* 0x0050 IP Powerdown Exception Control Register */ + u8 res0054[16]; + __be32 cpmimr; /* 0x0064 Core PM IRQ Mask Register */ + u8 res0068[4]; + __be32 cpmcimr; /* 0x006c Core PM Critical IRQ Mask Register */ + u8 res0070[4]; + __be32 cpmmcmr; /* 0x0074 Core PM Machine Check Mask Register */ + u8 res0078[4]; + __be32 cpmnmimr; /* 0x007c Core PM NMI Mask Register */ + u8 res0080[4]; + __be32 ctbenr; /* 0x0084 Core Time Base Enable Register */ + u8 res0088[4]; + __be32 ctbckselr; /* 0x008c Core Time Base Clock Select Register */ + u8 res0090[4]; + __be32 ctbhltcr; /* 0x0094 Core Time Base Halt Control Register */ + u8 res0098[4]; + __be32 cmcpmaskcr; /* 0x00a4 Core Machine Check Mask Register */ +}; + +struct ccsr_rcpm_v2 { + u8 res_00[12]; + __be32 tph10sr0; /* Thread PH10 Status Register */ + u8 res_10[12]; + __be32 tph10setr0; /* Thread PH10 Set Control Register */ + u8 res_20[12]; + __be32 tph10clrr0; /* Thread PH10 Clear Control Register */ + u8 res_30[12]; + __be32 tph10psr0; /* Thread PH10 Previous Status Register */ + u8 res_40[12]; + __be32 twaitsr0; /* Thread Wait Status Register */ + u8 res_50[96]; + __be32 pcph15sr; /* Physical Core PH15 Status Register */ + __be32 pcph15setr; /* Physical Core PH15 Set Control Register */ + __be32 pcph15clrr; /* Physical Core PH15 Clear Control Register */ + __be32 pcph15psr; /* Physical Core PH15 Prev Status Register */ + u8 res_c0[16]; + __be32 pcph20sr; /* Physical Core PH20 Status Register */ + __be32 pcph20setr; /* Physical Core PH20 Set Control Register */ + __be32 pcph20clrr; /* Physical Core PH20 Clear Control Register */ + __be32 pcph20psr; /* Physical Core PH20 Prev Status Register */ + __be32 pcpw20sr; /* Physical Core PW20 Status Register */ + u8 res_e0[12]; + __be32 pcph30sr; /* Physical Core PH30 Status Register */ + __be32 pcph30setr; /* Physical Core PH30 Set Control Register */ + __be32 pcph30clrr; /* Physical Core PH30 Clear Control Register */ + __be32 pcph30psr; /* Physical Core PH30 Prev Status Register */ + u8 res_100[32]; + __be32 ippwrgatecr; /* IP Power Gating Control Register */ + u8 res_124[12]; + __be32 powmgtcsr; /* Power Management Control & Status Reg */ +#define RCPM_POWMGTCSR_LPM20_RQ 0x00100000 +#define RCPM_POWMGTCSR_LPM20_ST 0x00000200 +#define RCPM_POWMGTCSR_P_LPM20_ST 0x00000100 + u8 res_134[12]; + __be32 ippdexpcr[4]; /* IP Powerdown Exception Control Reg */ + u8 res_150[12]; + __be32 tpmimr0; /* Thread PM Interrupt Mask Reg */ + u8 res_160[12]; + __be32 tpmcimr0; /* Thread PM Crit Interrupt Mask Reg */ + u8 res_170[12]; + __be32 tpmmcmr0; /* Thread PM Machine Check Interrupt Mask Reg */ + u8 res_180[12]; + __be32 tpmnmimr0; /* Thread PM NMI Mask Reg */ + u8 res_190[12]; + __be32 tmcpmaskcr0; /* Thread Machine Check Mask Control Reg */ + __be32 pctbenr; /* Physical Core Time Base Enable Reg */ + __be32 pctbclkselr; /* Physical Core Time Base Clock Select */ + __be32 tbclkdivr; /* Time Base Clock Divider Register */ + u8 res_1ac[4]; + __be32 ttbhltcr[4]; /* Thread Time Base Halt Control Register */ + __be32 clpcl10sr; /* Cluster PCL10 Status Register */ + __be32 clpcl10setr; /* Cluster PCL30 Set Control Register */ + __be32 clpcl10clrr; /* Cluster PCL30 Clear Control Register */ + __be32 clpcl10psr; /* Cluster PCL30 Prev Status Register */ + __be32 cddslpsetr; /* Core Domain Deep Sleep Set Register */ + __be32 cddslpclrr; /* Core Domain Deep Sleep Clear Register */ + __be32 cdpwroksetr; /* Core Domain Power OK Set Register */ + __be32 cdpwrokclrr; /* Core Domain Power OK Clear Register */ + __be32 cdpwrensr; /* Core Domain Power Enable Status Register */ + __be32 cddslsr; /* Core Domain Deep Sleep Status Register */ + u8 res_1e8[8]; + __be32 dslpcntcr[8]; /* Deep Sleep Counter Cfg Register */ + u8 res_300[3568]; +}; + #endif #endif diff --git a/arch/powerpc/include/asm/fsl_pm.h b/arch/powerpc/include/asm/fsl_pm.h new file mode 100644 index 0000000..4b09f09 --- /dev/null +++ b/arch/powerpc/include/asm/fsl_pm.h @@ -0,0 +1,48 @@ +/* + * Support Power Management + * + * Copyright 2014-2015 Freescale Semiconductor Inc. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by the + * Free Software Foundation; either version 2 of the License, or (at your + * option) any later version. + */ +#ifndef __PPC_FSL_PM_H +#define __PPC_FSL_PM_H +#ifdef __KERNEL__ + +#define E500_PM_PH10 1 +#define E500_PM_PH15 2 +#define E500_PM_PH20 3 +#define E500_PM_PH30 4 +#define E500_PM_DOZE E500_PM_PH10 +#define E500_PM_NAP E500_PM_PH15 + +#define PLAT_PM_SLEEP 20 +#define PLAT_PM_LPM20 30 + +#define FSL_PM_SLEEP (1 << 0) +#define FSL_PM_DEEP_SLEEP (1 << 1) + +struct fsl_pm_ops { + /* mask pending interrupts to the RCPM from MPIC */ + void (*irq_mask)(int cpu); + + /* unmask pending interrupts to the RCPM from MPIC */ + void (*irq_unmask)(int cpu); + void (*cpu_enter_state)(int cpu, int state); + void (*cpu_exit_state)(int cpu, int state); + int (*plat_enter_sleep)(void); + void (*freeze_time_base)(bool freeze); + + /* keep the power of IP blocks during sleep/deep sleep */ + void (*set_ip_power)(bool enable, u32 *mask); + + /* get platform supported power management modes */ + unsigned int (*get_pm_modes)(void); +}; + +extern const struct fsl_pm_ops *qoriq_pm_ops; +#endif /* __KERNEL__ */ +#endif /* __PPC_FSL_PM_H */ diff --git a/arch/powerpc/platforms/85xx/Kconfig b/arch/powerpc/platforms/85xx/Kconfig index 2fb4b24..ae1b8a2 100644 --- a/arch/powerpc/platforms/85xx/Kconfig +++ b/arch/powerpc/platforms/85xx/Kconfig @@ -8,6 +8,7 @@ menuconfig FSL_SOC_BOOKE select FSL_PCI if PCI select SERIAL_8250_EXTENDED if SERIAL_8250 select SERIAL_8250_SHARE_IRQ if SERIAL_8250 + select FSL_CORENET_RCPM if PPC_E500MC default y if FSL_SOC_BOOKE diff --git a/arch/powerpc/sysdev/Kconfig b/arch/powerpc/sysdev/Kconfig index a19332a..52dc165 100644 --- a/arch/powerpc/sysdev/Kconfig +++ b/arch/powerpc/sysdev/Kconfig @@ -40,3 +40,8 @@ config SCOM_DEBUGFS config GE_FPGA bool default n + +config FSL_CORENET_RCPM + bool + help + This option enables support for RCPM (Run Control/Power Management). diff --git a/arch/powerpc/sysdev/Makefile b/arch/powerpc/sysdev/Makefile index f7cb2a1..8bbedfd 100644 --- a/arch/powerpc/sysdev/Makefile +++ b/arch/powerpc/sysdev/Makefile @@ -20,6 +20,7 @@ obj-$(CONFIG_MMIO_NVRAM) += mmio_nvram.o obj-$(CONFIG_FSL_SOC) += fsl_soc.o fsl_mpic_err.o obj-$(CONFIG_FSL_PCI) += fsl_pci.o $(fsl-msi-obj-y) obj-$(CONFIG_FSL_PMC) += fsl_pmc.o +obj-$(CONFIG_FSL_CORENET_RCPM) += fsl_rcpm.o obj-$(CONFIG_FSL_LBC) += fsl_lbc.o obj-$(CONFIG_FSL_GTM) += fsl_gtm.o obj-$(CONFIG_FSL_85XX_CACHE_SRAM) += fsl_85xx_l2ctlr.o fsl_85xx_cache_sram.o diff --git a/arch/powerpc/sysdev/fsl_rcpm.c b/arch/powerpc/sysdev/fsl_rcpm.c new file mode 100644 index 0000000..4bd2d64 --- /dev/null +++ b/arch/powerpc/sysdev/fsl_rcpm.c @@ -0,0 +1,338 @@ +/* + * RCPM(Run Control/Power Management) support + * + * Copyright 2012-2015 Freescale Semiconductor Inc. + * + * Author: Chenhui Zhao + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by the + * Free Software Foundation; either version 2 of the License, or (at your + * option) any later version. + */ +#include +#include +#include +#include + +#include +#include +#include +#include + +const struct fsl_pm_ops *qoriq_pm_ops; + +static struct ccsr_rcpm_v1 __iomem *rcpm_v1_regs; +static struct ccsr_rcpm_v2 __iomem *rcpm_v2_regs; +static unsigned int fsl_supported_pm_modes; + +static void rcpm_v1_irq_mask(int cpu) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + unsigned int mask = 1 << hw_cpu; + + setbits32(&rcpm_v1_regs->cpmimr, mask); + setbits32(&rcpm_v1_regs->cpmcimr, mask); + setbits32(&rcpm_v1_regs->cpmmcmr, mask); + setbits32(&rcpm_v1_regs->cpmnmimr, mask); +} + +static void rcpm_v2_irq_mask(int cpu) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + unsigned int mask = 1 << hw_cpu; + + setbits32(&rcpm_v2_regs->tpmimr0, mask); + setbits32(&rcpm_v2_regs->tpmcimr0, mask); + setbits32(&rcpm_v2_regs->tpmmcmr0, mask); + setbits32(&rcpm_v2_regs->tpmnmimr0, mask); +} + +static void rcpm_v1_irq_unmask(int cpu) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + unsigned int mask = 1 << hw_cpu; + + clrbits32(&rcpm_v1_regs->cpmimr, mask); + clrbits32(&rcpm_v1_regs->cpmcimr, mask); + clrbits32(&rcpm_v1_regs->cpmmcmr, mask); + clrbits32(&rcpm_v1_regs->cpmnmimr, mask); +} + +static void rcpm_v2_irq_unmask(int cpu) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + unsigned int mask = 1 << hw_cpu; + + clrbits32(&rcpm_v2_regs->tpmimr0, mask); + clrbits32(&rcpm_v2_regs->tpmcimr0, mask); + clrbits32(&rcpm_v2_regs->tpmmcmr0, mask); + clrbits32(&rcpm_v2_regs->tpmnmimr0, mask); +} + +static void rcpm_v1_set_ip_power(bool enable, u32 *mask) +{ + if (enable) + setbits32(&rcpm_v1_regs->ippdexpcr, *mask); + else + clrbits32(&rcpm_v1_regs->ippdexpcr, *mask); +} + +static void rcpm_v2_set_ip_power(bool enable, u32 *mask) +{ + if (enable) + setbits32(&rcpm_v2_regs->ippdexpcr[0], *mask); + else + clrbits32(&rcpm_v2_regs->ippdexpcr[0], *mask); +} + +static void rcpm_v1_cpu_enter_state(int cpu, int state) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + unsigned int mask = 1 << hw_cpu; + + switch (state) { + case E500_PM_PH10: + setbits32(&rcpm_v1_regs->cdozcr, mask); + break; + case E500_PM_PH15: + setbits32(&rcpm_v1_regs->cnapcr, mask); + break; + default: + pr_warn("%s: Unknown cpu PM state (%d)\n", __func__, state); + break; + } +} + +static void rcpm_v2_cpu_enter_state(int cpu, int state) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + u32 mask = 1 << cpu_core_index_of_thread(hw_cpu); + + switch (state) { + case E500_PM_PH10: + /* one bit corresponds to one thread for PH10 of 6500 */ + setbits32(&rcpm_v2_regs->tph10setr0, 1 << hw_cpu); + break; + case E500_PM_PH15: + setbits32(&rcpm_v2_regs->pcph15setr, mask); + break; + case E500_PM_PH20: + setbits32(&rcpm_v2_regs->pcph20setr, mask); + break; + case E500_PM_PH30: + setbits32(&rcpm_v2_regs->pcph30setr, mask); + break; + default: + pr_warn("%s: Unknown cpu PM state (%d)\n", __func__, state); + } +} + +static void rcpm_v1_cpu_exit_state(int cpu, int state) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + unsigned int mask = 1 << hw_cpu; + + switch (state) { + case E500_PM_PH10: + clrbits32(&rcpm_v1_regs->cdozcr, mask); + break; + case E500_PM_PH15: + clrbits32(&rcpm_v1_regs->cnapcr, mask); + break; + default: + pr_warn("%s: Unknown cpu PM state (%d)\n", __func__, state); + break; + } +} + +static void rcpm_v2_cpu_exit_state(int cpu, int state) +{ + int hw_cpu = get_hard_smp_processor_id(cpu); + u32 mask = 1 << cpu_core_index_of_thread(hw_cpu); + + switch (state) { + case E500_PM_PH10: + setbits32(&rcpm_v2_regs->tph10clrr0, 1 << hw_cpu); + break; + case E500_PM_PH15: + setbits32(&rcpm_v2_regs->pcph15clrr, mask); + break; + case E500_PM_PH20: + setbits32(&rcpm_v2_regs->pcph20clrr, mask); + break; + case E500_PM_PH30: + setbits32(&rcpm_v2_regs->pcph30clrr, mask); + break; + default: + pr_warn("%s: Unknown cpu PM state (%d)\n", __func__, state); + } +} + +static int rcpm_v1_plat_enter_state(int state) +{ + u32 *pmcsr_reg = &rcpm_v1_regs->powmgtcsr; + int ret = 0; + int result; + + switch (state) { + case PLAT_PM_SLEEP: + setbits32(pmcsr_reg, RCPM_POWMGTCSR_SLP); + + /* Upon resume, wait for RCPM_POWMGTCSR_SLP bit to be clear. */ + result = spin_event_timeout( + !(in_be32(pmcsr_reg) & RCPM_POWMGTCSR_SLP), 10000, 10); + if (!result) { + pr_err("%s: timeout waiting for SLP bit to be cleared\n", + __func__); + ret = -ETIMEDOUT; + } + break; + default: + pr_warn("%s: Unknown platform PM state (%d)\n", + __func__, state); + ret = -EINVAL; + } + + return ret; +} + +static int rcpm_v2_plat_enter_state(int state) +{ + u32 *pmcsr_reg = &rcpm_v2_regs->powmgtcsr; + int ret = 0; + int result; + + switch (state) { + case PLAT_PM_LPM20: + /* clear previous LPM20 status */ + setbits32(pmcsr_reg, RCPM_POWMGTCSR_P_LPM20_ST); + /* enter LPM20 status */ + setbits32(pmcsr_reg, RCPM_POWMGTCSR_LPM20_RQ); + + /* At this point, the device is in LPM20 status. */ + + /* resume ... */ + result = spin_event_timeout( + !(in_be32(pmcsr_reg) & RCPM_POWMGTCSR_LPM20_ST), 10000, 10); + if (!result) { + pr_err("%s: timeout waiting for LPM20 bit to be cleared\n", + __func__); + ret = -ETIMEDOUT; + } + break; + default: + pr_warn("%s: Unknown platform PM state (%d)\n", + __func__, state); + ret = -EINVAL; + } + + return ret; +} + +static int rcpm_v1_plat_enter_sleep(void) +{ + return rcpm_v1_plat_enter_state(PLAT_PM_SLEEP); +} + +static int rcpm_v2_plat_enter_sleep(void) +{ + return rcpm_v2_plat_enter_state(PLAT_PM_LPM20); +} + +static void rcpm_common_freeze_time_base(u32 *tben_reg, int freeze) +{ + static u32 mask; + + if (freeze) { + mask = in_be32(tben_reg); + clrbits32(tben_reg, mask); + } else { + setbits32(tben_reg, mask); + } + + /* read back to push the previous write */ + in_be32(tben_reg); +} + +static void rcpm_v1_freeze_time_base(bool freeze) +{ + rcpm_common_freeze_time_base(&rcpm_v1_regs->ctbenr, freeze); +} + +static void rcpm_v2_freeze_time_base(bool freeze) +{ + rcpm_common_freeze_time_base(&rcpm_v2_regs->pctbenr, freeze); +} + +static unsigned int rcpm_get_pm_modes(void) +{ + return fsl_supported_pm_modes; +} + +static const struct fsl_pm_ops qoriq_rcpm_v1_ops = { + .irq_mask = rcpm_v1_irq_mask, + .irq_unmask = rcpm_v1_irq_unmask, + .cpu_enter_state = rcpm_v1_cpu_enter_state, + .cpu_exit_state = rcpm_v1_cpu_exit_state, + .plat_enter_sleep = rcpm_v1_plat_enter_sleep, + .set_ip_power = rcpm_v1_set_ip_power, + .freeze_time_base = rcpm_v1_freeze_time_base, + .get_pm_modes = rcpm_get_pm_modes, +}; + +static const struct fsl_pm_ops qoriq_rcpm_v2_ops = { + .irq_mask = rcpm_v2_irq_mask, + .irq_unmask = rcpm_v2_irq_unmask, + .cpu_enter_state = rcpm_v2_cpu_enter_state, + .cpu_exit_state = rcpm_v2_cpu_exit_state, + .plat_enter_sleep = rcpm_v2_plat_enter_sleep, + .set_ip_power = rcpm_v2_set_ip_power, + .freeze_time_base = rcpm_v2_freeze_time_base, + .get_pm_modes = rcpm_get_pm_modes, +}; + +static const struct of_device_id rcpm_matches[] = { + { + .compatible = "fsl,qoriq-rcpm-1.0", + .data = (void *)&qoriq_rcpm_v1_ops, + }, + { + .compatible = "fsl,qoriq-rcpm-2.0", + .data = (void *)&qoriq_rcpm_v2_ops, + }, + {}, +}; + +static int __init fsl_rcpm_init(void) +{ + struct device_node *np; + const struct of_device_id *match; + void __iomem *base; + + np = of_find_matching_node_and_match(NULL, rcpm_matches, &match); + if (!np) { + pr_err("%s: can't find the rcpm node.\n", __func__); + return -ENODEV; + } + + base = of_iomap(np, 0); + if (!base) { + pr_err("%s: of_iomap() error.\n", __func__); + return -ENOMEM; + } + + /* support sleep by default */ + fsl_supported_pm_modes = FSL_PM_SLEEP; + of_node_put(np); + + rcpm_v1_regs = base; + rcpm_v2_regs = base; + + qoriq_pm_ops = match->data; + + return 0; +} + +/* need to call this before SMP init */ +early_initcall(fsl_rcpm_init); -- 2.1.0.27.g96db324 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/