blob: de8d805566aacdddfb19a04f17f7d6548968c0c6 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Felix Brack9493f0e2017-11-30 13:52:37 +01002/*
3 * Copyright (C) EETS GmbH, 2017, Felix Brack <f.brack@eets.ch>
Felix Brack9493f0e2017-11-30 13:52:37 +01004 */
5
Felix Brack9493f0e2017-11-30 13:52:37 +01006#include <dm.h>
Svyatoslav Ryhelab97ca22023-10-24 10:49:07 +03007#include <dm/lists.h>
Felix Brack9493f0e2017-11-30 13:52:37 +01008#include <i2c.h>
Simon Glass0f2af882020-05-10 11:40:05 -06009#include <log.h>
Simon Glassbdd5f812023-09-14 18:21:46 -060010#include <linux/printk.h>
Felix Brack9493f0e2017-11-30 13:52:37 +010011#include <power/pmic.h>
12#include <power/regulator.h>
13#include <power/tps65910_pmic.h>
14
Svyatoslav Ryhel04081b52023-10-27 11:26:14 +030015static const struct pmic_child_info tps65910_children_info[] = {
Felix Brack9493f0e2017-11-30 13:52:37 +010016 { .prefix = "ldo_", .driver = TPS65910_LDO_DRIVER },
17 { .prefix = "buck_", .driver = TPS65910_BUCK_DRIVER },
18 { .prefix = "boost_", .driver = TPS65910_BOOST_DRIVER },
19 { },
20};
21
Svyatoslav Ryhel04081b52023-10-27 11:26:14 +030022static const struct pmic_child_info tps65911_children_info[] = {
23 { .prefix = "ldo", .driver = TPS65911_LDO_DRIVER },
24 { .prefix = "vdd", .driver = TPS65911_VDD_DRIVER },
25 { },
26};
27
Felix Brack9493f0e2017-11-30 13:52:37 +010028static int pmic_tps65910_reg_count(struct udevice *dev)
29{
30 return TPS65910_NUM_REGS;
31}
32
33static int pmic_tps65910_write(struct udevice *dev, uint reg, const u8 *buffer,
34 int len)
35{
36 int ret;
37
38 ret = dm_i2c_write(dev, reg, buffer, len);
39 if (ret)
Felix Brackabdd51d2017-12-18 15:38:28 +010040 pr_err("%s write error on register %02x\n", dev->name, reg);
Felix Brack9493f0e2017-11-30 13:52:37 +010041
42 return ret;
43}
44
45static int pmic_tps65910_read(struct udevice *dev, uint reg, u8 *buffer,
46 int len)
47{
48 int ret;
49
50 ret = dm_i2c_read(dev, reg, buffer, len);
51 if (ret)
Felix Brackabdd51d2017-12-18 15:38:28 +010052 pr_err("%s read error on register %02x\n", dev->name, reg);
Felix Brack9493f0e2017-11-30 13:52:37 +010053
54 return ret;
55}
56
57static int pmic_tps65910_bind(struct udevice *dev)
58{
Svyatoslav Ryhel04081b52023-10-27 11:26:14 +030059 const struct pmic_child_info *tps6591x_children_info =
60 (struct pmic_child_info *)dev_get_driver_data(dev);
Felix Brack9493f0e2017-11-30 13:52:37 +010061 ofnode regulators_node;
Svyatoslav Ryhelab97ca22023-10-24 10:49:07 +030062 int children, ret;
63
64 if (IS_ENABLED(CONFIG_SYSRESET_TPS65910)) {
65 ret = device_bind_driver(dev, TPS65910_RST_DRIVER,
66 "sysreset", NULL);
67 if (ret) {
68 log_err("cannot bind SYSRESET (ret = %d)\n", ret);
69 return ret;
70 }
71 }
Felix Brack9493f0e2017-11-30 13:52:37 +010072
73 regulators_node = dev_read_subnode(dev, "regulators");
74 if (!ofnode_valid(regulators_node)) {
75 debug("%s regulators subnode not found\n", dev->name);
76 return -EINVAL;
77 }
78
Svyatoslav Ryhel04081b52023-10-27 11:26:14 +030079 children = pmic_bind_children(dev, regulators_node, tps6591x_children_info);
Felix Brack9493f0e2017-11-30 13:52:37 +010080 if (!children)
81 debug("%s has no children (regulators)\n", dev->name);
82
83 return 0;
84}
85
86static int pmic_tps65910_probe(struct udevice *dev)
87{
88 /* use I2C control interface instead of I2C smartreflex interface to
89 * access smartrefelex registers VDD1_OP_REG, VDD1_SR_REG, VDD2_OP_REG
90 * and VDD2_SR_REG
91 */
92 return pmic_clrsetbits(dev, TPS65910_REG_DEVICE_CTRL, 0,
93 TPS65910_I2C_SEL_MASK);
94}
95
96static struct dm_pmic_ops pmic_tps65910_ops = {
97 .reg_count = pmic_tps65910_reg_count,
98 .read = pmic_tps65910_read,
99 .write = pmic_tps65910_write,
100};
101
102static const struct udevice_id pmic_tps65910_match[] = {
Svyatoslav Ryhel04081b52023-10-27 11:26:14 +0300103 { .compatible = "ti,tps65910", .data = (ulong)&tps65910_children_info },
104 { .compatible = "ti,tps65911", .data = (ulong)&tps65911_children_info },
Felix Brack9493f0e2017-11-30 13:52:37 +0100105 { /* sentinel */ }
106};
107
108U_BOOT_DRIVER(pmic_tps65910) = {
109 .name = "pmic_tps65910",
110 .id = UCLASS_PMIC,
111 .of_match = pmic_tps65910_match,
112 .bind = pmic_tps65910_bind,
113 .probe = pmic_tps65910_probe,
114 .ops = &pmic_tps65910_ops,
115};