All of lore.kernel.org
 help / color / mirror / Atom feed
* [PATCH v4 0/2] soc: loongson2_pm: add power management support
@ 2023-06-20  9:08 Yinbo Zhu
  2023-06-20  9:08 ` [PATCH v4 1/2] soc: dt-bindings: add loongson-2 pm Yinbo Zhu
  2023-06-20  9:08 ` [PATCH v4 2/2] soc: loongson2_pm: add power management support Yinbo Zhu
  0 siblings, 2 replies; 7+ messages in thread
From: Yinbo Zhu @ 2023-06-20  9:08 UTC (permalink / raw)
  To: Rob Herring, Krzysztof Kozlowski, Conor Dooley, Arnd Bergmann,
	linux-pm, devicetree, linux-kernel, Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel, Yinbo Zhu

Loongson-2 platform support Power Management Controller (ACPI) and this
series patch was to add PM driver that base on dts and PM binding support.

Change in v4:
		1. The patch "[PATCH v3 1/3] loongarch: export some arch-specific
		   pm interfaces" had been merged into linux-next tree thus this
		   v4 series patch need drop it and need depend on it and it's
		   patch link was:
https://lore.kernel.org/all/20230615091757.24686-2-zhuyinbo@loongson.cn/
		2. Remove the pmc label in dt-binding patch.
		3. Add the Co-developed-by for driver patch.
		4. Simplify the loongson2_suspend_valid_state that "return
		   (state == PM_SUSPEND_MEM)".
		5. Use Using loongson2_pm_irq_enable() to replace.
		   loongson2_power_button_irq_enable().
		6. Remove the "oneOf" in dt-bindings patch.
		7. Replace "suspend-address" that use "loongson,suspend-address".
		8. Use u64 type that for "loongson,suspend-address".
		9. Rename "pm" to "power-mangement" in dt-bindings patch.
		10. Add the reivewed-by for dt-bindings patch.
Change in v3:
		1. Reword the [1/3] patch commit log and title.
		2. Use the old naming for suspend interface for the [1/3] and
		   [3/3] patch.
		3. Combine some small function in the driver patch.
		4. Rename 'pwrbt' to 'button' in the driver patch.
		5. Use the specific compatible in yaml file.
Change in v2:
		1. Fixup the "suspend-address" description.
		2. Remove the "return -EINVAL" in PM driver probe when firmware
		   no configure "suspend-address" property in dts in oder to
		   other PM state to work.

Yinbo Zhu (2):
  soc: dt-bindings: add loongson-2 pm
  soc: loongson2_pm: add power management support

 .../soc/loongson/loongson,ls2k-pmc.yaml       |  52 +++++
 MAINTAINERS                                   |   7 +
 drivers/soc/loongson/Kconfig                  |  10 +
 drivers/soc/loongson/Makefile                 |   1 +
 drivers/soc/loongson/loongson2_pm.c           | 215 ++++++++++++++++++
 5 files changed, 285 insertions(+)

-- 
2.20.1


^ permalink raw reply	[flat|nested] 7+ messages in thread

* [PATCH v4 1/2] soc: dt-bindings: add loongson-2 pm
  2023-06-20  9:08 [PATCH v4 0/2] soc: loongson2_pm: add power management support Yinbo Zhu
@ 2023-06-20  9:08 ` Yinbo Zhu
  2023-06-20  9:08 ` [PATCH v4 2/2] soc: loongson2_pm: add power management support Yinbo Zhu
  1 sibling, 0 replies; 7+ messages in thread
From: Yinbo Zhu @ 2023-06-20  9:08 UTC (permalink / raw)
  To: Rob Herring, Krzysztof Kozlowski, Conor Dooley, Arnd Bergmann,
	linux-pm, devicetree, linux-kernel, Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel,
	Yinbo Zhu, Krzysztof Kozlowski

Add the Loongson-2 SoC Power Management Controller binding with DT
schema format using json-schema.

Signed-off-by: Yinbo Zhu <zhuyinbo@loongson.cn>
Reviewed-by: Krzysztof Kozlowski <krzysztof.kozlowski@linaro.org>
---
 .../soc/loongson/loongson,ls2k-pmc.yaml       | 52 +++++++++++++++++++
 MAINTAINERS                                   |  6 +++
 2 files changed, 58 insertions(+)
 create mode 100644 Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml

diff --git a/Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml b/Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml
new file mode 100644
index 000000000000..3c3af6d15eb2
--- /dev/null
+++ b/Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml
@@ -0,0 +1,52 @@
+# SPDX-License-Identifier: (GPL-2.0 OR BSD-2-Clause)
+%YAML 1.2
+---
+$id: http://devicetree.org/schemas/soc/loongson/loongson,ls2k-pmc.yaml#
+$schema: http://devicetree.org/meta-schemas/core.yaml#
+
+title: Loongson-2 Power Manager controller
+
+maintainers:
+  - Yinbo Zhu <zhuyinbo@loongson.cn>
+
+properties:
+  compatible:
+    items:
+      - enum:
+          - loongson,ls2k1000-pmc
+          - loongson,ls2k0500-pmc
+      - const: syscon
+
+  reg:
+    maxItems: 1
+
+  interrupts:
+    maxItems: 1
+
+  loongson,suspend-address:
+    $ref: /schemas/types.yaml#/definitions/uint64
+    description:
+      The "loongson,suspend-address" is a deep sleep state (Suspend To
+      RAM) firmware entry address which was jumped from kernel and it's
+      value was dependent on specific platform firmware code. In
+      addition, the PM need according to it to indicate that current
+      SoC whether support Suspend To RAM.
+
+required:
+  - compatible
+  - reg
+  - interrupts
+
+additionalProperties: false
+
+examples:
+  - |
+    #include <dt-bindings/interrupt-controller/irq.h>
+
+    power-management@1fe27000 {
+        compatible = "loongson,ls2k1000-pmc", "syscon";
+        reg = <0x1fe27000 0x58>;
+        interrupt-parent = <&liointc1>;
+        interrupts = <11 IRQ_TYPE_LEVEL_LOW>;
+        loongson,suspend-address = <0x0 0x1c000500>;
+    };
diff --git a/MAINTAINERS b/MAINTAINERS
index 7a91f14cad2e..bcd05f1fa5c1 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -12190,6 +12190,12 @@ S:	Maintained
 F:	Documentation/devicetree/bindings/hwinfo/loongson,ls2k-chipid.yaml
 F:	drivers/soc/loongson/loongson2_guts.c
 
