From: Konrad Dybcio <konrad.dybcio@linaro.org>
To: Dmitry Baryshkov <dmitry.baryshkov@linaro.org>,
Rob Herring <robh+dt@kernel.org>,
Krzysztof Kozlowski <krzysztof.kozlowski+dt@linaro.org>,
Conor Dooley <conor+dt@kernel.org>,
Andy Gross <agross@kernel.org>,
Bjorn Andersson <andersson@kernel.org>,
Ilia Lin <ilia.lin@kernel.org>, Viresh Kumar <vireshk@kernel.org>,
Nishanth Menon <nm@ti.com>, Stephen Boyd <sboyd@kernel.org>,
Michael Turquette <mturquette@baylibre.com>,
"Rafael J. Wysocki" <rafael@kernel.org>,
Georgi Djakov <djakov@kernel.org>
Cc: linux-arm-msm@vger.kernel.org, devicetree@vger.kernel.org,
linux-pm@vger.kernel.org, linux-clk@vger.kernel.org,
Christian Marangi <ansuelsmth@gmail.com>,
Stephan Gerhold <stephan@gerhold.net>
Subject: Re: [PATCH v3 08/28] soc: qcom: spm: add support for voltage regulator
Date: Mon, 3 Jul 2023 13:21:23 +0200 [thread overview]
Message-ID: <e6eacaa5-c505-b4ab-2854-b1feec563128@linaro.org> (raw)
In-Reply-To: <20230702174246.121656-9-dmitry.baryshkov@linaro.org>
On 2.07.2023 19:42, Dmitry Baryshkov wrote:
> The SPM / SAW2 device also provides a voltage regulator functionality
> with optional AVS (Adaptive Voltage Scaling) support. The exact register
> sequence and voltage ranges differs from device to device.
>
> Signed-off-by: Dmitry Baryshkov <dmitry.baryshkov@linaro.org>
> ---
Looks like you didn't update this patch from v2
Konrad
> drivers/soc/qcom/spm.c | 205 ++++++++++++++++++++++++++++++++++++++++-
> include/soc/qcom/spm.h | 9 ++
> 2 files changed, 212 insertions(+), 2 deletions(-)
>
> diff --git a/drivers/soc/qcom/spm.c b/drivers/soc/qcom/spm.c
> index a6cbeb40831b..3c16a7e1710c 100644
> --- a/drivers/soc/qcom/spm.c
> +++ b/drivers/soc/qcom/spm.c
> @@ -9,19 +9,31 @@
> #include <linux/kernel.h>
> #include <linux/init.h>
> #include <linux/io.h>
> +#include <linux/iopoll.h>
> +#include <linux/linear_range.h>
> #include <linux/module.h>
> #include <linux/slab.h>
> #include <linux/of.h>
> #include <linux/of_address.h>
> #include <linux/of_device.h>
> +#include <linux/bitfield.h>
> #include <linux/err.h>
> #include <linux/platform_device.h>
> +#include <linux/regulator/driver.h>
> +#include <linux/smp.h>
> #include <soc/qcom/spm.h>
>
> +#define FIELD_SET(current, mask, val) \
> + (((current) & ~(mask)) | FIELD_PREP((mask), (val)))
> +
> #define SPM_CTL_INDEX 0x7f
> #define SPM_CTL_INDEX_SHIFT 4
> #define SPM_CTL_EN BIT(0)
>
> +#define SPM_1_1_AVS_CTL_AVS_ENABLED BIT(27)
> +#define SPM_AVS_CTL_MIN_VLVL (0x3f << 10)
> +#define SPM_AVS_CTL_MAX_VLVL (0x3f << 17)
> +
> enum spm_reg {
> SPM_REG_CFG,
> SPM_REG_SPM_CTL,
> @@ -31,10 +43,12 @@ enum spm_reg {
> SPM_REG_PMIC_DATA_1,
> SPM_REG_VCTL,
> SPM_REG_SEQ_ENTRY,
> - SPM_REG_SPM_STS,
> + SPM_REG_STS0,
> + SPM_REG_STS1,
> SPM_REG_PMIC_STS,
> SPM_REG_AVS_CTL,
> SPM_REG_AVS_LIMIT,
> + SPM_REG_RST,
> SPM_REG_NR,
> };
>
> @@ -171,6 +185,10 @@ static const struct spm_reg_data spm_reg_8226_cpu = {
>
> static const u16 spm_reg_offset_v1_1[SPM_REG_NR] = {
> [SPM_REG_CFG] = 0x08,
> + [SPM_REG_STS0] = 0x0c,
> + [SPM_REG_STS1] = 0x10,
> + [SPM_REG_VCTL] = 0x14,
> + [SPM_REG_AVS_CTL] = 0x18,
> [SPM_REG_SPM_CTL] = 0x20,
> [SPM_REG_PMIC_DLY] = 0x24,
> [SPM_REG_PMIC_DATA_0] = 0x28,
> @@ -178,7 +196,12 @@ static const u16 spm_reg_offset_v1_1[SPM_REG_NR] = {
> [SPM_REG_SEQ_ENTRY] = 0x80,
> };
>
> +static void smp_set_vdd_v1_1(void *data);
> +
> /* SPM register data for 8064 */
> +static struct linear_range spm_v1_1_regulator_range =
> + REGULATOR_LINEAR_RANGE(700000, 0, 56, 12500);
> +
> static const struct spm_reg_data spm_reg_8064_cpu = {
> .reg_offset = spm_reg_offset_v1_1,
> .spm_cfg = 0x1F,
> @@ -189,6 +212,10 @@ static const struct spm_reg_data spm_reg_8064_cpu = {
> 0x10, 0x54, 0x30, 0x0C, 0x24, 0x30, 0x0F },
> .start_index[PM_SLEEP_MODE_STBY] = 0,
> .start_index[PM_SLEEP_MODE_SPC] = 2,
> + .set_vdd = smp_set_vdd_v1_1,
> + .range = &spm_v1_1_regulator_range,
> + .init_uV = 1300000,
> + .ramp_delay = 1250,
> };
>
> static inline void spm_register_write(struct spm_driver_data *drv,
> @@ -240,6 +267,179 @@ void spm_set_low_power_mode(struct spm_driver_data *drv,
> spm_register_write_sync(drv, SPM_REG_SPM_CTL, ctl_val);
> }
>
> +static int spm_set_voltage_sel(struct regulator_dev *rdev, unsigned int selector)
> +{
> + struct spm_driver_data *drv = rdev_get_drvdata(rdev);
> +
> + drv->volt_sel = selector;
> +
> + /* Always do the SAW register writes on the corresponding CPU */
> + return smp_call_function_single(drv->reg_cpu, drv->reg_data->set_vdd, drv, true);
> +}
> +
> +static int spm_get_voltage_sel(struct regulator_dev *rdev)
> +{
> + struct spm_driver_data *drv = rdev_get_drvdata(rdev);
> +
> + return drv->volt_sel;
> +}
> +
> +static const struct regulator_ops spm_reg_ops = {
> + .set_voltage_sel = spm_set_voltage_sel,
> + .get_voltage_sel = spm_get_voltage_sel,
> + .list_voltage = regulator_list_voltage_linear_range,
> + .set_voltage_time_sel = regulator_set_voltage_time_sel,
> +};
> +
> +static void smp_set_vdd_v1_1(void *data)
> +{
> + struct spm_driver_data *drv = data;
> + unsigned int vlevel = drv->volt_sel;
> + unsigned int vctl, data0, data1, avs_ctl, sts;
> + bool avs_enabled;
> +
> + vlevel |= 0x80; /* band */
> +
> + avs_ctl = spm_register_read(drv, SPM_REG_AVS_CTL);
> + vctl = spm_register_read(drv, SPM_REG_VCTL);
> + data0 = spm_register_read(drv, SPM_REG_PMIC_DATA_0);
> + data1 = spm_register_read(drv, SPM_REG_PMIC_DATA_1);
> +
> + avs_enabled = avs_ctl & SPM_1_1_AVS_CTL_AVS_ENABLED;
> +
> + /* If AVS is enabled, switch it off during the voltage change */
> + if (avs_enabled) {
> + avs_ctl &= ~SPM_1_1_AVS_CTL_AVS_ENABLED;
> + spm_register_write(drv, SPM_REG_AVS_CTL, avs_ctl);
> + }
> +
> + /* Kick the state machine back to idle */
> + spm_register_write(drv, SPM_REG_RST, 1);
> +
> + vctl = FIELD_SET(vctl, 0xff, vlevel);
> + data0 = FIELD_SET(data0, 0xff, vlevel);
> + data1 = FIELD_SET(data1, 0x3f, vlevel);
> + data1 = FIELD_SET(data1, 0x3f << 16, vlevel);
> +
> + spm_register_write(drv, SPM_REG_VCTL, vctl);
> + spm_register_write(drv, SPM_REG_PMIC_DATA_0, data0);
> + spm_register_write(drv, SPM_REG_PMIC_DATA_1, data1);
> +
> + if (read_poll_timeout_atomic(spm_register_read,
> + sts, sts == vlevel,
> + 1, 200, false,
> + drv, SPM_REG_STS1)) {
> + dev_err_ratelimited(drv->dev, "timeout setting the voltage (%x %x)!\n", sts, vlevel);
> + goto enable_avs;
> + }
> +
> + if (avs_enabled) {
> + unsigned int max_avs = vlevel & 0x3f;
> + unsigned int min_avs = max(max_avs, 4U) - 4;
> + avs_ctl = FIELD_SET(avs_ctl, SPM_AVS_CTL_MIN_VLVL, min_avs);
> + avs_ctl = FIELD_SET(avs_ctl, SPM_AVS_CTL_MAX_VLVL, max_avs);
> + spm_register_write(drv, SPM_REG_AVS_CTL, avs_ctl);
> + }
> +
> +enable_avs:
> + if (avs_enabled) {
> + avs_ctl |= SPM_1_1_AVS_CTL_AVS_ENABLED;
> + spm_register_write(drv, SPM_REG_AVS_CTL, avs_ctl);
> + }
> +}
> +
> +static int spm_get_cpu(struct device *dev)
> +{
> + int cpu;
> + bool found;
> +
> + for_each_possible_cpu(cpu) {
> + struct device_node *cpu_node, *saw_node;
> +
> + cpu_node = of_cpu_device_node_get(cpu);
> + if (!cpu_node)
> + continue;
> +
> + saw_node = of_parse_phandle(cpu_node, "qcom,saw", 0);
> + found = (saw_node == dev->of_node);
> + of_node_put(saw_node);
> + of_node_put(cpu_node);
> +
> + if (found)
> + return cpu;
> + }
> +
> + /* L2 SPM is not bound to any CPU, tie it to CPU0 */
> +
> + return 0;
> +}
> +
> +#ifdef CONFIG_REGULATOR
> +static int spm_register_regulator(struct device *dev, struct spm_driver_data *drv)
> +{
> + struct regulator_config config = {
> + .dev = dev,
> + .driver_data = drv,
> + };
> + struct regulator_desc *rdesc;
> + struct regulator_dev *rdev;
> + int ret;
> + bool found;
> +
> + if (!drv->reg_data->set_vdd)
> + return 0;
> +
> + rdesc = devm_kzalloc(dev, sizeof(*rdesc), GFP_KERNEL);
> + if (!rdesc)
> + return -ENOMEM;
> +
> + rdesc->name = "spm";
> + rdesc->of_match = of_match_ptr("regulator");
> + rdesc->type = REGULATOR_VOLTAGE;
> + rdesc->owner = THIS_MODULE;
> + rdesc->ops = &spm_reg_ops;
> +
> + rdesc->linear_ranges = drv->reg_data->range;
> + rdesc->n_linear_ranges = 1;
> + rdesc->n_voltages = rdesc->linear_ranges[rdesc->n_linear_ranges - 1].max_sel + 1;
> + rdesc->ramp_delay = drv->reg_data->ramp_delay;
> +
> + drv->reg_cpu = spm_get_cpu(dev);
> + dev_dbg(dev, "SAW2 bound to CPU %d\n", drv->reg_cpu);
> +
> + /*
> + * Program initial voltage, otherwise registration will also try
> + * setting the voltage, which might result in undervolting the CPU.
> + */
> + drv->volt_sel = DIV_ROUND_UP(drv->reg_data->init_uV - rdesc->min_uV,
> + rdesc->uV_step);
> + ret = linear_range_get_selector_high(drv->reg_data->range,
> + drv->reg_data->init_uV,
> + &drv->volt_sel,
> + &found);
> + if (ret) {
> + dev_err(dev, "Initial uV value out of bounds\n");
> + return ret;
> + }
> +
> + /* Always do the SAW register writes on the corresponding CPU */
> + smp_call_function_single(drv->reg_cpu, drv->reg_data->set_vdd, drv, true);
> +
> + rdev = devm_regulator_register(dev, rdesc, &config);
> + if (IS_ERR(rdev)) {
> + dev_err(dev, "failed to register regulator\n");
> + return PTR_ERR(rdev);
> + }
> +
> + return 0;
> +}
> +#else
> +static int spm_register_regulator(struct device *dev, struct spm_driver_data *drv)
> +{
> + return 0;
> +}
> +#endif
> +
> static const struct of_device_id spm_match_table[] = {
> { .compatible = "qcom,sdm660-gold-saw2-v4.1-l2",
> .data = &spm_reg_660_gold_l2 },
> @@ -292,6 +492,7 @@ static int spm_dev_probe(struct platform_device *pdev)
> return -ENODEV;
>
> drv->reg_data = match_id->data;
> + drv->dev = &pdev->dev;
> platform_set_drvdata(pdev, drv);
>
> /* Write the SPM sequences first.. */
> @@ -319,7 +520,7 @@ static int spm_dev_probe(struct platform_device *pdev)
> if (drv->reg_data->reg_offset[SPM_REG_SPM_CTL])
> spm_set_low_power_mode(drv, PM_SLEEP_MODE_STBY);
>
> - return 0;
> + return spm_register_regulator(&pdev->dev, drv);
> }
>
> static struct platform_driver spm_driver = {
> diff --git a/include/soc/qcom/spm.h b/include/soc/qcom/spm.h
> index 4951f9d8b0bd..9859ebe42003 100644
> --- a/include/soc/qcom/spm.h
> +++ b/include/soc/qcom/spm.h
> @@ -30,11 +30,20 @@ struct spm_reg_data {
> u32 avs_limit;
> u8 seq[MAX_SEQ_DATA];
> u8 start_index[PM_SLEEP_MODE_NR];
> +
> + smp_call_func_t set_vdd;
> + /* for now we support only a single range */
> + struct linear_range *range;
> + unsigned int ramp_delay;
> + unsigned int init_uV;
> };
>
> struct spm_driver_data {
> void __iomem *reg_base;
> const struct spm_reg_data *reg_data;
> + struct device *dev;
> + unsigned int volt_sel;
> + int reg_cpu;
> };
>
> void spm_set_low_power_mode(struct spm_driver_data *drv,
next prev parent reply other threads:[~2023-07-03 11:21 UTC|newest]
Thread overview: 43+ messages / expand[flat|nested] mbox.gz Atom feed top
2023-07-02 17:42 [PATCH v3 00/28] ARM: qcom: apq8064: support CPU frequency scaling Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 01/28] dt-bindings: opp: opp-v2-kryo-cpu: support Qualcomm Krait SoCs Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 02/28] dt-bindings: soc: qcom: merge qcom,saw2.txt into qcom,spm.yaml Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 03/28] dt-bindings: soc: qcom: qcom,saw2: define optional regulator node Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 04/28] dt-bindings: clock: qcom,krait-cc: Krait core clock controller Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 05/28] dt-bindings: cache: describe L2 cache on Qualcomm Krait platforms Dmitry Baryshkov
2023-07-02 18:48 ` Rob Herring
2023-07-02 20:09 ` Krzysztof Kozlowski
2023-07-02 20:46 ` Krzysztof Kozlowski
2023-07-02 17:42 ` [PATCH v3 06/28] interconnect: icc-clk: add support for scaling using OPP Dmitry Baryshkov
2023-07-03 11:17 ` Konrad Dybcio
2023-07-02 17:42 ` [PATCH v3 07/28] clk: qcom: krait-cc: rewrite driver to use clk_hw instead of clk Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 08/28] soc: qcom: spm: add support for voltage regulator Dmitry Baryshkov
2023-07-03 11:21 ` Konrad Dybcio [this message]
2023-07-02 17:42 ` [PATCH v3 09/28] cpufreq: qcom-nvmem: create L2 cache device Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 10/28] cpufreq: qcom-nvmem: also accept operating-points-v2-krait-cpu Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 11/28] cpufreq: qcom-nvmem: drop pvs_ver for format a fuses Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 12/28] cpufreq: qcom-nvmem: provide separate configuration data for apq8064 Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 13/28] soc: qcom: Add driver for Qualcomm Krait L2 cache scaling Dmitry Baryshkov
2023-07-03 11:25 ` Konrad Dybcio
2023-07-02 17:42 ` [PATCH v3 14/28] ARM: dts: qcom: apq8064-asus-nexus7-flo: constraint cpufreq regulators Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 15/28] ARM: dts: qcom: apq8064-cm-qs600: " Dmitry Baryshkov
2023-07-03 11:26 ` Konrad Dybcio
2023-07-02 17:42 ` [PATCH v3 16/28] ARM: dts: qcom: apq8064-ifc6410: " Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 17/28] ARM: dts: qcom: apq8064-sony-xperia-lagan-yuga: " Dmitry Baryshkov
2023-07-03 11:26 ` Konrad Dybcio
2023-07-02 17:42 ` [PATCH v3 18/28] ARM: dts: qcom: apq8064: rename SAW nodes to power-manager Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 19/28] ARM: dts: qcom: apq8064: declare SAW2 regulators Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 20/28] ARM: dts: qcom: apq8064: add Krait clock controller Dmitry Baryshkov
2023-07-03 11:27 ` Konrad Dybcio
2023-07-02 17:42 ` [PATCH v3 21/28] ARM: dts: qcom: apq8064: add L2 cache scaling Dmitry Baryshkov
2023-07-02 20:09 ` Krzysztof Kozlowski
2023-07-02 17:42 ` [PATCH v3 22/28] ARM: dts: qcom: apq8064: add simple CPUFreq support Dmitry Baryshkov
2023-07-03 11:30 ` Konrad Dybcio
2023-07-02 17:42 ` [PATCH v3 23/28] ARM: dts: qcom: apq8064: enable passive CPU cooling Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 24/28] ARM: dts: qcom: msm8960: declare SAW2 regulators Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 25/28] ARM: dts: qcom: apq8084: drop 'regulator' property from SAW2 device Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 26/28] ARM: dts: qcom: msm8974: " Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 27/28] ARM: dts: qcom: ipq4019: drop 'regulator' property from SAW2 devices Dmitry Baryshkov
2023-07-02 17:42 ` [PATCH v3 28/28] ARM: dts: qcom: ipq8064: " Dmitry Baryshkov
2023-07-03 4:39 ` [PATCH v3 00/28] ARM: qcom: apq8064: support CPU frequency scaling Viresh Kumar
2023-07-07 9:34 ` Linus Walleij
2023-07-07 10:59 ` Dmitry Baryshkov
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=e6eacaa5-c505-b4ab-2854-b1feec563128@linaro.org \
--to=konrad.dybcio@linaro.org \
--cc=agross@kernel.org \
--cc=andersson@kernel.org \
--cc=ansuelsmth@gmail.com \
--cc=conor+dt@kernel.org \
--cc=devicetree@vger.kernel.org \
--cc=djakov@kernel.org \
--cc=dmitry.baryshkov@linaro.org \
--cc=ilia.lin@kernel.org \
--cc=krzysztof.kozlowski+dt@linaro.org \
--cc=linux-arm-msm@vger.kernel.org \
--cc=linux-clk@vger.kernel.org \
--cc=linux-pm@vger.kernel.org \
--cc=mturquette@baylibre.com \
--cc=nm@ti.com \
--cc=rafael@kernel.org \
--cc=robh+dt@kernel.org \
--cc=sboyd@kernel.org \
--cc=stephan@gerhold.net \
--cc=vireshk@kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox