| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * Copyright (c) 2022 Analog Devices, Inc. |
| * ADI Regulator driver for the MAX77540 and MAX77541 |
| */ |
| |
| #include <linux/mfd/max77541.h> |
| #include <linux/mod_devicetable.h> |
| #include <linux/platform_device.h> |
| #include <linux/regmap.h> |
| #include <linux/regulator/driver.h> |
| |
| static const struct regulator_ops max77541_buck_ops = { |
| .enable = regulator_enable_regmap, |
| .disable = regulator_disable_regmap, |
| .is_enabled = regulator_is_enabled_regmap, |
| .list_voltage = regulator_list_voltage_pickable_linear_range, |
| .get_voltage_sel = regulator_get_voltage_sel_pickable_regmap, |
| .set_voltage_sel = regulator_set_voltage_sel_pickable_regmap, |
| }; |
| |
| static const struct linear_range max77540_buck_ranges[] = { |
| /* Ranges when VOLT_SEL bits are 0x00 */ |
| REGULATOR_LINEAR_RANGE(500000, 0x00, 0x8B, 5000), |
| REGULATOR_LINEAR_RANGE(1200000, 0x8C, 0xFF, 0), |
| /* Ranges when VOLT_SEL bits are 0x40 */ |
| REGULATOR_LINEAR_RANGE(1200000, 0x00, 0x8B, 10000), |
| REGULATOR_LINEAR_RANGE(2400000, 0x8C, 0xFF, 0), |
| /* Ranges when VOLT_SEL bits are 0x80 */ |
| REGULATOR_LINEAR_RANGE(2000000, 0x00, 0x9F, 20000), |
| REGULATOR_LINEAR_RANGE(5200000, 0xA0, 0xFF, 0), |
| }; |
| |
| static const struct linear_range max77541_buck_ranges[] = { |
| /* Ranges when VOLT_SEL bits are 0x00 */ |
| REGULATOR_LINEAR_RANGE(300000, 0x00, 0xB3, 5000), |
| REGULATOR_LINEAR_RANGE(1200000, 0xB4, 0xFF, 0), |
| /* Ranges when VOLT_SEL bits are 0x40 */ |
| REGULATOR_LINEAR_RANGE(1200000, 0x00, 0x8B, 10000), |
| REGULATOR_LINEAR_RANGE(2400000, 0x8C, 0xFF, 0), |
| /* Ranges when VOLT_SEL bits are 0x80 */ |
| REGULATOR_LINEAR_RANGE(2000000, 0x00, 0x9F, 20000), |
| REGULATOR_LINEAR_RANGE(5200000, 0xA0, 0xFF, 0), |
| }; |
| |
| static const unsigned int max77541_buck_volt_range_sel[] = { |
| 0x0, 0x0, 0x1, 0x1, 0x2, 0x2, |
| }; |
| |
| enum max77541_regulators { |
| MAX77541_BUCK1 = 1, |
| MAX77541_BUCK2, |
| }; |
| |
| #define MAX77540_BUCK(_id, _ops) \ |
| { .id = MAX77541_BUCK ## _id, \ |
| .name = "buck"#_id, \ |
| .of_match = "buck"#_id, \ |
| .regulators_node = "regulators", \ |
| .enable_reg = MAX77541_REG_EN_CTRL, \ |
| .enable_mask = MAX77541_BIT_M ## _id ## _EN, \ |
| .ops = &(_ops), \ |
| .type = REGULATOR_VOLTAGE, \ |
| .linear_ranges = max77540_buck_ranges, \ |
| .n_linear_ranges = ARRAY_SIZE(max77540_buck_ranges), \ |
| .vsel_reg = MAX77541_REG_M ## _id ## _VOUT, \ |
| .vsel_mask = MAX77541_BITS_MX_VOUT, \ |
| .vsel_range_reg = MAX77541_REG_M ## _id ## _CFG1, \ |
| .vsel_range_mask = MAX77541_BITS_MX_CFG1_RNG, \ |
| .linear_range_selectors_bitfield = max77541_buck_volt_range_sel, \ |
| .owner = THIS_MODULE, \ |
| } |
| |
| #define MAX77541_BUCK(_id, _ops) \ |
| { .id = MAX77541_BUCK ## _id, \ |
| .name = "buck"#_id, \ |
| .of_match = "buck"#_id, \ |
| .regulators_node = "regulators", \ |
| .enable_reg = MAX77541_REG_EN_CTRL, \ |
| .enable_mask = MAX77541_BIT_M ## _id ## _EN, \ |
| .ops = &(_ops), \ |
| .type = REGULATOR_VOLTAGE, \ |
| .linear_ranges = max77541_buck_ranges, \ |
| .n_linear_ranges = ARRAY_SIZE(max77541_buck_ranges), \ |
| .vsel_reg = MAX77541_REG_M ## _id ## _VOUT, \ |
| .vsel_mask = MAX77541_BITS_MX_VOUT, \ |
| .vsel_range_reg = MAX77541_REG_M ## _id ## _CFG1, \ |
| .vsel_range_mask = MAX77541_BITS_MX_CFG1_RNG, \ |
| .linear_range_selectors_bitfield = max77541_buck_volt_range_sel, \ |
| .owner = THIS_MODULE, \ |
| } |
| |
| static const struct regulator_desc max77540_regulators_desc[] = { |
| MAX77540_BUCK(1, max77541_buck_ops), |
| MAX77540_BUCK(2, max77541_buck_ops), |
| }; |
| |
| static const struct regulator_desc max77541_regulators_desc[] = { |
| MAX77541_BUCK(1, max77541_buck_ops), |
| MAX77541_BUCK(2, max77541_buck_ops), |
| }; |
| |
| static int max77541_regulator_probe(struct platform_device *pdev) |
| { |
| struct regulator_config config = {}; |
| const struct regulator_desc *desc; |
| struct device *dev = &pdev->dev; |
| struct regulator_dev *rdev; |
| struct max77541 *max77541 = dev_get_drvdata(dev->parent); |
| unsigned int i; |
| |
| config.dev = dev->parent; |
| |
| switch (max77541->id) { |
| case MAX77540: |
| desc = max77540_regulators_desc; |
| break; |
| case MAX77541: |
| desc = max77541_regulators_desc; |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| for (i = 0; i < MAX77541_MAX_REGULATORS; i++) { |
| rdev = devm_regulator_register(dev, &desc[i], &config); |
| if (IS_ERR(rdev)) |
| return dev_err_probe(dev, PTR_ERR(rdev), |
| "Failed to register regulator\n"); |
| } |
| |
| return 0; |
| } |
| |
| static const struct platform_device_id max77541_regulator_platform_id[] = { |
| { "max77540-regulator" }, |
| { "max77541-regulator" }, |
| { } |
| }; |
| MODULE_DEVICE_TABLE(platform, max77541_regulator_platform_id); |
| |
| static struct platform_driver max77541_regulator_driver = { |
| .driver = { |
| .name = "max77541-regulator", |
| }, |
| .probe = max77541_regulator_probe, |
| .id_table = max77541_regulator_platform_id, |
| }; |
| module_platform_driver(max77541_regulator_driver); |
| |
| MODULE_AUTHOR("Okan Sahin <Okan.Sahin@analog.com>"); |
| MODULE_DESCRIPTION("MAX77540/MAX77541 regulator driver"); |
| MODULE_LICENSE("GPL"); |