+LOONGSON-2 SOC SERIES PM DRIVER
+M:	Yinbo Zhu <zhuyinbo@loongson.cn>
+L:	linux-pm@vger.kernel.org
+S:	Maintained
+F:	Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml
+
 LOONGSON-2 SOC SERIES PINCTRL DRIVER
 M:	zhanghongchen <zhanghongchen@loongson.cn>
 M:	Yinbo Zhu <zhuyinbo@loongson.cn>
-- 
2.20.1


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH v4 2/2] soc: loongson2_pm: add power management support
  2023-06-20  9:08 [PATCH v4 0/2] soc: loongson2_pm: add power management support Yinbo Zhu
  2023-06-20  9:08 ` [PATCH v4 1/2] soc: dt-bindings: add loongson-2 pm Yinbo Zhu
@ 2023-06-20  9:08 ` Yinbo Zhu
  2023-07-03  1:30   ` zhuyinbo
  2023-07-27  2:58   ` Yinbo Zhu
  1 sibling, 2 replies; 7+ messages in thread
From: Yinbo Zhu @ 2023-06-20  9:08 UTC (permalink / raw)
  To: Rob Herring, Krzysztof Kozlowski, Conor Dooley, Arnd Bergmann,
	linux-pm, devicetree, linux-kernel, Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel, Yinbo Zhu

The Loongson-2's power management controller was ACPI, supports ACPI
S2Idle (Suspend To Idle), ACPI S3 (Suspend To RAM), ACPI S4 (Suspend To
Disk), ACPI S5 (Soft Shutdown) and supports multiple wake-up methods
(USB, GMAC, PWRBTN, etc.). This driver was to add power management
controller support that base on dts for Loongson-2 series SoCs.

Co-developed-by: Liu Yun <liuyun@loongson.cn>
Signed-off-by: Liu Yun <liuyun@loongson.cn>
Co-developed-by: Liu Peibao <liupeibao@loongson.cn>
Signed-off-by: Liu Peibao <liupeibao@loongson.cn>
Signed-off-by: Yinbo Zhu <zhuyinbo@loongson.cn>
---
 MAINTAINERS                         |   1 +
 drivers/soc/loongson/Kconfig        |  10 ++
 drivers/soc/loongson/Makefile       |   1 +
 drivers/soc/loongson/loongson2_pm.c | 215 ++++++++++++++++++++++++++++
 4 files changed, 227 insertions(+)
 create mode 100644 drivers/soc/loongson/loongson2_pm.c

diff --git a/MAINTAINERS b/MAINTAINERS
index bcd05f1fa5c1..7c4ad0cbaeff 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -12195,6 +12195,7 @@ M:	Yinbo Zhu <zhuyinbo@loongson.cn>
 L:	linux-pm@vger.kernel.org
 S:	Maintained
 F:	Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml
+F:	drivers/soc/loongson/loongson2_pm.c
 
 LOONGSON-2 SOC SERIES PINCTRL DRIVER
 M:	zhanghongchen <zhanghongchen@loongson.cn>
diff --git a/drivers/soc/loongson/Kconfig b/drivers/soc/loongson/Kconfig
index 707f56358dc4..4f3ce9eb7520 100644
--- a/drivers/soc/loongson/Kconfig
+++ b/drivers/soc/loongson/Kconfig
@@ -16,3 +16,13 @@ config LOONGSON2_GUTS
 	 SoCs. Initially only reading SVR and registering soc device are
 	 supported. Other guts accesses, such as reading firmware configuration
 	 by default, should eventually be added into this driver as well.
+
+config LOONGSON2_PM
+	bool "Loongson-2 SoC Power Management Controller Driver"
+	depends on LOONGARCH && OF
+	help
+	 The Loongson-2's power management controller was ACPI, supports ACPI
+	 S2Idle (Suspend To Idle), ACPI S3 (Suspend To RAM), ACPI S4 (Suspend To
+	 Disk), ACPI S5 (Soft Shutdown) and supports multiple wake-up methods
+	 (USB, GMAC, PWRBTN, etc.). This driver was to add power management
+	 controller support that base on dts for Loongson-2 series SoCs.
diff --git a/drivers/soc/loongson/Makefile b/drivers/soc/loongson/Makefile
index 263c486df638..4118f50f55e2 100644
--- a/drivers/soc/loongson/Makefile
+++ b/drivers/soc/loongson/Makefile
@@ -4,3 +4,4 @@
 #
 
 obj-$(CONFIG_LOONGSON2_GUTS)		+= loongson2_guts.o
