Received: by 2002:a05:6a10:9848:0:0:0:0 with SMTP id x8csp1437059pxf; Fri, 2 Apr 2021 10:25:21 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyh4dGWV5YE3Vl1noclFJfFwXkuKwmdJrzS9B+B8vm2eWMUaapAw5JzoSlJB7bTk0a+aFKk X-Received: by 2002:a05:6e02:4:: with SMTP id h4mr1622062ilr.137.1617384321092; Fri, 02 Apr 2021 10:25:21 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1617384321; cv=none; d=google.com; s=arc-20160816; b=WsSbS5IacgWhvQPo9l1J1/OmznziZjlHbvMBRXXXzG/vv5j99Im2UkEgGtXjD1ea/o +MrjSHXQVWYTTN7bKRuNjfVWS1WQufM/3wRGGSLd7gSLippVLO+kv4gX4qeutW/YdChZ itICNdtjDJZDlu7+yfRsGQQVWpuam0L3Q3axc70l5TACXsaB+QCoWVdC8s1KbyppWwAN /JpLmG8CRT4d05sUrDNS/Gfa8FUg6S/AYZ5RnGSIXKLViP3TSnHV6ENDGbYf7vNze5f7 4r97+9TICK2v5ooftKdQfwfrTx29PvrZxIDRiuVwHMB6lPU3yWLmJBK0vzd5BAih+Bgf jSJA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:user-agent:message-id:references:in-reply-to :subject:cc:to:from:date:content-transfer-encoding:mime-version :sender:dkim-signature; bh=cQ+cbp24NuhwEIrZj0yvy1gVXW0X9jtwWhlgMPpel8o=; b=jmyJJkeThOzawpSgCnkvTDFN6Ihl0vmiC16exElnvjI0YTWPEGSIxvWxU4dlQWX7ca AWET4B21la1uWaBOZamBBjHAOrji16FsDwAI+0BDKR8qCyE0KFL64SUPoenSkx0MY+iT 9E0iNrR15BQT00+QYhGhQnE/7roxN1Df1LxyuuZTDaI0AQhRo2a92cCy1OoyjhCIgGAd t9yppSByBxlWmwXKGEqm3hDPxPKr+lYF0iLIONnMCKrvfzo20AyIATz75xPAnE/lapnh GNapGMzu4EMGADqERVcuWENS6btVrO73V4AbGx504oWQfBvIoBnIWkwAa+oeqedHbqCa nSOg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@mg.codeaurora.org header.s=smtp header.b=Q5T00SJK; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id o3si363007ill.74.2021.04.02.10.25.04; Fri, 02 Apr 2021 10:25:21 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@mg.codeaurora.org header.s=smtp header.b=Q5T00SJK; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S229553AbhDBRYt (ORCPT + 99 others); Fri, 2 Apr 2021 13:24:49 -0400 Received: from so254-9.mailgun.net ([198.61.254.9]:46047 "EHLO so254-9.mailgun.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229605AbhDBRYt (ORCPT ); Fri, 2 Apr 2021 13:24:49 -0400 DKIM-Signature: a=rsa-sha256; v=1; c=relaxed/relaxed; d=mg.codeaurora.org; q=dns/txt; s=smtp; t=1617384288; h=Message-ID: References: In-Reply-To: Subject: Cc: To: From: Date: Content-Transfer-Encoding: Content-Type: MIME-Version: Sender; bh=cQ+cbp24NuhwEIrZj0yvy1gVXW0X9jtwWhlgMPpel8o=; b=Q5T00SJKHTHKB/LSJLxhPJYoDy9co91by2O7OuITTV88DOYmHtCd3fmECzw9KxaycF59sUnC 4CUzgX8yvQdJO1MCoM8w3MBsb59OGnkUUkFqoaWUkBcz9ECDvQ09mT38e6ClSR443zQSsPcn UhIOCpPx/EEABenpHgr9ewQ5v/4= X-Mailgun-Sending-Ip: 198.61.254.9 X-Mailgun-Sid: WyI0MWYwYSIsICJsaW51eC1rZXJuZWxAdmdlci5rZXJuZWwub3JnIiwgImJlOWU0YSJd Received: from smtp.codeaurora.org (ec2-35-166-182-171.us-west-2.compute.amazonaws.com [35.166.182.171]) by smtp-out-n04.prod.us-west-2.postgun.com with SMTP id 6067535a9a9ff96d9591b230 (version=TLS1.2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256); Fri, 02 Apr 2021 17:24:42 GMT Sender: manafm=codeaurora.org@mg.codeaurora.org Received: by smtp.codeaurora.org (Postfix, from userid 1001) id 0B1EAC43462; Fri, 2 Apr 2021 17:24:41 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-caf-mail-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-2.9 required=2.0 tests=ALL_TRUSTED,BAYES_00 autolearn=unavailable autolearn_force=no version=3.4.0 Received: from mail.codeaurora.org (localhost.localdomain [127.0.0.1]) (using TLSv1 with cipher ECDHE-RSA-AES256-SHA (256/256 bits)) (No client certificate requested) (Authenticated sender: manafm) by smtp.codeaurora.org (Postfix) with ESMTPSA id 9C112C433CA; Fri, 2 Apr 2021 17:24:40 +0000 (UTC) MIME-Version: 1.0 Content-Type: text/plain; charset=US-ASCII; format=flowed Content-Transfer-Encoding: 7bit Date: Fri, 02 Apr 2021 22:54:40 +0530 From: manafm@codeaurora.org To: AngeloGioacchino Del Regno Cc: linux-arm-msm@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, Andy Gross , Bjorn Andersson , Rob Herring , Ram Chandrasekar , Taniya Das , Akhil P Oommen Subject: Re: [RFC PATCH 2/2] drivers: soc: qcom: Add SDPM clock monitor driver In-Reply-To: <56471ee3-5bde-4169-2fb9-f33dd144ae48@somainline.org> References: <1617346747-8611-1-git-send-email-manafm@codeaurora.org> <1617346747-8611-3-git-send-email-manafm@codeaurora.org> <56471ee3-5bde-4169-2fb9-f33dd144ae48@somainline.org> Message-ID: <4460ccd912fb3ec805b9a050c7444347@codeaurora.org> X-Sender: manafm@codeaurora.org User-Agent: Roundcube Webmail/1.3.9 Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 2021-04-02 15:49, AngeloGioacchino Del Regno wrote: > Il 02/04/21 08:59, Manaf Meethalavalappu Pallikunhi ha scritto: >> Add SDPM clock monitor driver, which will register for clock rate >> change notification and write the clock rate into SDPM CSR register. >> >> Signed-off-by: Ram Chandrasekar >> Signed-off-by: Manaf Meethalavalappu Pallikunhi >> >> --- >> drivers/soc/qcom/Kconfig | 8 ++ >> drivers/soc/qcom/Makefile | 1 + >> drivers/soc/qcom/sdpm_clk_monitor.c | 217 >> ++++++++++++++++++++++++++++++++++++ >> 3 files changed, 226 insertions(+) >> create mode 100644 drivers/soc/qcom/sdpm_clk_monitor.c >> >> diff --git a/drivers/soc/qcom/Kconfig b/drivers/soc/qcom/Kconfig >> index 9464ff4..1f04d69 100644 >> --- a/drivers/soc/qcom/Kconfig >> +++ b/drivers/soc/qcom/Kconfig >> @@ -237,4 +237,12 @@ config QCOM_APR >> used by audio driver to configure QDSP6 >> ASM, ADM and AFE modules. >> +config QCOM_SDPM_CLOCK_MONITOR >> + tristate "Qualcomm SDPM Clock Monitor" >> + depends on COMMON_CLK >> + help >> + This enables the Qualcomm SDPM Clock Monitor. This driver can >> register >> + for different clock rate change notifications and write the >> clock >> + rate into the SDPM CSR register. This driver will receive the >> clock >> + list and the CSR details from devicetree. >> endmenu >> diff --git a/drivers/soc/qcom/Makefile b/drivers/soc/qcom/Makefile >> index d658a10..4eef767 100644 >> --- a/drivers/soc/qcom/Makefile >> +++ b/drivers/soc/qcom/Makefile >> @@ -29,3 +29,4 @@ obj-$(CONFIG_QCOM_RPMPD) += rpmpd.o >> obj-$(CONFIG_QCOM_KRYO_L2_ACCESSORS) += kryo-l2-accessors.o >> obj-$(CONFIG_QCOM_LLCC_PERFMON) += llcc_perfmon.o >> obj-$(CONFIG_QCOM_MEMORY_DUMP_V2) += memory_dump_v2.o >> +obj-$(CONFIG_QCOM_SDPM_CLOCK_MONITOR) += sdpm_clk_monitor.o >> diff --git a/drivers/soc/qcom/sdpm_clk_monitor.c >> b/drivers/soc/qcom/sdpm_clk_monitor.c >> new file mode 100644 >> index 00000000..1aee119 >> --- /dev/null >> +++ b/drivers/soc/qcom/sdpm_clk_monitor.c >> @@ -0,0 +1,217 @@ >> +// SPDX-License-Identifier: GPL-2.0-only >> +/* >> + * Copyright (c) 2021, The Linux Foundation. All rights reserved. >> + */ >> + >> +#define pr_fmt(fmt) "%s " fmt, KBUILD_MODNAME >> + >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> + >> +#define SDPM_DRIVER "sdpm-clk-monitor" >> +#define CSR_MAX_VAL 7 >> +#define CSR_OFFSET 0xF00 >> +#define FREQ_HZ_TO_MHZ(f) DIV_ROUND_UP((f), 1000000) >> +#define SDPM_CSR_OFFSET(id) (CSR_OFFSET + (id * 4)) >> + >> +struct sdpm_clk_instance; >> +struct sdpm_clk_data { >> + struct list_head sdpm_node; >> + struct notifier_block clk_rate_nb; >> + struct clk *clk; >> + const char *clock_name; >> + uint32_t csr_id; >> + struct sdpm_clk_instance *sdpm_inst; >> +}; >> + >> +struct sdpm_clk_instance { >> + struct device *dev; >> + void __iomem *regmap; >> + struct list_head sdpm_instances; >> +}; >> + >> +static void sdpm_csr_write(struct sdpm_clk_data *sdpm_data, >> + unsigned long clk_rate) >> +{ >> + struct sdpm_clk_instance *sdpm_inst = sdpm_data->sdpm_inst; >> + >> + writel_relaxed(clk_rate, sdpm_inst->regmap + >> + SDPM_CSR_OFFSET(sdpm_data->csr_id)); >> + dev_dbg(sdpm_inst->dev, "clock:%s offset:0x%x frequency:%u\n", >> + sdpm_data->clock_name, >> + SDPM_CSR_OFFSET(sdpm_data->csr_id), >> + clk_rate); >> +} >> + >> +static int sdpm_clock_notifier(struct notifier_block *nb, >> + unsigned long event, void *data) >> +{ >> + struct clk_notifier_data *ndata = data; >> + struct sdpm_clk_data *sdpm_data = container_of(nb, >> + struct sdpm_clk_data, clk_rate_nb); >> + >> + dev_dbg(sdpm_data->sdpm_inst->dev, "clock:%s event:%lu\n", >> + sdpm_data->clock_name, event); >> + switch (event) { >> + case PRE_RATE_CHANGE: >> + if (ndata->new_rate > ndata->old_rate) >> + sdpm_csr_write(sdpm_data, >> + FREQ_HZ_TO_MHZ(ndata->new_rate)); >> + return NOTIFY_DONE; >> + case POST_RATE_CHANGE: >> + if (ndata->new_rate < ndata->old_rate) >> + sdpm_csr_write(sdpm_data, >> + FREQ_HZ_TO_MHZ(ndata->new_rate)); >> + return NOTIFY_DONE; >> + case ABORT_RATE_CHANGE: >> + if (ndata->new_rate > ndata->old_rate) >> + sdpm_csr_write(sdpm_data, >> + FREQ_HZ_TO_MHZ(ndata->old_rate)); >> + return NOTIFY_DONE; >> + default: >> + return NOTIFY_DONE; >> + } >> +} >> + >> +static int sdpm_clk_device_remove(struct platform_device *pdev) >> +{ >> + struct sdpm_clk_instance *sdpm_inst = >> + (struct sdpm_clk_instance *)dev_get_drvdata(&pdev->dev); >> + struct sdpm_clk_data *sdpm_data = NULL, *next = NULL; >> + >> + list_for_each_entry_safe(sdpm_data, next, >> + &sdpm_inst->sdpm_instances, sdpm_node) { >> + clk_notifier_unregister(sdpm_data->clk, >> + &sdpm_data->clk_rate_nb); >> + list_del(&sdpm_data->sdpm_node); >> + } >> + >> + return 0; >> +} >> + >> +static int sdpm_clk_device_probe(struct platform_device *pdev) >> +{ >> + struct device *dev = &pdev->dev; >> + struct device_node *dev_node = dev->of_node; >> + int ret = 0, idx = 0, clk_ct = 0, csr = 0, csr_ct = 0; >> + struct sdpm_clk_instance *sdpm_inst = NULL; >> + struct sdpm_clk_data *sdpm_data = NULL; >> + struct resource *res; >> + >> + sdpm_inst = devm_kzalloc(dev, sizeof(*sdpm_inst), GFP_KERNEL); >> + if (!sdpm_inst) >> + return -ENOMEM; >> + sdpm_inst->dev = dev; >> + >> + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); >> + if (!res) { >> + dev_err(dev, "Couldn't get MEM resource\n"); >> + return -EINVAL; >> + } >> + dev_dbg(dev, "sdpm@0x%x size:%d\n", res->start, >> + resource_size(res)); >> + dev_set_drvdata(dev, sdpm_inst); >> + >> + sdpm_inst->regmap = devm_ioremap_resource(dev, res); >> + if (!sdpm_inst->regmap) { >> + dev_err(dev, "Couldn't get regmap\n"); >> + return -EINVAL; >> + } >> + >> + ret = of_property_count_strings(dev_node, "clock-names"); >> + if (ret <= 0) { >> + dev_err(dev, "Couldn't get clock names. %d\n", ret); >> + return ret; >> + } >> + clk_ct = ret; >> + ret = of_property_count_u32_elems(dev_node, "csr-id"); >> + if (ret <= 0) { >> + dev_err(dev, "Couldn't get csr ID array. %d\n", ret); >> + return ret; >> + } >> + csr_ct = ret; >> + >> + if (csr_ct != clk_ct) { >> + dev_err(dev, "Invalid csr:%d and clk:%d count.\n", csr_ct, >> + clk_ct); >> + return -EINVAL; >> + } >> + >> + INIT_LIST_HEAD(&sdpm_inst->sdpm_instances); >> + >> + for (idx = 0; idx < clk_ct; idx++) { >> + >> + sdpm_data = devm_kzalloc(dev, sizeof(*sdpm_data), GFP_KERNEL); >> + if (!sdpm_data) { >> + ret = -ENOMEM; >> + goto clk_err_exit; >> + } >> + >> + ret = of_property_read_string_index(dev_node, "clock-names", >> + idx, &sdpm_data->clock_name); >> + if (ret < 0) { >> + dev_err(dev, "Couldn't get clk name index:%d. %d\n", >> + idx, ret); >> + goto clk_err_exit; >> + } >> + >> + sdpm_data->clk = devm_clk_get(dev, sdpm_data->clock_name); >> + if (IS_ERR(sdpm_data->clk)) { >> + ret = PTR_ERR(sdpm_data->clk); >> + goto clk_err_exit; >> + } >> + >> + ret = of_property_read_u32_index(dev_node, "csr-id", idx, &csr); >> + if (ret < 0) { >> + dev_err(dev, "Couldn't get CSR for index:%d. %d\n", >> + idx, ret); >> + goto clk_err_exit; >> + } >> + >> + if (ret > CSR_MAX_VAL) { >> + dev_err(dev, "Invalid CSR %d\n", csr); >> + ret = -EINVAL; >> + goto clk_err_exit; >> + } >> + >> + dev_dbg(dev, "SDPM clock:%s csr:%d initialized\n", >> + sdpm_data->clock_name, csr); >> + sdpm_data->csr_id = csr; >> + sdpm_data->sdpm_inst = sdpm_inst; >> + sdpm_data->clk_rate_nb.notifier_call = sdpm_clock_notifier; >> + sdpm_csr_write(sdpm_data, >> + FREQ_HZ_TO_MHZ(clk_get_rate(sdpm_data->clk))); >> + clk_notifier_register(sdpm_data->clk, &sdpm_data->clk_rate_nb); >> + list_add(&sdpm_data->sdpm_node, &sdpm_inst->sdpm_instances); >> + } >> + >> + return 0; >> + >> +clk_err_exit: >> + sdpm_clk_device_remove(pdev); >> + >> + return ret; >> +} >> + >> +static const struct of_device_id sdpm_clk_device_match[] = { >> + {.compatible = "qcom,sdpm"}, >> + {} >> +}; >> + >> +static struct platform_driver sdpm_clk_device_driver = { >> + .probe = sdpm_clk_device_probe, >> + .remove = sdpm_clk_device_remove, >> + .driver = { >> + .name = SDPM_DRIVER, >> + .of_match_table = sdpm_clk_device_match, >> + }, >> +}; >> +module_platform_driver(sdpm_clk_device_driver); >> + >> +MODULE_DESCRIPTION("Qualcomm Technologies, Inc. SDPM Clock Monitor >> Driver"); >> +MODULE_LICENSE("GPL v2"); >> > Hello! > > Thanks for the patch. A few things to point out: > > Before getting into a big lack of time situation, I have written a > functional driver for the Qualcomm System Performance Dynamic > Monitoring (SPDM), which has to yet be cleaned up before sending. > This driver appears to be "SDPM" instead, but the finalities of this > new block seem to be very similar - if not the same - as the SPDM. > > My SPDM implementation was done as an interconnect driver, as that is > actually controlling the bandwidth of the CCI on the SoC. > > > Is this block doing the same and/or anything similar? > In case it does, I suggest to look at my *still a bit dirty* (sorry!) > driver for the SPDM-TZ. > > Please, look at the last three commits from this branch: > https://github.com/SoMainline/linux/commits/angelo/v5.12-rc4-feat-spdmtz > > Yours, > - Angelo Hi Angelo, I have gone through above commit and description. I don't see any relation with this SDPM (Simple Digital power meter) clock monitor driver. The SPDM (System Performance Dynamic Monitoring) is trying to control CCI bandwidth. But this driver monitors different IPs operating clock those are handled by linux clock framework and informs the same to Qualcomm RDPM hardware. Thank You, Manaf