162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/* Author: Dan Scally <djrscally@gmail.com> */
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#include <linux/acpi.h>
562306a36Sopenharmony_ci#include <linux/i2c.h>
662306a36Sopenharmony_ci#include <linux/kernel.h>
762306a36Sopenharmony_ci#include <linux/mfd/core.h>
862306a36Sopenharmony_ci#include <linux/mfd/tps68470.h>
962306a36Sopenharmony_ci#include <linux/platform_device.h>
1062306a36Sopenharmony_ci#include <linux/platform_data/tps68470.h>
1162306a36Sopenharmony_ci#include <linux/regmap.h>
1262306a36Sopenharmony_ci#include <linux/string.h>
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include "common.h"
1562306a36Sopenharmony_ci#include "tps68470.h"
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci#define DESIGNED_FOR_CHROMEOS		1
1862306a36Sopenharmony_ci#define DESIGNED_FOR_WINDOWS		2
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#define TPS68470_WIN_MFD_CELL_COUNT	3
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic const struct mfd_cell tps68470_cros[] = {
2362306a36Sopenharmony_ci	{ .name = "tps68470-gpio" },
2462306a36Sopenharmony_ci	{ .name = "tps68470_pmic_opregion" },
2562306a36Sopenharmony_ci};
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_cistatic const struct regmap_config tps68470_regmap_config = {
2862306a36Sopenharmony_ci	.reg_bits = 8,
2962306a36Sopenharmony_ci	.val_bits = 8,
3062306a36Sopenharmony_ci	.max_register = TPS68470_REG_MAX,
3162306a36Sopenharmony_ci};
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_cistatic int tps68470_chip_init(struct device *dev, struct regmap *regmap)
3462306a36Sopenharmony_ci{
3562306a36Sopenharmony_ci	unsigned int version;
3662306a36Sopenharmony_ci	int ret;
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci	/* Force software reset */
3962306a36Sopenharmony_ci	ret = regmap_write(regmap, TPS68470_REG_RESET, TPS68470_REG_RESET_MASK);
4062306a36Sopenharmony_ci	if (ret)
4162306a36Sopenharmony_ci		return ret;
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	ret = regmap_read(regmap, TPS68470_REG_REVID, &version);
4462306a36Sopenharmony_ci	if (ret) {
4562306a36Sopenharmony_ci		dev_err(dev, "Failed to read revision register: %d\n", ret);
4662306a36Sopenharmony_ci		return ret;
4762306a36Sopenharmony_ci	}
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	dev_info(dev, "TPS68470 REVID: 0x%02x\n", version);
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	return 0;
5262306a36Sopenharmony_ci}
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci/** skl_int3472_tps68470_calc_type: Check what platform a device is designed for
5562306a36Sopenharmony_ci * @adev: A pointer to a &struct acpi_device
5662306a36Sopenharmony_ci *
5762306a36Sopenharmony_ci * Check CLDB buffer against the PMIC's adev. If present, then we check
5862306a36Sopenharmony_ci * the value of control_logic_type field and follow one of the
5962306a36Sopenharmony_ci * following scenarios:
6062306a36Sopenharmony_ci *
6162306a36Sopenharmony_ci *	1. No CLDB - likely ACPI tables designed for ChromeOS. We
6262306a36Sopenharmony_ci *	create platform devices for the GPIOs and OpRegion drivers.
6362306a36Sopenharmony_ci *
6462306a36Sopenharmony_ci *	2. CLDB, with control_logic_type = 2 - probably ACPI tables
6562306a36Sopenharmony_ci *	made for Windows 2-in-1 platforms. Register pdevs for GPIO,
6662306a36Sopenharmony_ci *	Clock and Regulator drivers to bind to.
6762306a36Sopenharmony_ci *
6862306a36Sopenharmony_ci *	3. Any other value in control_logic_type, we should never have
6962306a36Sopenharmony_ci *	gotten to this point; fail probe and return.
7062306a36Sopenharmony_ci *
7162306a36Sopenharmony_ci * Return:
7262306a36Sopenharmony_ci * * 1		Device intended for ChromeOS
7362306a36Sopenharmony_ci * * 2		Device intended for Windows
7462306a36Sopenharmony_ci * * -EINVAL	Where @adev has an object named CLDB but it does not conform to
7562306a36Sopenharmony_ci *		our expectations
7662306a36Sopenharmony_ci */
7762306a36Sopenharmony_cistatic int skl_int3472_tps68470_calc_type(struct acpi_device *adev)
7862306a36Sopenharmony_ci{
7962306a36Sopenharmony_ci	struct int3472_cldb cldb = { 0 };
8062306a36Sopenharmony_ci	int ret;
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ci	/*
8362306a36Sopenharmony_ci	 * A CLDB buffer that exists, but which does not match our expectations
8462306a36Sopenharmony_ci	 * should trigger an error so we don't blindly continue.
8562306a36Sopenharmony_ci	 */
8662306a36Sopenharmony_ci	ret = skl_int3472_fill_cldb(adev, &cldb);
8762306a36Sopenharmony_ci	if (ret && ret != -ENODEV)
8862306a36Sopenharmony_ci		return ret;
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	if (ret)
9162306a36Sopenharmony_ci		return DESIGNED_FOR_CHROMEOS;
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci	if (cldb.control_logic_type != 2)
9462306a36Sopenharmony_ci		return -EINVAL;
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci	return DESIGNED_FOR_WINDOWS;
9762306a36Sopenharmony_ci}
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci/*
10062306a36Sopenharmony_ci * Return the size of the flexible array member, because we'll need that later
10162306a36Sopenharmony_ci * on to pass .pdata_size to cells.
10262306a36Sopenharmony_ci */
10362306a36Sopenharmony_cistatic int
10462306a36Sopenharmony_ciskl_int3472_fill_clk_pdata(struct device *dev, struct tps68470_clk_platform_data **clk_pdata)
10562306a36Sopenharmony_ci{
10662306a36Sopenharmony_ci	struct acpi_device *adev = ACPI_COMPANION(dev);
10762306a36Sopenharmony_ci	struct acpi_device *consumer;
10862306a36Sopenharmony_ci	unsigned int n_consumers = 0;
10962306a36Sopenharmony_ci	const char *sensor_name;
11062306a36Sopenharmony_ci	unsigned int i = 0;
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	for_each_acpi_consumer_dev(adev, consumer)
11362306a36Sopenharmony_ci		n_consumers++;
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci	if (!n_consumers) {
11662306a36Sopenharmony_ci		dev_err(dev, "INT3472 seems to have no dependents\n");
11762306a36Sopenharmony_ci		return -ENODEV;
11862306a36Sopenharmony_ci	}
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	*clk_pdata = devm_kzalloc(dev, struct_size(*clk_pdata, consumers, n_consumers),
12162306a36Sopenharmony_ci				  GFP_KERNEL);
12262306a36Sopenharmony_ci	if (!*clk_pdata)
12362306a36Sopenharmony_ci		return -ENOMEM;
12462306a36Sopenharmony_ci
12562306a36Sopenharmony_ci	(*clk_pdata)->n_consumers = n_consumers;
12662306a36Sopenharmony_ci	i = 0;
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	for_each_acpi_consumer_dev(adev, consumer) {
12962306a36Sopenharmony_ci		sensor_name = devm_kasprintf(dev, GFP_KERNEL, I2C_DEV_NAME_FORMAT,
13062306a36Sopenharmony_ci					     acpi_dev_name(consumer));
13162306a36Sopenharmony_ci		if (!sensor_name) {
13262306a36Sopenharmony_ci			acpi_dev_put(consumer);
13362306a36Sopenharmony_ci			return -ENOMEM;
13462306a36Sopenharmony_ci		}
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci		(*clk_pdata)->consumers[i].consumer_dev_name = sensor_name;
13762306a36Sopenharmony_ci		i++;
13862306a36Sopenharmony_ci	}
13962306a36Sopenharmony_ci
14062306a36Sopenharmony_ci	return n_consumers;
14162306a36Sopenharmony_ci}
14262306a36Sopenharmony_ci
14362306a36Sopenharmony_cistatic int skl_int3472_tps68470_probe(struct i2c_client *client)
14462306a36Sopenharmony_ci{
14562306a36Sopenharmony_ci	struct acpi_device *adev = ACPI_COMPANION(&client->dev);
14662306a36Sopenharmony_ci	const struct int3472_tps68470_board_data *board_data;
14762306a36Sopenharmony_ci	struct tps68470_clk_platform_data *clk_pdata;
14862306a36Sopenharmony_ci	struct mfd_cell *cells;
14962306a36Sopenharmony_ci	struct regmap *regmap;
15062306a36Sopenharmony_ci	int n_consumers;
15162306a36Sopenharmony_ci	int device_type;
15262306a36Sopenharmony_ci	int ret;
15362306a36Sopenharmony_ci	int i;
15462306a36Sopenharmony_ci
15562306a36Sopenharmony_ci	n_consumers = skl_int3472_fill_clk_pdata(&client->dev, &clk_pdata);
15662306a36Sopenharmony_ci	if (n_consumers < 0)
15762306a36Sopenharmony_ci		return n_consumers;
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_ci	regmap = devm_regmap_init_i2c(client, &tps68470_regmap_config);
16062306a36Sopenharmony_ci	if (IS_ERR(regmap)) {
16162306a36Sopenharmony_ci		dev_err(&client->dev, "Failed to create regmap: %ld\n", PTR_ERR(regmap));
16262306a36Sopenharmony_ci		return PTR_ERR(regmap);
16362306a36Sopenharmony_ci	}
16462306a36Sopenharmony_ci
16562306a36Sopenharmony_ci	i2c_set_clientdata(client, regmap);
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_ci	ret = tps68470_chip_init(&client->dev, regmap);
16862306a36Sopenharmony_ci	if (ret < 0) {
16962306a36Sopenharmony_ci		dev_err(&client->dev, "TPS68470 init error %d\n", ret);
17062306a36Sopenharmony_ci		return ret;
17162306a36Sopenharmony_ci	}
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_ci	device_type = skl_int3472_tps68470_calc_type(adev);
17462306a36Sopenharmony_ci	switch (device_type) {
17562306a36Sopenharmony_ci	case DESIGNED_FOR_WINDOWS:
17662306a36Sopenharmony_ci		board_data = int3472_tps68470_get_board_data(dev_name(&client->dev));
17762306a36Sopenharmony_ci		if (!board_data)
17862306a36Sopenharmony_ci			return dev_err_probe(&client->dev, -ENODEV, "No board-data found for this model\n");
17962306a36Sopenharmony_ci
18062306a36Sopenharmony_ci		cells = kcalloc(TPS68470_WIN_MFD_CELL_COUNT, sizeof(*cells), GFP_KERNEL);
18162306a36Sopenharmony_ci		if (!cells)
18262306a36Sopenharmony_ci			return -ENOMEM;
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci		/*
18562306a36Sopenharmony_ci		 * The order of the cells matters here! The clk must be first
18662306a36Sopenharmony_ci		 * because the regulator depends on it. The gpios must be last,
18762306a36Sopenharmony_ci		 * acpi_gpiochip_add() calls acpi_dev_clear_dependencies() and
18862306a36Sopenharmony_ci		 * the clk + regulators must be ready when this happens.
18962306a36Sopenharmony_ci		 */
19062306a36Sopenharmony_ci		cells[0].name = "tps68470-clk";
19162306a36Sopenharmony_ci		cells[0].platform_data = clk_pdata;
19262306a36Sopenharmony_ci		cells[0].pdata_size = struct_size(clk_pdata, consumers, n_consumers);
19362306a36Sopenharmony_ci		cells[1].name = "tps68470-regulator";
19462306a36Sopenharmony_ci		cells[1].platform_data = (void *)board_data->tps68470_regulator_pdata;
19562306a36Sopenharmony_ci		cells[1].pdata_size = sizeof(struct tps68470_regulator_platform_data);
19662306a36Sopenharmony_ci		cells[2].name = "tps68470-gpio";
19762306a36Sopenharmony_ci
19862306a36Sopenharmony_ci		for (i = 0; i < board_data->n_gpiod_lookups; i++)
19962306a36Sopenharmony_ci			gpiod_add_lookup_table(board_data->tps68470_gpio_lookup_tables[i]);
20062306a36Sopenharmony_ci
20162306a36Sopenharmony_ci		ret = devm_mfd_add_devices(&client->dev, PLATFORM_DEVID_NONE,
20262306a36Sopenharmony_ci					   cells, TPS68470_WIN_MFD_CELL_COUNT,
20362306a36Sopenharmony_ci					   NULL, 0, NULL);
20462306a36Sopenharmony_ci		kfree(cells);
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci		if (ret) {
20762306a36Sopenharmony_ci			for (i = 0; i < board_data->n_gpiod_lookups; i++)
20862306a36Sopenharmony_ci				gpiod_remove_lookup_table(board_data->tps68470_gpio_lookup_tables[i]);
20962306a36Sopenharmony_ci		}
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_ci		break;
21262306a36Sopenharmony_ci	case DESIGNED_FOR_CHROMEOS:
21362306a36Sopenharmony_ci		ret = devm_mfd_add_devices(&client->dev, PLATFORM_DEVID_NONE,
21462306a36Sopenharmony_ci					   tps68470_cros, ARRAY_SIZE(tps68470_cros),
21562306a36Sopenharmony_ci					   NULL, 0, NULL);
21662306a36Sopenharmony_ci		break;
21762306a36Sopenharmony_ci	default:
21862306a36Sopenharmony_ci		dev_err(&client->dev, "Failed to add MFD devices\n");
21962306a36Sopenharmony_ci		return device_type;
22062306a36Sopenharmony_ci	}
22162306a36Sopenharmony_ci
22262306a36Sopenharmony_ci	/*
22362306a36Sopenharmony_ci	 * No acpi_dev_clear_dependencies() here, since the acpi_gpiochip_add()
22462306a36Sopenharmony_ci	 * for the GPIO cell already does this.
22562306a36Sopenharmony_ci	 */
22662306a36Sopenharmony_ci
22762306a36Sopenharmony_ci	return ret;
22862306a36Sopenharmony_ci}
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_cistatic void skl_int3472_tps68470_remove(struct i2c_client *client)
23162306a36Sopenharmony_ci{
23262306a36Sopenharmony_ci	const struct int3472_tps68470_board_data *board_data;
23362306a36Sopenharmony_ci	int i;
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	board_data = int3472_tps68470_get_board_data(dev_name(&client->dev));
23662306a36Sopenharmony_ci	if (board_data) {
23762306a36Sopenharmony_ci		for (i = 0; i < board_data->n_gpiod_lookups; i++)
23862306a36Sopenharmony_ci			gpiod_remove_lookup_table(board_data->tps68470_gpio_lookup_tables[i]);
23962306a36Sopenharmony_ci	}
24062306a36Sopenharmony_ci}
24162306a36Sopenharmony_ci
24262306a36Sopenharmony_cistatic const struct acpi_device_id int3472_device_id[] = {
24362306a36Sopenharmony_ci	{ "INT3472", 0 },
24462306a36Sopenharmony_ci	{ }
24562306a36Sopenharmony_ci};
24662306a36Sopenharmony_ciMODULE_DEVICE_TABLE(acpi, int3472_device_id);
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_cistatic struct i2c_driver int3472_tps68470 = {
24962306a36Sopenharmony_ci	.driver = {
25062306a36Sopenharmony_ci		.name = "int3472-tps68470",
25162306a36Sopenharmony_ci		.acpi_match_table = int3472_device_id,
25262306a36Sopenharmony_ci	},
25362306a36Sopenharmony_ci	.probe = skl_int3472_tps68470_probe,
25462306a36Sopenharmony_ci	.remove = skl_int3472_tps68470_remove,
25562306a36Sopenharmony_ci};
25662306a36Sopenharmony_cimodule_i2c_driver(int3472_tps68470);
25762306a36Sopenharmony_ci
25862306a36Sopenharmony_ciMODULE_DESCRIPTION("Intel SkyLake INT3472 ACPI TPS68470 Device Driver");
25962306a36Sopenharmony_ciMODULE_AUTHOR("Daniel Scally <djrscally@gmail.com>");
26062306a36Sopenharmony_ciMODULE_LICENSE("GPL v2");
26162306a36Sopenharmony_ciMODULE_SOFTDEP("pre: clk-tps68470 tps68470-regulator");
262