xref: /freebsd/sys/arm/ti/twl/twl.c (revision 11a9117871e6037ae7b8011b243939322efce569)
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 = &reg;
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