| // SPDX-License-Identifier: GPL-2.0-only | 
 | /* | 
 |  * isl6271a-regulator.c | 
 |  * | 
 |  * Support for Intersil ISL6271A voltage regulator | 
 |  * | 
 |  * Copyright (C) 2010 Marek Vasut <marek.vasut@gmail.com> | 
 |  */ | 
 |  | 
 | #include <linux/kernel.h> | 
 | #include <linux/module.h> | 
 | #include <linux/init.h> | 
 | #include <linux/err.h> | 
 | #include <linux/platform_device.h> | 
 | #include <linux/regulator/driver.h> | 
 | #include <linux/i2c.h> | 
 | #include <linux/slab.h> | 
 |  | 
 | #define	ISL6271A_VOLTAGE_MIN	850000 | 
 | #define	ISL6271A_VOLTAGE_MAX	1600000 | 
 | #define	ISL6271A_VOLTAGE_STEP	50000 | 
 |  | 
 | /* PMIC details */ | 
 | struct isl_pmic { | 
 | 	struct i2c_client	*client; | 
 | 	struct mutex		mtx; | 
 | }; | 
 |  | 
 | static int isl6271a_get_voltage_sel(struct regulator_dev *dev) | 
 | { | 
 | 	struct isl_pmic *pmic = rdev_get_drvdata(dev); | 
 | 	int idx; | 
 |  | 
 | 	mutex_lock(&pmic->mtx); | 
 |  | 
 | 	idx = i2c_smbus_read_byte(pmic->client); | 
 | 	if (idx < 0) | 
 | 		dev_err(&pmic->client->dev, "Error getting voltage\n"); | 
 |  | 
 | 	mutex_unlock(&pmic->mtx); | 
 | 	return idx; | 
 | } | 
 |  | 
 | static int isl6271a_set_voltage_sel(struct regulator_dev *dev, | 
 | 				    unsigned selector) | 
 | { | 
 | 	struct isl_pmic *pmic = rdev_get_drvdata(dev); | 
 | 	int err; | 
 |  | 
 | 	mutex_lock(&pmic->mtx); | 
 |  | 
 | 	err = i2c_smbus_write_byte(pmic->client, selector); | 
 | 	if (err < 0) | 
 | 		dev_err(&pmic->client->dev, "Error setting voltage\n"); | 
 |  | 
 | 	mutex_unlock(&pmic->mtx); | 
 | 	return err; | 
 | } | 
 |  | 
 | static const struct regulator_ops isl_core_ops = { | 
 | 	.get_voltage_sel = isl6271a_get_voltage_sel, | 
 | 	.set_voltage_sel = isl6271a_set_voltage_sel, | 
 | 	.list_voltage	= regulator_list_voltage_linear, | 
 | 	.map_voltage	= regulator_map_voltage_linear, | 
 | }; | 
 |  | 
 | static const struct regulator_ops isl_fixed_ops = { | 
 | 	.list_voltage	= regulator_list_voltage_linear, | 
 | }; | 
 |  | 
 | static const struct regulator_desc isl_rd[] = { | 
 | 	{ | 
 | 		.name		= "Core Buck", | 
 | 		.id		= 0, | 
 | 		.n_voltages	= 16, | 
 | 		.ops		= &isl_core_ops, | 
 | 		.type		= REGULATOR_VOLTAGE, | 
 | 		.owner		= THIS_MODULE, | 
 | 		.min_uV		= ISL6271A_VOLTAGE_MIN, | 
 | 		.uV_step	= ISL6271A_VOLTAGE_STEP, | 
 | 	}, { | 
 | 		.name		= "LDO1", | 
 | 		.id		= 1, | 
 | 		.n_voltages	= 1, | 
 | 		.ops		= &isl_fixed_ops, | 
 | 		.type		= REGULATOR_VOLTAGE, | 
 | 		.owner		= THIS_MODULE, | 
 | 		.min_uV		= 1100000, | 
 | 	}, { | 
 | 		.name		= "LDO2", | 
 | 		.id		= 2, | 
 | 		.n_voltages	= 1, | 
 | 		.ops		= &isl_fixed_ops, | 
 | 		.type		= REGULATOR_VOLTAGE, | 
 | 		.owner		= THIS_MODULE, | 
 | 		.min_uV		= 1300000, | 
 | 	}, | 
 | }; | 
 |  | 
 | static int isl6271a_probe(struct i2c_client *i2c) | 
 | { | 
 | 	const struct i2c_device_id *id = i2c_client_get_device_id(i2c); | 
 | 	struct regulator_dev *rdev; | 
 | 	struct regulator_config config = { }; | 
 | 	struct regulator_init_data *init_data	= dev_get_platdata(&i2c->dev); | 
 | 	struct isl_pmic *pmic; | 
 | 	int i; | 
 |  | 
 | 	if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_SMBUS_BYTE_DATA)) | 
 | 		return -EIO; | 
 |  | 
 | 	pmic = devm_kzalloc(&i2c->dev, sizeof(struct isl_pmic), GFP_KERNEL); | 
 | 	if (!pmic) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	pmic->client = i2c; | 
 |  | 
 | 	mutex_init(&pmic->mtx); | 
 |  | 
 | 	for (i = 0; i < 3; i++) { | 
 | 		config.dev = &i2c->dev; | 
 | 		if (i == 0) | 
 | 			config.init_data = init_data; | 
 | 		else | 
 | 			config.init_data = NULL; | 
 | 		config.driver_data = pmic; | 
 |  | 
 | 		rdev = devm_regulator_register(&i2c->dev, &isl_rd[i], &config); | 
 | 		if (IS_ERR(rdev)) { | 
 | 			dev_err(&i2c->dev, "failed to register %s\n", id->name); | 
 | 			return PTR_ERR(rdev); | 
 | 		} | 
 | 	} | 
 |  | 
 | 	i2c_set_clientdata(i2c, pmic); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct i2c_device_id isl6271a_id[] = { | 
 | 	{.name = "isl6271a", 0 }, | 
 | 	{ }, | 
 | }; | 
 |  | 
 | MODULE_DEVICE_TABLE(i2c, isl6271a_id); | 
 |  | 
 | static struct i2c_driver isl6271a_i2c_driver = { | 
 | 	.driver = { | 
 | 		.name = "isl6271a", | 
 | 		.probe_type = PROBE_PREFER_ASYNCHRONOUS, | 
 | 	}, | 
 | 	.probe = isl6271a_probe, | 
 | 	.id_table = isl6271a_id, | 
 | }; | 
 |  | 
 | static int __init isl6271a_init(void) | 
 | { | 
 | 	return i2c_add_driver(&isl6271a_i2c_driver); | 
 | } | 
 |  | 
 | static void __exit isl6271a_cleanup(void) | 
 | { | 
 | 	i2c_del_driver(&isl6271a_i2c_driver); | 
 | } | 
 |  | 
 | subsys_initcall(isl6271a_init); | 
 | module_exit(isl6271a_cleanup); | 
 |  | 
 | MODULE_AUTHOR("Marek Vasut <marek.vasut@gmail.com>"); | 
 | MODULE_DESCRIPTION("Intersil ISL6271A voltage regulator driver"); | 
 | MODULE_LICENSE("GPL v2"); |