Received: by 2002:a05:6358:3188:b0:123:57c1:9b43 with SMTP id q8csp25492578rwd; Sun, 2 Jul 2023 18:58:02 -0700 (PDT) X-Google-Smtp-Source: APBJJlF7W7mXfY5EHLJHqD0XmFfJbI0tkIXzD3Zm8/w4dVMKOpivoto86uz/Td/wwJIXWuxwY+zF X-Received: by 2002:a05:6214:d03:b0:636:e0be:14fd with SMTP id 3-20020a0562140d0300b00636e0be14fdmr1679780qvh.49.1688349482458; Sun, 02 Jul 2023 18:58:02 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1688349482; cv=none; d=google.com; s=arc-20160816; b=JZ3blem+OCTWsG7RL1At3zzufCu9v7O5Q6RDWF2c4ZJ/eejsw+aM5PnPEtQr4yEIdP XIIvicdBoeQ9yPzXbHDQXiN1gGOamx27TnlCut46PGSulD+spYWjQJOA3vCI/9CebOiB DOpGWMfEtepmEoFD/tk6azB8qfJjfvNWtCIN0hvhek15DloYxEShR3mTo28I2VOQBYpA f8r+zZRFnXfFOT0jz3q3LQo/IWEMOv3T/2umknwJn4UC9aYSq2kI4zzuTpF80Qm5PD1O 8d+eB+3A9mnHd7X2NoFLY1PQhRqeATuZjl8+KiAId6IeJhnP++22f690sDqHTH/Eb7Gh JBWQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:content-language :in-reply-to:mime-version:user-agent:date:message-id:from:references :cc:to:subject; bh=sEclxXVr+nP0otsEHCi8J6yq75b2zvFX0jx4FKhNB/4=; fh=X253pFYKujkjNN5+epQO59i3QXCny7cf5SPcU6+hNNQ=; b=MgglSsLM7FCp6QdwzD8Sfij92NCoJb+iW09Nb2w1Zb1cwXLolhoe8hq2QK/1mdzBdT C6iaO+caR6hegxaFrnbU2KbcsoAs2WA2uhmffzlDYxP1HyXBn8zN4KAtwy85FByg8Vya H4wPEkm8NlC7vhm8a7ba1WtH1ASfBrVqs/j8vwMKQubtJvP7hnQusibyNBm8901DoIkQ 3rJZOpdGQWUhXCzexlm7o1MAjb4msn7D4SzB8smTzyxvbOjvy5Ok2AMeud2OAK06EgLj Z1ulRdm2+ppkZITKxHLGAbEK7YEA8au5AbAkJpJhA3vwc7Hqnruvi/KZ+PYP5sko/ehh FIeA== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id s4-20020a63d044000000b0052c73367c13si16362126pgi.871.2023.07.02.18.57.46; Sun, 02 Jul 2023 18:58:02 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S229988AbjGCBau (ORCPT + 99 others); Sun, 2 Jul 2023 21:30:50 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:43070 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229777AbjGCBat (ORCPT ); Sun, 2 Jul 2023 21:30:49 -0400 Received: from mail.loongson.cn (mail.loongson.cn [114.242.206.163]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id B0A5A12E; Sun, 2 Jul 2023 18:30:46 -0700 (PDT) Received: from loongson.cn (unknown [10.20.42.35]) by gateway (Coremail) with SMTP id _____8AxCsbFJKJkZREFAA--.8271S3; Mon, 03 Jul 2023 09:30:45 +0800 (CST) Received: from [10.20.42.35] (unknown [10.20.42.35]) by localhost.localdomain (Coremail) with SMTP id AQAAf8DxJ8y+JKJkfkUZAA--.38104S3; Mon, 03 Jul 2023 09:30:44 +0800 (CST) Subject: Re: [PATCH v4 2/2] soc: loongson2_pm: add power management support To: Rob Herring , Krzysztof Kozlowski , Conor Dooley , Arnd Bergmann , linux-pm@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, Liu Yun Cc: Jianmin Lv , wanghongliang@loongson.cn, Liu Peibao , loongson-kernel@lists.loongnix.cn, zhuyinbo@loongson.cn References: <20230620090811.721-1-zhuyinbo@loongson.cn> <20230620090811.721-3-zhuyinbo@loongson.cn> From: zhuyinbo Message-ID: Date: Mon, 3 Jul 2023 09:30:38 +0800 User-Agent: Mozilla/5.0 (X11; Linux loongarch64; rv:68.0) Gecko/20100101 Thunderbird/68.7.0 MIME-Version: 1.0 In-Reply-To: <20230620090811.721-3-zhuyinbo@loongson.cn> Content-Type: text/plain; charset=gbk; format=flowed Content-Language: en-US Content-Transfer-Encoding: 8bit X-CM-TRANSID: AQAAf8DxJ8y+JKJkfkUZAA--.38104S3 X-CM-SenderInfo: 52kx5xhqerqz5rrqw2lrqou0/ X-Coremail-Antispam: 1Uk129KBjDUn29KB7ZKAUJUUUUU529EdanIXcx71UUUUU7KY7 ZEXasCq-sGcSsGvfJ3UbIjqfuFe4nvWSU5nxnvy29KBjDU0xBIdaVrnUUvcSsGvfC2Kfnx nUUI43ZEXa7xR_UUUUUUUUU== X-Spam-Status: No, score=-2.0 required=5.0 tests=BAYES_00,NICE_REPLY_A, SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org 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 > Signed-off-by: Liu Yun > Co-developed-by: Liu Peibao > Signed-off-by: Liu Peibao > Signed-off-by: Yinbo Zhu > --- > 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 > 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 > 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 > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#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"); >