xref: /openbsd-src/sys/arch/macppc/dev/i2s.c (revision 28d09237077354eff565f9f3ef991865a249ce6c)
1*28d09237Sjsg /*	$OpenBSD: i2s.c,v 1.39 2024/05/22 05:51:49 jsg Exp $	*/
2da53b1f9Sjoris /*	$NetBSD: i2s.c,v 1.1 2003/12/27 02:19:34 grant Exp $	*/
3da53b1f9Sjoris 
4da53b1f9Sjoris /*-
5da53b1f9Sjoris  * Copyright (c) 2002 Tsubai Masanari.  All rights reserved.
6da53b1f9Sjoris  *
7da53b1f9Sjoris  * Redistribution and use in source and binary forms, with or without
8da53b1f9Sjoris  * modification, are permitted provided that the following conditions
9da53b1f9Sjoris  * are met:
10da53b1f9Sjoris  * 1. Redistributions of source code must retain the above copyright
11da53b1f9Sjoris  *    notice, this list of conditions and the following disclaimer.
12da53b1f9Sjoris  * 2. Redistributions in binary form must reproduce the above copyright
13da53b1f9Sjoris  *    notice, this list of conditions and the following disclaimer in the
14da53b1f9Sjoris  *    documentation and/or other materials provided with the distribution.
15da53b1f9Sjoris  * 3. The name of the author may not be used to endorse or promote products
16da53b1f9Sjoris  *    derived from this software without specific prior written permission.
17da53b1f9Sjoris  *
18da53b1f9Sjoris  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19da53b1f9Sjoris  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20da53b1f9Sjoris  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21da53b1f9Sjoris  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22da53b1f9Sjoris  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23da53b1f9Sjoris  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24da53b1f9Sjoris  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25da53b1f9Sjoris  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26da53b1f9Sjoris  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27da53b1f9Sjoris  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28da53b1f9Sjoris  */
29da53b1f9Sjoris 
30da53b1f9Sjoris #include <sys/param.h>
31da53b1f9Sjoris #include <sys/audioio.h>
32da53b1f9Sjoris #include <sys/device.h>
33da53b1f9Sjoris #include <sys/malloc.h>
34da53b1f9Sjoris #include <sys/systm.h>
35da53b1f9Sjoris 
36da53b1f9Sjoris #include <dev/audio_if.h>
37da53b1f9Sjoris #include <dev/ofw/openfirm.h>
38da53b1f9Sjoris #include <macppc/dev/dbdma.h>
39da53b1f9Sjoris 
40da53b1f9Sjoris #include <uvm/uvm_extern.h>
41da53b1f9Sjoris 
42da53b1f9Sjoris #include <machine/autoconf.h>
43da53b1f9Sjoris #include <machine/pio.h>
44da53b1f9Sjoris 
45da53b1f9Sjoris #include <macppc/dev/i2svar.h>
46daa2e9a1Sgwk #include <macppc/dev/i2sreg.h>
4739dd359fSmpi #include <macppc/pci/macobio.h>
48da53b1f9Sjoris 
49da53b1f9Sjoris #ifdef I2S_DEBUG
50da53b1f9Sjoris # define DPRINTF(x) printf x
51da53b1f9Sjoris #else
52da53b1f9Sjoris # define DPRINTF(x)
53da53b1f9Sjoris #endif
54da53b1f9Sjoris 
5555882323Smpi void	i2s_mute(u_int, int);
56da53b1f9Sjoris int	i2s_cint(void *);
5755882323Smpi u_int	i2s_gpio_offset(struct i2s_softc *, char *, int *);
58da53b1f9Sjoris 
59f0d5bc97Stodd int	i2s_intr(void *);
60f0d5bc97Stodd int	i2s_iintr(void *);
61f0d5bc97Stodd 
62da53b1f9Sjoris struct cfdriver i2s_cd = {
63da53b1f9Sjoris 	NULL, "i2s", DV_DULL
64da53b1f9Sjoris };
65da53b1f9Sjoris 
66da53b1f9Sjoris void
i2s_attach(struct device * parent,struct i2s_softc * sc,struct confargs * ca)67da53b1f9Sjoris i2s_attach(struct device *parent, struct i2s_softc *sc, struct confargs *ca)
68da53b1f9Sjoris {
69da53b1f9Sjoris 	int cirq, oirq, iirq, cirq_type, oirq_type, iirq_type;
70eea27875Skettenis 	u_int32_t reg[6], intr[6];
71a05e39a7Smpi 	char compat[32];
72a05e39a7Smpi 	int child;
73da53b1f9Sjoris 
74eea27875Skettenis 	sc->sc_node = OF_child(ca->ca_node);
75eea27875Skettenis 	sc->sc_baseaddr = ca->ca_baseaddr;
76da53b1f9Sjoris 
77eea27875Skettenis 	OF_getprop(sc->sc_node, "reg", reg, sizeof reg);
78a05e39a7Smpi 
79a05e39a7Smpi 	child = OF_child(sc->sc_node);
80a05e39a7Smpi 	memset(compat, 0, sizeof(compat));
81a05e39a7Smpi 	OF_getprop(child, "compatible", compat, sizeof(compat));
82a05e39a7Smpi 
83a05e39a7Smpi 	/* Deal with broken device-tree on PowerMac7,2 and 7,3. */
84a05e39a7Smpi 	if (strcmp(compat, "AOAK2") == 0) {
85a05e39a7Smpi 		reg[0] += ca->ca_reg[0];
86a05e39a7Smpi 		reg[2] += ca->ca_reg[2];
87a05e39a7Smpi 		reg[4] += ca->ca_reg[2];
88a05e39a7Smpi 	}
89a05e39a7Smpi 
90eea27875Skettenis 	reg[0] += sc->sc_baseaddr;
91eea27875Skettenis 	reg[2] += sc->sc_baseaddr;
92eea27875Skettenis 	reg[4] += sc->sc_baseaddr;
93da53b1f9Sjoris 
94eea27875Skettenis 	sc->sc_reg = mapiodev(reg[0], reg[1]);
95da53b1f9Sjoris 
96da53b1f9Sjoris 	sc->sc_dmat = ca->ca_dmat;
97eea27875Skettenis 	sc->sc_odma = mapiodev(reg[2], reg[3]); /* out */
98eea27875Skettenis 	sc->sc_idma = mapiodev(reg[4], reg[5]); /* in */
99da53b1f9Sjoris 	sc->sc_odbdma = dbdma_alloc(sc->sc_dmat, I2S_DMALIST_MAX);
100da53b1f9Sjoris 	sc->sc_odmacmd = sc->sc_odbdma->d_addr;
101da53b1f9Sjoris 	sc->sc_idbdma = dbdma_alloc(sc->sc_dmat, I2S_DMALIST_MAX);
102da53b1f9Sjoris 	sc->sc_idmacmd = sc->sc_idbdma->d_addr;
103da53b1f9Sjoris 
104eea27875Skettenis 	OF_getprop(sc->sc_node, "interrupts", intr, sizeof intr);
105da53b1f9Sjoris 	cirq = intr[0];
106da53b1f9Sjoris 	oirq = intr[2];
107da53b1f9Sjoris 	iirq = intr[4];
108a74f2f29Smpi 	cirq_type = (intr[1] & 1) ? IST_LEVEL : IST_EDGE;
109a74f2f29Smpi 	oirq_type = (intr[3] & 1) ? IST_LEVEL : IST_EDGE;
110a74f2f29Smpi 	iirq_type = (intr[5] & 1) ? IST_LEVEL : IST_EDGE;
111da53b1f9Sjoris 
112da53b1f9Sjoris 	/* intr_establish(cirq, cirq_type, IPL_AUDIO, i2s_intr, sc); */
1133b53f463Smpi 	mac_intr_establish(parent, oirq, oirq_type, IPL_AUDIO | IPL_MPSAFE,
1143b53f463Smpi 	    i2s_intr, sc, sc->sc_dev.dv_xname);
1153b53f463Smpi 	mac_intr_establish(parent, iirq, iirq_type, IPL_AUDIO | IPL_MPSAFE,
1163b53f463Smpi 	    i2s_iintr, sc, sc->sc_dev.dv_xname);
117da53b1f9Sjoris 
118da53b1f9Sjoris 	printf(": irq %d,%d,%d\n", cirq, oirq, iirq);
119da53b1f9Sjoris 
120a05e39a7Smpi 	/* Need to be explicitly turned on some G5. */
121a05e39a7Smpi 	macobio_enable(I2SClockOffset, I2S0CLKEN|I2S0EN);
122a05e39a7Smpi 
123da53b1f9Sjoris 	i2s_set_rate(sc, 44100);
124041a1952Sjasper 	sc->sc_mute = 0;
125eea27875Skettenis 	i2s_gpio_init(sc, ca->ca_node, parent);
126da53b1f9Sjoris }
127da53b1f9Sjoris 
128da53b1f9Sjoris int
i2s_intr(void * v)12947d381f7Sjsg i2s_intr(void *v)
130da53b1f9Sjoris {
131da53b1f9Sjoris 	struct i2s_softc *sc = v;
132da53b1f9Sjoris 	struct dbdma_command *cmd = sc->sc_odmap;
133da53b1f9Sjoris 	u_int16_t c, status;
134da53b1f9Sjoris 
135886882aaSratchov 	mtx_enter(&audio_lock);
136886882aaSratchov 
137da53b1f9Sjoris 	/* if not set we are not running */
138886882aaSratchov 	if (!cmd) {
139886882aaSratchov 		mtx_leave(&audio_lock);
140da53b1f9Sjoris 		return (0);
141886882aaSratchov 	}
14274be753dSmglocker 	DPRINTF(("i2s_intr: cmd %p\n", cmd));
143da53b1f9Sjoris 
144da53b1f9Sjoris 	c = in16rb(&cmd->d_command);
145da53b1f9Sjoris 	status = in16rb(&cmd->d_status);
146da53b1f9Sjoris 
147da53b1f9Sjoris 	if (c >> 12 == DBDMA_CMD_OUT_LAST)
148da53b1f9Sjoris 		sc->sc_odmap = sc->sc_odmacmd;
149da53b1f9Sjoris 	else
150da53b1f9Sjoris 		sc->sc_odmap++;
151da53b1f9Sjoris 
152da53b1f9Sjoris 	if (c & (DBDMA_INT_ALWAYS << 4)) {
153da53b1f9Sjoris 		cmd->d_status = 0;
154da53b1f9Sjoris 		if (status)	/* status == 0x8400 */
155da53b1f9Sjoris 			if (sc->sc_ointr)
156da53b1f9Sjoris 				(*sc->sc_ointr)(sc->sc_oarg);
157da53b1f9Sjoris 	}
158886882aaSratchov 	mtx_leave(&audio_lock);
159da53b1f9Sjoris 	return 1;
160da53b1f9Sjoris }
161da53b1f9Sjoris 
162da53b1f9Sjoris int
i2s_iintr(void * v)16347d381f7Sjsg i2s_iintr(void *v)
164f0d5bc97Stodd {
165f0d5bc97Stodd 	struct i2s_softc *sc = v;
166f0d5bc97Stodd 	struct dbdma_command *cmd = sc->sc_idmap;
167f0d5bc97Stodd 	u_int16_t c, status;
168f0d5bc97Stodd 
169886882aaSratchov 	mtx_enter(&audio_lock);
170886882aaSratchov 
171f0d5bc97Stodd 	/* if not set we are not running */
172886882aaSratchov 	if (!cmd) {
173886882aaSratchov 		mtx_leave(&audio_lock);
174f0d5bc97Stodd 		return (0);
175886882aaSratchov 	}
17674be753dSmglocker 	DPRINTF(("i2s_intr: cmd %p\n", cmd));
177f0d5bc97Stodd 
178f0d5bc97Stodd 	c = in16rb(&cmd->d_command);
179f0d5bc97Stodd 	status = in16rb(&cmd->d_status);
180f0d5bc97Stodd 
181f0d5bc97Stodd 	if (c >> 12 == DBDMA_CMD_IN_LAST)
182f0d5bc97Stodd 		sc->sc_idmap = sc->sc_idmacmd;
183f0d5bc97Stodd 	else
184f0d5bc97Stodd 		sc->sc_idmap++;
185f0d5bc97Stodd 
186f0d5bc97Stodd 	if (c & (DBDMA_INT_ALWAYS << 4)) {
187f0d5bc97Stodd 		cmd->d_status = 0;
188f0d5bc97Stodd 		if (status)	/* status == 0x8400 */
189f0d5bc97Stodd 			if (sc->sc_iintr)
190f0d5bc97Stodd 				(*sc->sc_iintr)(sc->sc_iarg);
191f0d5bc97Stodd 	}
192886882aaSratchov 	mtx_leave(&audio_lock);
193f0d5bc97Stodd 	return 1;
194f0d5bc97Stodd }
195f0d5bc97Stodd 
196f0d5bc97Stodd int
i2s_open(void * h,int flags)19747d381f7Sjsg i2s_open(void *h, int flags)
198da53b1f9Sjoris {
199da53b1f9Sjoris 	return 0;
200da53b1f9Sjoris }
201da53b1f9Sjoris 
202da53b1f9Sjoris /*
203da53b1f9Sjoris  * Close function is called at splaudio().
204da53b1f9Sjoris  */
205da53b1f9Sjoris void
i2s_close(void * h)20647d381f7Sjsg i2s_close(void *h)
207da53b1f9Sjoris {
208da53b1f9Sjoris 	struct i2s_softc *sc = h;
209da53b1f9Sjoris 
210da53b1f9Sjoris 	i2s_halt_output(sc);
211da53b1f9Sjoris 	i2s_halt_input(sc);
212da53b1f9Sjoris 
213da53b1f9Sjoris 	sc->sc_ointr = 0;
214da53b1f9Sjoris 	sc->sc_iintr = 0;
215da53b1f9Sjoris }
216da53b1f9Sjoris 
217da53b1f9Sjoris int
i2s_set_params(void * h,int setmode,int usemode,struct audio_params * play,struct audio_params * rec)21847d381f7Sjsg i2s_set_params(void *h, int setmode, int usemode, struct audio_params *play,
21947d381f7Sjsg     struct audio_params  *rec)
220da53b1f9Sjoris {
221da53b1f9Sjoris 	struct i2s_softc *sc = h;
222da53b1f9Sjoris 	struct audio_params *p;
223da53b1f9Sjoris 	int mode;
224da53b1f9Sjoris 
225da53b1f9Sjoris 	p = play; /* default to play */
226da53b1f9Sjoris 
227da53b1f9Sjoris 	/*
228da53b1f9Sjoris 	 * This device only has one clock, so make the sample rates match.
229da53b1f9Sjoris 	 */
230da53b1f9Sjoris 	if (play->sample_rate != rec->sample_rate &&
231da53b1f9Sjoris 	    usemode == (AUMODE_PLAY | AUMODE_RECORD)) {
232da53b1f9Sjoris 		if (setmode == AUMODE_PLAY) {
233da53b1f9Sjoris 			rec->sample_rate = play->sample_rate;
234da53b1f9Sjoris 			setmode |= AUMODE_RECORD;
235da53b1f9Sjoris 		} else if (setmode == AUMODE_RECORD) {
236da53b1f9Sjoris 			play->sample_rate = rec->sample_rate;
237da53b1f9Sjoris 			setmode |= AUMODE_PLAY;
238da53b1f9Sjoris 		} else
239da53b1f9Sjoris 			return EINVAL;
240da53b1f9Sjoris 	}
241da53b1f9Sjoris 
242da53b1f9Sjoris 	for (mode = AUMODE_RECORD; mode != -1;
243da53b1f9Sjoris 	     mode = mode == AUMODE_RECORD ? AUMODE_PLAY : -1) {
244da53b1f9Sjoris 		if ((setmode & mode) == 0)
245da53b1f9Sjoris 			continue;
246da53b1f9Sjoris 
247da53b1f9Sjoris 		p = mode == AUMODE_PLAY ? play : rec;
248da53b1f9Sjoris 
2499f8ca1f5Stodd 		if (p->sample_rate < 4000)
2509f8ca1f5Stodd 			p->sample_rate = 4000;
2519f8ca1f5Stodd 		if (p->sample_rate > 50000)
2529f8ca1f5Stodd 			p->sample_rate = 50000;
2539f8ca1f5Stodd 		if (p->precision > 16)
2549f8ca1f5Stodd 			p->precision = 16;
2559f8ca1f5Stodd 		if (p->channels > 2)
2569f8ca1f5Stodd 			p->channels = 2;
257347c0a49Sratchov 		p->bps = AUDIO_BPS(p->precision);
258347c0a49Sratchov 		p->msb = 1;
259347c0a49Sratchov 		p->encoding = AUDIO_ENCODING_SLINEAR_BE;
260da53b1f9Sjoris 	}
261da53b1f9Sjoris 
262da53b1f9Sjoris 	/* Set the speed */
263da53b1f9Sjoris 	if (i2s_set_rate(sc, play->sample_rate))
264da53b1f9Sjoris 		return EINVAL;
265da53b1f9Sjoris 
266da53b1f9Sjoris 	p->sample_rate = sc->sc_rate;
267da53b1f9Sjoris 	return 0;
268da53b1f9Sjoris }
269da53b1f9Sjoris 
270da53b1f9Sjoris int
i2s_round_blocksize(void * h,int size)27147d381f7Sjsg i2s_round_blocksize(void *h, int size)
272da53b1f9Sjoris {
273da53b1f9Sjoris 	if (size < NBPG)
274da53b1f9Sjoris 		size = NBPG;
275da53b1f9Sjoris 	return size & ~PGOFSET;
276da53b1f9Sjoris }
277da53b1f9Sjoris 
278da53b1f9Sjoris int
i2s_halt_output(void * h)27947d381f7Sjsg i2s_halt_output(void *h)
280da53b1f9Sjoris {
281da53b1f9Sjoris 	struct i2s_softc *sc = h;
282da53b1f9Sjoris 
283da53b1f9Sjoris 	dbdma_stop(sc->sc_odma);
284da53b1f9Sjoris 	dbdma_reset(sc->sc_odma);
285da53b1f9Sjoris 	return 0;
286da53b1f9Sjoris }
287da53b1f9Sjoris 
288da53b1f9Sjoris int
i2s_halt_input(void * h)28947d381f7Sjsg i2s_halt_input(void *h)
290da53b1f9Sjoris {
291da53b1f9Sjoris 	struct i2s_softc *sc = h;
292da53b1f9Sjoris 
293da53b1f9Sjoris 	dbdma_stop(sc->sc_idma);
294da53b1f9Sjoris 	dbdma_reset(sc->sc_idma);
295da53b1f9Sjoris 	return 0;
296da53b1f9Sjoris }
297da53b1f9Sjoris 
298da53b1f9Sjoris enum {
299da53b1f9Sjoris 	I2S_OUTPUT_CLASS,
300da53b1f9Sjoris 	I2S_RECORD_CLASS,
301da53b1f9Sjoris 	I2S_OUTPUT_SELECT,
302da53b1f9Sjoris 	I2S_VOL_OUTPUT,
303da53b1f9Sjoris 	I2S_INPUT_SELECT,
304da53b1f9Sjoris 	I2S_VOL_INPUT,
305041a1952Sjasper 	I2S_MUTE, 		/* should be before bass/treble */
306ddfa3b0dSjoris 	I2S_BASS,
307ddfa3b0dSjoris 	I2S_TREBLE,
308da53b1f9Sjoris 	I2S_ENUM_LAST
309da53b1f9Sjoris };
310da53b1f9Sjoris 
311da53b1f9Sjoris int
i2s_set_port(void * h,mixer_ctrl_t * mc)312041a1952Sjasper i2s_set_port(void *h, mixer_ctrl_t *mc)
313da53b1f9Sjoris {
314da53b1f9Sjoris 	struct i2s_softc *sc = h;
315da53b1f9Sjoris 	int l, r;
316da53b1f9Sjoris 
317da53b1f9Sjoris 	DPRINTF(("i2s_set_port dev = %d, type = %d\n", mc->dev, mc->type));
318da53b1f9Sjoris 
319da53b1f9Sjoris 	l = mc->un.value.level[AUDIO_MIXER_LEVEL_LEFT];
320da53b1f9Sjoris 	r = mc->un.value.level[AUDIO_MIXER_LEVEL_RIGHT];
321da53b1f9Sjoris 
322da53b1f9Sjoris 	switch (mc->dev) {
323da53b1f9Sjoris 	case I2S_OUTPUT_SELECT:
324da53b1f9Sjoris 		/* No change necessary? */
325da53b1f9Sjoris 		if (mc->un.mask == sc->sc_output_mask)
326da53b1f9Sjoris 			return 0;
327da53b1f9Sjoris 
32855882323Smpi 		i2s_mute(sc->sc_spkr, 1);
32955882323Smpi 		i2s_mute(sc->sc_hp, 1);
33055882323Smpi 		i2s_mute(sc->sc_line, 1);
33155882323Smpi 		if (mc->un.mask & I2S_SELECT_SPEAKER)
33255882323Smpi 			i2s_mute(sc->sc_spkr, 0);
33355882323Smpi 		if (mc->un.mask & I2S_SELECT_HEADPHONE)
33455882323Smpi 			i2s_mute(sc->sc_hp, 0);
33555882323Smpi 		if (mc->un.mask & I2S_SELECT_LINEOUT)
33655882323Smpi 			i2s_mute(sc->sc_line, 0);
337da53b1f9Sjoris 
338da53b1f9Sjoris 		sc->sc_output_mask = mc->un.mask;
339da53b1f9Sjoris 		return 0;
340da53b1f9Sjoris 
341da53b1f9Sjoris 	case I2S_VOL_OUTPUT:
342da53b1f9Sjoris 		(*sc->sc_setvolume)(sc, l, r);
343da53b1f9Sjoris 		return 0;
344da53b1f9Sjoris 
345041a1952Sjasper 	case I2S_MUTE:
346041a1952Sjasper 		if (mc->type != AUDIO_MIXER_ENUM)
347041a1952Sjasper 			return (EINVAL);
348041a1952Sjasper 
349041a1952Sjasper 		sc->sc_mute = (mc->un.ord != 0);
350041a1952Sjasper 
351041a1952Sjasper 		if (sc->sc_mute) {
35255882323Smpi 			if (sc->sc_output_mask & I2S_SELECT_SPEAKER)
35355882323Smpi 				i2s_mute(sc->sc_spkr, 1);
35455882323Smpi 			if (sc->sc_output_mask & I2S_SELECT_HEADPHONE)
35555882323Smpi 				i2s_mute(sc->sc_hp, 1);
35655882323Smpi 			if (sc->sc_output_mask & I2S_SELECT_LINEOUT)
35755882323Smpi 				i2s_mute(sc->sc_line, 1);
358041a1952Sjasper 		} else {
35955882323Smpi 			if (sc->sc_output_mask & I2S_SELECT_SPEAKER)
36055882323Smpi 				i2s_mute(sc->sc_spkr, 0);
36155882323Smpi 			if (sc->sc_output_mask & I2S_SELECT_HEADPHONE)
36255882323Smpi 				i2s_mute(sc->sc_hp, 0);
36355882323Smpi 			if (sc->sc_output_mask & I2S_SELECT_LINEOUT)
36455882323Smpi 				i2s_mute(sc->sc_line, 0);
365041a1952Sjasper 		}
366041a1952Sjasper 
367041a1952Sjasper 		return (0);
368041a1952Sjasper 
369ddfa3b0dSjoris 	case I2S_BASS:
370ddfa3b0dSjoris 		if (sc->sc_setbass != NULL)
371ddfa3b0dSjoris 			(*sc->sc_setbass)(sc, l);
372ddfa3b0dSjoris 		return (0);
373ddfa3b0dSjoris 
374ddfa3b0dSjoris 	case I2S_TREBLE:
375ddfa3b0dSjoris 		if (sc->sc_settreble != NULL)
376ddfa3b0dSjoris 			(*sc->sc_settreble)(sc, l);
377ddfa3b0dSjoris 		return (0);
378ddfa3b0dSjoris 
379da53b1f9Sjoris 	case I2S_INPUT_SELECT:
380da53b1f9Sjoris 		/* no change necessary? */
381da53b1f9Sjoris 		if (mc->un.mask == sc->sc_record_source)
382da53b1f9Sjoris 			return 0;
383da53b1f9Sjoris 		switch (mc->un.mask) {
38455882323Smpi 		case I2S_SELECT_SPEAKER:
38555882323Smpi 		case I2S_SELECT_HEADPHONE:
386da53b1f9Sjoris 			/* XXX TO BE DONE */
387da53b1f9Sjoris 			break;
388da53b1f9Sjoris 		default: /* invalid argument */
389da53b1f9Sjoris 			return EINVAL;
390da53b1f9Sjoris 		}
391f0d5bc97Stodd 		if (sc->sc_setinput != NULL)
392f0d5bc97Stodd 			(*sc->sc_setinput)(sc, mc->un.mask);
393da53b1f9Sjoris 		sc->sc_record_source = mc->un.mask;
394da53b1f9Sjoris 		return 0;
395da53b1f9Sjoris 
396da53b1f9Sjoris 	case I2S_VOL_INPUT:
397da53b1f9Sjoris 		/* XXX TO BE DONE */
398da53b1f9Sjoris 		return 0;
399da53b1f9Sjoris 	}
400da53b1f9Sjoris 
401da53b1f9Sjoris 	return ENXIO;
402da53b1f9Sjoris }
403da53b1f9Sjoris 
404da53b1f9Sjoris int
i2s_get_port(void * h,mixer_ctrl_t * mc)405041a1952Sjasper i2s_get_port(void *h, mixer_ctrl_t *mc)
406da53b1f9Sjoris {
407da53b1f9Sjoris 	struct i2s_softc *sc = h;
408da53b1f9Sjoris 
409da53b1f9Sjoris 	DPRINTF(("i2s_get_port dev = %d, type = %d\n", mc->dev, mc->type));
410da53b1f9Sjoris 
411da53b1f9Sjoris 	switch (mc->dev) {
412da53b1f9Sjoris 	case I2S_OUTPUT_SELECT:
413da53b1f9Sjoris 		mc->un.mask = sc->sc_output_mask;
414da53b1f9Sjoris 		return 0;
415da53b1f9Sjoris 
416da53b1f9Sjoris 	case I2S_VOL_OUTPUT:
417da53b1f9Sjoris 		mc->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = sc->sc_vol_l;
418da53b1f9Sjoris 		mc->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = sc->sc_vol_r;
419da53b1f9Sjoris 		return 0;
420da53b1f9Sjoris 
421041a1952Sjasper 	case I2S_MUTE:
422041a1952Sjasper 		mc->un.ord = sc->sc_mute;
423041a1952Sjasper 		return (0);
424041a1952Sjasper 
425da53b1f9Sjoris 	case I2S_INPUT_SELECT:
426da53b1f9Sjoris 		mc->un.mask = sc->sc_record_source;
427da53b1f9Sjoris 		return 0;
428da53b1f9Sjoris 
429ddfa3b0dSjoris 	case I2S_BASS:
430ce98e251Sjakemsr 		if (mc->un.value.num_channels != 1)
431ce98e251Sjakemsr 			return ENXIO;
432ddfa3b0dSjoris 		mc->un.value.level[AUDIO_MIXER_LEVEL_MONO] = sc->sc_bass;
433ce98e251Sjakemsr 		return 0;
434ddfa3b0dSjoris 
435ddfa3b0dSjoris 	case I2S_TREBLE:
436ce98e251Sjakemsr 		if (mc->un.value.num_channels != 1)
437ce98e251Sjakemsr 			return ENXIO;
438ddfa3b0dSjoris 		mc->un.value.level[AUDIO_MIXER_LEVEL_MONO] = sc->sc_treble;
439ce98e251Sjakemsr 		return 0;
440ddfa3b0dSjoris 
441da53b1f9Sjoris 	case I2S_VOL_INPUT:
442da53b1f9Sjoris 		/* XXX TO BE DONE */
443da53b1f9Sjoris 		mc->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = 0;
444da53b1f9Sjoris 		mc->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = 0;
445da53b1f9Sjoris 		return 0;
446da53b1f9Sjoris 
447da53b1f9Sjoris 	default:
448da53b1f9Sjoris 		return ENXIO;
449da53b1f9Sjoris 	}
450da53b1f9Sjoris 
451da53b1f9Sjoris 	return 0;
452da53b1f9Sjoris }
453da53b1f9Sjoris 
454da53b1f9Sjoris int
i2s_query_devinfo(void * h,mixer_devinfo_t * dip)45590633bbeSjasper i2s_query_devinfo(void *h, mixer_devinfo_t *dip)
456da53b1f9Sjoris {
4576a340eeeSkettenis 	struct i2s_softc *sc = h;
458c8d53f6dSkettenis 	int n = 0;
459c8d53f6dSkettenis 
460da53b1f9Sjoris 	switch (dip->index) {
461da53b1f9Sjoris 
462da53b1f9Sjoris 	case I2S_OUTPUT_SELECT:
463a7f41e16Sbrad 		dip->mixer_class = I2S_OUTPUT_CLASS;
464a7f41e16Sbrad 		strlcpy(dip->label.name, AudioNselect, sizeof(dip->label.name));
465da53b1f9Sjoris 		dip->type = AUDIO_MIXER_SET;
466da53b1f9Sjoris 		dip->prev = dip->next = AUDIO_MIXER_LAST;
467c8d53f6dSkettenis 		strlcpy(dip->un.s.member[n].label.name, AudioNspeaker,
468c8d53f6dSkettenis 		    sizeof(dip->un.s.member[n].label.name));
46955882323Smpi 		dip->un.s.member[n++].mask = I2S_SELECT_SPEAKER;
47055882323Smpi 		if (sc->sc_hp) {
471c8d53f6dSkettenis 			strlcpy(dip->un.s.member[n].label.name,
472c8d53f6dSkettenis 			    AudioNheadphone,
473c8d53f6dSkettenis 			    sizeof(dip->un.s.member[n].label.name));
47455882323Smpi 			dip->un.s.member[n++].mask = I2S_SELECT_HEADPHONE;
475c8d53f6dSkettenis 		}
47655882323Smpi 		if (sc->sc_line) {
477c8d53f6dSkettenis 			strlcpy(dip->un.s.member[n].label.name,	AudioNline,
478c8d53f6dSkettenis 			    sizeof(dip->un.s.member[n].label.name));
47955882323Smpi 			dip->un.s.member[n++].mask = I2S_SELECT_LINEOUT;
480c8d53f6dSkettenis 		}
481c8d53f6dSkettenis 		dip->un.s.num_mem = n;
482da53b1f9Sjoris 		return 0;
483da53b1f9Sjoris 
484da53b1f9Sjoris 	case I2S_VOL_OUTPUT:
485a7f41e16Sbrad 		dip->mixer_class = I2S_OUTPUT_CLASS;
486da53b1f9Sjoris 		strlcpy(dip->label.name, AudioNmaster, sizeof(dip->label.name));
487da53b1f9Sjoris 		dip->type = AUDIO_MIXER_VALUE;
488041a1952Sjasper 		dip->prev = AUDIO_MIXER_LAST;
489041a1952Sjasper 		dip->next = I2S_MUTE;
490da53b1f9Sjoris 		dip->un.v.num_channels = 2;
49190633bbeSjasper 		dip->un.v.delta = 8;
492da53b1f9Sjoris 		strlcpy(dip->un.v.units.name, AudioNvolume,
493da53b1f9Sjoris 		    sizeof(dip->un.v.units.name));
494da53b1f9Sjoris 		return 0;
495da53b1f9Sjoris 
496041a1952Sjasper 	case I2S_MUTE:
497041a1952Sjasper 		dip->mixer_class = I2S_OUTPUT_CLASS;
498041a1952Sjasper 		dip->prev = I2S_VOL_OUTPUT;
499041a1952Sjasper 		dip->next = AUDIO_MIXER_LAST;
500041a1952Sjasper 		strlcpy(dip->label.name, AudioNmute, sizeof(dip->label.name));
501041a1952Sjasper 		dip->type = AUDIO_MIXER_ENUM;
502041a1952Sjasper 		dip->un.e.num_mem = 2;
503041a1952Sjasper 		strlcpy(dip->un.e.member[0].label.name, AudioNoff,
504041a1952Sjasper 		    sizeof dip->un.e.member[0].label.name);
505041a1952Sjasper 		dip->un.e.member[0].ord = 0;
506041a1952Sjasper 		strlcpy(dip->un.e.member[1].label.name, AudioNon,
507041a1952Sjasper 		    sizeof dip->un.e.member[1].label.name);
508041a1952Sjasper 		dip->un.e.member[1].ord = 1;
509041a1952Sjasper 		return (0);
510041a1952Sjasper 
511da53b1f9Sjoris 	case I2S_INPUT_SELECT:
512da53b1f9Sjoris 		dip->mixer_class = I2S_RECORD_CLASS;
513da53b1f9Sjoris 		strlcpy(dip->label.name, AudioNsource, sizeof(dip->label.name));
514da53b1f9Sjoris 		dip->type = AUDIO_MIXER_SET;
515da53b1f9Sjoris 		dip->prev = dip->next = AUDIO_MIXER_LAST;
516f0d5bc97Stodd 		dip->un.s.num_mem = 2;
517f0d5bc97Stodd 		strlcpy(dip->un.s.member[0].label.name, AudioNmicrophone,
518da53b1f9Sjoris 		    sizeof(dip->un.s.member[0].label.name));
51955882323Smpi 		dip->un.s.member[0].mask = I2S_SELECT_SPEAKER;
520f0d5bc97Stodd 		strlcpy(dip->un.s.member[1].label.name, AudioNline,
521da53b1f9Sjoris 		    sizeof(dip->un.s.member[1].label.name));
52255882323Smpi 		dip->un.s.member[1].mask = I2S_SELECT_HEADPHONE;
523da53b1f9Sjoris 		return 0;
524da53b1f9Sjoris 
525da53b1f9Sjoris 	case I2S_VOL_INPUT:
526da53b1f9Sjoris 		dip->mixer_class = I2S_RECORD_CLASS;
527da53b1f9Sjoris 		strlcpy(dip->label.name, AudioNrecord, sizeof(dip->label.name));
528da53b1f9Sjoris 		dip->type = AUDIO_MIXER_VALUE;
529da53b1f9Sjoris 		dip->prev = dip->next = AUDIO_MIXER_LAST;
530da53b1f9Sjoris 		dip->un.v.num_channels = 2;
531da53b1f9Sjoris 		strlcpy(dip->un.v.units.name, AudioNvolume,
532da53b1f9Sjoris 		    sizeof(dip->un.v.units.name));
533da53b1f9Sjoris 		return 0;
534da53b1f9Sjoris 
535da53b1f9Sjoris 	case I2S_OUTPUT_CLASS:
536da53b1f9Sjoris 		dip->mixer_class = I2S_OUTPUT_CLASS;
537da53b1f9Sjoris 		strlcpy(dip->label.name, AudioCoutputs,
538da53b1f9Sjoris 		    sizeof(dip->label.name));
539da53b1f9Sjoris 		dip->type = AUDIO_MIXER_CLASS;
540da53b1f9Sjoris 		dip->next = dip->prev = AUDIO_MIXER_LAST;
541da53b1f9Sjoris 		return 0;
542da53b1f9Sjoris 
543da53b1f9Sjoris 	case I2S_RECORD_CLASS:
544da53b1f9Sjoris 		dip->mixer_class = I2S_RECORD_CLASS;
545da53b1f9Sjoris 		strlcpy(dip->label.name, AudioCrecord, sizeof(dip->label.name));
546da53b1f9Sjoris 		dip->type = AUDIO_MIXER_CLASS;
547da53b1f9Sjoris 		dip->next = dip->prev = AUDIO_MIXER_LAST;
548da53b1f9Sjoris 		return 0;
549ddfa3b0dSjoris 
550ddfa3b0dSjoris 	case I2S_BASS:
5516a340eeeSkettenis 		if (sc->sc_setbass == NULL)
5526a340eeeSkettenis 			return (ENXIO);
5536a340eeeSkettenis 		dip->mixer_class = I2S_OUTPUT_CLASS;
554ddfa3b0dSjoris 		strlcpy(dip->label.name, AudioNbass, sizeof(dip->label.name));
555ddfa3b0dSjoris 		dip->type = AUDIO_MIXER_VALUE;
556ddfa3b0dSjoris 		dip->prev = dip->next = AUDIO_MIXER_LAST;
557ddfa3b0dSjoris 		dip->un.v.num_channels = 1;
558ddfa3b0dSjoris 		return (0);
559ddfa3b0dSjoris 
560ddfa3b0dSjoris 	case I2S_TREBLE:
5616a340eeeSkettenis 		if (sc->sc_settreble == NULL)
5626a340eeeSkettenis 			return (ENXIO);
5636a340eeeSkettenis 		dip->mixer_class = I2S_OUTPUT_CLASS;
564ddfa3b0dSjoris 		strlcpy(dip->label.name, AudioNtreble, sizeof(dip->label.name));
565ddfa3b0dSjoris 		dip->type = AUDIO_MIXER_VALUE;
566ddfa3b0dSjoris 		dip->prev = dip->next = AUDIO_MIXER_LAST;
567ddfa3b0dSjoris 		dip->un.v.num_channels = 1;
568ddfa3b0dSjoris 		return (0);
569da53b1f9Sjoris 	}
570da53b1f9Sjoris 
571da53b1f9Sjoris 	return ENXIO;
572da53b1f9Sjoris }
573da53b1f9Sjoris 
574da53b1f9Sjoris size_t
i2s_round_buffersize(void * h,int dir,size_t size)57547d381f7Sjsg i2s_round_buffersize(void *h, int dir, size_t size)
576da53b1f9Sjoris {
577da53b1f9Sjoris 	if (size > 65536)
578da53b1f9Sjoris 		size = 65536;
579da53b1f9Sjoris 	return size;
580da53b1f9Sjoris }
581da53b1f9Sjoris 
582da53b1f9Sjoris int
i2s_trigger_output(void * h,void * start,void * end,int bsize,void (* intr)(void *),void * arg,struct audio_params * param)58347d381f7Sjsg i2s_trigger_output(void *h, void *start, void *end, int bsize,
58447d381f7Sjsg     void (*intr)(void *), void *arg, struct audio_params *param)
585da53b1f9Sjoris {
586da53b1f9Sjoris 	struct i2s_softc *sc = h;
587da53b1f9Sjoris 	struct i2s_dma *p;
588da53b1f9Sjoris 	struct dbdma_command *cmd = sc->sc_odmacmd;
589da53b1f9Sjoris 	vaddr_t spa, pa, epa;
590da53b1f9Sjoris 	int c;
591da53b1f9Sjoris 
592da53b1f9Sjoris 	DPRINTF(("trigger_output %p %p 0x%x\n", start, end, bsize));
593da53b1f9Sjoris 
59436dba039Sjsg 	for (p = sc->sc_dmas; p && p->addr != start; p = p->next)
59536dba039Sjsg 		;
596da53b1f9Sjoris 	if (!p)
597da53b1f9Sjoris 		return -1;
598da53b1f9Sjoris 
599da53b1f9Sjoris 	sc->sc_ointr = intr;
600da53b1f9Sjoris 	sc->sc_oarg = arg;
601da53b1f9Sjoris 	sc->sc_odmap = sc->sc_odmacmd;
602da53b1f9Sjoris 
603da53b1f9Sjoris 	spa = p->segs[0].ds_addr;
604da53b1f9Sjoris 	c = DBDMA_CMD_OUT_MORE;
605da53b1f9Sjoris 	for (pa = spa, epa = spa + (end - start);
606da53b1f9Sjoris 	    pa < epa; pa += bsize, cmd++) {
607da53b1f9Sjoris 
608da53b1f9Sjoris 		if (pa + bsize == epa)
609da53b1f9Sjoris 			c = DBDMA_CMD_OUT_LAST;
610da53b1f9Sjoris 
611da53b1f9Sjoris 		DBDMA_BUILD(cmd, c, 0, bsize, pa, DBDMA_INT_ALWAYS,
612da53b1f9Sjoris 			DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER);
613da53b1f9Sjoris 	}
614da53b1f9Sjoris 
615da53b1f9Sjoris 	DBDMA_BUILD(cmd, DBDMA_CMD_NOP, 0, 0, 0,
616da53b1f9Sjoris 		DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_ALWAYS);
617da53b1f9Sjoris 	dbdma_st32(&cmd->d_cmddep, sc->sc_odbdma->d_paddr);
618da53b1f9Sjoris 
619da53b1f9Sjoris 	dbdma_start(sc->sc_odma, sc->sc_odbdma);
620da53b1f9Sjoris 
621da53b1f9Sjoris 	return 0;
622da53b1f9Sjoris }
623da53b1f9Sjoris 
624da53b1f9Sjoris int
i2s_trigger_input(void * h,void * start,void * end,int bsize,void (* intr)(void *),void * arg,struct audio_params * param)62547d381f7Sjsg i2s_trigger_input(void *h, void *start, void *end, int bsize,
62647d381f7Sjsg     void (*intr)(void *), void *arg, struct audio_params *param)
627da53b1f9Sjoris {
628f0d5bc97Stodd 	struct i2s_softc *sc = h;
629f0d5bc97Stodd 	struct i2s_dma *p;
630f0d5bc97Stodd 	struct dbdma_command *cmd = sc->sc_idmacmd;
631f0d5bc97Stodd 	vaddr_t spa, pa, epa;
632f0d5bc97Stodd 	int c;
633da53b1f9Sjoris 
634f0d5bc97Stodd 	DPRINTF(("trigger_input %p %p 0x%x\n", start, end, bsize));
635f0d5bc97Stodd 
63636dba039Sjsg 	for (p = sc->sc_dmas; p && p->addr != start; p = p->next)
63736dba039Sjsg 		;
638f0d5bc97Stodd 	if (!p)
639f0d5bc97Stodd 		return -1;
640f0d5bc97Stodd 
641f0d5bc97Stodd 	sc->sc_iintr = intr;
642f0d5bc97Stodd 	sc->sc_iarg = arg;
643f0d5bc97Stodd 	sc->sc_idmap = sc->sc_idmacmd;
644f0d5bc97Stodd 
645f0d5bc97Stodd 	spa = p->segs[0].ds_addr;
646f0d5bc97Stodd 	c = DBDMA_CMD_IN_MORE;
647f0d5bc97Stodd 	for (pa = spa, epa = spa + (end - start);
648f0d5bc97Stodd 	    pa < epa; pa += bsize, cmd++) {
649f0d5bc97Stodd 
650f0d5bc97Stodd 		if (pa + bsize == epa)
651f0d5bc97Stodd 			c = DBDMA_CMD_IN_LAST;
652f0d5bc97Stodd 
653f0d5bc97Stodd 		DBDMA_BUILD(cmd, c, 0, bsize, pa, DBDMA_INT_ALWAYS,
654f0d5bc97Stodd 			DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER);
655f0d5bc97Stodd 	}
656f0d5bc97Stodd 
657f0d5bc97Stodd 	DBDMA_BUILD(cmd, DBDMA_CMD_NOP, 0, 0, 0,
658f0d5bc97Stodd 		DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_ALWAYS);
659f0d5bc97Stodd 	dbdma_st32(&cmd->d_cmddep, sc->sc_idbdma->d_paddr);
660f0d5bc97Stodd 
661f0d5bc97Stodd 	dbdma_start(sc->sc_idma, sc->sc_idbdma);
662f0d5bc97Stodd 
663f0d5bc97Stodd 	return 0;
664da53b1f9Sjoris }
665da53b1f9Sjoris 
666da53b1f9Sjoris 
667daa2e9a1Sgwk /* rate = fs = LRCLK
668daa2e9a1Sgwk  * SCLK = 64*LRCLK (I2S)
669daa2e9a1Sgwk  * MCLK = 256fs (typ. -- changeable)
670daa2e9a1Sgwk  * MCLK = clksrc / mdiv
671daa2e9a1Sgwk  *  SCLK = MCLK / sdiv
672daa2e9a1Sgwk  * rate = SCLK / 64    ( = LRCLK = fs)
673daa2e9a1Sgwk  */
674da53b1f9Sjoris int
i2s_set_rate(struct i2s_softc * sc,int rate)67547d381f7Sjsg i2s_set_rate(struct i2s_softc *sc, int rate)
676da53b1f9Sjoris {
677da53b1f9Sjoris 	u_int reg = 0;
678da53b1f9Sjoris 	int MCLK;
679da53b1f9Sjoris 	int clksrc, mdiv, sdiv;
680da53b1f9Sjoris 	int mclk_fs;
681da53b1f9Sjoris 	int timo;
682da53b1f9Sjoris 
683da53b1f9Sjoris 	/* sanify */
6849f8ca1f5Stodd 	if (rate > (48000 + 44100) / 2)
685da53b1f9Sjoris 		rate = 48000;
6869f8ca1f5Stodd 	else
687b26a3ef9Sjakemsr 		rate = 44100;
688da53b1f9Sjoris 
689da53b1f9Sjoris 	switch (rate) {
690da53b1f9Sjoris 	case 44100:
691da53b1f9Sjoris 		clksrc = 45158400;		/* 45MHz */
692da53b1f9Sjoris 		reg = CLKSRC_45MHz;
693da53b1f9Sjoris 		mclk_fs = 256;
694da53b1f9Sjoris 		break;
695da53b1f9Sjoris 
696da53b1f9Sjoris 	case 48000:
697da53b1f9Sjoris 		clksrc = 49152000;		/* 49MHz */
698da53b1f9Sjoris 		reg = CLKSRC_49MHz;
699da53b1f9Sjoris 		mclk_fs = 256;
700da53b1f9Sjoris 		break;
701da53b1f9Sjoris 
702da53b1f9Sjoris 	default:
703da53b1f9Sjoris 		return EINVAL;
704da53b1f9Sjoris 	}
705da53b1f9Sjoris 
706da53b1f9Sjoris 	MCLK = rate * mclk_fs;
707daa2e9a1Sgwk 	mdiv = clksrc / MCLK;			/* 4 */
708daa2e9a1Sgwk 	sdiv = mclk_fs / 64;			/* 4 */
709da53b1f9Sjoris 
710da53b1f9Sjoris 	switch (mdiv) {
711da53b1f9Sjoris 	case 1:
712da53b1f9Sjoris 		reg |= MCLK_DIV1;
713da53b1f9Sjoris 		break;
714da53b1f9Sjoris 	case 3:
715da53b1f9Sjoris 		reg |= MCLK_DIV3;
716da53b1f9Sjoris 		break;
717da53b1f9Sjoris 	case 5:
718da53b1f9Sjoris 		reg |= MCLK_DIV5;
719da53b1f9Sjoris 		break;
720da53b1f9Sjoris 	default:
721da53b1f9Sjoris 		reg |= ((mdiv / 2 - 1) << 24) & 0x1f000000;
722da53b1f9Sjoris 		break;
723da53b1f9Sjoris 	}
724da53b1f9Sjoris 
725da53b1f9Sjoris 	switch (sdiv) {
726da53b1f9Sjoris 	case 1:
727da53b1f9Sjoris 		reg |= SCLK_DIV1;
728da53b1f9Sjoris 		break;
729da53b1f9Sjoris 	case 3:
730da53b1f9Sjoris 		reg |= SCLK_DIV3;
731da53b1f9Sjoris 		break;
732da53b1f9Sjoris 	default:
733da53b1f9Sjoris 		reg |= ((sdiv / 2 - 1) << 20) & 0x00f00000;
734da53b1f9Sjoris 		break;
735da53b1f9Sjoris 	}
736da53b1f9Sjoris 
737da53b1f9Sjoris 	reg |= SCLK_MASTER;	/* XXX master mode */
738da53b1f9Sjoris 
739da53b1f9Sjoris 	reg |= SERIAL_64x;
740da53b1f9Sjoris 
741da53b1f9Sjoris 	if (sc->sc_rate == rate)
742da53b1f9Sjoris 		return (0);
743da53b1f9Sjoris 
744da53b1f9Sjoris 	/* stereo input and output */
745da53b1f9Sjoris 	DPRINTF(("I2SSetDataWordSizeReg 0x%08x -> 0x%08x\n",
746da53b1f9Sjoris 	    in32rb(sc->sc_reg + I2S_WORDSIZE), 0x02000200));
747da53b1f9Sjoris 	out32rb(sc->sc_reg + I2S_WORDSIZE, 0x02000200);
748da53b1f9Sjoris 
749da53b1f9Sjoris 	/* Clear CLKSTOPPEND */
750da53b1f9Sjoris 	out32rb(sc->sc_reg + I2S_INT, I2S_INT_CLKSTOPPEND);
751da53b1f9Sjoris 
75239dd359fSmpi 	macobio_disable(I2SClockOffset, I2S0CLKEN);
753da53b1f9Sjoris 
754da53b1f9Sjoris 	/* Wait until clock is stopped */
75529097c86Stodd 	for (timo = 50; timo > 0; timo--) {
756da53b1f9Sjoris 		if (in32rb(sc->sc_reg + I2S_INT) & I2S_INT_CLKSTOPPEND)
757da53b1f9Sjoris 			goto done;
75829097c86Stodd 		delay(10);
759da53b1f9Sjoris 	}
760da53b1f9Sjoris 
761da53b1f9Sjoris 	printf("i2s_set_rate: timeout\n");
762da53b1f9Sjoris 
763da53b1f9Sjoris done:
764da53b1f9Sjoris 	DPRINTF(("I2SSetSerialFormatReg 0x%x -> 0x%x\n",
765da53b1f9Sjoris 	    in32rb(sc->sc_reg + I2S_FORMAT), reg));
766da53b1f9Sjoris 	out32rb(sc->sc_reg + I2S_FORMAT, reg);
767da53b1f9Sjoris 
76839dd359fSmpi 	macobio_enable(I2SClockOffset, I2S0CLKEN);
769da53b1f9Sjoris 
770da53b1f9Sjoris 	sc->sc_rate = rate;
771da53b1f9Sjoris 
772da53b1f9Sjoris 	return 0;
773da53b1f9Sjoris }
774da53b1f9Sjoris 
775da53b1f9Sjoris void
i2s_mute(u_int offset,int mute)77655882323Smpi i2s_mute(u_int offset, int mute)
777da53b1f9Sjoris {
77855882323Smpi 	if (offset == 0)
779c8d53f6dSkettenis 		return;
780c8d53f6dSkettenis 
78155882323Smpi 	DPRINTF(("gpio: %x, %d -> ", offset, macobio_read(offset) & GPIO_DATA));
782da53b1f9Sjoris 
78355882323Smpi 	/* 0 means mute */
78455882323Smpi 	if (mute == (macobio_read(offset) & GPIO_DATA))
78555882323Smpi 		macobio_write(offset, !mute | GPIO_DDR_OUTPUT);
786da53b1f9Sjoris 
78755882323Smpi 	DPRINTF(("%d\n", macobio_read(offset) & GPIO_DATA));
788c8d53f6dSkettenis }
789c8d53f6dSkettenis 
790da53b1f9Sjoris int
i2s_cint(void * v)79155882323Smpi i2s_cint(void *v)
792da53b1f9Sjoris {
793da53b1f9Sjoris 	struct i2s_softc *sc = v;
794da53b1f9Sjoris 	u_int sense;
795da53b1f9Sjoris 
796c8d53f6dSkettenis 	sc->sc_output_mask = 0;
79755882323Smpi 	i2s_mute(sc->sc_spkr, 1);
79855882323Smpi 	i2s_mute(sc->sc_hp, 1);
79955882323Smpi 	i2s_mute(sc->sc_line, 1);
800c8d53f6dSkettenis 
80155882323Smpi 	if (sc->sc_hp_detect)
80255882323Smpi 		sense = macobio_read(sc->sc_hp_detect);
803c8d53f6dSkettenis 	else
80455882323Smpi 		sense = !sc->sc_hp_active << 1;
805da53b1f9Sjoris 	DPRINTF(("headphone detect = 0x%x\n", sense));
806da53b1f9Sjoris 
80755882323Smpi 	if (((sense & 0x02) >> 1) == sc->sc_hp_active) {
808da53b1f9Sjoris 		DPRINTF(("headphone is inserted\n"));
80955882323Smpi 		sc->sc_output_mask |= I2S_SELECT_HEADPHONE;
810041a1952Sjasper 		if (!sc->sc_mute)
81155882323Smpi 			i2s_mute(sc->sc_hp, 0);
812da53b1f9Sjoris 	} else {
813da53b1f9Sjoris 		DPRINTF(("headphone is NOT inserted\n"));
814c8d53f6dSkettenis 	}
815c8d53f6dSkettenis 
81655882323Smpi 	if (sc->sc_line_detect)
81755882323Smpi 		sense = macobio_read(sc->sc_line_detect);
818c8d53f6dSkettenis 	else
81955882323Smpi 		sense = !sc->sc_line_active << 1;
820c8d53f6dSkettenis 	DPRINTF(("lineout detect = 0x%x\n", sense));
821c8d53f6dSkettenis 
82255882323Smpi 	if (((sense & 0x02) >> 1) == sc->sc_line_active) {
823c8d53f6dSkettenis 		DPRINTF(("lineout is inserted\n"));
82455882323Smpi 		sc->sc_output_mask |= I2S_SELECT_LINEOUT;
825041a1952Sjasper 		if (!sc->sc_mute)
82655882323Smpi 			i2s_mute(sc->sc_line, 0);
827c8d53f6dSkettenis 	} else {
828c8d53f6dSkettenis 		DPRINTF(("lineout is NOT inserted\n"));
829c8d53f6dSkettenis 	}
830c8d53f6dSkettenis 
831c8d53f6dSkettenis 	if (sc->sc_output_mask == 0) {
83255882323Smpi 		sc->sc_output_mask |= I2S_SELECT_SPEAKER;
833041a1952Sjasper 		if (!sc->sc_mute)
83455882323Smpi 			i2s_mute(sc->sc_spkr, 0);
835da53b1f9Sjoris 	}
836da53b1f9Sjoris 
837da53b1f9Sjoris 	return 1;
838da53b1f9Sjoris }
839da53b1f9Sjoris 
84055882323Smpi u_int
i2s_gpio_offset(struct i2s_softc * sc,char * name,int * irq)84155882323Smpi i2s_gpio_offset(struct i2s_softc *sc, char *name, int *irq)
842eea27875Skettenis {
843eea27875Skettenis 	u_int32_t reg[2];
844c8d53f6dSkettenis 	u_int32_t intr[2];
845eea27875Skettenis 	int gpio;
846eea27875Skettenis 
847eea27875Skettenis 	if (OF_getprop(sc->sc_node, name, &gpio,
848eea27875Skettenis             sizeof(gpio)) != sizeof(gpio) ||
849eea27875Skettenis 	    OF_getprop(gpio, "reg", &reg[0],
850eea27875Skettenis 	    sizeof(reg[0])) != sizeof(reg[0]) ||
851eea27875Skettenis 	    OF_getprop(OF_parent(gpio), "reg", &reg[1],
852eea27875Skettenis 	    sizeof(reg[1])) != sizeof(reg[1]))
85355882323Smpi 		return (0);
854eea27875Skettenis 
855c8d53f6dSkettenis 	if (irq && OF_getprop(gpio, "interrupts",
856c8d53f6dSkettenis 	    intr, sizeof(intr)) == sizeof(intr)) {
857c8d53f6dSkettenis 		*irq = intr[0];
858c8d53f6dSkettenis 	}
859c8d53f6dSkettenis 
86055882323Smpi 	return (reg[0] + reg[1]);
861eea27875Skettenis }
862eea27875Skettenis 
863da53b1f9Sjoris void
i2s_gpio_init(struct i2s_softc * sc,int node,struct device * parent)86455882323Smpi i2s_gpio_init(struct i2s_softc *sc, int node, struct device *parent)
865da53b1f9Sjoris {
866da53b1f9Sjoris 	int gpio;
86755882323Smpi 	int hp_detect_intr = -1, line_detect_intr = -1;
868da53b1f9Sjoris 
86955882323Smpi 	sc->sc_spkr = i2s_gpio_offset(sc, "platform-amp-mute", NULL);
87055882323Smpi 	sc->sc_hp = i2s_gpio_offset(sc, "platform-headphone-mute", NULL);
87155882323Smpi 	sc->sc_hp_detect = i2s_gpio_offset(sc, "platform-headphone-detect",
87255882323Smpi 	    &hp_detect_intr);
87355882323Smpi 	sc->sc_line = i2s_gpio_offset(sc, "platform-lineout-mute", NULL);
87455882323Smpi 	sc->sc_line_detect = i2s_gpio_offset(sc, "platform-lineout-detect",
87555882323Smpi 	    &line_detect_intr);
87655882323Smpi 	sc->sc_hw_reset = i2s_gpio_offset(sc, "platform-hw-reset", NULL);
877eea27875Skettenis 
878da53b1f9Sjoris 	gpio = OF_getnodebyname(OF_parent(node), "gpio");
879da53b1f9Sjoris 	DPRINTF((" /gpio 0x%x\n", gpio));
88055882323Smpi 	for (gpio = OF_child(gpio); gpio; gpio = OF_peer(gpio)) {
881da53b1f9Sjoris 		char name[64], audio_gpio[64];
882da53b1f9Sjoris 		int intr[2];
88355882323Smpi 		uint32_t reg;
884da53b1f9Sjoris 
88555882323Smpi 		reg = 0;
886da53b1f9Sjoris 		bzero(name, sizeof name);
887da53b1f9Sjoris 		bzero(audio_gpio, sizeof audio_gpio);
888da53b1f9Sjoris 		OF_getprop(gpio, "name", name, sizeof name);
889da53b1f9Sjoris 		OF_getprop(gpio, "audio-gpio", audio_gpio, sizeof audio_gpio);
8907d1fae03Smpi 		if (OF_getprop(gpio, "reg", &reg, sizeof(reg)) == -1)
8917d1fae03Smpi 			OF_getprop(gpio, "AAPL,address", &reg, sizeof(reg));
8927d1fae03Smpi 
8937d1fae03Smpi 		if (reg > sc->sc_baseaddr)
8947d1fae03Smpi 			reg = (reg - sc->sc_baseaddr);
895da53b1f9Sjoris 
896da53b1f9Sjoris 		/* gpio5 */
89755882323Smpi 		if (sc->sc_hp == 0 && strcmp(audio_gpio, "headphone-mute") == 0)
89855882323Smpi 			sc->sc_hp = reg;
899da53b1f9Sjoris 
900da53b1f9Sjoris 		/* gpio6 */
90155882323Smpi 		if (sc->sc_spkr == 0 && strcmp(audio_gpio, "amp-mute") == 0)
90255882323Smpi 			sc->sc_spkr = reg;
903da53b1f9Sjoris 
904da53b1f9Sjoris 		/* extint-gpio15 */
90555882323Smpi 		if (sc->sc_hp_detect == 0 &&
906da53b1f9Sjoris 		    strcmp(audio_gpio, "headphone-detect") == 0) {
90755882323Smpi 			sc->sc_hp_detect = reg;
908da53b1f9Sjoris 			OF_getprop(gpio, "audio-gpio-active-state",
90955882323Smpi 			    &sc->sc_hp_active, 4);
910da53b1f9Sjoris 			OF_getprop(gpio, "interrupts", intr, 8);
91155882323Smpi 			hp_detect_intr = intr[0];
912da53b1f9Sjoris 		}
913da53b1f9Sjoris 
914da53b1f9Sjoris 		/* gpio11 (keywest-11) */
91555882323Smpi 		if (sc->sc_hw_reset == 0 &&
916da53b1f9Sjoris 		    strcmp(audio_gpio, "audio-hw-reset") == 0)
91755882323Smpi 			sc->sc_hw_reset = reg;
918da53b1f9Sjoris 	}
91955882323Smpi 	DPRINTF((" amp-mute 0x%x\n", sc->sc_spkr));
92055882323Smpi 	DPRINTF((" headphone-mute 0x%x\n", sc->sc_hp));
92155882323Smpi 	DPRINTF((" headphone-detect 0x%x\n", sc->sc_hp_detect));
92255882323Smpi 	DPRINTF((" headphone-detect active %x\n", sc->sc_hp_active));
92355882323Smpi 	DPRINTF((" headphone-detect intr %x\n", hp_detect_intr));
92455882323Smpi 	DPRINTF((" lineout-mute 0x%x\n", sc->sc_line));
92555882323Smpi 	DPRINTF((" lineout-detect 0x%x\n", sc->sc_line_detect));
92655882323Smpi 	DPRINTF((" lineout-detect active 0x%x\n", sc->sc_line_active));
92755882323Smpi 	DPRINTF((" lineout-detect intr 0x%x\n", line_detect_intr));
92855882323Smpi 	DPRINTF((" audio-hw-reset 0x%x\n", sc->sc_hw_reset));
929da53b1f9Sjoris 
93055882323Smpi 	if (hp_detect_intr != -1)
93155882323Smpi 		mac_intr_establish(parent, hp_detect_intr, IST_EDGE,
9323b53f463Smpi 		    IPL_AUDIO | IPL_MPSAFE, i2s_cint, sc, sc->sc_dev.dv_xname);
933da53b1f9Sjoris 
93455882323Smpi 	if (line_detect_intr != -1)
93555882323Smpi 		mac_intr_establish(parent, line_detect_intr, IST_EDGE,
9363b53f463Smpi 		    IPL_AUDIO | IPL_MPSAFE, i2s_cint, sc, sc->sc_dev.dv_xname);
937c8d53f6dSkettenis 
938da53b1f9Sjoris 	/* Enable headphone interrupt? */
93955882323Smpi 	macobio_write(sc->sc_hp_detect, 0x80);
940da53b1f9Sjoris 
941da53b1f9Sjoris 	/* Update headphone status. */
942da53b1f9Sjoris 	i2s_cint(sc);
943da53b1f9Sjoris }
944da53b1f9Sjoris 
945da53b1f9Sjoris void *
i2s_allocm(void * h,int dir,size_t size,int type,int flags)946da53b1f9Sjoris i2s_allocm(void *h, int dir, size_t size, int type, int flags)
947da53b1f9Sjoris {
948da53b1f9Sjoris 	struct i2s_softc *sc = h;
949da53b1f9Sjoris 	struct i2s_dma *p;
950da53b1f9Sjoris 	int error;
951da53b1f9Sjoris 
952da53b1f9Sjoris 	if (size > I2S_DMALIST_MAX * I2S_DMASEG_MAX)
953da53b1f9Sjoris 		return (NULL);
954da53b1f9Sjoris 
955bc9397c2Skrw 	p = malloc(sizeof(*p), type, flags | M_ZERO);
956da53b1f9Sjoris 	if (!p)
957da53b1f9Sjoris 		return (NULL);
958da53b1f9Sjoris 
959da53b1f9Sjoris 	/* convert to the bus.h style, not used otherwise */
960da53b1f9Sjoris 	if (flags & M_NOWAIT)
961da53b1f9Sjoris 		flags = BUS_DMA_NOWAIT;
962da53b1f9Sjoris 
963da53b1f9Sjoris 	p->size = size;
964da53b1f9Sjoris 	if ((error = bus_dmamem_alloc(sc->sc_dmat, p->size, NBPG, 0, p->segs,
965da53b1f9Sjoris 	    1, &p->nsegs, flags)) != 0) {
966da53b1f9Sjoris 		printf("%s: unable to allocate dma, error = %d\n",
967da53b1f9Sjoris 		    sc->sc_dev.dv_xname, error);
96858d5cf8aSderaadt 		free(p, type, sizeof *p);
969da53b1f9Sjoris 		return NULL;
970da53b1f9Sjoris 	}
971da53b1f9Sjoris 
972da53b1f9Sjoris 	if ((error = bus_dmamem_map(sc->sc_dmat, p->segs, p->nsegs, p->size,
973da53b1f9Sjoris 	    &p->addr, flags | BUS_DMA_COHERENT)) != 0) {
974da53b1f9Sjoris 		printf("%s: unable to map dma, error = %d\n",
975da53b1f9Sjoris 		    sc->sc_dev.dv_xname, error);
976da53b1f9Sjoris 		bus_dmamem_free(sc->sc_dmat, p->segs, p->nsegs);
97758d5cf8aSderaadt 		free(p, type, sizeof *p);
978da53b1f9Sjoris 		return NULL;
979da53b1f9Sjoris 	}
980da53b1f9Sjoris 
981da53b1f9Sjoris 	if ((error = bus_dmamap_create(sc->sc_dmat, p->size, 1,
982da53b1f9Sjoris 	    p->size, 0, flags, &p->map)) != 0) {
983da53b1f9Sjoris 		printf("%s: unable to create dma map, error = %d\n",
984da53b1f9Sjoris 		    sc->sc_dev.dv_xname, error);
985da53b1f9Sjoris 		bus_dmamem_unmap(sc->sc_dmat, p->addr, size);
986da53b1f9Sjoris 		bus_dmamem_free(sc->sc_dmat, p->segs, p->nsegs);
98758d5cf8aSderaadt 		free(p, type, sizeof *p);
988da53b1f9Sjoris 		return NULL;
989da53b1f9Sjoris 	}
990da53b1f9Sjoris 
991da53b1f9Sjoris 	if ((error = bus_dmamap_load(sc->sc_dmat, p->map, p->addr, p->size,
992da53b1f9Sjoris 	    NULL, flags)) != 0) {
993da53b1f9Sjoris 		printf("%s: unable to load dma map, error = %d\n",
994da53b1f9Sjoris 		    sc->sc_dev.dv_xname, error);
995da53b1f9Sjoris 		bus_dmamap_destroy(sc->sc_dmat, p->map);
996da53b1f9Sjoris 		bus_dmamem_unmap(sc->sc_dmat, p->addr, size);
997da53b1f9Sjoris 		bus_dmamem_free(sc->sc_dmat, p->segs, p->nsegs);
99858d5cf8aSderaadt 		free(p, type, sizeof *p);
999da53b1f9Sjoris 		return NULL;
1000da53b1f9Sjoris 	}
1001da53b1f9Sjoris 
1002da53b1f9Sjoris 	p->next = sc->sc_dmas;
1003da53b1f9Sjoris 	sc->sc_dmas = p;
1004da53b1f9Sjoris 
1005da53b1f9Sjoris 	return p->addr;
1006da53b1f9Sjoris }
1007da53b1f9Sjoris 
1008da53b1f9Sjoris #define reset_active 0
1009da53b1f9Sjoris 
1010da53b1f9Sjoris int
deq_reset(struct i2s_softc * sc)1011da53b1f9Sjoris deq_reset(struct i2s_softc *sc)
1012da53b1f9Sjoris {
101355882323Smpi 	if (sc->sc_hw_reset == 0)
1014da53b1f9Sjoris 		return (-1);
1015da53b1f9Sjoris 
101655882323Smpi 	macobio_write(sc->sc_hw_reset, !reset_active | GPIO_DDR_OUTPUT);
1017da53b1f9Sjoris 	delay(1000000);
1018da53b1f9Sjoris 
101955882323Smpi 	macobio_write(sc->sc_hw_reset, reset_active | GPIO_DDR_OUTPUT);
1020da53b1f9Sjoris 	delay(1);
1021da53b1f9Sjoris 
102255882323Smpi 	macobio_write(sc->sc_hw_reset, !reset_active | GPIO_DDR_OUTPUT);
1023da53b1f9Sjoris 	delay(10000);
1024da53b1f9Sjoris 
1025da53b1f9Sjoris 	return (0);
1026da53b1f9Sjoris }
1027