+obj-$(CONFIG_LOONGSON2_PM)		+= loongson2_pm.o
diff --git a/drivers/soc/loongson/loongson2_pm.c b/drivers/soc/loongson/loongson2_pm.c
new file mode 100644
index 000000000000..cd022a1d5b33
--- /dev/null
+++ b/drivers/soc/loongson/loongson2_pm.c
@@ -0,0 +1,215 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Loongson-2 PM Support
+ *
+ * Copyright (C) 2023 Loongson Technology Corporation Limited
+ */
+
+#include <linux/io.h>
+#include <linux/of.h>
+#include <linux/init.h>
+#include <linux/input.h>
+#include <linux/suspend.h>
+#include <linux/interrupt.h>
+#include <linux/pm_wakeirq.h>
+#include <linux/platform_device.h>
+#include <asm/bootinfo.h>
+#include <asm/suspend.h>
+
+#define LOONGSON2_PM1_CNT_REG		0x14
+#define LOONGSON2_PM1_STS_REG		0x0c
+#define LOONGSON2_PM1_ENA_REG		0x10
+#define LOONGSON2_GPE0_STS_REG		0x28
+#define LOONGSON2_GPE0_ENA_REG		0x2c
+
+#define LOONGSON2_PM1_PWRBTN_STS	BIT(8)
+#define LOONGSON2_PM1_PCIEXP_WAKE_STS	BIT(14)
+#define LOONGSON2_PM1_WAKE_STS		BIT(15)
+#define LOONGSON2_PM1_CNT_INT_EN	BIT(0)
+#define LOONGSON2_PM1_PWRBTN_EN		LOONGSON2_PM1_PWRBTN_STS
+
+static struct loongson2_pm {
+	void __iomem			*base;
+	struct input_dev		*dev;
+	bool				suspended;
+} loongson2_pm;
+
+#define loongson2_pm_readw(reg)		readw(loongson2_pm.base + reg)
+#define loongson2_pm_readl(reg)		readl(loongson2_pm.base + reg)
+#define loongson2_pm_writew(val, reg)	writew(val, loongson2_pm.base + reg)
+#define loongson2_pm_writel(val, reg)	writel(val, loongson2_pm.base + reg)
+
+static void loongson2_pm_status_clear(void)
+{
+	u16 value;
+
+	value = loongson2_pm_readw(LOONGSON2_PM1_STS_REG);
+	value |= (LOONGSON2_PM1_PWRBTN_STS | LOONGSON2_PM1_PCIEXP_WAKE_STS |
+		  LOONGSON2_PM1_WAKE_STS);
+	loongson2_pm_writew(value, LOONGSON2_PM1_STS_REG);
+	loongson2_pm_writel(loongson2_pm_readl(LOONGSON2_GPE0_STS_REG), LOONGSON2_GPE0_STS_REG);
+}
+
+static void loongson2_pm_irq_enable(void)
+{
+	u16 value;
+
+	value = loongson2_pm_readw(LOONGSON2_PM1_CNT_REG);
+	value |= LOONGSON2_PM1_CNT_INT_EN;
+	loongson2_pm_writew(value, LOONGSON2_PM1_CNT_REG);
+
+	value = loongson2_pm_readw(LOONGSON2_PM1_ENA_REG);
+	value |= LOONGSON2_PM1_PWRBTN_EN;
+	loongson2_pm_writew(value, LOONGSON2_PM1_ENA_REG);
+}
+
+static int loongson2_suspend_enter(suspend_state_t state)
+{
+	loongson2_pm_status_clear();
+	loongarch_common_suspend();
+	loongarch_suspend_enter();
+	loongarch_common_resume();
+	loongson2_pm_irq_enable();
+	pm_set_resume_via_firmware();
+
+	return 0;
+}
+
+static int loongson2_suspend_begin(suspend_state_t state)
+{
+	pm_set_suspend_via_firmware();
+
+	return 0;
+}
+
+static int loongson2_suspend_valid_state(suspend_state_t state)
+{
+	return (state == PM_SUSPEND_MEM);
+}
+
+static const struct platform_suspend_ops loongson2_suspend_ops = {
+	.valid	= loongson2_suspend_valid_state,
+	.begin	= loongson2_suspend_begin,
+	.enter	= loongson2_suspend_enter,
+};
+
+static int loongson2_power_button_init(struct device *dev, int irq)
+{
+	int ret;
+	struct input_dev *button;
+
+	button = input_allocate_device();
+	if (!dev)
+		return -ENOMEM;
+
+	button->name = "Power Button";
+	button->phys = "pm/button/input0";
+	button->id.bustype = BUS_HOST;
+	button->dev.parent = NULL;
+	input_set_capability(button, EV_KEY, KEY_POWER);
+
+	ret = input_register_device(button);
+	if (ret)
+		goto free_dev;
+
+	dev_pm_set_wake_irq(&button->dev, irq);
+	device_set_wakeup_capable(&button->dev, true);
+	device_set_wakeup_enable(&button->dev, true);
+
+	loongson2_pm.dev = button;
+	dev_info(dev, "Power Button: Init successful!\n");
+
+	return 0;
+
+free_dev:
+	input_free_device(button);
+
+	return ret;
+}
+
+static irqreturn_t loongson2_pm_irq_handler(int irq, void *dev_id)
+{
+	u16 status = loongson2_pm_readw(LOONGSON2_PM1_STS_REG);
+
+	if (!loongson2_pm.suspended && (status & LOONGSON2_PM1_PWRBTN_STS)) {
+		pr_info("Power Button pressed...\n");
+		input_report_key(loongson2_pm.dev, KEY_POWER, 1);
+		input_sync(loongson2_pm.dev);
+		input_report_key(loongson2_pm.dev, KEY_POWER, 0);
+		input_sync(loongson2_pm.dev);
+	}
+
+	loongson2_pm_status_clear();
+
+	return IRQ_HANDLED;
+}
+
+static int __maybe_unused loongson2_pm_suspend(struct device *dev)
+{
+	loongson2_pm.suspended = true;
+
+	return 0;
+}
+
+static int __maybe_unused loongson2_pm_resume(struct device *dev)
+{
+	loongson2_pm.suspended = false;
+
+	return 0;
+}
+static SIMPLE_DEV_PM_OPS(loongson2_pm_ops, loongson2_pm_suspend, loongson2_pm_resume);
+
+static int loongson2_pm_probe(struct platform_device *pdev)
+{
+	int irq, retval;
+	u64 suspend_addr;
+	struct device *dev = &pdev->dev;
+
+	loongson2_pm.base = devm_platform_ioremap_resource(pdev, 0);
+	if (IS_ERR(loongson2_pm.base))
+		return PTR_ERR(loongson2_pm.base);
+
+	irq = platform_get_irq(pdev, 0);
+	if (irq < 0)
+		return irq;
+
+	if (!device_property_read_u64(dev, "loongson,suspend-address", &suspend_addr))
+		loongson_sysconf.suspend_addr = (u64)phys_to_virt(suspend_addr);
+	else
+		dev_err(dev, "No loongson,suspend-address, could not support S3!\n");
+
+	if (loongson2_power_button_init(dev, irq))
+		return -EINVAL;
+
+	retval = devm_request_irq(&pdev->dev, irq, loongson2_pm_irq_handler,
+				  IRQF_SHARED, "pm_irq", &loongson2_pm);
+	if (retval)
+		return retval;
+
+	loongson2_pm_irq_enable();
+	loongson2_pm_status_clear();
+
+	if (loongson_sysconf.suspend_addr)
+		suspend_set_ops(&loongson2_suspend_ops);
+
+	return 0;
+}
+
+static const struct of_device_id loongson2_pm_match[] = {
+	{ .compatible = "loongson,ls2k1000-pmc", },
+	{ .compatible = "loongson,ls2k0500-pmc", },
+	{},
+};
+
+static struct platform_driver loongson2_pm_driver = {
+	.driver = {
+		.name = "ls2k-pm",
+		.pm = &loongson2_pm_ops,
+		.of_match_table = loongson2_pm_match,
+	},
+	.probe = loongson2_pm_probe,
+};
+module_platform_driver(loongson2_pm_driver);
+
+MODULE_DESCRIPTION("Loongson-2 PM driver");
+MODULE_LICENSE("GPL");
-- 
2.20.1


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* Re: [PATCH v4 2/2] soc: loongson2_pm: add power management support
  2023-06-20  9:08 ` [PATCH v4 2/2] soc: loongson2_pm: add power management support Yinbo Zhu
@ 2023-07-03  1:30   ` zhuyinbo
  2023-07-03 13:47     ` Krzysztof Kozlowski
  2023-07-27  2:58   ` Yinbo Zhu
  1 sibling, 1 reply; 7+ messages in thread
