From 786d7b987c829d864e7037901e5d07af57eff75b Mon Sep 17 00:00:00 2001 From: Alexandre Ratchov Date: Fri, 8 May 2009 16:53:46 +0000 Subject: add support for the ak4358 DAC used in ESI Julia cards. There are two mono line0 and line1 knobs rather than a single stereo knob because that's how the routing control (not exposed for ht chips yet) works. --- sys/dev/pci/envy.c | 76 +++++++++++++++++++++++++++++++++++++++++++++------ sys/dev/pci/envyreg.h | 4 +-- sys/dev/pci/envyvar.h | 8 ++---- 3 files changed, 71 insertions(+), 17 deletions(-) (limited to 'sys') diff --git a/sys/dev/pci/envy.c b/sys/dev/pci/envy.c index 094021d92d9..56d4d234d37 100644 --- a/sys/dev/pci/envy.c +++ b/sys/dev/pci/envy.c @@ -1,4 +1,4 @@ -/* $OpenBSD: envy.c,v 1.25 2009/05/08 16:12:23 ratchov Exp $ */ +/* $OpenBSD: envy.c,v 1.26 2009/05/08 16:53:45 ratchov Exp $ */ /* * Copyright (c) 2007 Alexandre Ratchov * @@ -119,6 +119,11 @@ void ak4524_adc_devinfo(struct envy_softc *, struct mixer_devinfo *, int); void ak4524_adc_get(struct envy_softc *, struct mixer_ctrl *, int); int ak4524_adc_set(struct envy_softc *, struct mixer_ctrl *, int); +int ak4358_dac_ndev(struct envy_softc *); +void ak4358_dac_devinfo(struct envy_softc *, struct mixer_devinfo *, int); +void ak4358_dac_get(struct envy_softc *, struct mixer_ctrl *, int); +int ak4358_dac_set(struct envy_softc *, struct mixer_ctrl *, int); + struct cfattach envy_ca = { sizeof(struct envy_softc), envymatch, envyattach, envydetach }; @@ -190,6 +195,8 @@ struct envy_codec ak4524_dac = { "ak4524 dac", ak4524_dac_ndev, ak4524_dac_devinfo, ak4524_dac_get, ak4524_dac_set }, ak4524_adc = { "ak4524 adc", ak4524_adc_ndev, ak4524_adc_devinfo, ak4524_adc_get, ak4524_adc_set +}, ak4358_dac = { + "ak4358 dac", ak4358_dac_ndev, ak4358_dac_devinfo, ak4358_dac_get, ak4358_dac_set }, unkenvy_codec = { "unknown codec", unkenvy_codec_ndev, NULL, NULL, NULL }; @@ -237,7 +244,7 @@ struct envy_card envy_cards[] = { { PCI_ID_CODE(0x3031, 0x4553), "ESI Julia", - 2, &unkenvy_codec, 2, &unkenvy_codec, + 2, &unkenvy_codec, 2, &ak4358_dac, julia_init, julia_codec_write, julia_eeprom @@ -267,11 +274,11 @@ delta_init(struct envy_softc *sc) AK4524_RST_AD | AK4524_RST_DA); envy_codec_write(sc, dev, AK4524_FMT, AK4524_FMT_IIS24); - sc->ak[dev].reg[AK4524_DEEMVOL] = AK4524_DEEM_OFF; - sc->ak[dev].reg[AK4524_ADC_GAIN0] = 0x7f; - sc->ak[dev].reg[AK4524_ADC_GAIN1] = 0x7f; - sc->ak[dev].reg[AK4524_DAC_GAIN0] = 0x7f; - sc->ak[dev].reg[AK4524_DAC_GAIN1] = 0x7f; + sc->shadow[dev][AK4524_DEEMVOL] = AK4524_DEEM_OFF; + sc->shadow[dev][AK4524_ADC_GAIN0] = 0x7f; + sc->shadow[dev][AK4524_ADC_GAIN1] = 0x7f; + sc->shadow[dev][AK4524_DAC_GAIN0] = 0x7f; + sc->shadow[dev][AK4524_DAC_GAIN1] = 0x7f; } } @@ -312,9 +319,14 @@ delta_codec_write(struct envy_softc *sc, int dev, int addr, int data) void julia_init(struct envy_softc *sc) { + int i; + envy_codec_write(sc, 0, 0, 0); /* reset */ delay(300); envy_codec_write(sc, 0, 0, 0x87); /* i2s mode */ + for (i = 0; i < sc->card->noch; i++) { + sc->shadow[0][AK4358_ATT(i)] = 0xff; + } } void @@ -345,6 +357,52 @@ unkenvy_codec_ndev(struct envy_softc *sc) return 0; } +/* + * AK 4358 DAC specific code + */ +int +ak4358_dac_ndev(struct envy_softc *sc) +{ + /* 1 volume knob per channel */ + return sc->card->noch; +} + + +void +ak4358_dac_devinfo(struct envy_softc *sc, struct mixer_devinfo *dev, int idx) +{ + dev->type = AUDIO_MIXER_VALUE; + dev->mixer_class = ENVY_MIX_CLASSOUT; + dev->un.v.delta = 2; + dev->un.v.num_channels = 1; + snprintf(dev->label.name, MAX_AUDIO_DEV_LEN, + AudioNline "%d", idx); + strlcpy(dev->un.v.units.name, AudioNvolume, + MAX_AUDIO_DEV_LEN); +} + +void +ak4358_dac_get(struct envy_softc *sc, struct mixer_ctrl *ctl, int idx) +{ + int val; + + val = envy_codec_read(sc, 0, AK4358_ATT(idx)) & ~AK4358_ATT_EN; + ctl->un.value.num_channels = 1; + ctl->un.value.level[0] = 2 * val; +} + +int +ak4358_dac_set(struct envy_softc *sc, struct mixer_ctrl *ctl, int idx) +{ + int val; + + if (ctl->un.value.num_channels != 1) + return EINVAL; + val = ctl->un.value.level[0] / 2; + envy_codec_write(sc, 0, AK4358_ATT(idx), val | AK4358_ATT_EN); + return 0; +} + /* * AK 4524 DAC specific code */ @@ -614,14 +672,14 @@ envy_i2c_write(struct envy_softc *sc, int dev, int addr, int data) int envy_codec_read(struct envy_softc *sc, int dev, int addr) { - return sc->ak[dev].reg[addr]; + return sc->shadow[dev][addr]; } void envy_codec_write(struct envy_softc *sc, int dev, int addr, int data) { DPRINTFN(2, "envy_codec_write: %d, %d, 0x%x\n", dev, addr, data); - sc->ak[dev].reg[addr] = data; + sc->shadow[dev][addr] = data; sc->card->codec_write(sc, dev, addr, data); } diff --git a/sys/dev/pci/envyreg.h b/sys/dev/pci/envyreg.h index 42f8c2d33d8..81516f6682e 100644 --- a/sys/dev/pci/envyreg.h +++ b/sys/dev/pci/envyreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: envyreg.h,v 1.7 2009/05/08 15:31:16 ratchov Exp $ */ +/* $OpenBSD: envyreg.h,v 1.8 2009/05/08 16:53:45 ratchov Exp $ */ /* * Copyright (c) 2007 Alexandre Ratchov * @@ -184,7 +184,7 @@ * AK4358 control registers */ #define AK4358_ATT(chan) ((chan) <= 5 ? 0x4 + (chan) : 0xb - 6 + (chan)) - +#define AK4358_ATT_EN 0x80 /* * default formats diff --git a/sys/dev/pci/envyvar.h b/sys/dev/pci/envyvar.h index cf3e44f66d3..ed23eb70a02 100644 --- a/sys/dev/pci/envyvar.h +++ b/sys/dev/pci/envyvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: envyvar.h,v 1.11 2009/05/08 15:31:16 ratchov Exp $ */ +/* $OpenBSD: envyvar.h,v 1.12 2009/05/08 16:53:45 ratchov Exp $ */ /* * Copyright (c) 2007 Alexandre Ratchov * @@ -51,10 +51,6 @@ struct envy_card { unsigned char *eeprom; }; -struct envy_ak { - unsigned char reg[16]; -}; - struct envy_softc { struct device dev; struct device *audio; @@ -71,7 +67,7 @@ struct envy_softc { bus_space_handle_t mt_ioh; bus_size_t mt_iosz; struct envy_card *card; - struct envy_ak ak[4]; + unsigned char shadow[4][16]; #define ENVY_EEPROM_MAXSZ 32 unsigned char eeprom[ENVY_EEPROM_MAXSZ]; void (*iintr)(void *); -- cgit v1.2.3