]> git.ipfire.org Git - thirdparty/u-boot.git/blame - drivers/power/power_core.c
Revert "Merge patch series "arm: dts: am62-beagleplay: Fix Beagleplay Ethernet""
[thirdparty/u-boot.git] / drivers / power / power_core.c
CommitLineData
83d290c5 1// SPDX-License-Identifier: GPL-2.0+
e542b7f0
ŁM
2/*
3 * Copyright (C) 2011 Samsung Electronics
4 * Lukasz Majewski <l.majewski@samsung.com>
5 *
6 * (C) Copyright 2010
7 * Stefano Babic, DENX Software Engineering, sbabic@denx.de
8 *
9 * (C) Copyright 2008-2009 Freescale Semiconductor, Inc.
e542b7f0
ŁM
10 */
11
d678a59d 12#include <common.h>
09140113 13#include <command.h>
f7ae49fc 14#include <log.h>
c7336815 15#include <malloc.h>
e542b7f0 16#include <linux/types.h>
c7336815
ŁM
17#include <linux/list.h>
18#include <power/pmic.h>
e542b7f0 19
c7336815 20static LIST_HEAD(pmic_list);
e542b7f0 21
c7336815 22int check_reg(struct pmic *p, u32 reg)
e542b7f0 23{
c7336815 24 if (reg >= p->number_of_regs) {
e542b7f0 25 printf("<reg num> = %d is invalid. Should be less than %d\n",
c7336815 26 reg, p->number_of_regs);
505cf475 27 return -EINVAL;
e542b7f0 28 }
c7336815 29
e542b7f0
ŁM
30 return 0;
31}
32
33int pmic_set_output(struct pmic *p, u32 reg, int out, int on)
34{
35 u32 val;
36
37 if (pmic_reg_read(p, reg, &val))
505cf475 38 return -ENOTSUPP;
e542b7f0
ŁM
39
40 if (on)
41 val |= out;
42 else
43 val &= ~out;
44
45 if (pmic_reg_write(p, reg, val))
505cf475 46 return -ENOTSUPP;
e542b7f0
ŁM
47
48 return 0;
49}
50
c7336815 51struct pmic *pmic_alloc(void)
e542b7f0 52{
c7336815
ŁM
53 struct pmic *p;
54
55 p = calloc(sizeof(*p), 1);
56 if (!p) {
57 printf("%s: No available memory for allocation!\n", __func__);
58 return NULL;
59 }
60
61 list_add_tail(&p->list, &pmic_list);
62
63 debug("%s: new pmic struct: 0x%p\n", __func__, p);
64
65 return p;
66}
67
68struct pmic *pmic_get(const char *s)
69{
70 struct pmic *p;
71
72 list_for_each_entry(p, &pmic_list, list) {
73 if (strcmp(p->name, s) == 0) {
74 debug("%s: pmic %s -> 0x%p\n", __func__, p->name, p);
75 return p;
76 }
77 }
78
79 return NULL;
80}
81
8c0bb858
TR
82#ifndef CONFIG_SPL_BUILD
83static int pmic_dump(struct pmic *p)
84{
85 int i, ret;
86 u32 val;
87
88 if (!p) {
89 puts("Wrong PMIC name!\n");
90 return -ENODEV;
91 }
92
93 printf("PMIC: %s\n", p->name);
94 for (i = 0; i < p->number_of_regs; i++) {
95 ret = pmic_reg_read(p, i, &val);
96 if (ret)
97 puts("PMIC: Registers dump failed\n");
98
99 if (!(i % 8))
100 printf("\n0x%02x: ", i);
101
102 printf("%08x ", val);
103 }
104 puts("\n");
105 return 0;
106}
107
108static const char *power_get_interface(int interface)
7d19629f
ŁM
109{
110 const char *power_interface[] = {"I2C", "SPI", "|+|-|"};
111 return power_interface[interface];
112}
113
c7336815
ŁM
114static void pmic_list_names(void)
115{
116 struct pmic *p;
117
118 puts("PMIC devices:\n");
119 list_for_each_entry(p, &pmic_list, list) {
7d19629f
ŁM
120 printf("name: %s bus: %s_%d\n", p->name,
121 power_get_interface(p->interface), p->bus);
c7336815 122 }
e542b7f0
ŁM
123}
124
09140113
SG
125static int do_pmic(struct cmd_tbl *cmdtp, int flag, int argc,
126 char *const argv[])
e542b7f0
ŁM
127{
128 u32 ret, reg, val;
7d19629f 129 char *cmd, *name;
c7336815 130 struct pmic *p;
e542b7f0 131
e542b7f0
ŁM
132 /* at least two arguments please */
133 if (argc < 2)
c7336815 134 return CMD_RET_USAGE;
e542b7f0 135
7d19629f 136 if (strcmp(argv[1], "list") == 0) {
c7336815
ŁM
137 pmic_list_names();
138 return CMD_RET_SUCCESS;
139 }
140
eae74537
ŁM
141 if (argc < 3)
142 return CMD_RET_USAGE;
143
7d19629f
ŁM
144 name = argv[1];
145 cmd = argv[2];
146
147 debug("%s: name: %s cmd: %s\n", __func__, name, cmd);
148 p = pmic_get(name);
149 if (!p)
150 return CMD_RET_FAILURE;
151
e542b7f0 152 if (strcmp(cmd, "dump") == 0) {
c7336815
ŁM
153 if (pmic_dump(p))
154 return CMD_RET_FAILURE;
155 return CMD_RET_SUCCESS;
e542b7f0
ŁM
156 }
157
158 if (strcmp(cmd, "read") == 0) {
c7336815
ŁM
159 if (argc < 4)
160 return CMD_RET_USAGE;
e542b7f0 161
7e5f460e 162 reg = hextoul(argv[3], NULL);
e542b7f0
ŁM
163 ret = pmic_reg_read(p, reg, &val);
164
165 if (ret)
166 puts("PMIC: Register read failed\n");
167
168 printf("\n0x%02x: 0x%08x\n", reg, val);
169
c7336815 170 return CMD_RET_SUCCESS;
e542b7f0
ŁM
171 }
172
173 if (strcmp(cmd, "write") == 0) {
c7336815
ŁM
174 if (argc < 5)
175 return CMD_RET_USAGE;
176
7e5f460e
SG
177 reg = hextoul(argv[3], NULL);
178 val = hextoul(argv[4], NULL);
e542b7f0
ŁM
179 pmic_reg_write(p, reg, val);
180
c7336815 181 return CMD_RET_SUCCESS;
e542b7f0
ŁM
182 }
183
7d19629f
ŁM
184 if (strcmp(cmd, "bat") == 0) {
185 if (argc < 4)
186 return CMD_RET_USAGE;
187
a6abaadc
PW
188 if (!p->pbat) {
189 printf("%s is not a battery\n", p->name);
190 return CMD_RET_FAILURE;
191 }
192
7d19629f
ŁM
193 if (strcmp(argv[3], "state") == 0)
194 p->fg->fg_battery_check(p->pbat->fg, p);
195
196 if (strcmp(argv[3], "charge") == 0) {
a6abaadc
PW
197 printf("BAT: %s charging (ctrl+c to break)\n",
198 p->name);
199 if (p->low_power_mode)
200 p->low_power_mode();
201 if (p->pbat->battery_charge)
202 p->pbat->battery_charge(p);
7d19629f
ŁM
203 }
204
205 return CMD_RET_SUCCESS;
206 }
207
e542b7f0 208 /* No subcommand found */
c7336815 209 return CMD_RET_SUCCESS;
e542b7f0
ŁM
210}
211
212U_BOOT_CMD(
213 pmic, CONFIG_SYS_MAXARGS, 1, do_pmic,
214 "PMIC",
c7336815 215 "list - list available PMICs\n"
7d19629f 216 "pmic name dump - dump named PMIC registers\n"
c7336815 217 "pmic name read <reg> - read register\n"
7d19629f
ŁM
218 "pmic name write <reg> <value> - write register\n"
219 "pmic name bat state - write register\n"
220 "pmic name bat charge - write register\n"
e542b7f0 221);
8c0bb858 222#endif