/* $OpenBSD: i2s.c,v 1.39 2024/05/22 05:51:49 jsg Exp $ */ /* $NetBSD: i2s.c,v 1.1 2003/12/27 02:19:34 grant Exp $ */ /*- * Copyright (c) 2002 Tsubai Masanari. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef I2S_DEBUG # define DPRINTF(x) printf x #else # define DPRINTF(x) #endif void i2s_mute(u_int, int); int i2s_cint(void *); u_int i2s_gpio_offset(struct i2s_softc *, char *, int *); int i2s_intr(void *); int i2s_iintr(void *); struct cfdriver i2s_cd = { NULL, "i2s", DV_DULL }; void i2s_attach(struct device *parent, struct i2s_softc *sc, struct confargs *ca) { int cirq, oirq, iirq, cirq_type, oirq_type, iirq_type; u_int32_t reg[6], intr[6]; char compat[32]; int child; sc->sc_node = OF_child(ca->ca_node); sc->sc_baseaddr = ca->ca_baseaddr; OF_getprop(sc->sc_node, "reg", reg, sizeof reg); child = OF_child(sc->sc_node); memset(compat, 0, sizeof(compat)); OF_getprop(child, "compatible", compat, sizeof(compat)); /* Deal with broken device-tree on PowerMac7,2 and 7,3. */ if (strcmp(compat, "AOAK2") == 0) { reg[0] += ca->ca_reg[0]; reg[2] += ca->ca_reg[2]; reg[4] += ca->ca_reg[2]; } reg[0] += sc->sc_baseaddr; reg[2] += sc->sc_baseaddr; reg[4] += sc->sc_baseaddr; sc->sc_reg = mapiodev(reg[0], reg[1]); sc->sc_dmat = ca->ca_dmat; sc->sc_odma = mapiodev(reg[2], reg[3]); /* out */ sc->sc_idma = mapiodev(reg[4], reg[5]); /* in */ sc->sc_odbdma = dbdma_alloc(sc->sc_dmat, I2S_DMALIST_MAX); sc->sc_odmacmd = sc->sc_odbdma->d_addr; sc->sc_idbdma = dbdma_alloc(sc->sc_dmat, I2S_DMALIST_MAX); sc->sc_idmacmd = sc->sc_idbdma->d_addr; OF_getprop(sc->sc_node, "interrupts", intr, sizeof intr); cirq = intr[0]; oirq = intr[2]; iirq = intr[4]; cirq_type = (intr[1] & 1) ? IST_LEVEL : IST_EDGE; oirq_type = (intr[3] & 1) ? IST_LEVEL : IST_EDGE; iirq_type = (intr[5] & 1) ? IST_LEVEL : IST_EDGE; /* intr_establish(cirq, cirq_type, IPL_AUDIO, i2s_intr, sc); */ mac_intr_establish(parent, oirq, oirq_type, IPL_AUDIO | IPL_MPSAFE, i2s_intr, sc, sc->sc_dev.dv_xname); mac_intr_establish(parent, iirq, iirq_type, IPL_AUDIO | IPL_MPSAFE, i2s_iintr, sc, sc->sc_dev.dv_xname); printf(": irq %d,%d,%d\n", cirq, oirq, iirq); /* Need to be explicitly turned on some G5. */ macobio_enable(I2SClockOffset, I2S0CLKEN|I2S0EN); i2s_set_rate(sc, 44100); sc->sc_mute = 0; i2s_gpio_init(sc, ca->ca_node, parent); } int i2s_intr(void *v) { struct i2s_softc *sc = v; struct dbdma_command *cmd = sc->sc_odmap; u_int16_t c, status; mtx_enter(&audio_lock); /* if not set we are not running */ if (!cmd) { mtx_leave(&audio_lock); return (0); } DPRINTF(("i2s_intr: cmd %p\n", cmd)); c = in16rb(&cmd->d_command); status = in16rb(&cmd->d_status); if (c >> 12 == DBDMA_CMD_OUT_LAST) sc->sc_odmap = sc->sc_odmacmd; else sc->sc_odmap++; if (c & (DBDMA_INT_ALWAYS << 4)) { cmd->d_status = 0; if (status) /* status == 0x8400 */ if (sc->sc_ointr) (*sc->sc_ointr)(sc->sc_oarg); } mtx_leave(&audio_lock); return 1; } int i2s_iintr(void *v) { struct i2s_softc *sc = v; struct dbdma_command *cmd = sc->sc_idmap; u_int16_t c, status; mtx_enter(&audio_lock); /* if not set we are not running */ if (!cmd) { mtx_leave(&audio_lock); return (0); } DPRINTF(("i2s_intr: cmd %p\n", cmd)); c = in16rb(&cmd->d_command); status = in16rb(&cmd->d_status); if (c >> 12 == DBDMA_CMD_IN_LAST) sc->sc_idmap = sc->sc_idmacmd; else sc->sc_idmap++; if (c & (DBDMA_INT_ALWAYS << 4)) { cmd->d_status = 0; if (status) /* status == 0x8400 */ if (sc->sc_iintr) (*sc->sc_iintr)(sc->sc_iarg); } mtx_leave(&audio_lock); return 1; } int i2s_open(void *h, int flags) { return 0; } /* * Close function is called at splaudio(). */ void i2s_close(void *h) { struct i2s_softc *sc = h; i2s_halt_output(sc); i2s_halt_input(sc); sc->sc_ointr = 0; sc->sc_iintr = 0; } int i2s_set_params(void *h, int setmode, int usemode, struct audio_params *play, struct audio_params *rec) { struct i2s_softc *sc = h; struct audio_params *p; int mode; p = play; /* default to play */ /* * This device only has one clock, so make the sample rates match. */ if (play->sample_rate != rec->sample_rate && usemode == (AUMODE_PLAY | AUMODE_RECORD)) { if (setmode == AUMODE_PLAY) { rec->sample_rate = play->sample_rate; setmode |= AUMODE_RECORD; } else if (setmode == AUMODE_RECORD) { play->sample_rate = rec->sample_rate; setmode |= AUMODE_PLAY; } else return EINVAL; } for (mode = AUMODE_RECORD; mode != -1; mode = mode == AUMODE_RECORD ? AUMODE_PLAY : -1) { if ((setmode & mode) == 0) continue; p = mode == AUMODE_PLAY ? play : rec; if (p->sample_rate < 4000) p->sample_rate = 4000; if (p->sample_rate > 50000) p->sample_rate = 50000; if (p->precision > 16) p->precision = 16; if (p->channels > 2) p->channels = 2; p->bps = AUDIO_BPS(p->precision); p->msb = 1; p->encoding = AUDIO_ENCODING_SLINEAR_BE; } /* Set the speed */ if (i2s_set_rate(sc, play->sample_rate)) return EINVAL; p->sample_rate = sc->sc_rate; return 0; } int i2s_round_blocksize(void *h, int size) { if (size < NBPG) size = NBPG; return size & ~PGOFSET; } int i2s_halt_output(void *h) { struct i2s_softc *sc = h; dbdma_stop(sc->sc_odma); dbdma_reset(sc->sc_odma); return 0; } int i2s_halt_input(void *h) { struct i2s_softc *sc = h; dbdma_stop(sc->sc_idma); dbdma_reset(sc->sc_idma); return 0; } enum { I2S_OUTPUT_CLASS, I2S_RECORD_CLASS, I2S_OUTPUT_SELECT, I2S_VOL_OUTPUT, I2S_INPUT_SELECT, I2S_VOL_INPUT, I2S_MUTE, /* should be before bass/treble */ I2S_BASS, I2S_TREBLE, I2S_ENUM_LAST }; int i2s_set_port(void *h, mixer_ctrl_t *mc) { struct i2s_softc *sc = h; int l, r; DPRINTF(("i2s_set_port dev = %d, type = %d\n", mc->dev, mc->type)); l = mc->un.value.level[AUDIO_MIXER_LEVEL_LEFT]; r = mc->un.value.level[AUDIO_MIXER_LEVEL_RIGHT]; switch (mc->dev) { case I2S_OUTPUT_SELECT: /* No change necessary? */ if (mc->un.mask == sc->sc_output_mask) return 0; i2s_mute(sc->sc_spkr, 1); i2s_mute(sc->sc_hp, 1); i2s_mute(sc->sc_line, 1); if (mc->un.mask & I2S_SELECT_SPEAKER) i2s_mute(sc->sc_spkr, 0); if (mc->un.mask & I2S_SELECT_HEADPHONE) i2s_mute(sc->sc_hp, 0); if (mc->un.mask & I2S_SELECT_LINEOUT) i2s_mute(sc->sc_line, 0); sc->sc_output_mask = mc->un.mask; return 0; case I2S_VOL_OUTPUT: (*sc->sc_setvolume)(sc, l, r); return 0; case I2S_MUTE: if (mc->type != AUDIO_MIXER_ENUM) return (EINVAL); sc->sc_mute = (mc->un.ord != 0); if (sc->sc_mute) { if (sc->sc_output_mask & I2S_SELECT_SPEAKER) i2s_mute(sc->sc_spkr, 1); if (sc->sc_output_mask & I2S_SELECT_HEADPHONE) i2s_mute(sc->sc_hp, 1); if (sc->sc_output_mask & I2S_SELECT_LINEOUT) i2s_mute(sc->sc_line, 1); } else { if (sc->sc_output_mask & I2S_SELECT_SPEAKER) i2s_mute(sc->sc_spkr, 0); if (sc->sc_output_mask & I2S_SELECT_HEADPHONE) i2s_mute(sc->sc_hp, 0); if (sc->sc_output_mask & I2S_SELECT_LINEOUT) i2s_mute(sc->sc_line, 0); } return (0); case I2S_BASS: if (sc->sc_setbass != NULL) (*sc->sc_setbass)(sc, l); return (0); case I2S_TREBLE: if (sc->sc_settreble != NULL) (*sc->sc_settreble)(sc, l); return (0); case I2S_INPUT_SELECT: /* no change necessary? */ if (mc->un.mask == sc->sc_record_source) return 0; switch (mc->un.mask) { case I2S_SELECT_SPEAKER: case I2S_SELECT_HEADPHONE: /* XXX TO BE DONE */ break; default: /* invalid argument */ return EINVAL; } if (sc->sc_setinput != NULL) (*sc->sc_setinput)(sc, mc->un.mask); sc->sc_record_source = mc->un.mask; return 0; case I2S_VOL_INPUT: /* XXX TO BE DONE */ return 0; } return ENXIO; } int i2s_get_port(void *h, mixer_ctrl_t *mc) { struct i2s_softc *sc = h; DPRINTF(("i2s_get_port dev = %d, type = %d\n", mc->dev, mc->type)); switch (mc->dev) { case I2S_OUTPUT_SELECT: mc->un.mask = sc->sc_output_mask; return 0; case I2S_VOL_OUTPUT: mc->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = sc->sc_vol_l; mc->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = sc->sc_vol_r; return 0; case I2S_MUTE: mc->un.ord = sc->sc_mute; return (0); case I2S_INPUT_SELECT: mc->un.mask = sc->sc_record_source; return 0; case I2S_BASS: if (mc->un.value.num_channels != 1) return ENXIO; mc->un.value.level[AUDIO_MIXER_LEVEL_MONO] = sc->sc_bass; return 0; case I2S_TREBLE: if (mc->un.value.num_channels != 1) return ENXIO; mc->un.value.level[AUDIO_MIXER_LEVEL_MONO] = sc->sc_treble; return 0; case I2S_VOL_INPUT: /* XXX TO BE DONE */ mc->un.value.level[AUDIO_MIXER_LEVEL_LEFT] = 0; mc->un.value.level[AUDIO_MIXER_LEVEL_RIGHT] = 0; return 0; default: return ENXIO; } return 0; } int i2s_query_devinfo(void *h, mixer_devinfo_t *dip) { struct i2s_softc *sc = h; int n = 0; switch (dip->index) { case I2S_OUTPUT_SELECT: dip->mixer_class = I2S_OUTPUT_CLASS; strlcpy(dip->label.name, AudioNselect, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_SET; dip->prev = dip->next = AUDIO_MIXER_LAST; strlcpy(dip->un.s.member[n].label.name, AudioNspeaker, sizeof(dip->un.s.member[n].label.name)); dip->un.s.member[n++].mask = I2S_SELECT_SPEAKER; if (sc->sc_hp) { strlcpy(dip->un.s.member[n].label.name, AudioNheadphone, sizeof(dip->un.s.member[n].label.name)); dip->un.s.member[n++].mask = I2S_SELECT_HEADPHONE; } if (sc->sc_line) { strlcpy(dip->un.s.member[n].label.name, AudioNline, sizeof(dip->un.s.member[n].label.name)); dip->un.s.member[n++].mask = I2S_SELECT_LINEOUT; } dip->un.s.num_mem = n; return 0; case I2S_VOL_OUTPUT: dip->mixer_class = I2S_OUTPUT_CLASS; strlcpy(dip->label.name, AudioNmaster, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_VALUE; dip->prev = AUDIO_MIXER_LAST; dip->next = I2S_MUTE; dip->un.v.num_channels = 2; dip->un.v.delta = 8; strlcpy(dip->un.v.units.name, AudioNvolume, sizeof(dip->un.v.units.name)); return 0; case I2S_MUTE: dip->mixer_class = I2S_OUTPUT_CLASS; dip->prev = I2S_VOL_OUTPUT; dip->next = AUDIO_MIXER_LAST; strlcpy(dip->label.name, AudioNmute, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_ENUM; dip->un.e.num_mem = 2; strlcpy(dip->un.e.member[0].label.name, AudioNoff, sizeof dip->un.e.member[0].label.name); dip->un.e.member[0].ord = 0; strlcpy(dip->un.e.member[1].label.name, AudioNon, sizeof dip->un.e.member[1].label.name); dip->un.e.member[1].ord = 1; return (0); case I2S_INPUT_SELECT: dip->mixer_class = I2S_RECORD_CLASS; strlcpy(dip->label.name, AudioNsource, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_SET; dip->prev = dip->next = AUDIO_MIXER_LAST; dip->un.s.num_mem = 2; strlcpy(dip->un.s.member[0].label.name, AudioNmicrophone, sizeof(dip->un.s.member[0].label.name)); dip->un.s.member[0].mask = I2S_SELECT_SPEAKER; strlcpy(dip->un.s.member[1].label.name, AudioNline, sizeof(dip->un.s.member[1].label.name)); dip->un.s.member[1].mask = I2S_SELECT_HEADPHONE; return 0; case I2S_VOL_INPUT: dip->mixer_class = I2S_RECORD_CLASS; strlcpy(dip->label.name, AudioNrecord, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_VALUE; dip->prev = dip->next = AUDIO_MIXER_LAST; dip->un.v.num_channels = 2; strlcpy(dip->un.v.units.name, AudioNvolume, sizeof(dip->un.v.units.name)); return 0; case I2S_OUTPUT_CLASS: dip->mixer_class = I2S_OUTPUT_CLASS; strlcpy(dip->label.name, AudioCoutputs, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_CLASS; dip->next = dip->prev = AUDIO_MIXER_LAST; return 0; case I2S_RECORD_CLASS: dip->mixer_class = I2S_RECORD_CLASS; strlcpy(dip->label.name, AudioCrecord, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_CLASS; dip->next = dip->prev = AUDIO_MIXER_LAST; return 0; case I2S_BASS: if (sc->sc_setbass == NULL) return (ENXIO); dip->mixer_class = I2S_OUTPUT_CLASS; strlcpy(dip->label.name, AudioNbass, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_VALUE; dip->prev = dip->next = AUDIO_MIXER_LAST; dip->un.v.num_channels = 1; return (0); case I2S_TREBLE: if (sc->sc_settreble == NULL) return (ENXIO); dip->mixer_class = I2S_OUTPUT_CLASS; strlcpy(dip->label.name, AudioNtreble, sizeof(dip->label.name)); dip->type = AUDIO_MIXER_VALUE; dip->prev = dip->next = AUDIO_MIXER_LAST; dip->un.v.num_channels = 1; return (0); } return ENXIO; } size_t i2s_round_buffersize(void *h, int dir, size_t size) { if (size > 65536) size = 65536; return size; } int i2s_trigger_output(void *h, void *start, void *end, int bsize, void (*intr)(void *), void *arg, struct audio_params *param) { struct i2s_softc *sc = h; struct i2s_dma *p; struct dbdma_command *cmd = sc->sc_odmacmd; vaddr_t spa, pa, epa; int c; DPRINTF(("trigger_output %p %p 0x%x\n", start, end, bsize)); for (p = sc->sc_dmas; p && p->addr != start; p = p->next) ; if (!p) return -1; sc->sc_ointr = intr; sc->sc_oarg = arg; sc->sc_odmap = sc->sc_odmacmd; spa = p->segs[0].ds_addr; c = DBDMA_CMD_OUT_MORE; for (pa = spa, epa = spa + (end - start); pa < epa; pa += bsize, cmd++) { if (pa + bsize == epa) c = DBDMA_CMD_OUT_LAST; DBDMA_BUILD(cmd, c, 0, bsize, pa, DBDMA_INT_ALWAYS, DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER); } DBDMA_BUILD(cmd, DBDMA_CMD_NOP, 0, 0, 0, DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_ALWAYS); dbdma_st32(&cmd->d_cmddep, sc->sc_odbdma->d_paddr); dbdma_start(sc->sc_odma, sc->sc_odbdma); return 0; } int i2s_trigger_input(void *h, void *start, void *end, int bsize, void (*intr)(void *), void *arg, struct audio_params *param) { struct i2s_softc *sc = h; struct i2s_dma *p; struct dbdma_command *cmd = sc->sc_idmacmd; vaddr_t spa, pa, epa; int c; DPRINTF(("trigger_input %p %p 0x%x\n", start, end, bsize)); for (p = sc->sc_dmas; p && p->addr != start; p = p->next) ; if (!p) return -1; sc->sc_iintr = intr; sc->sc_iarg = arg; sc->sc_idmap = sc->sc_idmacmd; spa = p->segs[0].ds_addr; c = DBDMA_CMD_IN_MORE; for (pa = spa, epa = spa + (end - start); pa < epa; pa += bsize, cmd++) { if (pa + bsize == epa) c = DBDMA_CMD_IN_LAST; DBDMA_BUILD(cmd, c, 0, bsize, pa, DBDMA_INT_ALWAYS, DBDMA_WAIT_NEVER, DBDMA_BRANCH_NEVER); } DBDMA_BUILD(cmd, DBDMA_CMD_NOP, 0, 0, 0, DBDMA_INT_NEVER, DBDMA_WAIT_NEVER, DBDMA_BRANCH_ALWAYS); dbdma_st32(&cmd->d_cmddep, sc->sc_idbdma->d_paddr); dbdma_start(sc->sc_idma, sc->sc_idbdma); return 0; } /* rate = fs = LRCLK * SCLK = 64*LRCLK (I2S) * MCLK = 256fs (typ. -- changeable) * MCLK = clksrc / mdiv * SCLK = MCLK / sdiv * rate = SCLK / 64 ( = LRCLK = fs) */ int i2s_set_rate(struct i2s_softc *sc, int rate) { u_int reg = 0; int MCLK; int clksrc, mdiv, sdiv; int mclk_fs; int timo; /* sanify */ if (rate > (48000 + 44100) / 2) rate = 48000; else rate = 44100; switch (rate) { case 44100: clksrc = 45158400; /* 45MHz */ reg = CLKSRC_45MHz; mclk_fs = 256; break; case 48000: clksrc = 49152000; /* 49MHz */ reg = CLKSRC_49MHz; mclk_fs = 256; break; default: return EINVAL; } MCLK = rate * mclk_fs; mdiv = clksrc / MCLK; /* 4 */ sdiv = mclk_fs / 64; /* 4 */ switch (mdiv) { case 1: reg |= MCLK_DIV1; break; case 3: reg |= MCLK_DIV3; break; case 5: reg |= MCLK_DIV5; break; default: reg |= ((mdiv / 2 - 1) << 24) & 0x1f000000; break; } switch (sdiv) { case 1: reg |= SCLK_DIV1; break; case 3: reg |= SCLK_DIV3; break; default: reg |= ((sdiv / 2 - 1) << 20) & 0x00f00000; break; } reg |= SCLK_MASTER; /* XXX master mode */ reg |= SERIAL_64x; if (sc->sc_rate == rate) return (0); /* stereo input and output */ DPRINTF(("I2SSetDataWordSizeReg 0x%08x -> 0x%08x\n", in32rb(sc->sc_reg + I2S_WORDSIZE), 0x02000200)); out32rb(sc->sc_reg + I2S_WORDSIZE, 0x02000200); /* Clear CLKSTOPPEND */ out32rb(sc->sc_reg + I2S_INT, I2S_INT_CLKSTOPPEND); macobio_disable(I2SClockOffset, I2S0CLKEN); /* Wait until clock is stopped */ for (timo = 50; timo > 0; timo--) { if (in32rb(sc->sc_reg + I2S_INT) & I2S_INT_CLKSTOPPEND) goto done; delay(10); } printf("i2s_set_rate: timeout\n"); done: DPRINTF(("I2SSetSerialFormatReg 0x%x -> 0x%x\n", in32rb(sc->sc_reg + I2S_FORMAT), reg)); out32rb(sc->sc_reg + I2S_FORMAT, reg); macobio_enable(I2SClockOffset, I2S0CLKEN); sc->sc_rate = rate; return 0; } void i2s_mute(u_int offset, int mute) { if (offset == 0) return; DPRINTF(("gpio: %x, %d -> ", offset, macobio_read(offset) & GPIO_DATA)); /* 0 means mute */ if (mute == (macobio_read(offset) & GPIO_DATA)) macobio_write(offset, !mute | GPIO_DDR_OUTPUT); DPRINTF(("%d\n", macobio_read(offset) & GPIO_DATA)); } int i2s_cint(void *v) { struct i2s_softc *sc = v; u_int sense; sc->sc_output_mask = 0; i2s_mute(sc->sc_spkr, 1); i2s_mute(sc->sc_hp, 1); i2s_mute(sc->sc_line, 1); if (sc->sc_hp_detect) sense = macobio_read(sc->sc_hp_detect); else sense = !sc->sc_hp_active << 1; DPRINTF(("headphone detect = 0x%x\n", sense)); if (((sense & 0x02) >> 1) == sc->sc_hp_active) { DPRINTF(("headphone is inserted\n")); sc->sc_output_mask |= I2S_SELECT_HEADPHONE; if (!sc->sc_mute) i2s_mute(sc->sc_hp, 0); } else { DPRINTF(("headphone is NOT inserted\n")); } if (sc->sc_line_detect) sense = macobio_read(sc->sc_line_detect); else sense = !sc->sc_line_active << 1; DPRINTF(("lineout detect = 0x%x\n", sense)); if (((sense & 0x02) >> 1) == sc->sc_line_active) { DPRINTF(("lineout is inserted\n")); sc->sc_output_mask |= I2S_SELECT_LINEOUT; if (!sc->sc_mute) i2s_mute(sc->sc_line, 0); } else { DPRINTF(("lineout is NOT inserted\n")); } if (sc->sc_output_mask == 0) { sc->sc_output_mask |= I2S_SELECT_SPEAKER; if (!sc->sc_mute) i2s_mute(sc->sc_spkr, 0); } return 1; } u_int i2s_gpio_offset(struct i2s_softc *sc, char *name, int *irq) { u_int32_t reg[2]; u_int32_t intr[2]; int gpio; if (OF_getprop(sc->sc_node, name, &gpio, sizeof(gpio)) != sizeof(gpio) || OF_getprop(gpio, "reg", ®[0], sizeof(reg[0])) != sizeof(reg[0]) || OF_getprop(OF_parent(gpio), "reg", ®[1], sizeof(reg[1])) != sizeof(reg[1])) return (0); if (irq && OF_getprop(gpio, "interrupts", intr, sizeof(intr)) == sizeof(intr)) { *irq = intr[0]; } return (reg[0] + reg[1]); } void i2s_gpio_init(struct i2s_softc *sc, int node, struct device *parent) { int gpio; int hp_detect_intr = -1, line_detect_intr = -1; sc->sc_spkr = i2s_gpio_offset(sc, "platform-amp-mute", NULL); sc->sc_hp = i2s_gpio_offset(sc, "platform-headphone-mute", NULL); sc->sc_hp_detect = i2s_gpio_offset(sc, "platform-headphone-detect", &hp_detect_intr); sc->sc_line = i2s_gpio_offset(sc, "platform-lineout-mute", NULL); sc->sc_line_detect = i2s_gpio_offset(sc, "platform-lineout-detect", &line_detect_intr); sc->sc_hw_reset = i2s_gpio_offset(sc, "platform-hw-reset", NULL); gpio = OF_getnodebyname(OF_parent(node), "gpio"); DPRINTF((" /gpio 0x%x\n", gpio)); for (gpio = OF_child(gpio); gpio; gpio = OF_peer(gpio)) { char name[64], audio_gpio[64]; int intr[2]; uint32_t reg; reg = 0; bzero(name, sizeof name); bzero(audio_gpio, sizeof audio_gpio); OF_getprop(gpio, "name", name, sizeof name); OF_getprop(gpio, "audio-gpio", audio_gpio, sizeof audio_gpio); if (OF_getprop(gpio, "reg", ®, sizeof(reg)) == -1) OF_getprop(gpio, "AAPL,address", ®, sizeof(reg)); if (reg > sc->sc_baseaddr) reg = (reg - sc->sc_baseaddr); /* gpio5 */ if (sc->sc_hp == 0 && strcmp(audio_gpio, "headphone-mute") == 0) sc->sc_hp = reg; /* gpio6 */ if (sc->sc_spkr == 0 && strcmp(audio_gpio, "amp-mute") == 0) sc->sc_spkr = reg; /* extint-gpio15 */ if (sc->sc_hp_detect == 0 && strcmp(audio_gpio, "headphone-detect") == 0) { sc->sc_hp_detect = reg; OF_getprop(gpio, "audio-gpio-active-state", &sc->sc_hp_active, 4); OF_getprop(gpio, "interrupts", intr, 8); hp_detect_intr = intr[0]; } /* gpio11 (keywest-11) */ if (sc->sc_hw_reset == 0 && strcmp(audio_gpio, "audio-hw-reset") == 0) sc->sc_hw_reset = reg; } DPRINTF((" amp-mute 0x%x\n", sc->sc_spkr)); DPRINTF((" headphone-mute 0x%x\n", sc->sc_hp)); DPRINTF((" headphone-detect 0x%x\n", sc->sc_hp_detect)); DPRINTF((" headphone-detect active %x\n", sc->sc_hp_active)); DPRINTF((" headphone-detect intr %x\n", hp_detect_intr)); DPRINTF((" lineout-mute 0x%x\n", sc->sc_line)); DPRINTF((" lineout-detect 0x%x\n", sc->sc_line_detect)); DPRINTF((" lineout-detect active 0x%x\n", sc->sc_line_active)); DPRINTF((" lineout-detect intr 0x%x\n", line_detect_intr)); DPRINTF((" audio-hw-reset 0x%x\n", sc->sc_hw_reset)); if (hp_detect_intr != -1) mac_intr_establish(parent, hp_detect_intr, IST_EDGE, IPL_AUDIO | IPL_MPSAFE, i2s_cint, sc, sc->sc_dev.dv_xname); if (line_detect_intr != -1) mac_intr_establish(parent, line_detect_intr, IST_EDGE, IPL_AUDIO | IPL_MPSAFE, i2s_cint, sc, sc->sc_dev.dv_xname); /* Enable headphone interrupt? */ macobio_write(sc->sc_hp_detect, 0x80); /* Update headphone status. */ i2s_cint(sc); } void * i2s_allocm(void *h, int dir, size_t size, int type, int flags) { struct i2s_softc *sc = h; struct i2s_dma *p; int error; if (size > I2S_DMALIST_MAX * I2S_DMASEG_MAX) return (NULL); p = malloc(sizeof(*p), type, flags | M_ZERO); if (!p) return (NULL); /* convert to the bus.h style, not used otherwise */ if (flags & M_NOWAIT) flags = BUS_DMA_NOWAIT; p->size = size; if ((error = bus_dmamem_alloc(sc->sc_dmat, p->size, NBPG, 0, p->segs, 1, &p->nsegs, flags)) != 0) { printf("%s: unable to allocate dma, error = %d\n", sc->sc_dev.dv_xname, error); free(p, type, sizeof *p); return NULL; } if ((error = bus_dmamem_map(sc->sc_dmat, p->segs, p->nsegs, p->size, &p->addr, flags | BUS_DMA_COHERENT)) != 0) { printf("%s: unable to map dma, error = %d\n", sc->sc_dev.dv_xname, error); bus_dmamem_free(sc->sc_dmat, p->segs, p->nsegs); free(p, type, sizeof *p); return NULL; } if ((error = bus_dmamap_create(sc->sc_dmat, p->size, 1, p->size, 0, flags, &p->map)) != 0) { printf("%s: unable to create dma map, error = %d\n", sc->sc_dev.dv_xname, error); bus_dmamem_unmap(sc->sc_dmat, p->addr, size); bus_dmamem_free(sc->sc_dmat, p->segs, p->nsegs); free(p, type, sizeof *p); return NULL; } if ((error = bus_dmamap_load(sc->sc_dmat, p->map, p->addr, p->size, NULL, flags)) != 0) { printf("%s: unable to load dma map, error = %d\n", sc->sc_dev.dv_xname, error); bus_dmamap_destroy(sc->sc_dmat, p->map); bus_dmamem_unmap(sc->sc_dmat, p->addr, size); bus_dmamem_free(sc->sc_dmat, p->segs, p->nsegs); free(p, type, sizeof *p); return NULL; } p->next = sc->sc_dmas; sc->sc_dmas = p; return p->addr; } #define reset_active 0 int deq_reset(struct i2s_softc *sc) { if (sc->sc_hw_reset == 0) return (-1); macobio_write(sc->sc_hw_reset, !reset_active | GPIO_DDR_OUTPUT); delay(1000000); macobio_write(sc->sc_hw_reset, reset_active | GPIO_DDR_OUTPUT); delay(1); macobio_write(sc->sc_hw_reset, !reset_active | GPIO_DDR_OUTPUT); delay(10000); return (0); }