162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+ 262306a36Sopenharmony_ci 362306a36Sopenharmony_ci#include <linux/bitops.h> 462306a36Sopenharmony_ci#include <linux/delay.h> 562306a36Sopenharmony_ci#include <linux/gpio/consumer.h> 662306a36Sopenharmony_ci#include <linux/i2c.h> 762306a36Sopenharmony_ci#include <linux/kernel.h> 862306a36Sopenharmony_ci#include <linux/module.h> 962306a36Sopenharmony_ci#include <linux/regmap.h> 1062306a36Sopenharmony_ci#include <linux/regulator/driver.h> 1162306a36Sopenharmony_ci#include <linux/regulator/of_regulator.h> 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#define RT6245_VIRT_OCLIMIT 0x00 1462306a36Sopenharmony_ci#define RT6245_VIRT_OTLEVEL 0x01 1562306a36Sopenharmony_ci#define RT6245_VIRT_PGDLYTIME 0x02 1662306a36Sopenharmony_ci#define RT6245_VIRT_SLEWRATE 0x03 1762306a36Sopenharmony_ci#define RT6245_VIRT_SWFREQ 0x04 1862306a36Sopenharmony_ci#define RT6245_VIRT_VOUT 0x05 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci#define RT6245_VOUT_MASK GENMASK(6, 0) 2162306a36Sopenharmony_ci#define RT6245_SLEW_MASK GENMASK(2, 0) 2262306a36Sopenharmony_ci#define RT6245_CHKSUM_MASK BIT(7) 2362306a36Sopenharmony_ci#define RT6245_CODE_MASK GENMASK(6, 0) 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci/* HW Enable + Soft start time */ 2662306a36Sopenharmony_ci#define RT6245_ENTIME_IN_US 5000 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci#define RT6245_VOUT_MINUV 437500 2962306a36Sopenharmony_ci#define RT6245_VOUT_MAXUV 1387500 3062306a36Sopenharmony_ci#define RT6245_VOUT_STEPUV 12500 3162306a36Sopenharmony_ci#define RT6245_NUM_VOUT ((RT6245_VOUT_MAXUV - RT6245_VOUT_MINUV) / RT6245_VOUT_STEPUV + 1) 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_cistruct rt6245_priv { 3462306a36Sopenharmony_ci struct gpio_desc *enable_gpio; 3562306a36Sopenharmony_ci bool enable_state; 3662306a36Sopenharmony_ci}; 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_cistatic int rt6245_enable(struct regulator_dev *rdev) 3962306a36Sopenharmony_ci{ 4062306a36Sopenharmony_ci struct rt6245_priv *priv = rdev_get_drvdata(rdev); 4162306a36Sopenharmony_ci struct regmap *regmap = rdev_get_regmap(rdev); 4262306a36Sopenharmony_ci int ret; 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci if (!priv->enable_gpio) 4562306a36Sopenharmony_ci return 0; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_ci gpiod_direction_output(priv->enable_gpio, 1); 4862306a36Sopenharmony_ci usleep_range(RT6245_ENTIME_IN_US, RT6245_ENTIME_IN_US + 1000); 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci regcache_cache_only(regmap, false); 5162306a36Sopenharmony_ci ret = regcache_sync(regmap); 5262306a36Sopenharmony_ci if (ret) 5362306a36Sopenharmony_ci return ret; 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci priv->enable_state = true; 5662306a36Sopenharmony_ci return 0; 5762306a36Sopenharmony_ci} 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_cistatic int rt6245_disable(struct regulator_dev *rdev) 6062306a36Sopenharmony_ci{ 6162306a36Sopenharmony_ci struct rt6245_priv *priv = rdev_get_drvdata(rdev); 6262306a36Sopenharmony_ci struct regmap *regmap = rdev_get_regmap(rdev); 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci if (!priv->enable_gpio) 6562306a36Sopenharmony_ci return -EINVAL; 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci regcache_cache_only(regmap, true); 6862306a36Sopenharmony_ci regcache_mark_dirty(regmap); 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci gpiod_direction_output(priv->enable_gpio, 0); 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci priv->enable_state = false; 7362306a36Sopenharmony_ci return 0; 7462306a36Sopenharmony_ci} 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_cistatic int rt6245_is_enabled(struct regulator_dev *rdev) 7762306a36Sopenharmony_ci{ 7862306a36Sopenharmony_ci struct rt6245_priv *priv = rdev_get_drvdata(rdev); 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ci return priv->enable_state ? 1 : 0; 8162306a36Sopenharmony_ci} 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_cistatic const struct regulator_ops rt6245_regulator_ops = { 8462306a36Sopenharmony_ci .list_voltage = regulator_list_voltage_linear, 8562306a36Sopenharmony_ci .set_voltage_sel = regulator_set_voltage_sel_regmap, 8662306a36Sopenharmony_ci .get_voltage_sel = regulator_get_voltage_sel_regmap, 8762306a36Sopenharmony_ci .set_ramp_delay = regulator_set_ramp_delay_regmap, 8862306a36Sopenharmony_ci .enable = rt6245_enable, 8962306a36Sopenharmony_ci .disable = rt6245_disable, 9062306a36Sopenharmony_ci .is_enabled = rt6245_is_enabled, 9162306a36Sopenharmony_ci}; 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci/* ramp delay dividend is 12500 uV/uS, and divisor from 1 to 8 */ 9462306a36Sopenharmony_cistatic const unsigned int rt6245_ramp_delay_table[] = { 9562306a36Sopenharmony_ci 12500, 6250, 4167, 3125, 2500, 2083, 1786, 1562 9662306a36Sopenharmony_ci}; 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_cistatic const struct regulator_desc rt6245_regulator_desc = { 9962306a36Sopenharmony_ci .name = "rt6245-regulator", 10062306a36Sopenharmony_ci .ops = &rt6245_regulator_ops, 10162306a36Sopenharmony_ci .type = REGULATOR_VOLTAGE, 10262306a36Sopenharmony_ci .min_uV = RT6245_VOUT_MINUV, 10362306a36Sopenharmony_ci .uV_step = RT6245_VOUT_STEPUV, 10462306a36Sopenharmony_ci .n_voltages = RT6245_NUM_VOUT, 10562306a36Sopenharmony_ci .ramp_delay_table = rt6245_ramp_delay_table, 10662306a36Sopenharmony_ci .n_ramp_values = ARRAY_SIZE(rt6245_ramp_delay_table), 10762306a36Sopenharmony_ci .owner = THIS_MODULE, 10862306a36Sopenharmony_ci .vsel_reg = RT6245_VIRT_VOUT, 10962306a36Sopenharmony_ci .vsel_mask = RT6245_VOUT_MASK, 11062306a36Sopenharmony_ci .ramp_reg = RT6245_VIRT_SLEWRATE, 11162306a36Sopenharmony_ci .ramp_mask = RT6245_SLEW_MASK, 11262306a36Sopenharmony_ci}; 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_cistatic int rt6245_init_device_properties(struct device *dev) 11562306a36Sopenharmony_ci{ 11662306a36Sopenharmony_ci const struct { 11762306a36Sopenharmony_ci const char *name; 11862306a36Sopenharmony_ci unsigned int reg; 11962306a36Sopenharmony_ci } rt6245_props[] = { 12062306a36Sopenharmony_ci { "richtek,oc-level-select", RT6245_VIRT_OCLIMIT }, 12162306a36Sopenharmony_ci { "richtek,ot-level-select", RT6245_VIRT_OTLEVEL }, 12262306a36Sopenharmony_ci { "richtek,pgdly-time-select", RT6245_VIRT_PGDLYTIME }, 12362306a36Sopenharmony_ci { "richtek,switch-freq-select", RT6245_VIRT_SWFREQ } 12462306a36Sopenharmony_ci }; 12562306a36Sopenharmony_ci struct regmap *regmap = dev_get_regmap(dev, NULL); 12662306a36Sopenharmony_ci u8 propval; 12762306a36Sopenharmony_ci int i, ret; 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(rt6245_props); i++) { 13062306a36Sopenharmony_ci ret = device_property_read_u8(dev, rt6245_props[i].name, &propval); 13162306a36Sopenharmony_ci if (ret) 13262306a36Sopenharmony_ci continue; 13362306a36Sopenharmony_ci 13462306a36Sopenharmony_ci ret = regmap_write(regmap, rt6245_props[i].reg, propval); 13562306a36Sopenharmony_ci if (ret) { 13662306a36Sopenharmony_ci dev_err(dev, "Fail to apply [%s:%d]\n", rt6245_props[i].name, propval); 13762306a36Sopenharmony_ci return ret; 13862306a36Sopenharmony_ci } 13962306a36Sopenharmony_ci } 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_ci return 0; 14262306a36Sopenharmony_ci} 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_cistatic int rt6245_reg_write(void *context, unsigned int reg, unsigned int val) 14562306a36Sopenharmony_ci{ 14662306a36Sopenharmony_ci struct i2c_client *i2c = context; 14762306a36Sopenharmony_ci static const u8 func_base[] = { 0x6F, 0x73, 0x78, 0x61, 0x7C, 0 }; 14862306a36Sopenharmony_ci unsigned int code, bit_count; 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_ci code = func_base[reg]; 15162306a36Sopenharmony_ci code += val; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci /* xor checksum for bit 6 to 0 */ 15462306a36Sopenharmony_ci bit_count = hweight8(code & RT6245_CODE_MASK); 15562306a36Sopenharmony_ci if (bit_count % 2) 15662306a36Sopenharmony_ci code |= RT6245_CHKSUM_MASK; 15762306a36Sopenharmony_ci else 15862306a36Sopenharmony_ci code &= ~RT6245_CHKSUM_MASK; 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci return i2c_smbus_write_byte(i2c, code); 16162306a36Sopenharmony_ci} 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_cistatic const struct reg_default rt6245_reg_defaults[] = { 16462306a36Sopenharmony_ci /* Default over current 14A */ 16562306a36Sopenharmony_ci { RT6245_VIRT_OCLIMIT, 2 }, 16662306a36Sopenharmony_ci /* Default over temperature 150'c */ 16762306a36Sopenharmony_ci { RT6245_VIRT_OTLEVEL, 0 }, 16862306a36Sopenharmony_ci /* Default power good delay time 10us */ 16962306a36Sopenharmony_ci { RT6245_VIRT_PGDLYTIME, 1 }, 17062306a36Sopenharmony_ci /* Default slewrate 12.5mV/uS */ 17162306a36Sopenharmony_ci { RT6245_VIRT_SLEWRATE, 0 }, 17262306a36Sopenharmony_ci /* Default switch frequency 800KHz */ 17362306a36Sopenharmony_ci { RT6245_VIRT_SWFREQ, 1 }, 17462306a36Sopenharmony_ci /* Default voltage 750mV */ 17562306a36Sopenharmony_ci { RT6245_VIRT_VOUT, 0x19 } 17662306a36Sopenharmony_ci}; 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_cistatic const struct regmap_config rt6245_regmap_config = { 17962306a36Sopenharmony_ci .reg_bits = 8, 18062306a36Sopenharmony_ci .val_bits = 8, 18162306a36Sopenharmony_ci .max_register = RT6245_VIRT_VOUT, 18262306a36Sopenharmony_ci .cache_type = REGCACHE_FLAT, 18362306a36Sopenharmony_ci .reg_defaults = rt6245_reg_defaults, 18462306a36Sopenharmony_ci .num_reg_defaults = ARRAY_SIZE(rt6245_reg_defaults), 18562306a36Sopenharmony_ci .reg_write = rt6245_reg_write, 18662306a36Sopenharmony_ci}; 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_cistatic int rt6245_probe(struct i2c_client *i2c) 18962306a36Sopenharmony_ci{ 19062306a36Sopenharmony_ci struct rt6245_priv *priv; 19162306a36Sopenharmony_ci struct regmap *regmap; 19262306a36Sopenharmony_ci struct regulator_config regulator_cfg = {}; 19362306a36Sopenharmony_ci struct regulator_dev *rdev; 19462306a36Sopenharmony_ci int ret; 19562306a36Sopenharmony_ci 19662306a36Sopenharmony_ci priv = devm_kzalloc(&i2c->dev, sizeof(*priv), GFP_KERNEL); 19762306a36Sopenharmony_ci if (!priv) 19862306a36Sopenharmony_ci return -ENOMEM; 19962306a36Sopenharmony_ci 20062306a36Sopenharmony_ci priv->enable_state = true; 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_ci priv->enable_gpio = devm_gpiod_get_optional(&i2c->dev, "enable", GPIOD_OUT_HIGH); 20362306a36Sopenharmony_ci if (IS_ERR(priv->enable_gpio)) { 20462306a36Sopenharmony_ci dev_err(&i2c->dev, "Failed to get 'enable' gpio\n"); 20562306a36Sopenharmony_ci return PTR_ERR(priv->enable_gpio); 20662306a36Sopenharmony_ci } 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci usleep_range(RT6245_ENTIME_IN_US, RT6245_ENTIME_IN_US + 1000); 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_ci regmap = devm_regmap_init(&i2c->dev, NULL, i2c, &rt6245_regmap_config); 21162306a36Sopenharmony_ci if (IS_ERR(regmap)) { 21262306a36Sopenharmony_ci dev_err(&i2c->dev, "Failed to initialize the regmap\n"); 21362306a36Sopenharmony_ci return PTR_ERR(regmap); 21462306a36Sopenharmony_ci } 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci ret = rt6245_init_device_properties(&i2c->dev); 21762306a36Sopenharmony_ci if (ret) { 21862306a36Sopenharmony_ci dev_err(&i2c->dev, "Failed to initialize device properties\n"); 21962306a36Sopenharmony_ci return ret; 22062306a36Sopenharmony_ci } 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_ci regulator_cfg.dev = &i2c->dev; 22362306a36Sopenharmony_ci regulator_cfg.of_node = i2c->dev.of_node; 22462306a36Sopenharmony_ci regulator_cfg.regmap = regmap; 22562306a36Sopenharmony_ci regulator_cfg.driver_data = priv; 22662306a36Sopenharmony_ci regulator_cfg.init_data = of_get_regulator_init_data(&i2c->dev, i2c->dev.of_node, 22762306a36Sopenharmony_ci &rt6245_regulator_desc); 22862306a36Sopenharmony_ci rdev = devm_regulator_register(&i2c->dev, &rt6245_regulator_desc, ®ulator_cfg); 22962306a36Sopenharmony_ci if (IS_ERR(rdev)) { 23062306a36Sopenharmony_ci dev_err(&i2c->dev, "Failed to register regulator\n"); 23162306a36Sopenharmony_ci return PTR_ERR(rdev); 23262306a36Sopenharmony_ci } 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_ci return 0; 23562306a36Sopenharmony_ci} 23662306a36Sopenharmony_ci 23762306a36Sopenharmony_cistatic const struct of_device_id __maybe_unused rt6245_of_match_table[] = { 23862306a36Sopenharmony_ci { .compatible = "richtek,rt6245", }, 23962306a36Sopenharmony_ci {} 24062306a36Sopenharmony_ci}; 24162306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, rt6245_of_match_table); 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_cistatic struct i2c_driver rt6245_driver = { 24462306a36Sopenharmony_ci .driver = { 24562306a36Sopenharmony_ci .name = "rt6245", 24662306a36Sopenharmony_ci .probe_type = PROBE_PREFER_ASYNCHRONOUS, 24762306a36Sopenharmony_ci .of_match_table = rt6245_of_match_table, 24862306a36Sopenharmony_ci }, 24962306a36Sopenharmony_ci .probe = rt6245_probe, 25062306a36Sopenharmony_ci}; 25162306a36Sopenharmony_cimodule_i2c_driver(rt6245_driver); 25262306a36Sopenharmony_ci 25362306a36Sopenharmony_ciMODULE_AUTHOR("ChiYuan Huang <cy_huang@richtek.com>"); 25462306a36Sopenharmony_ciMODULE_DESCRIPTION("Richtek RT6245 Regulator Driver"); 25562306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 256