[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <04d08bf8-d5a2-4262-8210-b5173b8c613b@linaro.org>
Date: Thu, 23 Jan 2025 17:15:31 +0100
From: Neil Armstrong <neil.armstrong@...aro.org>
To: chuan.liu@...ogic.com, Kevin Hilman <khilman@...libre.com>,
Jerome Brunet <jbrunet@...libre.com>,
Martin Blumenstingl <martin.blumenstingl@...glemail.com>
Cc: linux-arm-kernel@...ts.infradead.org, linux-amlogic@...ts.infradead.org,
linux-kernel@...r.kernel.org
Subject: Re: [PATCH] soc: amlogic: clk-measure: Optimize the memory size of
clk-measure
Hi,
On 23/01/2025 11:37, Chuan Liu via B4 Relay wrote:
> From: Chuan Liu <chuan.liu@...ogic.com>
>
> Define struct meson_msr as a static global variable and remove the
> "*priv" member from struct meson_msr_id.
>
> Define the size of the corresponding array based on the actual number of
> msr_id of the chip.
>
> The array corresponding to msr_id is defined as "__initdata" to reduce
> memory usage.
>
> Signed-off-by: Chuan Liu <chuan.liu@...ogic.com>
> ---
> Each msr_id defines a pointer(*priv), and all these pointers point to
> the same address.
>
> The number of msr_ids for each chip is inconsistent. Defining a
> fixed-size array for each chip to store msr_ids would waste memory.
> ---
> drivers/soc/amlogic/meson-clk-measure.c | 119 ++++++++++++++++++++------------
> 1 file changed, 74 insertions(+), 45 deletions(-)
>
> diff --git a/drivers/soc/amlogic/meson-clk-measure.c b/drivers/soc/amlogic/meson-clk-measure.c
> index a6453ffeb753..b52e9ce25ea8 100644
> --- a/drivers/soc/amlogic/meson-clk-measure.c
> +++ b/drivers/soc/amlogic/meson-clk-measure.c
> @@ -33,23 +33,27 @@ static DEFINE_MUTEX(measure_lock);
> #define DIV_STEP 32
> #define DIV_MAX 640
>
> -#define CLK_MSR_MAX 128
> -
> struct meson_msr_id {
> - struct meson_msr *priv;
> unsigned int id;
> const char *name;
> };
>
> +struct meson_msr_data {
> + struct meson_msr_id *msr_table;
> + unsigned int msr_count;
> +};
> +
> struct meson_msr {
> struct regmap *regmap;
> - struct meson_msr_id msr_table[CLK_MSR_MAX];
> + struct meson_msr_data data;
> };
>
> #define CLK_MSR_ID(__id, __name) \
> [__id] = {.id = __id, .name = __name,}
>
> -static struct meson_msr_id clk_msr_m8[CLK_MSR_MAX] = {
> +static struct meson_msr meson_msr;
The whole architecture was to avoid a global variable like this
> +
> +static struct meson_msr_id clk_msr_m8[] __initdata = {
> CLK_MSR_ID(0, "ring_osc_out_ee0"),
> CLK_MSR_ID(1, "ring_osc_out_ee1"),
> CLK_MSR_ID(2, "ring_osc_out_ee2"),
> @@ -98,7 +102,7 @@ static struct meson_msr_id clk_msr_m8[CLK_MSR_MAX] = {
> CLK_MSR_ID(63, "mipi_csi_cfg"),
> };
>
> -static struct meson_msr_id clk_msr_gx[CLK_MSR_MAX] = {
> +static struct meson_msr_id clk_msr_gx[] __initdata = {
> CLK_MSR_ID(0, "ring_osc_out_ee_0"),
> CLK_MSR_ID(1, "ring_osc_out_ee_1"),
> CLK_MSR_ID(2, "ring_osc_out_ee_2"),
> @@ -168,7 +172,7 @@ static struct meson_msr_id clk_msr_gx[CLK_MSR_MAX] = {
> CLK_MSR_ID(82, "ge2d"),
> };
>
> -static struct meson_msr_id clk_msr_axg[CLK_MSR_MAX] = {
> +static struct meson_msr_id clk_msr_axg[] __initdata = {
> CLK_MSR_ID(0, "ring_osc_out_ee_0"),
> CLK_MSR_ID(1, "ring_osc_out_ee_1"),
> CLK_MSR_ID(2, "ring_osc_out_ee_2"),
> @@ -242,7 +246,7 @@ static struct meson_msr_id clk_msr_axg[CLK_MSR_MAX] = {
> CLK_MSR_ID(109, "audio_locker_in"),
> };
>
> -static struct meson_msr_id clk_msr_g12a[CLK_MSR_MAX] = {
> +static struct meson_msr_id clk_msr_g12a[] __initdata = {
> CLK_MSR_ID(0, "ring_osc_out_ee_0"),
> CLK_MSR_ID(1, "ring_osc_out_ee_1"),
> CLK_MSR_ID(2, "ring_osc_out_ee_2"),
> @@ -358,7 +362,7 @@ static struct meson_msr_id clk_msr_g12a[CLK_MSR_MAX] = {
> CLK_MSR_ID(122, "audio_pdm_dclk"),
> };
>
> -static struct meson_msr_id clk_msr_sm1[CLK_MSR_MAX] = {
> +static struct meson_msr_id clk_msr_sm1[] __initdata = {
> CLK_MSR_ID(0, "ring_osc_out_ee_0"),
> CLK_MSR_ID(1, "ring_osc_out_ee_1"),
> CLK_MSR_ID(2, "ring_osc_out_ee_2"),
> @@ -489,9 +493,8 @@ static struct meson_msr_id clk_msr_sm1[CLK_MSR_MAX] = {
> };
>
> static int meson_measure_id(struct meson_msr_id *clk_msr_id,
> - unsigned int duration)
> + unsigned int duration)
> {
> - struct meson_msr *priv = clk_msr_id->priv;
> unsigned int val;
> int ret;
>
> @@ -499,22 +502,22 @@ static int meson_measure_id(struct meson_msr_id *clk_msr_id,
> if (ret)
> return ret;
>
> - regmap_write(priv->regmap, MSR_CLK_REG0, 0);
> + regmap_write(meson_msr.regmap, MSR_CLK_REG0, 0);
>
> /* Set measurement duration */
> - regmap_update_bits(priv->regmap, MSR_CLK_REG0, MSR_DURATION,
> + regmap_update_bits(meson_msr.regmap, MSR_CLK_REG0, MSR_DURATION,
> FIELD_PREP(MSR_DURATION, duration - 1));
>
> /* Set ID */
> - regmap_update_bits(priv->regmap, MSR_CLK_REG0, MSR_CLK_SRC,
> + regmap_update_bits(meson_msr.regmap, MSR_CLK_REG0, MSR_CLK_SRC,
> FIELD_PREP(MSR_CLK_SRC, clk_msr_id->id));
>
> /* Enable & Start */
> - regmap_update_bits(priv->regmap, MSR_CLK_REG0,
> + regmap_update_bits(meson_msr.regmap, MSR_CLK_REG0,
> MSR_RUN | MSR_ENABLE,
> MSR_RUN | MSR_ENABLE);
>
> - ret = regmap_read_poll_timeout(priv->regmap, MSR_CLK_REG0,
> + ret = regmap_read_poll_timeout(meson_msr.regmap, MSR_CLK_REG0,
> val, !(val & MSR_BUSY), 10, 10000);
> if (ret) {
> mutex_unlock(&measure_lock);
> @@ -522,10 +525,10 @@ static int meson_measure_id(struct meson_msr_id *clk_msr_id,
> }
>
> /* Disable */
> - regmap_update_bits(priv->regmap, MSR_CLK_REG0, MSR_ENABLE, 0);
> + regmap_update_bits(meson_msr.regmap, MSR_CLK_REG0, MSR_ENABLE, 0);
>
> /* Get the value in multiple of gate time counts */
> - regmap_read(priv->regmap, MSR_CLK_REG2, &val);
> + regmap_read(meson_msr.regmap, MSR_CLK_REG2, &val);
>
> mutex_unlock(&measure_lock);
>
> @@ -579,7 +582,7 @@ static int clk_msr_summary_show(struct seq_file *s, void *data)
> seq_puts(s, " clock rate precision\n");
> seq_puts(s, "---------------------------------------------\n");
>
> - for (i = 0 ; i < CLK_MSR_MAX ; ++i) {
> + for (i = 0 ; i < meson_msr.data.msr_count ; ++i) {
> if (!msr_table[i].name)
> continue;
>
> @@ -604,77 +607,103 @@ static const struct regmap_config meson_clk_msr_regmap_config = {
>
> static int meson_msr_probe(struct platform_device *pdev)
> {
> - const struct meson_msr_id *match_data;
> - struct meson_msr *priv;
> + const struct meson_msr_data *match_data;
> + struct meson_msr_id *msr_table;
> struct dentry *root, *clks;
> void __iomem *base;
> int i;
>
> - priv = devm_kzalloc(&pdev->dev, sizeof(struct meson_msr),
> - GFP_KERNEL);
> - if (!priv)
> - return -ENOMEM;
> -
> match_data = device_get_match_data(&pdev->dev);
> if (!match_data) {
> dev_err(&pdev->dev, "failed to get match data\n");
> return -ENODEV;
> }
>
> - memcpy(priv->msr_table, match_data, sizeof(priv->msr_table));
> + msr_table = devm_kcalloc(&pdev->dev, match_data->msr_count,
> + sizeof(struct meson_msr_id), GFP_KERNEL);
> + if (!msr_table)
> + return -ENOMEM;
> +
> + memcpy(msr_table, match_data->msr_table,
> + match_data->msr_count * sizeof(struct meson_msr_id));
> + meson_msr.data.msr_table = msr_table;
> + meson_msr.data.msr_count = match_data->msr_count;
>
> base = devm_platform_ioremap_resource(pdev, 0);
> if (IS_ERR(base))
> return PTR_ERR(base);
>
> - priv->regmap = devm_regmap_init_mmio(&pdev->dev, base,
> - &meson_clk_msr_regmap_config);
> - if (IS_ERR(priv->regmap))
> - return PTR_ERR(priv->regmap);
> + meson_msr.regmap = devm_regmap_init_mmio(&pdev->dev, base,
> + &meson_clk_msr_regmap_config);
> + if (IS_ERR(meson_msr.regmap))
> + return PTR_ERR(meson_msr.regmap);
>
> root = debugfs_create_dir("meson-clk-msr", NULL);
> clks = debugfs_create_dir("clks", root);
>
> - debugfs_create_file("measure_summary", 0444, root,
> - priv->msr_table, &clk_msr_summary_fops);
> + debugfs_create_file("measure_summary", 0444, root, msr_table,
> + &clk_msr_summary_fops);
>
> - for (i = 0 ; i < CLK_MSR_MAX ; ++i) {
> - if (!priv->msr_table[i].name)
> + for (i = 0 ; i < meson_msr.data.msr_count ; ++i) {
> + if (!msr_table[i].name)
> continue;
>
> - priv->msr_table[i].priv = priv;
> -
> - debugfs_create_file(priv->msr_table[i].name, 0444, clks,
> - &priv->msr_table[i], &clk_msr_fops);
> + debugfs_create_file(msr_table[i].name, 0444, clks,
> + &msr_table[i], &clk_msr_fops);
> }
>
> return 0;
> }
>
> +static const struct meson_msr_data clk_msr_gx_data = {
> + .msr_table = clk_msr_gx,
> + .msr_count = ARRAY_SIZE(clk_msr_gx),
> +};
> +
> +static const struct meson_msr_data clk_msr_m8_data = {
> + .msr_table = clk_msr_m8,
> + .msr_count = ARRAY_SIZE(clk_msr_m8),
> +};
> +
> +static const struct meson_msr_data clk_msr_axg_data = {
> + .msr_table = clk_msr_axg,
> + .msr_count = ARRAY_SIZE(clk_msr_axg),
> +};
> +
> +static const struct meson_msr_data clk_msr_g12a_data = {
> + .msr_table = clk_msr_g12a,
> + .msr_count = ARRAY_SIZE(clk_msr_g12a),
> +};
> +
> +static const struct meson_msr_data clk_msr_sm1_data = {
> + .msr_table = clk_msr_sm1,
> + .msr_count = ARRAY_SIZE(clk_msr_sm1),
> +};
> +
> static const struct of_device_id meson_msr_match_table[] = {
> {
> .compatible = "amlogic,meson-gx-clk-measure",
> - .data = (void *)clk_msr_gx,
> + .data = &clk_msr_gx_data,
> },
> {
> .compatible = "amlogic,meson8-clk-measure",
> - .data = (void *)clk_msr_m8,
> + .data = &clk_msr_m8_data,
> },
> {
> .compatible = "amlogic,meson8b-clk-measure",
> - .data = (void *)clk_msr_m8,
> + .data = &clk_msr_m8_data,
> },
> {
> .compatible = "amlogic,meson-axg-clk-measure",
> - .data = (void *)clk_msr_axg,
> + .data = &clk_msr_axg_data,
> },
> {
> .compatible = "amlogic,meson-g12a-clk-measure",
> - .data = (void *)clk_msr_g12a,
> + .data = &clk_msr_g12a_data,
> },
> {
> .compatible = "amlogic,meson-sm1-clk-measure",
> - .data = (void *)clk_msr_sm1,
> + .data = &clk_msr_sm1_data,
> },
> { /* sentinel */ }
> };
>
> ---
> base-commit: 1e1fd26ed4ca05cc1f0e5857918da4dd54967f7d
> change-id: 20250123-optimize_memory_size_of_clk_measure-f9c40e815794
>
> Best regards,
I think the goal is fine, but we should avoid any global variable to store
the state and avoid initdata, but yes I agree to drop the fixed size tables
and store the table size size in the compatible data.
The solution would be to:
- drop the MAX like you did
- add a msr_count with ARRAY_SIZE like you did
- but mark the compat data and table as const
- in probe, allocate priv with a large table inside, & copy the data into it
This should probably work and we could move the tables to read-only section.
Neil
Powered by blists - more mailing lists