Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

platform/x86: int3472: Support multiple clock consumers

At present, the tps68470.c only supports a single clock consumer when
passing platform data to the clock driver. In some devices multiple
sensors depend on the clock provided by a single TPS68470 and so all
need to be able to acquire the clock. Support passing multiple
consumers as platform data.

Reviewed-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Daniel Scally <djrscally@gmail.com>
Reviewed-by: Stephen Boyd <sboyd@kernel.org>
Acked-by: Stephen Boyd <sboyd@kernel.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>

authored by

Daniel Scally and committed by
Rafael J. Wysocki
43cf3697 62c8bc0d

+67 -12
+9 -4
drivers/clk/clk-tps68470.c
··· 200 200 .flags = CLK_SET_RATE_GATE, 201 201 }; 202 202 struct tps68470_clkdata *tps68470_clkdata; 203 + struct tps68470_clk_consumer *consumer; 203 204 int ret; 205 + int i; 204 206 205 207 tps68470_clkdata = devm_kzalloc(&pdev->dev, sizeof(*tps68470_clkdata), 206 208 GFP_KERNEL); ··· 225 223 return ret; 226 224 227 225 if (pdata) { 228 - ret = devm_clk_hw_register_clkdev(&pdev->dev, 229 - &tps68470_clkdata->clkout_hw, 230 - pdata->consumer_con_id, 231 - pdata->consumer_dev_name); 226 + for (i = 0; i < pdata->n_consumers; i++) { 227 + consumer = &pdata->consumers[i]; 228 + ret = devm_clk_hw_register_clkdev(&pdev->dev, 229 + &tps68470_clkdata->clkout_hw, 230 + consumer->consumer_con_id, 231 + consumer->consumer_dev_name); 232 + } 232 233 } 233 234 234 235 return ret;
+52 -7
drivers/platform/x86/intel/int3472/tps68470.c
··· 1 1 // SPDX-License-Identifier: GPL-2.0 2 2 /* Author: Dan Scally <djrscally@gmail.com> */ 3 3 4 + #include <linux/acpi.h> 4 5 #include <linux/i2c.h> 5 6 #include <linux/kernel.h> 6 7 #include <linux/mfd/core.h> ··· 96 95 return DESIGNED_FOR_WINDOWS; 97 96 } 98 97 98 + /* 99 + * Return the size of the flexible array member, because we'll need that later 100 + * on to pass .pdata_size to cells. 101 + */ 102 + static int 103 + skl_int3472_fill_clk_pdata(struct device *dev, struct tps68470_clk_platform_data **clk_pdata) 104 + { 105 + struct acpi_device *adev = ACPI_COMPANION(dev); 106 + struct acpi_device *consumer; 107 + unsigned int n_consumers = 0; 108 + const char *sensor_name; 109 + unsigned int i = 0; 110 + 111 + for_each_acpi_consumer_dev(adev, consumer) 112 + n_consumers++; 113 + 114 + if (!n_consumers) { 115 + dev_err(dev, "INT3472 seems to have no dependents\n"); 116 + return -ENODEV; 117 + } 118 + 119 + *clk_pdata = devm_kzalloc(dev, struct_size(*clk_pdata, consumers, n_consumers), 120 + GFP_KERNEL); 121 + if (!*clk_pdata) 122 + return -ENOMEM; 123 + 124 + (*clk_pdata)->n_consumers = n_consumers; 125 + i = 0; 126 + 127 + for_each_acpi_consumer_dev(adev, consumer) { 128 + sensor_name = devm_kasprintf(dev, GFP_KERNEL, I2C_DEV_NAME_FORMAT, 129 + acpi_dev_name(consumer)); 130 + if (!sensor_name) 131 + return -ENOMEM; 132 + 133 + (*clk_pdata)->consumers[i].consumer_dev_name = sensor_name; 134 + i++; 135 + } 136 + 137 + acpi_dev_put(consumer); 138 + 139 + return n_consumers; 140 + } 141 + 99 142 static int skl_int3472_tps68470_probe(struct i2c_client *client) 100 143 { 101 144 struct acpi_device *adev = ACPI_COMPANION(&client->dev); 102 145 const struct int3472_tps68470_board_data *board_data; 103 - struct tps68470_clk_platform_data clk_pdata = {}; 146 + struct tps68470_clk_platform_data *clk_pdata; 104 147 struct mfd_cell *cells; 105 148 struct regmap *regmap; 149 + int n_consumers; 106 150 int device_type; 107 151 int ret; 108 152 109 - ret = skl_int3472_get_sensor_adev_and_name(&client->dev, NULL, 110 - &clk_pdata.consumer_dev_name); 111 - if (ret) 112 - return ret; 153 + n_consumers = skl_int3472_fill_clk_pdata(&client->dev, &clk_pdata); 154 + if (n_consumers < 0) 155 + return n_consumers; 113 156 114 157 regmap = devm_regmap_init_i2c(client, &tps68470_regmap_config); 115 158 if (IS_ERR(regmap)) { ··· 187 142 * the clk + regulators must be ready when this happens. 188 143 */ 189 144 cells[0].name = "tps68470-clk"; 190 - cells[0].platform_data = &clk_pdata; 191 - cells[0].pdata_size = sizeof(clk_pdata); 145 + cells[0].platform_data = clk_pdata; 146 + cells[0].pdata_size = struct_size(clk_pdata, consumers, n_consumers); 192 147 cells[1].name = "tps68470-regulator"; 193 148 cells[1].platform_data = (void *)board_data->tps68470_regulator_pdata; 194 149 cells[1].pdata_size = sizeof(struct tps68470_regulator_platform_data);
+6 -1
include/linux/platform_data/tps68470.h
··· 27 27 const struct regulator_init_data *reg_init_data[TPS68470_NUM_REGULATORS]; 28 28 }; 29 29 30 - struct tps68470_clk_platform_data { 30 + struct tps68470_clk_consumer { 31 31 const char *consumer_dev_name; 32 32 const char *consumer_con_id; 33 + }; 34 + 35 + struct tps68470_clk_platform_data { 36 + unsigned int n_consumers; 37 + struct tps68470_clk_consumer consumers[]; 33 38 }; 34 39 35 40 #endif