From: zhuyinbo @ 2023-07-03  1:30 UTC (permalink / raw)
  To: Rob Herring, Krzysztof Kozlowski, Conor Dooley, Arnd Bergmann,
	linux-pm, devicetree, linux-kernel, Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel, zhuyinbo


Friendly ping ?


在 2023/6/20 下午5:08, Yinbo Zhu 写道:
> The Loongson-2's power management controller was ACPI, supports ACPI
> S2Idle (Suspend To Idle), ACPI S3 (Suspend To RAM), ACPI S4 (Suspend To
> Disk), ACPI S5 (Soft Shutdown) and supports multiple wake-up methods
> (USB, GMAC, PWRBTN, etc.). This driver was to add power management
> controller support that base on dts for Loongson-2 series SoCs.
> 
> Co-developed-by: Liu Yun <liuyun@loongson.cn>
> Signed-off-by: Liu Yun <liuyun@loongson.cn>
> Co-developed-by: Liu Peibao <liupeibao@loongson.cn>
> Signed-off-by: Liu Peibao <liupeibao@loongson.cn>
> Signed-off-by: Yinbo Zhu <zhuyinbo@loongson.cn>
> ---
>   MAINTAINERS                         |   1 +
>   drivers/soc/loongson/Kconfig        |  10 ++
>   drivers/soc/loongson/Makefile       |   1 +
>   drivers/soc/loongson/loongson2_pm.c | 215 ++++++++++++++++++++++++++++
>   4 files changed, 227 insertions(+)
>   create mode 100644 drivers/soc/loongson/loongson2_pm.c
> 
> diff --git a/MAINTAINERS b/MAINTAINERS
> index bcd05f1fa5c1..7c4ad0cbaeff 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -12195,6 +12195,7 @@ M:	Yinbo Zhu <zhuyinbo@loongson.cn>
>   L:	linux-pm@vger.kernel.org
>   S:	Maintained
>   F:	Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml
> +F:	drivers/soc/loongson/loongson2_pm.c
>   
>   LOONGSON-2 SOC SERIES PINCTRL DRIVER
>   M:	zhanghongchen <zhanghongchen@loongson.cn>
> diff --git a/drivers/soc/loongson/Kconfig b/drivers/soc/loongson/Kconfig
> index 707f56358dc4..4f3ce9eb7520 100644
> --- a/drivers/soc/loongson/Kconfig
> +++ b/drivers/soc/loongson/Kconfig
> @@ -16,3 +16,13 @@ config LOONGSON2_GUTS
>   	 SoCs. Initially only reading SVR and registering soc device are
>   	 supported. Other guts accesses, such as reading firmware configuration
>   	 by default, should eventually be added into this driver as well.
> +
> +config LOONGSON2_PM
> +	bool "Loongson-2 SoC Power Management Controller Driver"
> +	depends on LOONGARCH && OF
> +	help
> +	 The Loongson-2's power management controller was ACPI, supports ACPI
> +	 S2Idle (Suspend To Idle), ACPI S3 (Suspend To RAM), ACPI S4 (Suspend To
> +	 Disk), ACPI S5 (Soft Shutdown) and supports multiple wake-up methods
> +	 (USB, GMAC, PWRBTN, etc.). This driver was to add power management
> +	 controller support that base on dts for Loongson-2 series SoCs.
> diff --git a/drivers/soc/loongson/Makefile b/drivers/soc/loongson/Makefile
> index 263c486df638..4118f50f55e2 100644
> --- a/drivers/soc/loongson/Makefile
> +++ b/drivers/soc/loongson/Makefile
> @@ -4,3 +4,4 @@
>   #
>   
>   obj-$(CONFIG_LOONGSON2_GUTS)		+= loongson2_guts.o
> +obj-$(CONFIG_LOONGSON2_PM)		+= loongson2_pm.o
> diff --git a/drivers/soc/loongson/loongson2_pm.c b/drivers/soc/loongson/loongson2_pm.c
> new file mode 100644
> index 000000000000..cd022a1d5b33
> --- /dev/null
> +++ b/drivers/soc/loongson/loongson2_pm.c
> @@ -0,0 +1,215 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * Loongson-2 PM Support
> + *
> + * Copyright (C) 2023 Loongson Technology Corporation Limited
> + */
> +
> +#include <linux/io.h>
> +#include <linux/of.h>
> +#include <linux/init.h>
> +#include <linux/input.h>
> +#include <linux/suspend.h>
> +#include <linux/interrupt.h>
> +#include <linux/pm_wakeirq.h>
> +#include <linux/platform_device.h>
> +#include <asm/bootinfo.h>
> +#include <asm/suspend.h>
> +
> +#define LOONGSON2_PM1_CNT_REG		0x14
> +#define LOONGSON2_PM1_STS_REG		0x0c
> +#define LOONGSON2_PM1_ENA_REG		0x10
> +#define LOONGSON2_GPE0_STS_REG		0x28
> +#define LOONGSON2_GPE0_ENA_REG		0x2c
> +
> +#define LOONGSON2_PM1_PWRBTN_STS	BIT(8)
> +#define LOONGSON2_PM1_PCIEXP_WAKE_STS	BIT(14)
> +#define LOONGSON2_PM1_WAKE_STS		BIT(15)
> +#define LOONGSON2_PM1_CNT_INT_EN	BIT(0)
> +#define LOONGSON2_PM1_PWRBTN_EN		LOONGSON2_PM1_PWRBTN_STS
> +
> +static struct loongson2_pm {
> +	void __iomem			*base;
> +	struct input_dev		*dev;
> +	bool				suspended;
> +} loongson2_pm;
> +
> +#define loongson2_pm_readw(reg)		readw(loongson2_pm.base + reg)
> +#define loongson2_pm_readl(reg)		readl(loongson2_pm.base + reg)
> +#define loongson2_pm_writew(val, reg)	writew(val, loongson2_pm.base + reg)
> +#define loongson2_pm_writel(val, reg)	writel(val, loongson2_pm.base + reg)
> +
> +static void loongson2_pm_status_clear(void)
> +{
> +	u16 value;
> +
> +	value = loongson2_pm_readw(LOONGSON2_PM1_STS_REG);
> +	value |= (LOONGSON2_PM1_PWRBTN_STS | LOONGSON2_PM1_PCIEXP_WAKE_STS |
> +		  LOONGSON2_PM1_WAKE_STS);
> +	loongson2_pm_writew(value, LOONGSON2_PM1_STS_REG);
> +	loongson2_pm_writel(loongson2_pm_readl(LOONGSON2_GPE0_STS_REG), LOONGSON2_GPE0_STS_REG);
> +}
> +
> +static void loongson2_pm_irq_enable(void)
> +{
> +	u16 value;
> +
> +	value = loongson2_pm_readw(LOONGSON2_PM1_CNT_REG);
> +	value |= LOONGSON2_PM1_CNT_INT_EN;
> +	loongson2_pm_writew(value, LOONGSON2_PM1_CNT_REG);
> +
> +	value = loongson2_pm_readw(LOONGSON2_PM1_ENA_REG);
> +	value |= LOONGSON2_PM1_PWRBTN_EN;
> +	loongson2_pm_writew(value, LOONGSON2_PM1_ENA_REG);
> +}
> +
> +static int loongson2_suspend_enter(suspend_state_t state)
> +{
> +	loongson2_pm_status_clear();
> +	loongarch_common_suspend();
> +	loongarch_suspend_enter();
> +	loongarch_common_resume();
> +	loongson2_pm_irq_enable();
> +	pm_set_resume_via_firmware();
> +
> +	return 0;
> +}
> +
> +static int loongson2_suspend_begin(suspend_state_t state)
> +{
> +	pm_set_suspend_via_firmware();
> +
> +	return 0;
> +}
> +
> +static int loongson2_suspend_valid_state(suspend_state_t state)
> +{
> +	return (state == PM_SUSPEND_MEM);
> +}
> +
> +static const struct platform_suspend_ops loongson2_suspend_ops = {
> +	.valid	= loongson2_suspend_valid_state,
> +	.begin	= loongson2_suspend_begin,
> +	.enter	= loongson2_suspend_enter,
> +};
> +
> +static int loongson2_power_button_init(struct device *dev, int irq)
> +{
> +	int ret;
> +	struct input_dev *button;
> +
> +	button = input_allocate_device();
> +	if (!dev)
> +		return -ENOMEM;
> +
> +	button->name = "Power Button";
> +	button->phys = "pm/button/input0";
> +	button->id.bustype = BUS_HOST;
> +	button->dev.parent = NULL;
> +	input_set_capability(button, EV_KEY, KEY_POWER);
> +
> +	ret = input_register_device(button);
> +	if (ret)
> +		goto free_dev;
> +
> +	dev_pm_set_wake_irq(&button->dev, irq);
> +	device_set_wakeup_capable(&button->dev, true);
> +	device_set_wakeup_enable(&button->dev, true);
> +
> +	loongson2_pm.dev = button;
> +	dev_info(dev, "Power Button: Init successful!\n");
> +
> +	return 0;
> +
> +free_dev:
> +	input_free_device(button);
> +
> +	return ret;
> +}
> +
> +static irqreturn_t loongson2_pm_irq_handler(int irq, void *dev_id)
> +{
> +	u16 status = loongson2_pm_readw(LOONGSON2_PM1_STS_REG);
> +
> +	if (!loongson2_pm.suspended && (status & LOONGSON2_PM1_PWRBTN_STS)) {
> +		pr_info("Power Button pressed...\n");
> +		input_report_key(loongson2_pm.dev, KEY_POWER, 1);
> +		input_sync(loongson2_pm.dev);
> +		input_report_key(loongson2_pm.dev, KEY_POWER, 0);
> +		input_sync(loongson2_pm.dev);
> +	}
> +
> +	loongson2_pm_status_clear();
> +
> +	return IRQ_HANDLED;
> +}
> +
> +static int __maybe_unused loongson2_pm_suspend(struct device *dev)
> +{
> +	loongson2_pm.suspended = true;
> +
> +	return 0;
> +}
> +
> +static int __maybe_unused loongson2_pm_resume(struct device *dev)
> +{
> +	loongson2_pm.suspended = false;
> +
> +	return 0;
> +}
> +static SIMPLE_DEV_PM_OPS(loongson2_pm_ops, loongson2_pm_suspend, loongson2_pm_resume);
> +
> +static int loongson2_pm_probe(struct platform_device *pdev)
> +{
> +	int irq, retval;
> +	u64 suspend_addr;
> +	struct device *dev = &pdev->dev;
> +
> +	loongson2_pm.base = devm_platform_ioremap_resource(pdev, 0);
> +	if (IS_ERR(loongson2_pm.base))
> +		return PTR_ERR(loongson2_pm.base);
> +
> +	irq = platform_get_irq(pdev, 0);
> +	if (irq < 0)
> +		return irq;
> +
> +	if (!device_property_read_u64(dev, "loongson,suspend-address", &suspend_addr))
> +		loongson_sysconf.suspend_addr = (u64)phys_to_virt(suspend_addr);
> +	else
> +		dev_err(dev, "No loongson,suspend-address, could not support S3!\n");
> +
> +	if (loongson2_power_button_init(dev, irq))
> +		return -EINVAL;
> +
> +	retval = devm_request_irq(&pdev->dev, irq, loongson2_pm_irq_handler,
> +				  IRQF_SHARED, "pm_irq", &loongson2_pm);
> +	if (retval)
> +		return retval;
> +
> +	loongson2_pm_irq_enable();
> +	loongson2_pm_status_clear();
> +
> +	if (loongson_sysconf.suspend_addr)
> +		suspend_set_ops(&loongson2_suspend_ops);
> +
> +	return 0;
> +}
> +
> +static const struct of_device_id loongson2_pm_match[] = {
> +	{ .compatible = "loongson,ls2k1000-pmc", },
> +	{ .compatible = "loongson,ls2k0500-pmc", },
> +	{},
> +};
> +
> +static struct platform_driver loongson2_pm_driver = {
> +	.driver = {
> +		.name = "ls2k-pm",
> +		.pm = &loongson2_pm_ops,
> +		.of_match_table = loongson2_pm_match,
> +	},
> +	.probe = loongson2_pm_probe,
> +};
> +module_platform_driver(loongson2_pm_driver);
> +
> +MODULE_DESCRIPTION("Loongson-2 PM driver");
> +MODULE_LICENSE("GPL");
> 


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v4 2/2] soc: loongson2_pm: add power management support
  2023-07-03  1:30   ` zhuyinbo
@ 2023-07-03 13:47     ` Krzysztof Kozlowski
  2023-07-04  1:38       ` zhuyinbo
  0 siblings, 1 reply; 7+ messages in thread
From: Krzysztof Kozlowski @ 2023-07-03 13:47 UTC (permalink / raw)
  To: zhuyinbo, Rob Herring, Krzysztof Kozlowski, Conor Dooley,
	Arnd Bergmann, linux-pm, devicetree, linux-kernel, Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel

On 03/07/2023 03:30, zhuyinbo wrote:
> 
> Friendly ping ?
> 

Please avoid pings during merge window. Not much can happen with the
patchset anyway. Resend or ping afterwards.

Best regards,
Krzysztof


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v4 2/2] soc: loongson2_pm: add power management support
  2023-07-03 13:47     ` Krzysztof Kozlowski
