Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/macppc/dev various minor fixes and updates:
details: https://anonhg.NetBSD.org/src/rev/b469d2d1a462
branches: trunk
changeset: 464039:b469d2d1a462
user: macallan <macallan%NetBSD.org@localhost>
date: Fri Sep 20 21:24:34 2019 +0000
description:
various minor fixes and updates:
- add (very) basic ONYX support. Treat as 16bit codec with software volume
control. Makes basic audio work on PCIe G5
- correctly detect TAS3001 without 'compatible' property, now this works on
Quicksilver
- mute line input on TAS3001 - it's unconnected on Quicksilver and causes noise
- make more of an effort to match codecs to i2sbus instances, needed on G5 and
some other models with software 'modems'
diffstat:
sys/arch/macppc/dev/snapper.c | 204 +++++++++++++++++++++++++++++++----------
1 files changed, 155 insertions(+), 49 deletions(-)
diffs (truncated from 348 to 300 lines):
diff -r 59c1d695e110 -r b469d2d1a462 sys/arch/macppc/dev/snapper.c
--- a/sys/arch/macppc/dev/snapper.c Fri Sep 20 17:25:11 2019 +0000
+++ b/sys/arch/macppc/dev/snapper.c Fri Sep 20 21:24:34 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: snapper.c,v 1.52 2019/06/08 08:02:37 isaki Exp $ */
+/* $NetBSD: snapper.c,v 1.53 2019/09/20 21:24:34 macallan Exp $ */
/* Id: snapper.c,v 1.11 2002/10/31 17:42:13 tsubai Exp */
/* Id: i2s.c,v 1.12 2005/01/15 14:32:35 tsubai Exp */
@@ -35,7 +35,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: snapper.c,v 1.52 2019/06/08 08:02:37 isaki Exp $");
+__KERNEL_RCSID(0, "$NetBSD: snapper.c,v 1.53 2019/09/20 21:24:34 macallan Exp $");
#include <sys/param.h>
#include <sys/audioio.h>
@@ -55,7 +55,7 @@
#include <macppc/dev/deqvar.h>
#include <macppc/dev/obiovar.h>
-
+//#define SNAPPER_DEBUG
#ifdef SNAPPER_DEBUG
# define DPRINTF printf
#else
@@ -66,9 +66,12 @@
struct snapper_softc {
device_t sc_dev;
- int sc_mode; // 0 for TAS3004
+ int sc_mode;
+#define SNAPPER_IS_TAS3004 0 // codec is TAS3004
#define SNAPPER_IS_TAS3001 1 // codec is TAS3001
-#define SNAPPER_SWVOL 2 // software codec
+#define SNAPPER_IS_PCM3052 2 // codec is PCM3052
+#define SNAPPER_IS_CS8416 3 // codec is CS8416
+#define SNAPPER_SWVOL 4 // software codec
int sc_node;
@@ -475,6 +478,21 @@
};
#define TUMBLER_NFORMATS __arraycount(tumbler_formats)
+/* OF hands us the codec in 16bit mode, run with it for now */
+static const struct audio_format onyx_formats[] = {
+ {
+ .mode = AUMODE_PLAY | AUMODE_RECORD,
+ .encoding = AUDIO_ENCODING_SLINEAR_BE,
+ .validbits = 16,
+ .precision = 16,
+ .channels = 2,
+ .channel_mask = AUFMT_STEREO,
+ .frequency_type = 3,
+ .frequency = { 44100, 48000, 96000 },
+ },
+};
+#define ONYX_NFORMATS __arraycount(onyx_formats)
+
static bus_size_t amp_mute;
static bus_size_t headphone_mute;
static bus_size_t audio_hw_reset;
@@ -492,6 +510,16 @@
/* I2S_INT register definitions */
#define I2S_INT_CLKSTOPPEND 0x01000000 /* clock-stop interrupt pending */
+/* I2S_WORDSIZE register definitions */
+#define INPUT_STEREO (2 << 24)
+#define INPUT_MONO (1 << 24)
+#define INPUT_16BIT (0 << 16)
+#define INPUT_24BIT (3 << 16)
+#define OUTPUT_STEREO (2 << 8)
+#define OUTPUT_MONO (1 << 8)
+#define OUTPUT_16BIT (0 << 0)
+#define OUTPUT_24BIT (3 << 0)
+
/* FCR(0x3c) bits */
#define KEYLARGO_FCR1 0x3c
#define I2S0CLKEN 0x1000
@@ -639,6 +667,9 @@
if (strcmp(compat, "AOAK2") == 0)
return 1;
+ if (strcmp(compat, "AOAbase") == 0)
+ return 1;
+
if (OF_getprop(soundchip, "platform-tas-codec-ref",
&soundcodec, sizeof soundcodec) == sizeof soundcodec)
return 1;
@@ -664,6 +695,8 @@
memset(compat, 0, sizeof compat);
OF_getprop(OF_child(soundbus), "compatible", compat, sizeof compat);
+ sc->sc_mode = SNAPPER_IS_TAS3004;
+
if (strcmp(compat, "tumbler") == 0)
sc->sc_mode = SNAPPER_IS_TAS3001;
sc->sc_swvol_l = 255;
@@ -718,8 +751,8 @@
oirq = intr[2];
iirq = intr[4];
/* cirq_type = intr[1] ? IST_LEVEL : IST_EDGE; */
- oirq_type = intr[3] ? IST_LEVEL : IST_EDGE;
- iirq_type = intr[5] ? 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, snapper_intr, sc); */
intr_establish(oirq, oirq_type, IPL_AUDIO, snapper_intr, sc);
@@ -743,25 +776,79 @@
device_t dv;
deviter_t di;
struct deq_softc *deq;
+ char prop[64], next[64], codec[64], *cref;
+ int codec_node, soundbus, sound, ok, deqnode = 0;
sc = device_private(dev);
+
+ /* look for platform-*-codec-ref node */
+
+ /*
+ * XXX
+ * there can be more than one i2sbus, the one we want just so happens
+ * to be the first we see
+ */
+ soundbus = OF_child(sc->sc_node);
+ sound = OF_child(soundbus);
+ ok = OF_nextprop(sound, NULL, next);
+ codec_node = 0;
+ while (ok && (codec_node == 0)) {
+ DPRINTF("prop %d %s\n", ok, next);
+ strncpy(prop, next, 64);
+ if ((cref = strstr(next, "-codec-ref")) != NULL) {
+ OF_getprop(sound, next, &codec_node, 4);
+ if (codec_node != 0) {
+ OF_getprop(codec_node, "compatible", codec, 64);
+ DPRINTF("%08x %s\n", codec_node, codec);
+ }
+ }
+ ok = OF_nextprop(sound, prop, next);
+ }
+
for (dv = deviter_first(&di, DEVITER_F_ROOT_FIRST);
dv != NULL;
dv = deviter_next(&di)) {
if (device_is_a(dv, "deq")) {
deq = device_private(dv);
+ if (codec_node != 0) {
+ if (codec_node != deq->sc_node)
+ continue;
+ }
sc->sc_i2c = deq->sc_i2c;
sc->sc_deqaddr = deq->sc_address;
+ deqnode = deq->sc_node;
}
}
deviter_release(&di);
+ DPRINTF("deqnode: %08x\n", deqnode);
+
/* If we don't find a codec, it's not the end of the world;
* we can control the volume in software in this case.
*/
- if (sc->sc_i2c == NULL)
+ if (sc->sc_i2c == NULL) {
sc->sc_mode = SNAPPER_SWVOL;
+ } else if (deqnode != 0) {
+ int ret;
+ codec[0] = 0;
+ ret = OF_getprop(deqnode, "compatible", codec, 64);
+ DPRINTF("codec <%s> %d\n", codec, ret);
+
+ if (codec[0] == 0) {
+ if (sc->sc_deqaddr == 0x34) {
+ sc->sc_mode = SNAPPER_IS_TAS3001;
+ } else
+ sc->sc_mode = SNAPPER_IS_TAS3004;
+ } else if (strcmp(codec, "tas3004") == 0) {
+ sc->sc_mode = SNAPPER_IS_TAS3004;
+ } else if (strcmp(codec, "pcm3052") == 0) {
+ sc->sc_mode = SNAPPER_IS_PCM3052;
+ } else if (strcmp(codec, "cs8416") == 0) {
+ sc->sc_mode = SNAPPER_IS_CS8416;
+ }
+ }
+ DPRINTF("mode %d\n", sc->sc_mode);
switch (sc->sc_mode) {
case SNAPPER_SWVOL:
aprint_verbose("%s: software codec\n", device_xname(dev));
@@ -769,9 +856,15 @@
case SNAPPER_IS_TAS3001:
aprint_verbose("%s: codec: TAS3001\n", device_xname(dev));
break;
- case 0:
+ case SNAPPER_IS_TAS3004:
aprint_verbose("%s: codec: TAS3004\n", device_xname(dev));
break;
+ case SNAPPER_IS_PCM3052:
+ aprint_verbose("%s: codec: PCM3052 / ONYX\n", device_xname(dev));
+ break;
+ default:
+ aprint_error_dev(sc->sc_dev, "unsupported codec\n");
+ sc->sc_mode = SNAPPER_SWVOL;
}
snapper_init(sc, sc->sc_node);
@@ -826,13 +919,19 @@
{
struct snapper_softc *sc = h;
- if (sc->sc_mode == SNAPPER_IS_TAS3001) {
- return audio_query_format(tumbler_formats, TUMBLER_NFORMATS,
- afp);
- } else {
- return audio_query_format(snapper_formats, SNAPPER_NFORMATS,
- afp);
+ switch (sc->sc_mode) {
+ case SNAPPER_IS_TAS3001:
+ return audio_query_format(tumbler_formats,
+ TUMBLER_NFORMATS, afp);
+ case SNAPPER_SWVOL:
+ case SNAPPER_IS_TAS3004:
+ return audio_query_format(snapper_formats,
+ SNAPPER_NFORMATS, afp);
+ case SNAPPER_IS_PCM3052:
+ return audio_query_format(onyx_formats,
+ ONYX_NFORMATS, afp);
}
+ return -1;
}
static int
@@ -870,8 +969,8 @@
const audio_params_t *param)
{
- if (size < NBPG)
- size = NBPG;
+ if (size < (3 * NBPG))
+ size = (3 * NBPG);
return size & ~PGOFSET;
}
@@ -1476,7 +1575,7 @@
{
uint8_t regs[9] = {0, 0, 0, 0, 0, 0, 0, 0, 0};
int i;
-
+if (sc->sc_mode > 1) return;
/* Left channel of SDIN1 */
ADJUST(i, sc->mixer[0]);
regs[0] = snapper_mixer_gain[i][0];
@@ -1619,11 +1718,13 @@
DPRINTF("precision: %d\n", sc->sc_bitspersample);
switch(sc->sc_bitspersample) {
case 16:
- wordsize = 0x02000200;
+ wordsize = INPUT_STEREO | INPUT_16BIT |
+ OUTPUT_STEREO | OUTPUT_16BIT;
mcr1 = DEQ_MCR1_SC_64 | DEQ_MCR1_SM_I2S | DEQ_MCR1_W_16;
break;
case 24:
- wordsize = 0x03000300;
+ wordsize = INPUT_STEREO | INPUT_24BIT |
+ OUTPUT_STEREO | OUTPUT_24BIT;
mcr1 = DEQ_MCR1_SC_64 | DEQ_MCR1_SM_I2S | DEQ_MCR1_W_24;
break;
default:
@@ -1895,34 +1996,36 @@
delay(10000);
noreset:
- DEQ_WRITE(sc, DEQ_LB0, tas3004_initdata.LB0);
- DEQ_WRITE(sc, DEQ_LB1, tas3004_initdata.LB1);
- DEQ_WRITE(sc, DEQ_LB2, tas3004_initdata.LB2);
- DEQ_WRITE(sc, DEQ_LB3, tas3004_initdata.LB3);
- DEQ_WRITE(sc, DEQ_LB4, tas3004_initdata.LB4);
- DEQ_WRITE(sc, DEQ_LB5, tas3004_initdata.LB5);
- DEQ_WRITE(sc, DEQ_LB6, tas3004_initdata.LB6);
- DEQ_WRITE(sc, DEQ_RB0, tas3004_initdata.RB0);
- DEQ_WRITE(sc, DEQ_RB1, tas3004_initdata.RB1);
- DEQ_WRITE(sc, DEQ_RB1, tas3004_initdata.RB1);
- DEQ_WRITE(sc, DEQ_RB2, tas3004_initdata.RB2);
- DEQ_WRITE(sc, DEQ_RB3, tas3004_initdata.RB3);
- DEQ_WRITE(sc, DEQ_RB4, tas3004_initdata.RB4);
- DEQ_WRITE(sc, DEQ_RB5, tas3004_initdata.RB5);
- DEQ_WRITE(sc, DEQ_MCR1, tas3004_initdata.MCR1);
- DEQ_WRITE(sc, DEQ_MCR2, tas3004_initdata.MCR2);
- DEQ_WRITE(sc, DEQ_DRC, tas3004_initdata.DRC);
- DEQ_WRITE(sc, DEQ_VOLUME, tas3004_initdata.VOLUME);
- DEQ_WRITE(sc, DEQ_TREBLE, tas3004_initdata.TREBLE);
- DEQ_WRITE(sc, DEQ_BASS, tas3004_initdata.BASS);
- DEQ_WRITE(sc, DEQ_MIXER_L, tas3004_initdata.MIXER_L);
- DEQ_WRITE(sc, DEQ_MIXER_R, tas3004_initdata.MIXER_R);
- DEQ_WRITE(sc, DEQ_LLB, tas3004_initdata.LLB);
- DEQ_WRITE(sc, DEQ_RLB, tas3004_initdata.RLB);
- DEQ_WRITE(sc, DEQ_LLB_GAIN, tas3004_initdata.LLB_GAIN);
Home |
Main Index |
Thread Index |
Old Index