plumpower.c revision 1.13 1 /* $NetBSD: plumpower.c,v 1.13 2012/10/27 17:17:53 chs Exp $ */
2
3 /*-
4 * Copyright (c) 1999, 2000 The NetBSD Foundation, Inc.
5 * All rights reserved.
6 *
7 * This code is derived from software contributed to The NetBSD Foundation
8 * by UCHIYAMA Yasushi.
9 *
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
12 * are met:
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
30 */
31
32 #include <sys/cdefs.h>
33 __KERNEL_RCSID(0, "$NetBSD: plumpower.c,v 1.13 2012/10/27 17:17:53 chs Exp $");
34
35 #undef PLUMPOWERDEBUG
36
37 #include <sys/param.h>
38 #include <sys/systm.h>
39 #include <sys/device.h>
40 #include <sys/malloc.h>
41
42 #include <machine/bus.h>
43 #include <machine/intr.h>
44
45 #include <hpcmips/tx/tx39var.h>
46 #include <hpcmips/dev/plumvar.h>
47 #include <hpcmips/dev/plumpowervar.h>
48 #include <hpcmips/dev/plumpowerreg.h>
49
50 #ifdef PLUMPOWERDEBUG
51 int plumpower_debug = 1;
52 #define DPRINTF(arg) if (plumpower_debug) printf arg;
53 #define DPRINTFN(n, arg) if (plumpower_debug > (n)) printf arg;
54 #else
55 #define DPRINTF(arg)
56 #define DPRINTFN(n, arg)
57 #endif
58
59 int plumpower_match(device_t, cfdata_t, void *);
60 void plumpower_attach(device_t, device_t, void *);
61
62 struct plumpower_softc {
63 plum_chipset_tag_t sc_pc;
64 bus_space_tag_t sc_regt;
65 bus_space_handle_t sc_regh;
66 };
67
68 CFATTACH_DECL_NEW(plumpower, sizeof(struct plumpower_softc),
69 plumpower_match, plumpower_attach, NULL, NULL);
70
71 #ifdef PLUMPOWERDEBUG
72 static void plumpower_dump(struct plumpower_softc *);
73 #endif
74
75 int
76 plumpower_match(device_t parent, cfdata_t cf, void *aux)
77 {
78 return 2; /* 1st attach group */
79 }
80
81 void
82 plumpower_attach(device_t parent, device_t self, void *aux)
83 {
84 struct plum_attach_args *pa = aux;
85 struct plumpower_softc *sc = device_private(self);
86
87 printf("\n");
88 sc->sc_pc = pa->pa_pc;
89 sc->sc_regt = pa->pa_regt;
90
91 if (bus_space_map(sc->sc_regt, PLUM_POWER_REGBASE,
92 PLUM_POWER_REGSIZE, 0, &sc->sc_regh)) {
93 printf(": register map failed\n");
94 return;
95 }
96 plum_conf_register_power(sc->sc_pc, (void*)sc);
97 #ifdef PLUMPOWERDEBUG
98 plumpower_dump(sc);
99 #endif
100 /* disable all power/clock */
101 plum_conf_write(sc->sc_regt, sc->sc_regh,
102 PLUM_POWER_PWRCONT_REG, 0);
103 plum_conf_write(sc->sc_regt, sc->sc_regh,
104 PLUM_POWER_CLKCONT_REG, 0);
105
106 /* enable MCS interface from TX3922 */
107 plum_conf_write(sc->sc_regt, sc->sc_regh, PLUM_POWER_INPENA_REG,
108 PLUM_POWER_INPENA);
109 }
110
111 void
112 plum_power_ioreset(plum_chipset_tag_t pc)
113 {
114 struct plumpower_softc *sc = pc->pc_powert;
115 bus_space_tag_t regt = sc->sc_regt;
116 bus_space_handle_t regh = sc->sc_regh;
117
118 plum_conf_write(regt, regh, PLUM_POWER_RESETC_REG,
119 PLUM_POWER_RESETC_IO5CL1 |
120 PLUM_POWER_RESETC_IO5CL1);
121 plum_conf_write(regt, regh, PLUM_POWER_RESETC_REG, 0);
122 }
123
124 void*
125 plum_power_establish(plum_chipset_tag_t pc, int src)
126 {
127 struct plumpower_softc *sc = pc->pc_powert;
128 bus_space_tag_t regt = sc->sc_regt;
129 bus_space_handle_t regh = sc->sc_regh;
130 plumreg_t pwrreg, clkreg;
131
132 pwrreg = plum_conf_read(regt, regh, PLUM_POWER_PWRCONT_REG);
133 clkreg = plum_conf_read(regt, regh, PLUM_POWER_CLKCONT_REG);
134
135 switch(src) {
136 default:
137 panic("plum_power_establish: unknown power source");
138 case PLUM_PWR_LCD:
139 pwrreg |= PLUM_POWER_PWRCONT_LCDPWR;
140 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
141 pwrreg |= PLUM_POWER_PWRCONT_LCDDSP;
142 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
143 pwrreg |= PLUM_POWER_PWRCONT_LCDOE;
144 break;
145 case PLUM_PWR_BKL:
146 pwrreg |= PLUM_POWER_PWRCONT_BKLIGHT;
147 break;
148 case PLUM_PWR_IO5:
149 /* reset I/O bus (High/Low) */
150 plum_power_ioreset(pc);
151
152 /* supply power */
153 pwrreg |= PLUM_POWER_PWRCONT_IO5PWR;
154 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
155
156 /* output enable & supply clock */
157 pwrreg |= PLUM_POWER_PWRCONT_IO5OE;
158 clkreg |= PLUM_POWER_CLKCONT_IO5CLK;
159 break;
160 case PLUM_PWR_EXTPW0:
161 pwrreg |= PLUM_POWER_PWRCONT_EXTPW0;
162 break;
163 case PLUM_PWR_EXTPW1:
164 pwrreg |= PLUM_POWER_PWRCONT_EXTPW1;
165 break;
166 case PLUM_PWR_EXTPW2:
167 pwrreg |= PLUM_POWER_PWRCONT_EXTPW2;
168 break;
169 case PLUM_PWR_USB:
170 /* output enable */
171 pwrreg |= PLUM_POWER_PWRCONT_USBEN;
172 /* supply clock to the USB host controller */
173 clkreg |= PLUM_POWER_CLKCONT_USBCLK1;
174 /*
175 * clock supply is adaptively controlled by hardware
176 * (recommended)
177 */
178 clkreg &= ~PLUM_POWER_CLKCONT_USBCLK2;
179 break;
180 case PLUM_PWR_SM:
181 clkreg |= PLUM_POWER_CLKCONT_SMCLK;
182 break;
183 case PLUM_PWR_PCC1:
184 clkreg |= PLUM_POWER_CLKCONT_PCCCLK1;
185 break;
186 case PLUM_PWR_PCC2:
187 clkreg |= PLUM_POWER_CLKCONT_PCCCLK2;
188 break;
189 }
190
191 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
192 plum_conf_write(regt, regh, PLUM_POWER_CLKCONT_REG, clkreg);
193 #ifdef PLUMPOWERDEBUG
194 plumpower_dump(sc);
195 #endif
196 return (void*)src;
197 }
198
199 void
200 plum_power_disestablish(plum_chipset_tag_t pc, int ph)
201 {
202 struct plumpower_softc *sc = pc->pc_powert;
203 bus_space_tag_t regt = sc->sc_regt;
204 bus_space_handle_t regh = sc->sc_regh;
205 int src = (int)ph;
206 plumreg_t pwrreg, clkreg;
207
208 pwrreg = plum_conf_read(regt, regh, PLUM_POWER_PWRCONT_REG);
209 clkreg = plum_conf_read(regt, regh, PLUM_POWER_CLKCONT_REG);
210
211 switch(src) {
212 default:
213 panic("plum_power_disestablish: unknown power source");
214 case PLUM_PWR_LCD:
215 pwrreg &= ~PLUM_POWER_PWRCONT_LCDOE;
216 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
217 pwrreg &= ~PLUM_POWER_PWRCONT_LCDDSP;
218 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
219 pwrreg &= ~PLUM_POWER_PWRCONT_LCDPWR;
220 break;
221 case PLUM_PWR_BKL:
222 pwrreg &= ~PLUM_POWER_PWRCONT_BKLIGHT;
223 break;
224 case PLUM_PWR_IO5:
225 pwrreg &= ~(PLUM_POWER_PWRCONT_IO5PWR |
226 PLUM_POWER_PWRCONT_IO5OE);
227 clkreg &= ~PLUM_POWER_CLKCONT_IO5CLK;
228 break;
229 case PLUM_PWR_EXTPW0:
230 pwrreg &= ~PLUM_POWER_PWRCONT_EXTPW0;
231 break;
232 case PLUM_PWR_EXTPW1:
233 pwrreg &= ~PLUM_POWER_PWRCONT_EXTPW1;
234 break;
235 case PLUM_PWR_EXTPW2:
236 pwrreg &= ~PLUM_POWER_PWRCONT_EXTPW2;
237 break;
238 case PLUM_PWR_USB:
239 pwrreg &= ~PLUM_POWER_PWRCONT_USBEN;
240 clkreg &= ~(PLUM_POWER_CLKCONT_USBCLK1 |
241 PLUM_POWER_CLKCONT_USBCLK2);
242 break;
243 case PLUM_PWR_SM:
244 clkreg &= ~PLUM_POWER_CLKCONT_SMCLK;
245 break;
246 case PLUM_PWR_PCC1:
247 clkreg &= ~PLUM_POWER_CLKCONT_PCCCLK1;
248 break;
249 case PLUM_PWR_PCC2:
250 clkreg &= ~PLUM_POWER_CLKCONT_PCCCLK2;
251 break;
252 }
253
254 plum_conf_write(regt, regh, PLUM_POWER_PWRCONT_REG, pwrreg);
255 plum_conf_write(regt, regh, PLUM_POWER_CLKCONT_REG, clkreg);
256 #ifdef PLUMPOWERDEBUG
257 plumpower_dump(sc);
258 #endif
259 }
260
261 #ifdef PLUMPOWERDEBUG
262 #define ISPOWERSUPPLY(r, m) dbg_bitmask_print(r, PLUM_POWER_PWRCONT_##m, #m)
263 #define ISCLOCKSUPPLY(r, m) dbg_bitmask_print(r, PLUM_POWER_CLKCONT_##m, #m)
264 static void
265 plumpower_dump(struct plumpower_softc *sc)
266 {
267 bus_space_tag_t regt = sc->sc_regt;
268 bus_space_handle_t regh = sc->sc_regh;
269 plumreg_t reg;
270
271 reg = plum_conf_read(regt, regh, PLUM_POWER_PWRCONT_REG);
272 printf(" power:");
273 ISPOWERSUPPLY(reg, USBEN);
274 ISPOWERSUPPLY(reg, IO5OE);
275 ISPOWERSUPPLY(reg, LCDOE);
276 ISPOWERSUPPLY(reg, EXTPW2);
277 ISPOWERSUPPLY(reg, EXTPW1);
278 ISPOWERSUPPLY(reg, EXTPW0);
279 ISPOWERSUPPLY(reg, IO5PWR);
280 ISPOWERSUPPLY(reg, BKLIGHT);
281 ISPOWERSUPPLY(reg, LCDPWR);
282 ISPOWERSUPPLY(reg, LCDDSP);
283 reg = plum_conf_read(regt, regh, PLUM_POWER_CLKCONT_REG);
284 printf("\n clock:");
285 ISCLOCKSUPPLY(reg, USBCLK2);
286 ISCLOCKSUPPLY(reg, USBCLK1);
287 ISCLOCKSUPPLY(reg, IO5CLK);
288 ISCLOCKSUPPLY(reg, SMCLK);
289 ISCLOCKSUPPLY(reg, PCCCLK2);
290 ISCLOCKSUPPLY(reg, PCCCLK1);
291 reg = plum_conf_read(regt, regh, PLUM_POWER_INPENA_REG);
292 printf("\n MCS interface %sebled",
293 reg & PLUM_POWER_INPENA ? "en" : "dis");
294 reg = plum_conf_read(regt, regh, PLUM_POWER_RESETC_REG);
295 printf("\n IO5 reset:%s %s",
296 reg & PLUM_POWER_RESETC_IO5CL0 ? "CLRL" : "",
297 reg & PLUM_POWER_RESETC_IO5CL1 ? "CLRH" : "");
298 printf("\n");
299 }
300 #endif /* PLUMPOWERDEBUG */
301
302