1e53470feSOleksandr Tymoshenko /*-
24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
3af3dc4a7SPedro F. Giffuni *
4e53470feSOleksandr Tymoshenko * Copyright (c) 2011
5e53470feSOleksandr Tymoshenko * Ben Gray <ben.r.gray@gmail.com>.
6e53470feSOleksandr Tymoshenko * All rights reserved.
7e53470feSOleksandr Tymoshenko *
8e53470feSOleksandr Tymoshenko * Redistribution and use in source and binary forms, with or without
9e53470feSOleksandr Tymoshenko * modification, are permitted provided that the following conditions
10e53470feSOleksandr Tymoshenko * are met:
11e53470feSOleksandr Tymoshenko * 1. Redistributions of source code must retain the above copyright
12e53470feSOleksandr Tymoshenko * notice, this list of conditions and the following disclaimer.
13e53470feSOleksandr Tymoshenko * 2. Redistributions in binary form must reproduce the above copyright
14e53470feSOleksandr Tymoshenko * notice, this list of conditions and the following disclaimer in the
15e53470feSOleksandr Tymoshenko * documentation and/or other materials provided with the distribution.
16e53470feSOleksandr Tymoshenko *
17e53470feSOleksandr Tymoshenko * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18e53470feSOleksandr Tymoshenko * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19e53470feSOleksandr Tymoshenko * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20e53470feSOleksandr Tymoshenko * ARE DISCLAIMED. IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21e53470feSOleksandr Tymoshenko * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22e53470feSOleksandr Tymoshenko * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23e53470feSOleksandr Tymoshenko * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24e53470feSOleksandr Tymoshenko * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25e53470feSOleksandr Tymoshenko * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26e53470feSOleksandr Tymoshenko * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27e53470feSOleksandr Tymoshenko * SUCH DAMAGE.
28e53470feSOleksandr Tymoshenko */
29e53470feSOleksandr Tymoshenko
30e53470feSOleksandr Tymoshenko #include <sys/cdefs.h>
31e53470feSOleksandr Tymoshenko /*
32e53470feSOleksandr Tymoshenko * Texas Instruments TWL4030/TWL5030/TWL60x0/TPS659x0 Power Management and
33e53470feSOleksandr Tymoshenko * Audio CODEC devices.
34e53470feSOleksandr Tymoshenko *
35e53470feSOleksandr Tymoshenko * This code is based on the Linux TWL multifunctional device driver, which is
36e53470feSOleksandr Tymoshenko * copyright (C) 2005-2006 Texas Instruments, Inc.
37e53470feSOleksandr Tymoshenko *
38e53470feSOleksandr Tymoshenko * These chips are typically used as support ICs for the OMAP range of embedded
39e53470feSOleksandr Tymoshenko * ARM processes/SOC from Texas Instruments. They are typically used to control
40e53470feSOleksandr Tymoshenko * on board voltages, however some variants have other features like audio
41e53470feSOleksandr Tymoshenko * codecs, USB OTG transceivers, RTC, PWM, etc.
42e53470feSOleksandr Tymoshenko *
43e53470feSOleksandr Tymoshenko * This driver acts as a bus for more specific companion devices.
44e53470feSOleksandr Tymoshenko *
45e53470feSOleksandr Tymoshenko */
46e53470feSOleksandr Tymoshenko
47e53470feSOleksandr Tymoshenko #include <sys/param.h>
48e53470feSOleksandr Tymoshenko #include <sys/systm.h>
49e53470feSOleksandr Tymoshenko #include <sys/kernel.h>
50e53470feSOleksandr Tymoshenko #include <sys/lock.h>
51e53470feSOleksandr Tymoshenko #include <sys/module.h>
52e53470feSOleksandr Tymoshenko #include <sys/bus.h>
53e53470feSOleksandr Tymoshenko #include <sys/resource.h>
54e53470feSOleksandr Tymoshenko #include <sys/rman.h>
55e53470feSOleksandr Tymoshenko #include <sys/sysctl.h>
56e53470feSOleksandr Tymoshenko #include <sys/mutex.h>
57e53470feSOleksandr Tymoshenko #include <sys/malloc.h>
58e53470feSOleksandr Tymoshenko
59e53470feSOleksandr Tymoshenko #include <machine/bus.h>
60e53470feSOleksandr Tymoshenko #include <machine/resource.h>
61e53470feSOleksandr Tymoshenko #include <machine/intr.h>
62e53470feSOleksandr Tymoshenko
63e53470feSOleksandr Tymoshenko #include <dev/iicbus/iicbus.h>
64e53470feSOleksandr Tymoshenko #include <dev/iicbus/iiconf.h>
65e53470feSOleksandr Tymoshenko
66e53470feSOleksandr Tymoshenko #include <dev/ofw/openfirm.h>
67e53470feSOleksandr Tymoshenko #include <dev/ofw/ofw_bus.h>
68e53470feSOleksandr Tymoshenko #include <dev/ofw/ofw_bus_subr.h>
69e53470feSOleksandr Tymoshenko
70e53470feSOleksandr Tymoshenko #include "arm/ti/twl/twl.h"
71e53470feSOleksandr Tymoshenko
72e53470feSOleksandr Tymoshenko /* TWL device IDs */
73e53470feSOleksandr Tymoshenko #define TWL_DEVICE_UNKNOWN 0xffff
74e53470feSOleksandr Tymoshenko #define TWL_DEVICE_4030 0x4030
75e53470feSOleksandr Tymoshenko #define TWL_DEVICE_6025 0x6025
76e53470feSOleksandr Tymoshenko #define TWL_DEVICE_6030 0x6030
77e53470feSOleksandr Tymoshenko
78e53470feSOleksandr Tymoshenko /* Each TWL device typically has more than one I2C address */
79e53470feSOleksandr Tymoshenko #define TWL_MAX_SUBADDRS 4
80e53470feSOleksandr Tymoshenko
818b11850fSGordon Bergling /* The maximum number of bytes that can be written in one call */
82e53470feSOleksandr Tymoshenko #define TWL_MAX_IIC_DATA_SIZE 63
83e53470feSOleksandr Tymoshenko
84e53470feSOleksandr Tymoshenko /* The TWL devices typically use 4 I2C address for the different internal
85e53470feSOleksandr Tymoshenko * register sets, plus one SmartReflex I2C address.
86e53470feSOleksandr Tymoshenko */
87e53470feSOleksandr Tymoshenko #define TWL_CHIP_ID0 0x48
88e53470feSOleksandr Tymoshenko #define TWL_CHIP_ID1 0x49
89e53470feSOleksandr Tymoshenko #define TWL_CHIP_ID2 0x4A
90e53470feSOleksandr Tymoshenko #define TWL_CHIP_ID3 0x4B
91e53470feSOleksandr Tymoshenko
92e53470feSOleksandr Tymoshenko #define TWL_SMARTREFLEX_CHIP_ID 0x12
93e53470feSOleksandr Tymoshenko
94e53470feSOleksandr Tymoshenko #define TWL_INVALID_CHIP_ID 0xff
95e53470feSOleksandr Tymoshenko
96e53470feSOleksandr Tymoshenko struct twl_softc {
97e53470feSOleksandr Tymoshenko device_t sc_dev;
98e53470feSOleksandr Tymoshenko struct mtx sc_mtx;
99e53470feSOleksandr Tymoshenko unsigned int sc_type;
100e53470feSOleksandr Tymoshenko
101e53470feSOleksandr Tymoshenko uint8_t sc_subaddr_map[TWL_MAX_SUBADDRS];
102e53470feSOleksandr Tymoshenko
103e53470feSOleksandr Tymoshenko struct intr_config_hook sc_scan_hook;
104e53470feSOleksandr Tymoshenko
105e53470feSOleksandr Tymoshenko device_t sc_vreg;
106e53470feSOleksandr Tymoshenko device_t sc_clks;
107e53470feSOleksandr Tymoshenko };
108e53470feSOleksandr Tymoshenko
109e53470feSOleksandr Tymoshenko /**
110e53470feSOleksandr Tymoshenko * Macros for driver mutex locking
111e53470feSOleksandr Tymoshenko */
112e53470feSOleksandr Tymoshenko #define TWL_LOCK(_sc) mtx_lock(&(_sc)->sc_mtx)
113e53470feSOleksandr Tymoshenko #define TWL_UNLOCK(_sc) mtx_unlock(&(_sc)->sc_mtx)
114e53470feSOleksandr Tymoshenko #define TWL_LOCK_INIT(_sc) \
115e53470feSOleksandr Tymoshenko mtx_init(&_sc->sc_mtx, device_get_nameunit(_sc->sc_dev), \
116e53470feSOleksandr Tymoshenko "twl", MTX_DEF)
117e53470feSOleksandr Tymoshenko #define TWL_LOCK_DESTROY(_sc) mtx_destroy(&_sc->sc_mtx);
118e53470feSOleksandr Tymoshenko #define TWL_ASSERT_LOCKED(_sc) mtx_assert(&_sc->sc_mtx, MA_OWNED);
119e53470feSOleksandr Tymoshenko #define TWL_ASSERT_UNLOCKED(_sc) mtx_assert(&_sc->sc_mtx, MA_NOTOWNED);
120e53470feSOleksandr Tymoshenko
121e53470feSOleksandr Tymoshenko /**
122e53470feSOleksandr Tymoshenko * twl_is_4030 - returns true if the device is TWL4030
123e53470feSOleksandr Tymoshenko * twl_is_6025 - returns true if the device is TWL6025
124e53470feSOleksandr Tymoshenko * twl_is_6030 - returns true if the device is TWL6030
125e53470feSOleksandr Tymoshenko * @sc: device soft context
126e53470feSOleksandr Tymoshenko *
127e53470feSOleksandr Tymoshenko * Returns a non-zero value if the device matches.
128e53470feSOleksandr Tymoshenko *
129e53470feSOleksandr Tymoshenko * RETURNS:
130e53470feSOleksandr Tymoshenko * Returns a non-zero value if the device matches, otherwise zero.
131e53470feSOleksandr Tymoshenko */
132e53470feSOleksandr Tymoshenko int
twl_is_4030(device_t dev)133e53470feSOleksandr Tymoshenko twl_is_4030(device_t dev)
134e53470feSOleksandr Tymoshenko {
135e53470feSOleksandr Tymoshenko struct twl_softc *sc = device_get_softc(dev);
136e53470feSOleksandr Tymoshenko return (sc->sc_type == TWL_DEVICE_4030);
137e53470feSOleksandr Tymoshenko }
138e53470feSOleksandr Tymoshenko
139e53470feSOleksandr Tymoshenko int
twl_is_6025(device_t dev)140e53470feSOleksandr Tymoshenko twl_is_6025(device_t dev)
141e53470feSOleksandr Tymoshenko {
142e53470feSOleksandr Tymoshenko struct twl_softc *sc = device_get_softc(dev);
143e53470feSOleksandr Tymoshenko return (sc->sc_type == TWL_DEVICE_6025);
144e53470feSOleksandr Tymoshenko }
145e53470feSOleksandr Tymoshenko
146e53470feSOleksandr Tymoshenko int
twl_is_6030(device_t dev)147e53470feSOleksandr Tymoshenko twl_is_6030(device_t dev)
148e53470feSOleksandr Tymoshenko {
149e53470feSOleksandr Tymoshenko struct twl_softc *sc = device_get_softc(dev);
150e53470feSOleksandr Tymoshenko return (sc->sc_type == TWL_DEVICE_6030);
151e53470feSOleksandr Tymoshenko }
152e53470feSOleksandr Tymoshenko
153e53470feSOleksandr Tymoshenko /**
154e53470feSOleksandr Tymoshenko * twl_read - read one or more registers from the TWL device
155e53470feSOleksandr Tymoshenko * @sc: device soft context
156e53470feSOleksandr Tymoshenko * @nsub: the sub-module to read from
157e53470feSOleksandr Tymoshenko * @reg: the register offset within the module to read
158e53470feSOleksandr Tymoshenko * @buf: buffer to store the bytes in
159e53470feSOleksandr Tymoshenko * @cnt: the number of bytes to read
160e53470feSOleksandr Tymoshenko *
161e53470feSOleksandr Tymoshenko * Reads one or more registers and stores the result in the suppled buffer.
162e53470feSOleksandr Tymoshenko *
163e53470feSOleksandr Tymoshenko * RETURNS:
164e53470feSOleksandr Tymoshenko * Zero on success or an error code on failure.
165e53470feSOleksandr Tymoshenko */
166e53470feSOleksandr Tymoshenko int
twl_read(device_t dev,uint8_t nsub,uint8_t reg,uint8_t * buf,uint16_t cnt)167e53470feSOleksandr Tymoshenko twl_read(device_t dev, uint8_t nsub, uint8_t reg, uint8_t *buf, uint16_t cnt)
168e53470feSOleksandr Tymoshenko {
169e53470feSOleksandr Tymoshenko struct twl_softc *sc;
170e53470feSOleksandr Tymoshenko struct iic_msg msg[2];
171e53470feSOleksandr Tymoshenko uint8_t addr;
172e53470feSOleksandr Tymoshenko int rc;
173e53470feSOleksandr Tymoshenko
174e53470feSOleksandr Tymoshenko sc = device_get_softc(dev);
175e53470feSOleksandr Tymoshenko
176e53470feSOleksandr Tymoshenko TWL_LOCK(sc);
177e53470feSOleksandr Tymoshenko addr = sc->sc_subaddr_map[nsub];
178e53470feSOleksandr Tymoshenko TWL_UNLOCK(sc);
179e53470feSOleksandr Tymoshenko
180e53470feSOleksandr Tymoshenko if (addr == TWL_INVALID_CHIP_ID)
181e53470feSOleksandr Tymoshenko return (EIO);
182e53470feSOleksandr Tymoshenko
183e53470feSOleksandr Tymoshenko /* Set the address to read from */
184e53470feSOleksandr Tymoshenko msg[0].slave = addr;
185e53470feSOleksandr Tymoshenko msg[0].flags = IIC_M_WR | IIC_M_NOSTOP;
186e53470feSOleksandr Tymoshenko msg[0].len = 1;
187e53470feSOleksandr Tymoshenko msg[0].buf = ®
188e53470feSOleksandr Tymoshenko /* Read the data back */
189e53470feSOleksandr Tymoshenko msg[1].slave = addr;
190e53470feSOleksandr Tymoshenko msg[1].flags = IIC_M_RD;
191e53470feSOleksandr Tymoshenko msg[1].len = cnt;
192e53470feSOleksandr Tymoshenko msg[1].buf = buf;
193e53470feSOleksandr Tymoshenko
194e53470feSOleksandr Tymoshenko rc = iicbus_transfer(dev, msg, 2);
195e53470feSOleksandr Tymoshenko if (rc != 0) {
196e53470feSOleksandr Tymoshenko device_printf(dev, "iicbus read failed (adr:0x%02x, reg:0x%02x)\n",
197e53470feSOleksandr Tymoshenko addr, reg);
198e53470feSOleksandr Tymoshenko return (EIO);
199e53470feSOleksandr Tymoshenko }
200e53470feSOleksandr Tymoshenko
201e53470feSOleksandr Tymoshenko return (0);
202e53470feSOleksandr Tymoshenko }
203e53470feSOleksandr Tymoshenko
204e53470feSOleksandr Tymoshenko /**
205e53470feSOleksandr Tymoshenko * twl_write - writes one or more registers to the TWL device
206e53470feSOleksandr Tymoshenko * @sc: device soft context
207e53470feSOleksandr Tymoshenko * @nsub: the sub-module to read from
208e53470feSOleksandr Tymoshenko * @reg: the register offset within the module to read
209e53470feSOleksandr Tymoshenko * @buf: data to write
210e53470feSOleksandr Tymoshenko * @cnt: the number of bytes to write
211e53470feSOleksandr Tymoshenko *
212e53470feSOleksandr Tymoshenko * Writes one or more registers.
213e53470feSOleksandr Tymoshenko *
214e53470feSOleksandr Tymoshenko * RETURNS:
215e53470feSOleksandr Tymoshenko * Zero on success or a negative error code on failure.
216e53470feSOleksandr Tymoshenko */
217e53470feSOleksandr Tymoshenko int
twl_write(device_t dev,uint8_t nsub,uint8_t reg,uint8_t * buf,uint16_t cnt)218e53470feSOleksandr Tymoshenko twl_write(device_t dev, uint8_t nsub, uint8_t reg, uint8_t *buf, uint16_t cnt)
219e53470feSOleksandr Tymoshenko {
220e53470feSOleksandr Tymoshenko struct twl_softc *sc;
221e53470feSOleksandr Tymoshenko struct iic_msg msg;
222e53470feSOleksandr Tymoshenko uint8_t addr;
223e53470feSOleksandr Tymoshenko uint8_t tmp_buf[TWL_MAX_IIC_DATA_SIZE + 1];
224e53470feSOleksandr Tymoshenko int rc;
225e53470feSOleksandr Tymoshenko
226e53470feSOleksandr Tymoshenko if (cnt > TWL_MAX_IIC_DATA_SIZE)
227e53470feSOleksandr Tymoshenko return (ENOMEM);
228e53470feSOleksandr Tymoshenko
229e53470feSOleksandr Tymoshenko /* Set the register address as the first byte */
230e53470feSOleksandr Tymoshenko tmp_buf[0] = reg;
231e53470feSOleksandr Tymoshenko memcpy(&tmp_buf[1], buf, cnt);
232e53470feSOleksandr Tymoshenko
233e53470feSOleksandr Tymoshenko sc = device_get_softc(dev);
234e53470feSOleksandr Tymoshenko
235e53470feSOleksandr Tymoshenko TWL_LOCK(sc);
236e53470feSOleksandr Tymoshenko addr = sc->sc_subaddr_map[nsub];
237e53470feSOleksandr Tymoshenko TWL_UNLOCK(sc);
238e53470feSOleksandr Tymoshenko
239e53470feSOleksandr Tymoshenko if (addr == TWL_INVALID_CHIP_ID)
240e53470feSOleksandr Tymoshenko return (EIO);
241e53470feSOleksandr Tymoshenko
242e53470feSOleksandr Tymoshenko /* Setup the transfer and execute it */
243e53470feSOleksandr Tymoshenko msg.slave = addr;
244e53470feSOleksandr Tymoshenko msg.flags = IIC_M_WR;
245e53470feSOleksandr Tymoshenko msg.len = cnt + 1;
246e53470feSOleksandr Tymoshenko msg.buf = tmp_buf;
247e53470feSOleksandr Tymoshenko
248e53470feSOleksandr Tymoshenko rc = iicbus_transfer(dev, &msg, 1);
249e53470feSOleksandr Tymoshenko if (rc != 0) {
250e53470feSOleksandr Tymoshenko device_printf(sc->sc_dev, "iicbus write failed (adr:0x%02x, reg:0x%02x)\n",
251e53470feSOleksandr Tymoshenko addr, reg);
252e53470feSOleksandr Tymoshenko return (EIO);
253e53470feSOleksandr Tymoshenko }
254e53470feSOleksandr Tymoshenko
255e53470feSOleksandr Tymoshenko return (0);
256e53470feSOleksandr Tymoshenko }
257e53470feSOleksandr Tymoshenko
258e53470feSOleksandr Tymoshenko /**
259e53470feSOleksandr Tymoshenko * twl_test_present - checks if a device with given address is present
260e53470feSOleksandr Tymoshenko * @sc: device soft context
261e53470feSOleksandr Tymoshenko * @addr: the address of the device to scan for
262e53470feSOleksandr Tymoshenko *
263e53470feSOleksandr Tymoshenko * Sends just the address byte and checks for an ACK. If no ACK then device
264e53470feSOleksandr Tymoshenko * is assumed to not be present.
265e53470feSOleksandr Tymoshenko *
266e53470feSOleksandr Tymoshenko * RETURNS:
267e53470feSOleksandr Tymoshenko * EIO if device is not present, otherwise 0 is returned.
268e53470feSOleksandr Tymoshenko */
269e53470feSOleksandr Tymoshenko static int
twl_test_present(struct twl_softc * sc,uint8_t addr)270e53470feSOleksandr Tymoshenko twl_test_present(struct twl_softc *sc, uint8_t addr)
271e53470feSOleksandr Tymoshenko {
272e53470feSOleksandr Tymoshenko struct iic_msg msg;
273e53470feSOleksandr Tymoshenko uint8_t tmp;
274e53470feSOleksandr Tymoshenko
275e53470feSOleksandr Tymoshenko /* Set the address to read from */
276e53470feSOleksandr Tymoshenko msg.slave = addr;
277e53470feSOleksandr Tymoshenko msg.flags = IIC_M_RD;
278e53470feSOleksandr Tymoshenko msg.len = 1;
279e53470feSOleksandr Tymoshenko msg.buf = &tmp;
280e53470feSOleksandr Tymoshenko
281e53470feSOleksandr Tymoshenko if (iicbus_transfer(sc->sc_dev, &msg, 1) != 0)
282e53470feSOleksandr Tymoshenko return (EIO);
283e53470feSOleksandr Tymoshenko
284e53470feSOleksandr Tymoshenko return (0);
285e53470feSOleksandr Tymoshenko }
286e53470feSOleksandr Tymoshenko
287e53470feSOleksandr Tymoshenko /**
288e53470feSOleksandr Tymoshenko * twl_scan - scans the i2c bus for sub modules
289e53470feSOleksandr Tymoshenko * @dev: the twl device
290e53470feSOleksandr Tymoshenko *
291e53470feSOleksandr Tymoshenko * TWL devices don't just have one i2c slave address, rather they have up to
292e53470feSOleksandr Tymoshenko * 5 other addresses, each is for separate modules within the device. This
293e53470feSOleksandr Tymoshenko * function scans the bus for 4 possible sub-devices and stores the info
294e53470feSOleksandr Tymoshenko * internally.
295e53470feSOleksandr Tymoshenko *
296e53470feSOleksandr Tymoshenko */
297e53470feSOleksandr Tymoshenko static void
twl_scan(void * dev)298e53470feSOleksandr Tymoshenko twl_scan(void *dev)
299e53470feSOleksandr Tymoshenko {
300e53470feSOleksandr Tymoshenko struct twl_softc *sc;
301e53470feSOleksandr Tymoshenko unsigned i;
302e53470feSOleksandr Tymoshenko uint8_t devs[TWL_MAX_SUBADDRS];
303e53470feSOleksandr Tymoshenko uint8_t base = TWL_CHIP_ID0;
304e53470feSOleksandr Tymoshenko
305e53470feSOleksandr Tymoshenko sc = device_get_softc((device_t)dev);
306e53470feSOleksandr Tymoshenko
307e53470feSOleksandr Tymoshenko memset(devs, TWL_INVALID_CHIP_ID, TWL_MAX_SUBADDRS);
308e53470feSOleksandr Tymoshenko
309e53470feSOleksandr Tymoshenko /* Try each of the addresses (0x48, 0x49, 0x4a & 0x4b) to determine which
310e53470feSOleksandr Tymoshenko * sub modules we have.
311e53470feSOleksandr Tymoshenko */
312e53470feSOleksandr Tymoshenko for (i = 0; i < TWL_MAX_SUBADDRS; i++) {
313e53470feSOleksandr Tymoshenko if (twl_test_present(sc, (base + i)) == 0) {
314e53470feSOleksandr Tymoshenko devs[i] = (base + i);
315e53470feSOleksandr Tymoshenko device_printf(sc->sc_dev, "Found (sub)device at 0x%02x\n", (base + i));
316e53470feSOleksandr Tymoshenko }
317e53470feSOleksandr Tymoshenko }
318e53470feSOleksandr Tymoshenko
319e53470feSOleksandr Tymoshenko TWL_LOCK(sc);
320e53470feSOleksandr Tymoshenko memcpy(sc->sc_subaddr_map, devs, TWL_MAX_SUBADDRS);
321e53470feSOleksandr Tymoshenko TWL_UNLOCK(sc);
322e53470feSOleksandr Tymoshenko
323e53470feSOleksandr Tymoshenko /* Finished with the interrupt hook */
324e53470feSOleksandr Tymoshenko config_intrhook_disestablish(&sc->sc_scan_hook);
325e53470feSOleksandr Tymoshenko }
326e53470feSOleksandr Tymoshenko
327e53470feSOleksandr Tymoshenko /**
328e53470feSOleksandr Tymoshenko * twl_probe -
329e53470feSOleksandr Tymoshenko * @dev: the twl device
330e53470feSOleksandr Tymoshenko *
331e53470feSOleksandr Tymoshenko * Scans the FDT for a match for the device, possible compatible device
332e53470feSOleksandr Tymoshenko * strings are; "ti,twl6030", "ti,twl6025", "ti,twl4030".
333e53470feSOleksandr Tymoshenko *
334e53470feSOleksandr Tymoshenko * The FDT compat string also determines the type of device (it is currently
335e53470feSOleksandr Tymoshenko * not possible to dynamically determine the device type).
336e53470feSOleksandr Tymoshenko *
337e53470feSOleksandr Tymoshenko */
338e53470feSOleksandr Tymoshenko static int
twl_probe(device_t dev)339e53470feSOleksandr Tymoshenko twl_probe(device_t dev)
340e53470feSOleksandr Tymoshenko {
341e53470feSOleksandr Tymoshenko phandle_t node;
342e53470feSOleksandr Tymoshenko const char *compat;
343e53470feSOleksandr Tymoshenko int len, l;
344e53470feSOleksandr Tymoshenko struct twl_softc *sc;
345e53470feSOleksandr Tymoshenko
346e53470feSOleksandr Tymoshenko if ((compat = ofw_bus_get_compat(dev)) == NULL)
347e53470feSOleksandr Tymoshenko return (ENXIO);
348e53470feSOleksandr Tymoshenko
349e53470feSOleksandr Tymoshenko if ((node = ofw_bus_get_node(dev)) == 0)
350e53470feSOleksandr Tymoshenko return (ENXIO);
351e53470feSOleksandr Tymoshenko
352e53470feSOleksandr Tymoshenko /* Get total 'compatible' prop len */
353e53470feSOleksandr Tymoshenko if ((len = OF_getproplen(node, "compatible")) <= 0)
354e53470feSOleksandr Tymoshenko return (ENXIO);
355e53470feSOleksandr Tymoshenko
356e53470feSOleksandr Tymoshenko sc = device_get_softc(dev);
357e53470feSOleksandr Tymoshenko sc->sc_dev = dev;
358e53470feSOleksandr Tymoshenko sc->sc_type = TWL_DEVICE_UNKNOWN;
359e53470feSOleksandr Tymoshenko
360e53470feSOleksandr Tymoshenko while (len > 0) {
361e53470feSOleksandr Tymoshenko if (strncasecmp(compat, "ti,twl6030", 10) == 0)
362e53470feSOleksandr Tymoshenko sc->sc_type = TWL_DEVICE_6030;
363e53470feSOleksandr Tymoshenko else if (strncasecmp(compat, "ti,twl6025", 10) == 0)
364e53470feSOleksandr Tymoshenko sc->sc_type = TWL_DEVICE_6025;
365e53470feSOleksandr Tymoshenko else if (strncasecmp(compat, "ti,twl4030", 10) == 0)
366e53470feSOleksandr Tymoshenko sc->sc_type = TWL_DEVICE_4030;
367e53470feSOleksandr Tymoshenko
368e53470feSOleksandr Tymoshenko if (sc->sc_type != TWL_DEVICE_UNKNOWN)
369e53470feSOleksandr Tymoshenko break;
370e53470feSOleksandr Tymoshenko
371e53470feSOleksandr Tymoshenko /* Slide to the next sub-string. */
372e53470feSOleksandr Tymoshenko l = strlen(compat) + 1;
373e53470feSOleksandr Tymoshenko compat += l;
374e53470feSOleksandr Tymoshenko len -= l;
375e53470feSOleksandr Tymoshenko }
376e53470feSOleksandr Tymoshenko
377e53470feSOleksandr Tymoshenko switch (sc->sc_type) {
378e53470feSOleksandr Tymoshenko case TWL_DEVICE_4030:
379e53470feSOleksandr Tymoshenko device_set_desc(dev, "TI TWL4030/TPS659x0 Companion IC");
380e53470feSOleksandr Tymoshenko break;
381e53470feSOleksandr Tymoshenko case TWL_DEVICE_6025:
382e53470feSOleksandr Tymoshenko device_set_desc(dev, "TI TWL6025 Companion IC");
383e53470feSOleksandr Tymoshenko break;
384e53470feSOleksandr Tymoshenko case TWL_DEVICE_6030:
385e53470feSOleksandr Tymoshenko device_set_desc(dev, "TI TWL6030 Companion IC");
386e53470feSOleksandr Tymoshenko break;
387e53470feSOleksandr Tymoshenko case TWL_DEVICE_UNKNOWN:
388e53470feSOleksandr Tymoshenko default:
389e53470feSOleksandr Tymoshenko return (ENXIO);
390e53470feSOleksandr Tymoshenko }
391e53470feSOleksandr Tymoshenko
392e53470feSOleksandr Tymoshenko return (0);
393e53470feSOleksandr Tymoshenko }
394e53470feSOleksandr Tymoshenko
395e53470feSOleksandr Tymoshenko static int
twl_attach(device_t dev)396e53470feSOleksandr Tymoshenko twl_attach(device_t dev)
397e53470feSOleksandr Tymoshenko {
398e53470feSOleksandr Tymoshenko struct twl_softc *sc;
399e53470feSOleksandr Tymoshenko
400e53470feSOleksandr Tymoshenko sc = device_get_softc(dev);
401e53470feSOleksandr Tymoshenko sc->sc_dev = dev;
402e53470feSOleksandr Tymoshenko
403e53470feSOleksandr Tymoshenko TWL_LOCK_INIT(sc);
404e53470feSOleksandr Tymoshenko
405e53470feSOleksandr Tymoshenko /* We have to wait until interrupts are enabled. I2C read and write
406e53470feSOleksandr Tymoshenko * only works if the interrupts are available.
407e53470feSOleksandr Tymoshenko */
408e53470feSOleksandr Tymoshenko sc->sc_scan_hook.ich_func = twl_scan;
409e53470feSOleksandr Tymoshenko sc->sc_scan_hook.ich_arg = dev;
410e53470feSOleksandr Tymoshenko
411e53470feSOleksandr Tymoshenko if (config_intrhook_establish(&sc->sc_scan_hook) != 0)
412e53470feSOleksandr Tymoshenko return (ENOMEM);
413e53470feSOleksandr Tymoshenko
414e53470feSOleksandr Tymoshenko /* FIXME: should be in DTS file */
415e53470feSOleksandr Tymoshenko if ((sc->sc_vreg = device_add_child(dev, "twl_vreg", -1)) == NULL)
416e53470feSOleksandr Tymoshenko device_printf(dev, "could not allocate twl_vreg instance\n");
417e53470feSOleksandr Tymoshenko if ((sc->sc_clks = device_add_child(dev, "twl_clks", -1)) == NULL)
418e53470feSOleksandr Tymoshenko device_printf(dev, "could not allocate twl_clks instance\n");
419e53470feSOleksandr Tymoshenko
42018250ec6SJohn Baldwin bus_attach_children(dev);
42118250ec6SJohn Baldwin return (0);
422e53470feSOleksandr Tymoshenko }
423e53470feSOleksandr Tymoshenko
424e53470feSOleksandr Tymoshenko static int
twl_detach(device_t dev)425e53470feSOleksandr Tymoshenko twl_detach(device_t dev)
426e53470feSOleksandr Tymoshenko {
427e53470feSOleksandr Tymoshenko struct twl_softc *sc;
428*11a91178SJohn Baldwin int error;
429e53470feSOleksandr Tymoshenko
430e53470feSOleksandr Tymoshenko sc = device_get_softc(dev);
431e53470feSOleksandr Tymoshenko
432*11a91178SJohn Baldwin error = bus_generic_detach(dev);
433*11a91178SJohn Baldwin if (error != 0)
434*11a91178SJohn Baldwin return (error);
435e53470feSOleksandr Tymoshenko
436e53470feSOleksandr Tymoshenko TWL_LOCK_DESTROY(sc);
437e53470feSOleksandr Tymoshenko
438e53470feSOleksandr Tymoshenko return (0);
439e53470feSOleksandr Tymoshenko }
440e53470feSOleksandr Tymoshenko
441e53470feSOleksandr Tymoshenko static device_method_t twl_methods[] = {
442e53470feSOleksandr Tymoshenko DEVMETHOD(device_probe, twl_probe),
443e53470feSOleksandr Tymoshenko DEVMETHOD(device_attach, twl_attach),
444e53470feSOleksandr Tymoshenko DEVMETHOD(device_detach, twl_detach),
445e53470feSOleksandr Tymoshenko
446e53470feSOleksandr Tymoshenko {0, 0},
447e53470feSOleksandr Tymoshenko };
448e53470feSOleksandr Tymoshenko
449e53470feSOleksandr Tymoshenko static driver_t twl_driver = {
450e53470feSOleksandr Tymoshenko "twl",
451e53470feSOleksandr Tymoshenko twl_methods,
452e53470feSOleksandr Tymoshenko sizeof(struct twl_softc),
453e53470feSOleksandr Tymoshenko };
454e53470feSOleksandr Tymoshenko
4558537e671SJohn Baldwin DRIVER_MODULE(twl, iicbus, twl_driver, 0, 0);
456e53470feSOleksandr Tymoshenko MODULE_VERSION(twl, 1);
457