@ 2023-07-04  1:38       ` zhuyinbo
  0 siblings, 0 replies; 7+ messages in thread
From: zhuyinbo @ 2023-07-04  1:38 UTC (permalink / raw)
  To: Krzysztof Kozlowski, Rob Herring, Krzysztof Kozlowski,
	Conor Dooley, Arnd Bergmann, linux-pm, devicetree, linux-kernel,
	Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel, zhuyinbo



在 2023/7/3 下午9:47, Krzysztof Kozlowski 写道:
> On 03/07/2023 03:30, zhuyinbo wrote:
>>
>> Friendly ping ?
>>
> 
> Please avoid pings during merge window. Not much can happen with the
> patchset anyway. Resend or ping afterwards.
> 


okay, I got it.

Thanks


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v4 2/2] soc: loongson2_pm: add power management support
  2023-06-20  9:08 ` [PATCH v4 2/2] soc: loongson2_pm: add power management support Yinbo Zhu
  2023-07-03  1:30   ` zhuyinbo
@ 2023-07-27  2:58   ` Yinbo Zhu
  1 sibling, 0 replies; 7+ messages in thread
From: Yinbo Zhu @ 2023-07-27  2:58 UTC (permalink / raw)
  To: Arnd Bergmann, linux-pm, linux-kernel, Liu Yun
  Cc: Jianmin Lv, wanghongliang, Liu Peibao, loongson-kernel, zhuyinbo


Friendly ping ?

在 2023/6/20 下午5:08, Yinbo Zhu 写道:
> The Loongson-2's power management controller was ACPI, supports ACPI
> S2Idle (Suspend To Idle), ACPI S3 (Suspend To RAM), ACPI S4 (Suspend To
> Disk), ACPI S5 (Soft Shutdown) and supports multiple wake-up methods
> (USB, GMAC, PWRBTN, etc.). This driver was to add power management
> controller support that base on dts for Loongson-2 series SoCs.
> 
> Co-developed-by: Liu Yun <liuyun@loongson.cn>
> Signed-off-by: Liu Yun <liuyun@loongson.cn>
> Co-developed-by: Liu Peibao <liupeibao@loongson.cn>
> Signed-off-by: Liu Peibao <liupeibao@loongson.cn>
> Signed-off-by: Yinbo Zhu <zhuyinbo@loongson.cn>
> ---
>   MAINTAINERS                         |   1 +
>   drivers/soc/loongson/Kconfig        |  10 ++
>   drivers/soc/loongson/Makefile       |   1 +
>   drivers/soc/loongson/loongson2_pm.c | 215 ++++++++++++++++++++++++++++
>   4 files changed, 227 insertions(+)
>   create mode 100644 drivers/soc/loongson/loongson2_pm.c
> 
> diff --git a/MAINTAINERS b/MAINTAINERS
> index bcd05f1fa5c1..7c4ad0cbaeff 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -12195,6 +12195,7 @@ M:	Yinbo Zhu <zhuyinbo@loongson.cn>
>   L:	linux-pm@vger.kernel.org
>   S:	Maintained
>   F:	Documentation/devicetree/bindings/soc/loongson/loongson,ls2k-pmc.yaml
> +F:	drivers/soc/loongson/loongson2_pm.c
>   
>   LOONGSON-2 SOC SERIES PINCTRL DRIVER
>   M:	zhanghongchen <zhanghongchen@loongson.cn>
> diff --git a/drivers/soc/loongson/Kconfig b/drivers/soc/loongson/Kconfig
> index 707f56358dc4..4f3ce9eb7520 100644
> --- a/drivers/soc/loongson/Kconfig
> +++ b/drivers/soc/loongson/Kconfig
> @@ -16,3 +16,13 @@ config LOONGSON2_GUTS
>   	 SoCs. Initially only reading SVR and registering soc device are
>   	 supported. Other guts accesses, such as reading firmware configuration
>   	 by default, should eventually be added into this driver as well.
> +
> +config LOONGSON2_PM
> +	bool "Loongson-2 SoC Power Management Controller Driver"
> +	depends on LOONGARCH && OF
> +	help
> +	 The Loongson-2's power management controller was ACPI, supports ACPI
> +	 S2Idle (Suspend To Idle), ACPI S3 (Suspend To RAM), ACPI S4 (Suspend To
> +	 Disk), ACPI S5 (Soft Shutdown) and supports multiple wake-up methods
> +	 (USB, GMAC, PWRBTN, etc.). This driver was to add power management
> +	 controller support that base on dts for Loongson-2 series SoCs.
> diff --git a/drivers/soc/loongson/Makefile b/drivers/soc/loongson/Makefile
> index 263c486df638..4118f50f55e2 100644
> --- a/drivers/soc/loongson/Makefile
> +++ b/drivers/soc/loongson/Makefile
> @@ -4,3 +4,4 @@
>   #
>   
>   obj-$(CONFIG_LOONGSON2_GUTS)		+= loongson2_guts.o
> +obj-$(CONFIG_LOONGSON2_PM)		+= loongson2_pm.o
> diff --git a/drivers/soc/loongson/loongson2_pm.c b/drivers/soc/loongson/loongson2_pm.c
> new file mode 100644
> index 000000000000..cd022a1d5b33
> --- /dev/null
> +++ b/drivers/soc/loongson/loongson2_pm.c
> @@ -0,0 +1,215 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * Loongson-2 PM Support
> + *
> + * Copyright (C) 2023 Loongson Technology Corporation Limited
> + */
> +
> +#include <linux/io.h>
> +#include <linux/of.h>
> +#include <linux/init.h>
> +#include <linux/input.h>
> +#include <linux/suspend.h>
> +#include <linux/interrupt.h>
> +#include <linux/pm_wakeirq.h>
> +#include <linux/platform_device.h>
> +#include <asm/bootinfo.h>
> +#include <asm/suspend.h>
> +
> +#define LOONGSON2_PM1_CNT_REG		0x14
> +#define LOONGSON2_PM1_STS_REG		0x0c
> +#define LOONGSON2_PM1_ENA_REG		0x10
> +#define LOONGSON2_GPE0_STS_REG		0x28
> +#define LOONGSON2_GPE0_ENA_REG		0x2c
> +
> +#define LOONGSON2_PM1_PWRBTN_STS	BIT(8)
> +#define LOONGSON2_PM1_PCIEXP_WAKE_STS	BIT(14)
> +#define LOONGSON2_PM1_WAKE_STS		BIT(15)
> +#define LOONGSON2_PM1_CNT_INT_EN	BIT(0)
> +#define LOONGSON2_PM1_PWRBTN_EN		LOONGSON2_PM1_PWRBTN_STS
> +
> +static struct loongson2_pm {
> +	void __iomem			*base;
> +	struct input_dev		*dev;
> +	bool				suspended;
> +} loongson2_pm;
> +
> +#define loongson2_pm_readw(reg)		readw(loongson2_pm.base + reg)
> +#define loongson2_pm_readl(reg)		readl(loongson2_pm.base + reg)
> +#define loongson2_pm_writew(val, reg)	writew(val, loongson2_pm.base + reg)
> +#define loongson2_pm_writel(val, reg)	writel(val, loongson2_pm.base + reg)
> +
> +static void loongson2_pm_status_clear(void)
> +{
> +	u16 value;
> +
> +	value = loongson2_pm_readw(LOONGSON2_PM1_STS_REG);
> +	value |= (LOONGSON2_PM1_PWRBTN_STS | LOONGSON2_PM1_PCIEXP_WAKE_STS |
> +		  LOONGSON2_PM1_WAKE_STS);
> +	loongson2_pm_writew(value, LOONGSON2_PM1_STS_REG);
> +	loongson2_pm_writel(loongson2_pm_readl(LOONGSON2_GPE0_STS_REG), LOONGSON2_GPE0_STS_REG);
> +}
> +
> +static void loongson2_pm_irq_enable(void)
> +{
> +	u16 value;
> +
> +	value = loongson2_pm_readw(LOONGSON2_PM1_CNT_REG);
> +	value |= LOONGSON2_PM1_CNT_INT_EN;
> +	loongson2_pm_writew(value, LOONGSON2_PM1_CNT_REG);
> +
> +	value = loongson2_pm_readw(LOONGSON2_PM1_ENA_REG);
> +	value |= LOONGSON2_PM1_PWRBTN_EN;
> +	loongson2_pm_writew(value, LOONGSON2_PM1_ENA_REG);
> +}
> +
> +static int loongson2_suspend_enter(suspend_state_t state)
> +{
> +	loongson2_pm_status_clear();
> +	loongarch_common_suspend();
> +	loongarch_suspend_enter();
> +	loongarch_common_resume();
> +	loongson2_pm_irq_enable();
> +	pm_set_resume_via_firmware();
> +
> +	return 0;
> +}
> +
> +static int loongson2_suspend_begin(suspend_state_t state)
> +{
> +	pm_set_suspend_via_firmware();
> +
> +	return 0;
> +}
> +
> +static int loongson2_suspend_valid_state(suspend_state_t state)
> +{
> +	return (state == PM_SUSPEND_MEM);
> +}
> +
> +static const struct platform_suspend_ops loongson2_suspend_ops = {
> +	.valid	= loongson2_suspend_valid_state,
> +	.begin	= loongson2_suspend_begin,
> +	.enter	= loongson2_suspend_enter,
> +};
> +
> +static int loongson2_power_button_init(struct device *dev, int irq)
> +{
> +	int ret;
> +	struct input_dev *button;
> +
> +	button = input_allocate_device();
> +	if (!dev)
> +		return -ENOMEM;
> +
> +	button->name = "Power Button";
> +	button->phys = "pm/button/input0";
> +	button->id.bustype = BUS_HOST;
> +	button->dev.parent = NULL;
> +	input_set_capability(button, EV_KEY, KEY_POWER);
> +
> +	ret = input_register_device(button);
> +	if (ret)
> +		goto free_dev;
> +
> +	dev_pm_set_wake_irq(&button->dev, irq);
> +	device_set_wakeup_capable(&button->dev, true);
> +	device_set_wakeup_enable(&button->dev, true);
> +
> +	loongson2_pm.dev = button;
> +	dev_info(dev, "Power Button: Init successful!\n");
> +
> +	return 0;
> +
> +free_dev:
> +	input_free_device(button);
> +
> +	return ret;
> +}
> +
> +static irqreturn_t loongson2_pm_irq_handler(int irq, void *dev_id)
> +{
> +	u16 status = loongson2_pm_readw(LOONGSON2_PM1_STS_REG);
> +
> +	if (!loongson2_pm.suspended && (status & LOONGSON2_PM1_PWRBTN_STS)) {
> +		pr_info("Power Button pressed...\n");
> +		input_report_key(loongson2_pm.dev, KEY_POWER, 1);
> +		input_sync(loongson2_pm.dev);
> +		input_report_key(loongson2_pm.dev, KEY_POWER, 0);
> +		input_sync(loongson2_pm.dev);
> +	}
> +
> +	loongson2_pm_status_clear();
> +
> +	return IRQ_HANDLED;
> +}
> +
> +static int __maybe_unused loongson2_pm_suspend(struct device *dev)
> +{
> +	loongson2_pm.suspended = true;
> +
> +	return 0;
> +}
> +
> +static int __maybe_unused loongson2_pm_resume(struct device *dev)
> +{
> +	loongson2_pm.suspended = false;
> +
> +	return 0;
> +}
> +static SIMPLE_DEV_PM_OPS(loongson2_pm_ops, loongson2_pm_suspend, loongson2_pm_resume);
> +
> +static int loongson2_pm_probe(struct platform_device *pdev)
> +{
> +	int irq, retval;
> +	u64 suspend_addr;
> +	struct device *dev = &pdev->dev;
> +
> +	loongson2_pm.base = devm_platform_ioremap_resource(pdev, 0);
> +	if (IS_ERR(loongson2_pm.base))
> +		return PTR_ERR(loongson2_pm.base);
> +
> +	irq = platform_get_irq(pdev, 0);
> +	if (irq < 0)
> +		return irq;
> +
> +	if (!device_property_read_u64(dev, "loongson,suspend-address", &suspend_addr))
> +		loongson_sysconf.suspend_addr = (u64)phys_to_virt(suspend_addr);
> +	else
> +		dev_err(dev, "No loongson,suspend-address, could not support S3!\n");
> +
> +	if (loongson2_power_button_init(dev, irq))
> +		return -EINVAL;
> +
> +	retval = devm_request_irq(&pdev->dev, irq, loongson2_pm_irq_handler,
> +				  IRQF_SHARED, "pm_irq", &loongson2_pm);
> +	if (retval)
> +		return retval;
> +
> +	loongson2_pm_irq_enable();
> +	loongson2_pm_status_clear();
> +
> +	if (loongson_sysconf.suspend_addr)
> +		suspend_set_ops(&loongson2_suspend_ops);
> +
> +	return 0;
> +}
> +
> +static const struct of_device_id loongson2_pm_match[] = {
> +	{ .compatible = "loongson,ls2k1000-pmc", },
> +	{ .compatible = "loongson,ls2k0500-pmc", },
> +	{},
> +};
> +
> +static struct platform_driver loongson2_pm_driver = {
> +	.driver = {
> +		.name = "ls2k-pm",
> +		.pm = &loongson2_pm_ops,
> +		.of_match_table = loongson2_pm_match,
> +	},
> +	.probe = loongson2_pm_probe,
> +};
> +module_platform_driver(loongson2_pm_driver);
> +
> +MODULE_DESCRIPTION("Loongson-2 PM driver");
> +MODULE_LICENSE("GPL");
> 


^ permalink raw reply	[flat|nested] 7+ messages in thread

end of thread, other threads:[~2023-07-27  2:59 UTC | newest]

Thread overview: 7+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2023-06-20  9:08 [PATCH v4 0/2] soc: loongson2_pm: add power management support Yinbo Zhu
2023-06-20  9:08 ` [PATCH v4 1/2] soc: dt-bindings: add loongson-2 pm Yinbo Zhu
2023-06-20  9:08 ` [PATCH v4 2/2] soc: loongson2_pm: add power management support Yinbo Zhu
2023-07-03  1:30   ` zhuyinbo
2023-07-03 13:47     ` Krzysztof Kozlowski
2023-07-04  1:38       ` zhuyinbo
2023-07-27  2:58   ` Yinbo Zhu

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.