summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/dev/pci/eap.c1065
-rw-r--r--sys/dev/pci/eapreg.h284
2 files changed, 666 insertions, 683 deletions
diff --git a/sys/dev/pci/eap.c b/sys/dev/pci/eap.c
index 6d91a585323..6a28b517344 100644
--- a/sys/dev/pci/eap.c
+++ b/sys/dev/pci/eap.c
@@ -1,5 +1,5 @@
-/* $OpenBSD: eap.c,v 1.12 2001/08/25 10:13:29 art Exp $ */
-/* $NetBSD: eap.c,v 1.25 1999/02/18 07:59:30 mycroft Exp $ */
+/* $OpenBSD: eap.c,v 1.13 2001/09/06 14:47:07 naddy Exp $ */
+/* $NetBSD: eap.c,v 1.46 2001/09/03 15:07:37 reinoud Exp $ */
/*
* Copyright (c) 1998, 1999 The NetBSD Foundation, Inc.
@@ -41,27 +41,26 @@
* Debugging: Andreas Gustafsson <gson@araneus.fi>
* Testing: Chuck Cranor <chuck@maria.wustl.edu>
* Phil Nelson <phil@cs.wwu.edu>
+ *
+ * ES1371/AC97: Ezra Story <ezy@panix.com>
*/
/*
- * Ensoniq AudoiPCI ES1370 + AK4531 driver.
- * Data sheets can be found at
- * http://www.ensoniq.com/multimedia/semi_html/html/es1370.zip
- * and
- * http://www.akm.com/pdf/4531.pdf
+ * Ensoniq ES1370 + AK4531 and ES1371/ES1373 + AC97
*
- * Added Creative Ensoniq support: ES1371 + AC97 = hack city.
- * -- Ezra Story <ezy@panix.com>
- * Check es1371.zip from above, and the Audio Codec 97 spec from
- * intel.com.
+ * Documentation links:
+ *
+ * ftp://ftp.alsa-project.org/pub/manuals/ensoniq/
+ * ftp://ftp.alsa-project.org/pub/manuals/asahi_kasei/4531.pdf
+ * ftp://download.intel.com/ial/scalableplatforms/audio/ac97r21.pdf
*/
-
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/malloc.h>
#include <sys/device.h>
+#include <sys/proc.h>
#include <dev/pci/pcidevs.h>
#include <dev/pci/pcivar.h>
@@ -74,226 +73,14 @@
#include <machine/bus.h>
+#include <dev/pci/eapreg.h>
+
struct cfdriver eap_cd = {
NULL, "eap", DV_DULL
};
#define PCI_CBIO 0x10
-#define EAP_ICSC 0x00 /* interrupt / chip select control */
-#define EAP_SERR_DISABLE 0x00000001
-#define EAP_CDC_EN 0x00000002
-#define EAP_JYSTK_EN 0x00000004
-#define EAP_UART_EN 0x00000008
-#define EAP_ADC_EN 0x00000010
-#define EAP_DAC2_EN 0x00000020
-#define EAP_DAC1_EN 0x00000040
-#define EAP_BREQ 0x00000080
-#define EAP_XTCL0 0x00000100
-#define EAP_M_CB 0x00000200
-#define EAP_CCB_INTRM 0x00000400
-#define EAP_DAC_SYNC 0x00000800
-#define EAP_WTSRSEL 0x00003000
-#define EAP_WTSRSEL_5 0x00000000
-#define EAP_WTSRSEL_11 0x00001000
-#define EAP_WTSRSEL_22 0x00002000
-#define EAP_WTSRSEL_44 0x00003000
-#define EAP_M_SBB 0x00004000
-#define EAP_MSFMTSEL 0x00008000
-#define EAP_SET_PCLKDIV(n) (((n)&0x1fff)<<16)
-#define EAP_GET_PCLKDIV(n) (((n)>>16)&0x1fff)
-#define EAP_PCLKBITS 0x1fff0000
-#define EAP_XTCL1 0x40000000
-#define EAP_ADC_STOP 0x80000000
-#define E1371_SYNC_RES (1<<14)
-
-#define EAP_ICSS 0x04 /* interrupt / chip select status */
-#define EAP_I_ADC 0x00000001
-#define EAP_I_DAC2 0x00000002
-#define EAP_I_DAC1 0x00000004
-#define EAP_I_UART 0x00000008
-#define EAP_I_MCCB 0x00000010
-#define EAP_VC 0x00000060
-#define EAP_CWRIP 0x00000100
-#define EAP_CBUSY 0x00000200
-#define EAP_CSTAT 0x00000400
-#define CT5880_AC97_RESET 0x20000000
-#define EAP_INTR 0x80000000
-
-#define EAP_UART_DATA 0x08
-#define EAP_UART_STATUS 0x09
-#define EAP_UART_CONTROL 0x09
-#define EAP_MEMPAGE 0x0c
-#define EAP_CODEC 0x10
-#define EAP_SET_CODEC(a,d) (((a)<<8) | (d))
-
-/* ES1371 Registers */
-#define E1371_CODEC 0x14
-#define E1371_CODEC_WIP (1<<30)
-#define E1371_CODEC_VALID (1<<31)
-#define E1371_CODEC_READ (1<<23)
-#define E1371_SET_CODEC(a,d) (((a)<<16) | (d))
-#define E1371_SRC 0x10
-#define E1371_SRC_RAMWE (1<<24)
-#define E1371_SRC_RBUSY (1<<23)
-#define E1371_SRC_DISABLE (1<<22)
-#define E1371_SRC_DISP1 (1<<21)
-#define E1371_SRC_DISP2 (1<<20)
-#define E1371_SRC_DISREC (1<<19)
-#define E1371_SRC_ADDR(a) ((a)<<25)
-#define E1371_SRC_STATE_MASK 0x870000
-#define E1371_SRC_STATE_OK 0x010000
-#define E1371_SRC_DATA(d) (d)
-#define E1371_SRC_DATAMASK 0xffff
-#define E1371_LEGACY 0x18
-
-/* ES1371 Sample rate converter registers */
-#define ESRC_ADC 0x78
-#define ESRC_DAC1 0x74
-#define ESRC_DAC2 0x70
-#define ESRC_ADC_VOLL 0x6c
-#define ESRC_ADC_VOLR 0x6d
-#define ESRC_DAC1_VOLL 0x7c
-#define ESRC_DAC1_VOLR 0x7d
-#define ESRC_DAC2_VOLL 0x7e
-#define ESRC_DAC2_VOLR 0x7f
-#define ESRC_TRUNC_N 0x00
-#define ESRC_IREGS 0x01
-#define ESRC_ACF 0x02
-#define ESRC_VFF 0x03
-#define ESRC_SET_TRUNC(n) ((n)<<9)
-#define ESRC_SET_N(n) ((n)<<4)
-#define ESRC_SMF 0x8000
-#define ESRC_SET_VFI(n) ((n)<<10)
-#define ESRC_SET_ACI(n) (n)
-#define ESRC_SET_ADC_VOL(n) ((n)<<8)
-#define ESRC_SET_DAC_VOLI(n) ((n)<<12)
-#define ESRC_SET_DAC_VOLF(n) (n)
-#define SRC_MAGIC ((1<15)|(1<<13)|(1<<11)|(1<<9))
-
-
-#define EAP_SIC 0x20
-#define EAP_P1_S_MB 0x00000001
-#define EAP_P1_S_EB 0x00000002
-#define EAP_P2_S_MB 0x00000004
-#define EAP_P2_S_EB 0x00000008
-#define EAP_R1_S_MB 0x00000010
-#define EAP_R1_S_EB 0x00000020
-#define EAP_P2_DAC_SEN 0x00000040
-#define EAP_P1_SCT_RLD 0x00000080
-#define EAP_P1_INTR_EN 0x00000100
-#define EAP_P2_INTR_EN 0x00000200
-#define EAP_R1_INTR_EN 0x00000400
-#define EAP_P1_PAUSE 0x00000800
-#define EAP_P2_PAUSE 0x00001000
-#define EAP_P1_LOOP_SEL 0x00002000
-#define EAP_P2_LOOP_SEL 0x00004000
-#define EAP_R1_LOOP_SEL 0x00008000
-#define EAP_SET_P2_ST_INC(i) ((i) << 16)
-#define EAP_SET_P2_END_INC(i) ((i) << 19)
-#define EAP_INC_BITS 0x003f0000
-
-#define EAP_DAC1_CSR 0x24
-#define EAP_DAC2_CSR 0x28
-#define EAP_ADC_CSR 0x2c
-#define EAP_GET_CURRSAMP(r) ((r) >> 16)
-
-#define EAP_DAC_PAGE 0xc
-#define EAP_ADC_PAGE 0xd
-#define EAP_UART_PAGE1 0xe
-#define EAP_UART_PAGE2 0xf
-
-#define EAP_DAC1_ADDR 0x30
-#define EAP_DAC1_SIZE 0x34
-#define EAP_DAC2_ADDR 0x38
-#define EAP_DAC2_SIZE 0x3c
-#define EAP_ADC_ADDR 0x30
-#define EAP_ADC_SIZE 0x34
-#define EAP_SET_SIZE(c,s) (((c)<<16) | (s))
-
-#define EAP_READ_TIMEOUT 0x1000
-#define EAP_WRITE_TIMEOUT 0x1000
-
-
-#define EAP_XTAL_FREQ 1411200 /* 22.5792 / 16 MHz */
-
-/* AK4531 registers */
-#define AK_MASTER_L 0x00
-#define AK_MASTER_R 0x01
-#define AK_VOICE_L 0x02
-#define AK_VOICE_R 0x03
-#define AK_FM_L 0x04
-#define AK_FM_R 0x05
-#define AK_CD_L 0x06
-#define AK_CD_R 0x07
-#define AK_LINE_L 0x08
-#define AK_LINE_R 0x09
-#define AK_AUX_L 0x0a
-#define AK_AUX_R 0x0b
-#define AK_MONO1 0x0c
-#define AK_MONO2 0x0d
-#define AK_MIC 0x0e
-#define AK_MONO 0x0f
-#define AK_OUT_MIXER1 0x10
-#define AK_M_FM_L 0x40
-#define AK_M_FM_R 0x20
-#define AK_M_LINE_L 0x10
-#define AK_M_LINE_R 0x08
-#define AK_M_CD_L 0x04
-#define AK_M_CD_R 0x02
-#define AK_M_MIC 0x01
-#define AK_OUT_MIXER2 0x11
-#define AK_M_AUX_L 0x20
-#define AK_M_AUX_R 0x10
-#define AK_M_VOICE_L 0x08
-#define AK_M_VOICE_R 0x04
-#define AK_M_MONO2 0x02
-#define AK_M_MONO1 0x01
-#define AK_IN_MIXER1_L 0x12
-#define AK_IN_MIXER1_R 0x13
-#define AK_IN_MIXER2_L 0x14
-#define AK_IN_MIXER2_R 0x15
-#define AK_M_TMIC 0x80
-#define AK_M_TMONO1 0x40
-#define AK_M_TMONO2 0x20
-#define AK_M2_AUX_L 0x10
-#define AK_M2_AUX_R 0x08
-#define AK_M_VOICE 0x04
-#define AK_M2_MONO2 0x02
-#define AK_M2_MONO1 0x01
-#define AK_RESET 0x16
-#define AK_PD 0x02
-#define AK_NRST 0x01
-#define AK_CS 0x17
-#define AK_ADSEL 0x18
-#define AK_MGAIN 0x19
-#define AK_NPORTS 0x20
-
-#define MAX_NPORTS AK_NPORTS
-
-/* Not sensical for AC97? */
-#define VOL_TO_ATT5(v) (0x1f - ((v) >> 3))
-#define VOL_TO_GAIN5(v) VOL_TO_ATT5(v)
-#define ATT5_TO_VOL(v) ((0x1f - (v)) << 3)
-#define GAIN5_TO_VOL(v) ATT5_TO_VOL(v)
-#define VOL_0DB 200
-
-/* Futzable parms */
-#define EAP_MASTER_VOL 0
-#define EAP_VOICE_VOL 1
-#define EAP_FM_VOL 2
-#define EAP_VIDEO_VOL 2 /* ES1371 */
-#define EAP_CD_VOL 3
-#define EAP_LINE_VOL 4
-#define EAP_AUX_VOL 5
-#define EAP_MIC_VOL 6
-#define EAP_RECORD_SOURCE 7
-#define EAP_OUTPUT_SELECT 8
-#define EAP_MIC_PREAMP 9
-#define EAP_OUTPUT_CLASS 10
-#define EAP_RECORD_CLASS 11
-#define EAP_INPUT_CLASS 12
-
/* Debug */
#ifdef AUDIO_DEBUG
#define DPRINTF(x) if (eapdebug) printf x
@@ -304,9 +91,9 @@ int eapdebug = 20;
#define DPRINTFN(n,x)
#endif
-int eap_match __P((struct device *, void *, void *));
-void eap_attach __P((struct device *, struct device *, void *));
-int eap_intr __P((void *));
+int eap_match(struct device *, void *, void *);
+void eap_attach(struct device *, struct device *, void *);
+int eap_intr(void *);
struct eap_dma {
bus_dmamap_t map;
@@ -316,6 +103,7 @@ struct eap_dma {
size_t size;
struct eap_dma *next;
};
+
#define DMAADDR(p) ((p)->map->dm_segs[0].ds_addr)
#define KERNADDR(p) ((void *)((p)->addr))
@@ -340,18 +128,18 @@ struct eap_softc {
char sc_rrun;
#endif
- u_short sc_port[MAX_NPORTS]; /* mirror of the hardware setting */
+ u_short sc_port[AK_NPORTS]; /* mirror of the hardware setting */
u_int sc_record_source; /* recording source mask */
u_int sc_output_source; /* output source mask */
u_int sc_mic_preamp;
- char sc_1371; /* Using ES1371/AC97 codec */
+ char sc_1371; /* Using ES1371/AC97 codec */
struct ac97_codec_if *codec_if;
struct ac97_host_if host_if;
};
-int eap_allocmem __P((struct eap_softc *, size_t, size_t, struct eap_dma *));
-int eap_freemem __P((struct eap_softc *, struct eap_dma *));
+int eap_allocmem(struct eap_softc *, size_t, size_t, struct eap_dma *);
+int eap_freemem(struct eap_softc *, struct eap_dma *);
#define EWRITE2(sc, r, x) bus_space_write_2((sc)->iot, (sc)->ioh, (r), (x))
#define EWRITE4(sc, r, x) bus_space_write_4((sc)->iot, (sc)->ioh, (r), (x))
@@ -362,43 +150,43 @@ struct cfattach eap_ca = {
sizeof(struct eap_softc), eap_match, eap_attach
};
-int eap_open __P((void *, int));
-void eap_close __P((void *));
-int eap_query_encoding __P((void *, struct audio_encoding *));
-int eap_set_params __P((void *, int, int, struct audio_params *, struct audio_params *));
-int eap_round_blocksize __P((void *, int));
-int eap_trigger_output __P((void *, void *, void *, int, void (*)(void *),
- void *, struct audio_params *));
-int eap_trigger_input __P((void *, void *, void *, int, void (*)(void *),
- void *, struct audio_params *));
-int eap_halt_output __P((void *));
-int eap_halt_input __P((void *));
-void eap_write_codec __P((struct eap_softc *, int, int));
-int eap_getdev __P((void *, struct audio_device *));
-int eap_mixer_set_port __P((void *, mixer_ctrl_t *));
-int eap_mixer_get_port __P((void *, mixer_ctrl_t *));
-int eap1371_mixer_set_port __P((void *, mixer_ctrl_t *));
-int eap1371_mixer_get_port __P((void *, mixer_ctrl_t *));
-int eap_query_devinfo __P((void *, mixer_devinfo_t *));
-void *eap_malloc __P((void *, u_long, int, int));
-void eap_free __P((void *, void *, int));
-u_long eap_round_buffersize __P((void *, u_long));
-int eap_mappage __P((void *, void *, int, int));
-int eap_get_props __P((void *));
-void eap_set_mixer __P((struct eap_softc *sc, int a, int d));
-int eap1371_src_wait __P((struct eap_softc *sc));
-void eap1371_set_adc_rate __P((struct eap_softc *sc, int rate));
-void eap1371_set_dac_rate __P((struct eap_softc *sc, int rate, int which));
-int eap1371_src_read __P((struct eap_softc *sc, int a));
-void eap1371_src_write __P((struct eap_softc *sc, int a, int d));
-int eap1371_query_devinfo __P((void *addr, mixer_devinfo_t *dip));
-
-int eap1371_attach_codec __P((void *sc, struct ac97_codec_if *));
-int eap1371_read_codec __P((void *sc, u_int8_t a, u_int16_t *d));
-int eap1371_write_codec __P((void *sc, u_int8_t a, u_int16_t d));
-void eap1371_reset_codec __P((void *sc));
-int eap1371_get_portnum_by_name __P((struct eap_softc *, char *, char *,
- char *));
+int eap_open(void *, int);
+void eap_close(void *);
+int eap_query_encoding(void *, struct audio_encoding *);
+int eap_set_params(void *, int, int, struct audio_params *, struct audio_params *);
+int eap_round_blocksize(void *, int);
+int eap_trigger_output(void *, void *, void *, int, void (*)(void *),
+ void *, struct audio_params *);
+int eap_trigger_input(void *, void *, void *, int, void (*)(void *),
+ void *, struct audio_params *);
+int eap_halt_output(void *);
+int eap_halt_input(void *);
+void eap1370_write_codec(struct eap_softc *, int, int);
+int eap_getdev(void *, struct audio_device *);
+int eap1370_mixer_set_port(void *, mixer_ctrl_t *);
+int eap1370_mixer_get_port(void *, mixer_ctrl_t *);
+int eap1371_mixer_set_port(void *, mixer_ctrl_t *);
+int eap1371_mixer_get_port(void *, mixer_ctrl_t *);
+int eap1370_query_devinfo(void *, mixer_devinfo_t *);
+void *eap_malloc(void *, u_long, int, int);
+void eap_free(void *, void *, int);
+u_long eap_round_buffersize(void *, u_long);
+int eap_mappage(void *, void *, int, int);
+int eap_get_props(void *);
+void eap1370_set_mixer(struct eap_softc *sc, int a, int d);
+u_int32_t eap1371_src_wait(struct eap_softc *sc);
+void eap1371_set_adc_rate(struct eap_softc *sc, int rate);
+void eap1371_set_dac_rate(struct eap_softc *sc, int rate, int which);
+int eap1371_src_read(struct eap_softc *sc, int a);
+void eap1371_src_write(struct eap_softc *sc, int a, int d);
+int eap1371_query_devinfo(void *addr, mixer_devinfo_t *dip);
+
+int eap1371_attach_codec(void *sc, struct ac97_codec_if *);
+int eap1371_read_codec(void *sc, u_int8_t a, u_int16_t *d);
+int eap1371_write_codec(void *sc, u_int8_t a, u_int16_t d);
+void eap1371_reset_codec(void *sc);
+int eap1371_get_portnum_by_name(struct eap_softc *, char *, char *,
+ char *);
struct audio_hw_if eap1370_hw_if = {
eap_open,
@@ -417,9 +205,9 @@ struct audio_hw_if eap1370_hw_if = {
NULL,
eap_getdev,
NULL,
- eap_mixer_set_port,
- eap_mixer_get_port,
- eap_query_devinfo,
+ eap1370_mixer_set_port,
+ eap1370_mixer_get_port,
+ eap1370_query_devinfo,
eap_malloc,
eap_free,
eap_round_buffersize,
@@ -465,302 +253,268 @@ struct audio_device eap_device = {
};
int
-eap_match(parent, match, aux)
- struct device *parent;
- void *match;
- void *aux;
+eap_match(struct device *parent, void *match, void *aux)
{
struct pci_attach_args *pa = (struct pci_attach_args *) aux;
- if (PCI_VENDOR(pa->pa_id) != PCI_VENDOR_ENSONIQ &&
- PCI_VENDOR(pa->pa_id) != PCI_VENDOR_CREATIVELABS)
- return (0);
-
- if (PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_AUDIOPCI ||
- PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_AUDIOPCI97 ||
- PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_CT5880 ||
- PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_CREATIVELABS_EV1938)
- return (1);
+ switch (PCI_VENDOR(pa->pa_id)) {
+ case PCI_VENDOR_CREATIVELABS:
+ switch (PCI_PRODUCT(pa->pa_id)) {
+ case PCI_PRODUCT_CREATIVELABS_EV1938:
+ return (1);
+ }
+ break;
+ case PCI_VENDOR_ENSONIQ:
+ switch (PCI_PRODUCT(pa->pa_id)) {
+ case PCI_PRODUCT_ENSONIQ_AUDIOPCI:
+ case PCI_PRODUCT_ENSONIQ_AUDIOPCI97:
+ case PCI_PRODUCT_ENSONIQ_CT5880:
+ return (1);
+ }
+ break;
+ }
return (0);
}
void
-eap_write_codec(sc, a, d)
- struct eap_softc *sc;
- int a, d;
+eap1370_write_codec(struct eap_softc *sc, int a, int d)
{
-
int icss, to;
-
+
to = EAP_WRITE_TIMEOUT;
do {
icss = EREAD4(sc, EAP_ICSS);
DPRINTFN(5,("eap: codec %d prog: icss=0x%08x\n", a, icss));
- if (!to--) {
- printf("eap: timeout writing to codec\n");
- return;
- }
+ if (!to--) {
+ printf("eap: timeout writing to codec\n");
+ return;
+ }
} while (icss & EAP_CWRIP); /* XXX could use CSTAT here */
- EWRITE4(sc, EAP_CODEC, EAP_SET_CODEC(a, d));
+ EWRITE4(sc, EAP_CODEC, EAP_SET_CODEC(a, d));
}
+/*
+ * Reading and writing the CODEC is very convoluted. This mimics the
+ * FreeBSD and Linux drivers.
+ */
-int
-eap1371_read_codec(sc_, a, d)
- void *sc_;
- u_int8_t a;
- u_int16_t *d;
+static __inline void
+eap1371_ready_codec(struct eap_softc *sc, u_int8_t a, u_int32_t wd)
{
- struct eap_softc *sc = sc_;
- int to;
- int cdc, src;
-
- to = EAP_WRITE_TIMEOUT;
- do {
- cdc = EREAD4(sc, E1371_CODEC);
- if (!to--) {
- printf("eap: timeout writing to codec\n");
- return 1;
- }
- } while (cdc & E1371_CODEC_WIP);
-
- /* just do it */
- src = (eap1371_src_wait(sc) & (E1371_SRC_DISABLE | E1371_SRC_DISP1 |
- E1371_SRC_DISP2 | E1371_SRC_DISREC)) ;
-
- EWRITE4(sc, E1371_SRC, src | 0x10000);
-
- for (to = 0; to < EAP_READ_TIMEOUT; to++) {
- if (!(EREAD4(sc, E1371_SRC) & E1371_SRC_STATE_MASK))
- break;
-
- delay(1);
- }
+ int to, s;
+ u_int32_t src, t;
for (to = 0; to < EAP_WRITE_TIMEOUT; to++) {
- if ((EREAD4(sc, E1371_SRC) & E1371_SRC_STATE_MASK) ==
- E1371_SRC_STATE_OK)
+ if (!(EREAD4(sc, E1371_CODEC) & E1371_CODEC_WIP))
break;
-
delay(1);
}
-
- EWRITE4(sc, E1371_CODEC, E1371_SET_CODEC(a, 0) | E1371_CODEC_READ);
+ if (to >= EAP_WRITE_TIMEOUT)
+ printf("%s: eap1371_ready_codec timeout 1\n",
+ sc->sc_dev.dv_xname);
- eap1371_src_wait(sc);
- EWRITE4(sc, E1371_SRC, src);
+ s = splaudio();
+ src = eap1371_src_wait(sc) & E1371_SRC_CTLMASK;
+ EWRITE4(sc, E1371_SRC, src | E1371_SRC_STATE_OK);
for (to = 0; to < EAP_READ_TIMEOUT; to++) {
- if ((cdc = EREAD4(sc, E1371_CODEC)) & E1371_CODEC_VALID)
+ t = EREAD4(sc, E1371_SRC);
+ if ((t & E1371_SRC_STATE_MASK) == 0)
break;
+ delay(1);
}
+ if (to >= EAP_READ_TIMEOUT)
+ printf("%s: eap1371_ready_codec timeout 2\n",
+ sc->sc_dev.dv_xname);
- if (to == EAP_WRITE_TIMEOUT) {
- DPRINTF(("eap1371: read codec timeout\n"));
+ for (to = 0; to < EAP_READ_TIMEOUT; to++) {
+ t = EREAD4(sc, E1371_SRC);
+ if ((t & E1371_SRC_STATE_MASK) == E1371_SRC_STATE_OK)
+ break;
+ delay(1);
}
+ if (to >= EAP_READ_TIMEOUT)
+ printf("%s: eap1371_ready_codec timeout 3\n",
+ sc->sc_dev.dv_xname);
- *d = cdc & 0xffff;
+ EWRITE4(sc, E1371_CODEC, wd);
- DPRINTFN(10, ("eap1371: reading codec (%x) = %x\n", a, *d));
+ eap1371_src_wait(sc);
+ EWRITE4(sc, E1371_SRC, src);
- return (0);
+ splx(s);
}
int
-eap1371_write_codec(sc_, a, d)
- void *sc_;
- u_int8_t a;
- u_int16_t d;
+eap1371_read_codec(void *sc_, u_int8_t a, u_int16_t *d)
{
struct eap_softc *sc = sc_;
- int to;
- int cdc, src;
-
- to = EAP_WRITE_TIMEOUT;
- do {
- cdc = EREAD4(sc, E1371_CODEC);
- if (!to--) {
- printf("eap: timeout writing to codec\n");
- return 1;
- }
- } while (cdc & E1371_CODEC_WIP);
+ int to;
+ u_int32_t t;
- /* just do it */
- src = (eap1371_src_wait(sc) & (E1371_SRC_DISABLE | E1371_SRC_DISP1 |
- E1371_SRC_DISP2 | E1371_SRC_DISREC)) ;
-
- EWRITE4(sc, E1371_SRC, src | 0x10000);
+ eap1371_ready_codec(sc, a, E1371_SET_CODEC(a, 0) | E1371_CODEC_READ);
for (to = 0; to < EAP_WRITE_TIMEOUT; to++) {
- if (!(EREAD4(sc, E1371_SRC) & E1371_SRC_STATE_MASK))
+ if (!(EREAD4(sc, E1371_CODEC) & E1371_CODEC_WIP))
break;
-
- delay(1);
}
-
+ if (to > EAP_WRITE_TIMEOUT)
+ printf("%s: eap1371_read_codec timeout 1\n",
+ sc->sc_dev.dv_xname);
+
for (to = 0; to < EAP_WRITE_TIMEOUT; to++) {
- if ((EREAD4(sc, E1371_SRC) & E1371_SRC_STATE_MASK) ==
- E1371_SRC_STATE_OK)
+ t = EREAD4(sc, E1371_CODEC);
+ if (t & E1371_CODEC_VALID)
break;
-
- delay(1);
}
+ if (to > EAP_WRITE_TIMEOUT)
+ printf("%s: eap1371_read_codec timeout 2\n",
+ sc->sc_dev.dv_xname);
- EWRITE4(sc, E1371_CODEC, E1371_SET_CODEC(a, d));
+ *d = (u_int16_t)t;
- eap1371_src_wait(sc);
- EWRITE4(sc, E1371_SRC, src);
+ DPRINTFN(10, ("eap1371: reading codec (%x) = %x\n", a, *d));
- DPRINTFN(10, ("eap1371: writing codec %x --> %x\n", d, a));
-
- return (0);
+ return (0);
}
int
-eap1371_src_wait(sc)
- struct eap_softc *sc;
+eap1371_write_codec(void *sc_, u_int8_t a, u_int16_t d)
{
- int to;
- int src;
+ struct eap_softc *sc = sc_;
+
+ eap1371_ready_codec(sc, a, E1371_SET_CODEC(a, d));
+
+ DPRINTFN(10, ("eap1371: writing codec %x --> %x\n", d, a));
+
+ return (0);
+}
+u_int32_t
+eap1371_src_wait(struct eap_softc *sc)
+{
+ int to;
+ u_int32_t src;
+
for (to = 0; to < EAP_READ_TIMEOUT; to++) {
src = EREAD4(sc, E1371_SRC);
if (!(src & E1371_SRC_RBUSY))
- return src;
+ return (src);
delay(1);
}
-
-
- printf("eap: timeout waiting for sample rate"
- "converter\n");
-
- return src;
+ printf("%s: eap1371_src_wait timeout\n", sc->sc_dev.dv_xname);
+ return (src);
}
int
-eap1371_src_read(sc, a)
- struct eap_softc *sc;
- int a;
+eap1371_src_read(struct eap_softc *sc, int a)
{
- int r, to;
- int src;
+ int to;
+ u_int32_t src, t;
- src = eap1371_src_wait(sc);
+ src = eap1371_src_wait(sc) & E1371_SRC_CTLMASK;
+ src |= E1371_SRC_ADDR(a);
+ EWRITE4(sc, E1371_SRC, src | E1371_SRC_STATE_OK);
- EWRITE4(sc, E1371_SRC,
- (src & (E1371_SRC_DISABLE | E1371_SRC_DISP1 |
- E1371_SRC_DISP2 | E1371_SRC_DISREC)) |
- E1371_SRC_ADDR(a) | 0x10000UL);
-
- r = eap1371_src_wait(sc);
-
- if ((r & E1371_SRC_STATE_MASK) != E1371_SRC_STATE_OK) {
+ if ((eap1371_src_wait(sc) & E1371_SRC_STATE_MASK) != E1371_SRC_STATE_OK) {
for (to = 0; to < EAP_READ_TIMEOUT; to++) {
- r = EREAD4(sc, E1371_SRC);
- if ((r & E1371_SRC_STATE_MASK) ==
- E1371_SRC_STATE_OK) break;
+ t = EREAD4(sc, E1371_SRC);
+ if ((t & E1371_SRC_STATE_MASK) == E1371_SRC_STATE_OK)
+ break;
+ delay(1);
}
}
- EWRITE4 (sc, E1371_SRC,
- (src & (E1371_SRC_DISABLE | E1371_SRC_DISP1 |
- E1371_SRC_DISP2 | E1371_SRC_DISREC)) |
- E1371_SRC_ADDR(a));
+ EWRITE4(sc, E1371_SRC, src);
- return r;
+ return t & E1371_SRC_DATAMASK;
}
void
-eap1371_src_write(sc, a, d)
- struct eap_softc *sc;
- int a,d;
+eap1371_src_write(struct eap_softc *sc, int a, int d)
{
- int r;
+ u_int32_t r;
- r = eap1371_src_wait(sc);
- r &= (E1371_SRC_DISABLE | E1371_SRC_DISP1 |
- E1371_SRC_DISP2 | E1371_SRC_DISREC);
+ r = eap1371_src_wait(sc) & E1371_SRC_CTLMASK;
r |= E1371_SRC_RAMWE | E1371_SRC_ADDR(a) | E1371_SRC_DATA(d);
EWRITE4(sc, E1371_SRC, r);
}
void
-eap1371_set_adc_rate(sc, rate)
- struct eap_softc *sc;
- int rate;
+eap1371_set_adc_rate(struct eap_softc *sc, int rate)
{
int freq, n, truncm;
int out;
-
- /* Whatever, it works, so I'll leave it :) */
-
- if (rate > 48000)
- rate = 48000;
- if (rate < 4000)
- rate = 4000;
- n = rate / 3000;
- if ((1 << n) & SRC_MAGIC)
- n--;
- truncm = ((21 * n) - 1) | 1;
- freq = ((48000 << 15) / rate) * n;
- if (rate >= 24000) {
- if (truncm > 239)
- truncm = 239;
+ int s;
+
+ /* Whatever, it works, so I'll leave it :) */
+
+ if (rate > 48000)
+ rate = 48000;
+ if (rate < 4000)
+ rate = 4000;
+ n = rate / 3000;
+ if ((1 << n) & SRC_MAGIC)
+ n--;
+ truncm = ((21 * n) - 1) | 1;
+ freq = ((48000 << 15) / rate) * n;
+ if (rate >= 24000) {
+ if (truncm > 239)
+ truncm = 239;
out = ESRC_SET_TRUNC((239 - truncm) / 2);
- } else {
- if (truncm > 119)
- truncm = 119;
+ } else {
+ if (truncm > 119)
+ truncm = 119;
out = ESRC_SMF | ESRC_SET_TRUNC((119 - truncm) / 2);
- }
+ }
out |= ESRC_SET_N(n);
- eap1371_src_write(sc, ESRC_ADC+ESRC_TRUNC_N, out);
+ s = splaudio();
+ eap1371_src_write(sc, ESRC_ADC+ESRC_TRUNC_N, out);
- out = eap1371_src_read(sc, ESRC_ADC+ESRC_IREGS) & 0xff;
- eap1371_src_write(sc, ESRC_ADC+ESRC_IREGS, out |
+ out = eap1371_src_read(sc, ESRC_ADC+ESRC_IREGS) & 0xff;
+ eap1371_src_write(sc, ESRC_ADC+ESRC_IREGS, out |
ESRC_SET_VFI(freq >> 15));
- eap1371_src_write(sc, ESRC_ADC+ESRC_VFF, freq & 0x7fff);
- eap1371_src_write(sc, ESRC_ADC_VOLL, ESRC_SET_ADC_VOL(n));
- eap1371_src_write(sc, ESRC_ADC_VOLR, ESRC_SET_ADC_VOL(n));
+ eap1371_src_write(sc, ESRC_ADC+ESRC_VFF, freq & 0x7fff);
+ eap1371_src_write(sc, ESRC_ADC_VOLL, ESRC_SET_ADC_VOL(n));
+ eap1371_src_write(sc, ESRC_ADC_VOLR, ESRC_SET_ADC_VOL(n));
+ splx(s);
}
void
-eap1371_set_dac_rate(sc, rate, which)
- struct eap_softc *sc;
- int rate;
- int which;
+eap1371_set_dac_rate(struct eap_softc *sc, int rate, int which)
{
- int dac = (which == 1) ? ESRC_DAC1 : ESRC_DAC2;
+ int dac = which == 1 ? ESRC_DAC1 : ESRC_DAC2;
int freq, r;
+ int s;
- /* Whatever, it works, so I'll leave it :) */
-
- if (rate > 48000)
- rate = 48000;
- if (rate < 4000)
- rate = 4000;
- freq = ((rate << 15) + 1500) / 3000;
-
- eap1371_src_wait(sc);
- r = EREAD4(sc, E1371_SRC) & (E1371_SRC_DISABLE |
- E1371_SRC_DISP2 | E1371_SRC_DISP1 | E1371_SRC_DISREC);
- r |= (which == 1) ? E1371_SRC_DISP1 : E1371_SRC_DISP2;
- EWRITE4(sc, E1371_SRC, r);
- r = eap1371_src_read(sc, dac + ESRC_IREGS) & 0x00ff;
- eap1371_src_write(sc, dac + ESRC_IREGS, r | ((freq >> 5) & 0xfc00));
- eap1371_src_write(sc, dac + ESRC_VFF, freq & 0x7fff);
- r = EREAD4(sc, E1371_SRC) & (E1371_SRC_DISABLE |
- E1371_SRC_DISP2 | E1371_SRC_DISP1 | E1371_SRC_DISREC);
- r &= ~((which == 1) ? E1371_SRC_DISP1 : E1371_SRC_DISP2);
- EWRITE4(sc, E1371_SRC, r);
+ /* Whatever, it works, so I'll leave it :) */
+
+ if (rate > 48000)
+ rate = 48000;
+ if (rate < 4000)
+ rate = 4000;
+ freq = ((rate << 15) + 1500) / 3000;
+
+ s = splaudio();
+ eap1371_src_wait(sc);
+ r = EREAD4(sc, E1371_SRC) & (E1371_SRC_DISABLE |
+ E1371_SRC_DISP2 | E1371_SRC_DISP1 | E1371_SRC_DISREC);
+ r |= (which == 1) ? E1371_SRC_DISP1 : E1371_SRC_DISP2;
+ EWRITE4(sc, E1371_SRC, r);
+ r = eap1371_src_read(sc, dac + ESRC_IREGS) & 0x00ff;
+ eap1371_src_write(sc, dac + ESRC_IREGS, r | ((freq >> 5) & 0xfc00));
+ eap1371_src_write(sc, dac + ESRC_VFF, freq & 0x7fff);
+ r = EREAD4(sc, E1371_SRC) & (E1371_SRC_DISABLE |
+ E1371_SRC_DISP2 | E1371_SRC_DISP1 | E1371_SRC_DISREC);
+ r &= ~(which == 1 ? E1371_SRC_DISP1 : E1371_SRC_DISP2);
+ EWRITE4(sc, E1371_SRC, r);
+ splx(s);
}
void
-eap_attach(parent, self, aux)
- struct device *parent;
- struct device *self;
- void *aux;
+eap_attach(struct device *parent, struct device *self, void *aux)
{
struct eap_softc *sc = (struct eap_softc *)self;
struct pci_attach_args *pa = (struct pci_attach_args *)aux;
@@ -771,11 +525,22 @@ eap_attach(parent, self, aux)
pcireg_t csr;
mixer_ctrl_t ctl;
int i;
- int revision;
+ int revision, ct5880;
+
+ /* Flag if we're "creative" */
+ sc->sc_1371 = !(PCI_VENDOR(pa->pa_id) == PCI_VENDOR_ENSONIQ &&
+ PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_AUDIOPCI);
- sc->sc_1371 =
- !(PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_AUDIOPCI);
revision = PCI_REVISION(pa->pa_class);
+ if (sc->sc_1371) {
+ if (PCI_VENDOR(pa->pa_id) == PCI_VENDOR_ENSONIQ &&
+ ((PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_AUDIOPCI97 &&
+ (revision == EAP_ES1373_8 || revision == EAP_CT5880_A)) ||
+ PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_CT5880))
+ ct5880 = 1;
+ else
+ ct5880 = 0;
+ }
/* Map I/O register */
if (pci_mapreg_map(pa, PCI_CBIO, PCI_MAPREG_TYPE_IO, 0,
@@ -818,9 +583,9 @@ eap_attach(parent, self, aux)
/* reset codec */
/* normal operation */
/* select codec clocks */
- eap_write_codec(sc, AK_RESET, AK_PD);
- eap_write_codec(sc, AK_RESET, AK_PD | AK_NRST);
- eap_write_codec(sc, AK_CS, 0x0);
+ eap1370_write_codec(sc, AK_RESET, AK_PD);
+ eap1370_write_codec(sc, AK_RESET, AK_PD | AK_NRST);
+ eap1370_write_codec(sc, AK_CS, 0x0);
eap_hw_if = &eap1370_hw_if;
@@ -850,21 +615,16 @@ eap_attach(parent, self, aux)
ctl.un.mask = 1 << EAP_MIC_VOL;
eap_hw_if->set_port(sc, &ctl);
} else {
- int error;
+ /* clean slate */
- /* clean slate */
EWRITE4(sc, EAP_SIC, 0);
- EWRITE4(sc, EAP_ICSC, 0);
- EWRITE4(sc, E1371_LEGACY, 0);
-
- /* It seems that revision 7 and greater of the AUDIOPCI 97
- are actually CT5880 */
- if ((PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_CT5880) ||
- ((PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_ENSONIQ_AUDIOPCI97)
- && (revision >= 7))) {
- EWRITE4(sc, EAP_ICSS, CT5880_AC97_RESET);
+ EWRITE4(sc, EAP_ICSC, 0);
+ EWRITE4(sc, E1371_LEGACY, 0);
- delay(20000);
+ if (ct5880) {
+ EWRITE4(sc, EAP_ICSS, EAP_CT5880_AC97_RESET);
+ /* Let codec wake up */
+ tsleep(sc, PRIBIO, "eapcdc", hz / 20);
}
/* Reset from es1371's perspective */
@@ -872,40 +632,40 @@ eap_attach(parent, self, aux)
delay(20);
EWRITE4(sc, EAP_ICSC, 0);
- /* must properly reprogram sample rate converter,
- * or it locks up. Set some defaults for the life of the
- * machine, and set up a sb default sample rate.
- */
- EWRITE4(sc, E1371_SRC, E1371_SRC_DISABLE);
- for (i=0; i<0x80; i++)
- eap1371_src_write(sc, i, 0);
+ /*
+ * Must properly reprogram sample rate converter,
+ * or it locks up. Set some defaults for the life of the
+ * machine, and set up a sb default sample rate.
+ */
+ EWRITE4(sc, E1371_SRC, E1371_SRC_DISABLE);
+ for (i = 0; i < 0x80; i++)
+ eap1371_src_write(sc, i, 0);
eap1371_src_write(sc, ESRC_DAC1+ESRC_TRUNC_N, ESRC_SET_N(16));
eap1371_src_write(sc, ESRC_DAC2+ESRC_TRUNC_N, ESRC_SET_N(16));
- eap1371_src_write(sc, ESRC_DAC1+ESRC_IREGS, ESRC_SET_VFI(16));
- eap1371_src_write(sc, ESRC_DAC2+ESRC_IREGS, ESRC_SET_VFI(16));
- eap1371_src_write(sc, ESRC_ADC_VOLL, ESRC_SET_ADC_VOL(16));
- eap1371_src_write(sc, ESRC_ADC_VOLR, ESRC_SET_ADC_VOL(16));
+ eap1371_src_write(sc, ESRC_DAC1+ESRC_IREGS, ESRC_SET_VFI(16));
+ eap1371_src_write(sc, ESRC_DAC2+ESRC_IREGS, ESRC_SET_VFI(16));
+ eap1371_src_write(sc, ESRC_ADC_VOLL, ESRC_SET_ADC_VOL(16));
+ eap1371_src_write(sc, ESRC_ADC_VOLR, ESRC_SET_ADC_VOL(16));
eap1371_src_write(sc, ESRC_DAC1_VOLL, ESRC_SET_DAC_VOLI(1));
eap1371_src_write(sc, ESRC_DAC1_VOLR, ESRC_SET_DAC_VOLI(1));
eap1371_src_write(sc, ESRC_DAC2_VOLL, ESRC_SET_DAC_VOLI(1));
eap1371_src_write(sc, ESRC_DAC2_VOLR, ESRC_SET_DAC_VOLI(1));
- eap1371_set_adc_rate(sc, 22050);
- eap1371_set_dac_rate(sc, 22050, 1);
- eap1371_set_dac_rate(sc, 22050, 2);
-
- EWRITE4(sc, E1371_SRC, 0);
+ eap1371_set_adc_rate(sc, 22050);
+ eap1371_set_dac_rate(sc, 22050, 1);
+ eap1371_set_dac_rate(sc, 22050, 2);
+
+ EWRITE4(sc, E1371_SRC, 0);
- /* Reset codec */
+ /* Reset codec */
/* Interrupt enable */
sc->host_if.arg = sc;
sc->host_if.attach = eap1371_attach_codec;
sc->host_if.read = eap1371_read_codec;
-
sc->host_if.write = eap1371_write_codec;
sc->host_if.reset = eap1371_reset_codec;
- if ((error = ac97_attach(&sc->host_if)) == 0) {
+ if (ac97_attach(&sc->host_if) == 0) {
/* Interrupt enable */
EWRITE4(sc, EAP_SIC, EAP_P2_INTR_EN | EAP_R1_INTR_EN);
} else
@@ -926,22 +686,19 @@ eap_attach(parent, self, aux)
AudioNvolume, AudioNmute);
eap1371_mixer_set_port(sc, &ctl);
-
ctl.dev = eap1371_get_portnum_by_name(sc, AudioCrecord,
AudioNsource, NULL);
ctl.type = AUDIO_MIXER_ENUM;
ctl.un.ord = 0;
eap1371_mixer_set_port(sc, &ctl);
- }
+ }
audio_attach_mi(eap_hw_if, sc, &sc->sc_dev);
}
int
-eap1371_attach_codec(sc_, codec_if)
- void *sc_;
- struct ac97_codec_if *codec_if;
+eap1371_attach_codec(void *sc_, struct ac97_codec_if *codec_if)
{
struct eap_softc *sc = sc_;
@@ -950,22 +707,25 @@ eap1371_attach_codec(sc_, codec_if)
}
void
-eap1371_reset_codec(sc_)
- void *sc_;
+eap1371_reset_codec(void *sc_)
{
struct eap_softc *sc = sc_;
- u_int32_t icsc = EREAD4(sc, EAP_ICSC);
-
+ u_int32_t icsc;
+ int s;
+
+ s = splaudio();
+ icsc = EREAD4(sc, EAP_ICSC);
EWRITE4(sc, EAP_ICSC, icsc | E1371_SYNC_RES);
- delay(100);
+ delay(20);
EWRITE4(sc, EAP_ICSC, icsc & ~E1371_SYNC_RES);
+ delay(1);
+ splx(s);
return;
}
int
-eap_intr(p)
- void *p;
+eap_intr(void *p)
{
struct eap_softc *sc = p;
u_int32_t intr, sic;
@@ -976,6 +736,7 @@ eap_intr(p)
sic = EREAD4(sc, EAP_SIC);
DPRINTFN(5, ("eap_intr: ICSS=0x%08x, SIC=0x%08x\n", intr, sic));
if (intr & EAP_I_ADC) {
+#if 0
/*
* XXX This is a hack!
* The EAP chip sometimes generates the recording interrupt
@@ -997,6 +758,7 @@ eap_intr(p)
}
}
/* Continue with normal interrupt handling. */
+#endif
EWRITE4(sc, EAP_SIC, sic & ~EAP_R1_INTR_EN);
EWRITE4(sc, EAP_SIC, sic | EAP_R1_INTR_EN);
if (sc->sc_rintr)
@@ -1012,11 +774,7 @@ eap_intr(p)
}
int
-eap_allocmem(sc, size, align, p)
- struct eap_softc *sc;
- size_t size;
- size_t align;
- struct eap_dma *p;
+eap_allocmem(struct eap_softc *sc, size_t size, size_t align, struct eap_dma *p)
{
int error;
@@ -1053,9 +811,7 @@ free:
}
int
-eap_freemem(sc, p)
- struct eap_softc *sc;
- struct eap_dma *p;
+eap_freemem(struct eap_softc *sc, struct eap_dma *p)
{
bus_dmamap_unload(sc->sc_dmatag, p->map);
bus_dmamap_destroy(sc->sc_dmatag, p->map);
@@ -1065,9 +821,7 @@ eap_freemem(sc, p)
}
int
-eap_open(addr, flags)
- void *addr;
- int flags;
+eap_open(void *addr, int flags)
{
return (0);
}
@@ -1076,8 +830,7 @@ eap_open(addr, flags)
* Close function is called at splaudio().
*/
void
-eap_close(addr)
- void *addr;
+eap_close(void *addr)
{
struct eap_softc *sc = addr;
@@ -1089,9 +842,7 @@ eap_close(addr)
}
int
-eap_query_encoding(addr, fp)
- void *addr;
- struct audio_encoding *fp;
+eap_query_encoding(void *addr, struct audio_encoding *fp)
{
switch (fp->index) {
case 0:
@@ -1148,10 +899,8 @@ eap_query_encoding(addr, fp)
}
int
-eap_set_params(addr, setmode, usemode, play, rec)
- void *addr;
- int setmode, usemode;
- struct audio_params *play, *rec;
+eap_set_params(void *addr, int setmode, int usemode,
+ struct audio_params *play, struct audio_params *rec)
{
struct eap_softc *sc = addr;
struct audio_params *p;
@@ -1203,26 +952,26 @@ eap_set_params(addr, setmode, usemode, play, rec)
case AUDIO_ENCODING_ULINEAR_BE:
if (p->precision == 16) {
if (mode == AUMODE_PLAY)
- p->sw_code = swap_bytes_change_sign16;
+ p->sw_code = swap_bytes_change_sign16_le;
else
- p->sw_code = change_sign16_swap_bytes;
+ p->sw_code = change_sign16_swap_bytes_le;
}
break;
case AUDIO_ENCODING_ULINEAR_LE:
if (p->precision == 16)
- p->sw_code = change_sign16;
+ p->sw_code = change_sign16_le;
break;
case AUDIO_ENCODING_ULAW:
if (mode == AUMODE_PLAY) {
p->factor = 2;
- p->sw_code = mulaw_to_slinear16;
+ p->sw_code = mulaw_to_slinear16_le;
} else
p->sw_code = ulinear8_to_mulaw;
break;
case AUDIO_ENCODING_ALAW:
if (mode == AUMODE_PLAY) {
p->factor = 2;
- p->sw_code = alaw_to_slinear16;
+ p->sw_code = alaw_to_slinear16_le;
} else
p->sw_code = ulinear8_to_alaw;
break;
@@ -1231,57 +980,55 @@ eap_set_params(addr, setmode, usemode, play, rec)
}
}
- if (sc->sc_1371) {
+ if (sc->sc_1371) {
eap1371_set_dac_rate(sc, play->sample_rate, 1);
eap1371_set_dac_rate(sc, play->sample_rate, 2);
eap1371_set_adc_rate(sc, rec->sample_rate);
} else {
- /* Set the speed */
- DPRINTFN(2, ("eap_set_params: old ICSC = 0x%08x\n",
- EREAD4(sc, EAP_ICSC)));
- div = EREAD4(sc, EAP_ICSC) & ~EAP_PCLKBITS;
- /*
- * XXX
- * The -2 isn't documented, but seemed to make the wall
- * time match
- * what I expect. - mycroft
- */
- if (usemode == AUMODE_RECORD)
- div |= EAP_SET_PCLKDIV(EAP_XTAL_FREQ /
- rec->sample_rate - 2);
- else
- div |= EAP_SET_PCLKDIV(EAP_XTAL_FREQ /
- play->sample_rate - 2);
- div |= EAP_CCB_INTRM;
- EWRITE4(sc, EAP_ICSC, div);
- DPRINTFN(2, ("eap_set_params: set ICSC = 0x%08x\n", div));
- }
+ /* Set the speed */
+ DPRINTFN(2, ("eap_set_params: old ICSC = 0x%08x\n",
+ EREAD4(sc, EAP_ICSC)));
+ div = EREAD4(sc, EAP_ICSC) & ~EAP_PCLKBITS;
+ /*
+ * XXX
+ * The -2 isn't documented, but seemed to make the wall
+ * time match
+ * what I expect. - mycroft
+ */
+ if (usemode == AUMODE_RECORD)
+ div |= EAP_SET_PCLKDIV(EAP_XTAL_FREQ /
+ rec->sample_rate - 2);
+ else
+ div |= EAP_SET_PCLKDIV(EAP_XTAL_FREQ /
+ play->sample_rate - 2);
+ div |= EAP_CCB_INTRM;
+ EWRITE4(sc, EAP_ICSC, div);
+ DPRINTFN(2, ("eap_set_params: set ICSC = 0x%08x\n", div));
+ }
return (0);
}
int
-eap_round_blocksize(addr, blk)
- void *addr;
- int blk;
+eap_round_blocksize(void *addr, int blk)
{
return (blk & -32); /* keep good alignment */
}
int
-eap_trigger_output(addr, start, end, blksize, intr, arg, param)
- void *addr;
- void *start, *end;
- int blksize;
- void (*intr) __P((void *));
- void *arg;
- struct audio_params *param;
+eap_trigger_output(
+ void *addr,
+ void *start,
+ void *end,
+ int blksize,
+ void (*intr)(void *),
+ void *arg,
+ struct audio_params *param)
{
struct eap_softc *sc = addr;
struct eap_dma *p;
u_int32_t icsc, sic;
int sampshift;
- int idx;
#ifdef DIAGNOSTIC
if (sc->sc_prun)
@@ -1290,12 +1037,12 @@ eap_trigger_output(addr, start, end, blksize, intr, arg, param)
#endif
DPRINTFN(1, ("eap_trigger_output: sc=%p start=%p end=%p "
- "blksize=%d intr=%p(%p)\n", addr, start, end, blksize, intr, arg));
+ "blksize=%d intr=%p(%p)\n", addr, start, end, blksize, intr, arg));
sc->sc_pintr = intr;
sc->sc_parg = arg;
sic = EREAD4(sc, EAP_SIC);
- sic &= ~(EAP_P2_S_EB | EAP_P2_S_MB | EAP_INC_BITS | EAP_P2_INTR_EN);
+ sic &= ~(EAP_P2_S_EB | EAP_P2_S_MB | EAP_INC_BITS);
sic |= EAP_SET_P2_ST_INC(0) | EAP_SET_P2_END_INC(param->precision * param->factor / 8);
sampshift = 0;
if (param->precision * param->factor == 16) {
@@ -1306,9 +1053,7 @@ eap_trigger_output(addr, start, end, blksize, intr, arg, param)
sic |= EAP_P2_S_MB;
sampshift++;
}
-
- DPRINTFN(1, ("set SIC=0x%08x\n", sic));
- EWRITE4(sc, EAP_SIC, sic);
+ EWRITE4(sc, EAP_SIC, sic & ~EAP_P2_INTR_EN);
EWRITE4(sc, EAP_SIC, sic | EAP_P2_INTR_EN);
for (p = sc->sc_dmas; p && KERNADDR(p) != start; p = p->next)
@@ -1320,45 +1065,34 @@ eap_trigger_output(addr, start, end, blksize, intr, arg, param)
DPRINTF(("eap_trigger_output: DAC2_ADDR=0x%x, DAC2_SIZE=0x%x\n",
(int)DMAADDR(p),
- EAP_SET_SIZE(0, (((char *)end - (char *)start) >> 2) - 1)));
+ (int)EAP_SET_SIZE(0, (((char *)end - (char *)start) >> 2) - 1)));
EWRITE4(sc, EAP_MEMPAGE, EAP_DAC_PAGE);
EWRITE4(sc, EAP_DAC2_ADDR, DMAADDR(p));
EWRITE4(sc, EAP_DAC2_SIZE,
EAP_SET_SIZE(0, (((char *)end - (char *)start) >> 2) - 1));
EWRITE4(sc, EAP_DAC2_CSR, (blksize >> sampshift) - 1);
- icsc = EREAD4(sc, EAP_ICSC);
if (sc->sc_1371)
EWRITE4(sc, E1371_SRC, 0);
- EWRITE4(sc, EAP_ICSC, icsc | EAP_DAC2_EN);
-
-
- for (idx = 0; idx < 3; idx++) {
- DPRINTFN (1, ("icsc=%08x sic=%08x dac2csr=%08x icss=%08x src=%08x\n",
- EREAD4(sc, EAP_ICSC),
- EREAD4(sc, EAP_SIC),
- EREAD4(sc, EAP_DAC2_CSR),
- EREAD4(sc, EAP_ICSS),
- EREAD4(sc, E1371_SRC)));
+ icsc = EREAD4(sc, EAP_ICSC);
+ EWRITE4(sc, EAP_ICSC, icsc | EAP_DAC2_EN);
- delay(1000);
- }
+ DPRINTFN(1, ("eap_trigger_output: set ICSC = 0x%08x\n", icsc));
return (0);
}
-
-
int
-eap_trigger_input(addr, start, end, blksize, intr, arg, param)
- void *addr;
- void *start, *end;
- int blksize;
- void (*intr) __P((void *));
- void *arg;
- struct audio_params *param;
+eap_trigger_input(
+ void *addr,
+ void *start,
+ void *end,
+ int blksize,
+ void (*intr)(void *),
+ void *arg,
+ struct audio_params *param)
{
struct eap_softc *sc = addr;
struct eap_dma *p;
@@ -1387,9 +1121,8 @@ eap_trigger_input(addr, start, end, blksize, intr, arg, param)
sic |= EAP_R1_S_MB;
sampshift++;
}
- DPRINTFN(1, ("set SIC=0x%08x\n", sic));
-
- EWRITE4(sc, EAP_SIC, sic);
+ EWRITE4(sc, EAP_SIC, sic & ~EAP_R1_INTR_EN);
+ EWRITE4(sc, EAP_SIC, sic | EAP_R1_INTR_EN);
for (p = sc->sc_dmas; p && KERNADDR(p) != start; p = p->next)
;
@@ -1400,7 +1133,7 @@ eap_trigger_input(addr, start, end, blksize, intr, arg, param)
DPRINTF(("eap_trigger_input: ADC_ADDR=0x%x, ADC_SIZE=0x%x\n",
(int)DMAADDR(p),
- EAP_SET_SIZE(0, (((char *)end - (char *)start) >> 2) - 1)));
+ (int)EAP_SET_SIZE(0, (((char *)end - (char *)start) >> 2) - 1)));
EWRITE4(sc, EAP_MEMPAGE, EAP_ADC_PAGE);
EWRITE4(sc, EAP_ADC_ADDR, DMAADDR(p));
EWRITE4(sc, EAP_ADC_SIZE,
@@ -1420,8 +1153,7 @@ eap_trigger_input(addr, start, end, blksize, intr, arg, param)
}
int
-eap_halt_output(addr)
- void *addr;
+eap_halt_output(void *addr)
{
struct eap_softc *sc = addr;
u_int32_t icsc;
@@ -1436,8 +1168,7 @@ eap_halt_output(addr)
}
int
-eap_halt_input(addr)
- void *addr;
+eap_halt_input(void *addr)
{
struct eap_softc *sc = addr;
u_int32_t icsc;
@@ -1452,70 +1183,55 @@ eap_halt_input(addr)
}
int
-eap_getdev(addr, retp)
- void *addr;
- struct audio_device *retp;
+eap_getdev(void *addr, struct audio_device *retp)
{
*retp = eap_device;
return (0);
}
int
-eap1371_mixer_set_port(addr, cp)
- void *addr;
- mixer_ctrl_t *cp;
+eap1371_mixer_set_port(void *addr, mixer_ctrl_t *cp)
{
struct eap_softc *sc = addr;
- return ((sc->codec_if->vtbl->mixer_set_port)(sc->codec_if,
- cp));
+ return (sc->codec_if->vtbl->mixer_set_port(sc->codec_if, cp));
}
int
-eap1371_mixer_get_port(addr, cp)
- void *addr;
- mixer_ctrl_t *cp;
+eap1371_mixer_get_port(void *addr, mixer_ctrl_t *cp)
{
struct eap_softc *sc = addr;
- return ((sc->codec_if->vtbl->mixer_get_port)(sc->codec_if,
- cp));
+ return (sc->codec_if->vtbl->mixer_get_port(sc->codec_if, cp));
}
int
-eap1371_query_devinfo(addr, dip)
- void *addr;
- mixer_devinfo_t *dip;
+eap1371_query_devinfo(void *addr, mixer_devinfo_t *dip)
{
struct eap_softc *sc = addr;
- return ((sc->codec_if->vtbl->query_devinfo)(sc->codec_if, dip));
+ return (sc->codec_if->vtbl->query_devinfo(sc->codec_if, dip));
}
int
-eap1371_get_portnum_by_name(sc, class, device, qualifier)
- struct eap_softc *sc;
- char *class, *device, *qualifier;
+eap1371_get_portnum_by_name(struct eap_softc *sc,
+ char *class, char *device, char *qualifier)
{
- return ((sc->codec_if->vtbl->get_portnum_by_name)(sc->codec_if, class,
- device, qualifier));
+ return (sc->codec_if->vtbl->get_portnum_by_name(sc->codec_if, class,
+ device, qualifier));
}
void
-eap_set_mixer(sc, a, d)
- struct eap_softc *sc;
- int a, d;
+eap1370_set_mixer(struct eap_softc *sc, int a, int d)
{
- eap_write_codec(sc, a, d);
+ eap1370_write_codec(sc, a, d);
- sc->sc_port[a] = d;
- DPRINTFN(1, ("eap_mixer_set_port port 0x%02x = 0x%02x\n", a, d));
+ sc->sc_port[a] = d;
+ DPRINTFN(1, ("eap1370_mixer_set_port port 0x%02x = 0x%02x\n", a, d));
}
int
-eap_mixer_set_port(addr, cp)
- void *addr;
- mixer_ctrl_t *cp;
+eap1370_mixer_set_port(void *addr, mixer_ctrl_t *cp)
{
struct eap_softc *sc = addr;
int lval, rval, l, r, la, ra;
@@ -1538,10 +1254,10 @@ eap_mixer_set_port(addr, cp)
l2 |= AK_M2_AUX_L, r2 |= AK_M2_AUX_R;
if (m & (1 << EAP_MIC_VOL))
l2 |= AK_M_TMIC, r2 |= AK_M_TMIC;
- eap_set_mixer(sc, AK_IN_MIXER1_L, l1);
- eap_set_mixer(sc, AK_IN_MIXER1_R, r1);
- eap_set_mixer(sc, AK_IN_MIXER2_L, l2);
- eap_set_mixer(sc, AK_IN_MIXER2_R, r2);
+ eap1370_set_mixer(sc, AK_IN_MIXER1_L, l1);
+ eap1370_set_mixer(sc, AK_IN_MIXER1_R, r1);
+ eap1370_set_mixer(sc, AK_IN_MIXER2_L, l2);
+ eap1370_set_mixer(sc, AK_IN_MIXER2_R, r2);
return (0);
}
if (cp->dev == EAP_OUTPUT_SELECT) {
@@ -1561,8 +1277,8 @@ eap_mixer_set_port(addr, cp)
o2 |= AK_M_AUX_L | AK_M_AUX_R;
if (m & (1 << EAP_MIC_VOL))
o1 |= AK_M_MIC;
- eap_set_mixer(sc, AK_OUT_MIXER1, o1);
- eap_set_mixer(sc, AK_OUT_MIXER2, o2);
+ eap1370_set_mixer(sc, AK_OUT_MIXER1, o1);
+ eap1370_set_mixer(sc, AK_OUT_MIXER2, o2);
return (0);
}
if (cp->dev == EAP_MIC_PREAMP) {
@@ -1571,7 +1287,7 @@ eap_mixer_set_port(addr, cp)
if (cp->un.ord != 0 && cp->un.ord != 1)
return (EINVAL);
sc->sc_mic_preamp = cp->un.ord;
- eap_set_mixer(sc, AK_MGAIN, cp->un.ord);
+ eap1370_set_mixer(sc, AK_MGAIN, cp->un.ord);
return (0);
}
if (cp->type != AUDIO_MIXER_VALUE)
@@ -1622,17 +1338,15 @@ eap_mixer_set_port(addr, cp)
default:
return (EINVAL);
}
- eap_set_mixer(sc, la, l);
+ eap1370_set_mixer(sc, la, l);
if (ra >= 0) {
- eap_set_mixer(sc, ra, r);
+ eap1370_set_mixer(sc, ra, r);
}
return (0);
}
int
-eap_mixer_get_port(addr, cp)
- void *addr;
- mixer_ctrl_t *cp;
+eap1370_mixer_get_port(void *addr, mixer_ctrl_t *cp)
{
struct eap_softc *sc = addr;
int la, ra, l, r;
@@ -1699,9 +1413,7 @@ eap_mixer_get_port(addr, cp)
}
int
-eap_query_devinfo(addr, dip)
- void *addr;
- mixer_devinfo_t *dip;
+eap1370_query_devinfo(void *addr, mixer_devinfo_t *dip)
{
switch (dip->index) {
case EAP_MASTER_VOL:
@@ -1839,10 +1551,7 @@ eap_query_devinfo(addr, dip)
}
void *
-eap_malloc(addr, size, pool, flags)
- void *addr;
- u_long size;
- int pool, flags;
+eap_malloc(void *addr, u_long size, int pool, int flags)
{
struct eap_softc *sc = addr;
struct eap_dma *p;
@@ -1862,38 +1571,29 @@ eap_malloc(addr, size, pool, flags)
}
void
-eap_free(addr, ptr, pool)
- void *addr;
- void *ptr;
- int pool;
+eap_free(void *addr, void *ptr, int pool)
{
struct eap_softc *sc = addr;
- struct eap_dma **p;
+ struct eap_dma **pp, *p;
- for (p = &sc->sc_dmas; *p; p = &(*p)->next) {
- if (KERNADDR(*p) == ptr) {
- eap_freemem(sc, *p);
- *p = (*p)->next;
- free(*p, pool);
+ for (pp = &sc->sc_dmas; (p = *pp) != NULL; pp = &p->next) {
+ if (KERNADDR(p) == ptr) {
+ eap_freemem(sc, p);
+ *pp = p->next;
+ free(p, pool);
return;
}
}
}
u_long
-eap_round_buffersize(addr, size)
- void *addr;
- u_long size;
+eap_round_buffersize(void *addr, u_long size)
{
return (size);
}
int
-eap_mappage(addr, mem, off, prot)
- void *addr;
- void *mem;
- int off;
- int prot;
+eap_mappage(void *addr, void *mem, int off, int prot)
{
struct eap_softc *sc = addr;
struct eap_dma *p;
@@ -1909,9 +1609,8 @@ eap_mappage(addr, mem, off, prot)
}
int
-eap_get_props(addr)
- void *addr;
+eap_get_props(void *addr)
{
return (AUDIO_PROP_MMAP | AUDIO_PROP_INDEPENDENT |
- AUDIO_PROP_FULLDUPLEX);
+ AUDIO_PROP_FULLDUPLEX);
}
diff --git a/sys/dev/pci/eapreg.h b/sys/dev/pci/eapreg.h
new file mode 100644
index 00000000000..0997f83e8eb
--- /dev/null
+++ b/sys/dev/pci/eapreg.h
@@ -0,0 +1,284 @@
+/* $OpenBSD: eapreg.h,v 1.1 2001/09/06 14:47:07 naddy Exp $ */
+/* $NetBSD: eapreg.h,v 1.2 2000/04/30 21:59:58 augustss Exp $ */
+
+/*
+ * Copyright (c) 1998, 1999 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Lennart Augustsson <augustss@netbsd.org> and Charles M. Hannum.
+ *
+ * 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. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the NetBSD
+ * Foundation, Inc. and its contributors.
+ * 4. Neither the name of The NetBSD Foundation nor the names of its
+ * contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
+ * ``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 FOUNDATION OR CONTRIBUTORS
+ * 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.
+ */
+
+/*
+ * ES1370/ES1371/ES1373 registers
+ */
+
+#define EAP_ICSC 0x00 /* interrupt / chip select control */
+#define EAP_SERR_DISABLE 0x00000001
+#define EAP_CDC_EN 0x00000002
+#define EAP_JYSTK_EN 0x00000004
+#define EAP_UART_EN 0x00000008
+#define EAP_ADC_EN 0x00000010
+#define EAP_DAC2_EN 0x00000020
+#define EAP_DAC1_EN 0x00000040
+#define EAP_BREQ 0x00000080
+#define EAP_XTCL0 0x00000100
+#define EAP_M_CB 0x00000200
+#define EAP_CCB_INTRM 0x00000400
+#define EAP_DAC_SYNC 0x00000800
+#define EAP_WTSRSEL 0x00003000
+#define EAP_WTSRSEL_5 0x00000000
+#define EAP_WTSRSEL_11 0x00001000
+#define EAP_WTSRSEL_22 0x00002000
+#define EAP_WTSRSEL_44 0x00003000
+#define EAP_M_SBB 0x00004000
+#define E1371_SYNC_RES 0x00004000
+#define EAP_MSFMTSEL 0x00008000
+#define EAP_SET_PCLKDIV(n) (((n)&0x1fff)<<16)
+#define EAP_GET_PCLKDIV(n) (((n)>>16)&0x1fff)
+#define EAP_PCLKBITS 0x1fff0000
+#define EAP_XTCL1 0x40000000
+#define EAP_ADC_STOP 0x80000000
+
+#define EAP_ICSS 0x04 /* interrupt / chip select status */
+ /* on the 5880 control / status */
+#define EAP_I_ADC 0x00000001
+#define EAP_I_DAC2 0x00000002
+#define EAP_I_DAC1 0x00000004
+#define EAP_I_UART 0x00000008
+#define EAP_I_MCCB 0x00000010
+#define EAP_VC 0x00000060
+#define EAP_CWRIP 0x00000100
+#define EAP_CBUSY 0x00000200
+#define EAP_CSTAT 0x00000400
+#define EAP_CT5880_AC97_RESET 0x20000000
+#define EAP_INTR 0x80000000
+
+#define EAP_UART_DATA 0x08
+#define EAP_UART_STATUS 0x09
+#define EAP_US_RXRDY 0x01
+#define EAP_US_TXRDY 0x02
+#define EAP_US_TXINT 0x04
+#define EAP_US_RXINT 0x80
+#define EAP_UART_CONTROL 0x09
+#define EAP_UC_CNTRL 0x03
+#define EAP_UC_TXINTEN 0x20
+#define EAP_UC_RXINTEN 0x80
+#define EAP_MEMPAGE 0x0c
+#define EAP_CODEC 0x10
+#define EAP_SET_CODEC(a,d) (((a)<<8) | (d))
+
+/*
+ * ES1371/ES1373 registers
+ */
+
+#define E1371_CODEC 0x14
+#define E1371_CODEC_VALID 0x80000000
+#define E1371_CODEC_WIP 0x40000000
+#define E1371_CODEC_READ 0x00800000
+#define E1371_SET_CODEC(a,d) (((a)<<16) | (d))
+
+#define E1371_SRC 0x10
+#define E1371_SRC_RAMWE 0x01000000
+#define E1371_SRC_RBUSY 0x00800000
+#define E1371_SRC_DISABLE 0x00400000
+#define E1371_SRC_DISP1 0x00200000
+#define E1371_SRC_DISP2 0x00100000
+#define E1371_SRC_DISREC 0x00080000
+#define E1371_SRC_ADDR(a) ((a)<<25)
+#define E1371_SRC_DATA(d) (d)
+#define E1371_SRC_DATAMASK 0x0000ffff
+#define E1371_SRC_CTLMASK (E1371_SRC_DISABLE | E1371_SRC_DISP1 | \
+ E1371_SRC_DISP2 | E1371_SRC_DISREC)
+#define E1371_SRC_STATE_MASK 0x00870000
+#define E1371_SRC_STATE_OK 0x00010000
+
+#define E1371_LEGACY 0x18
+
+/*
+ * ES1371/ES1373 sample rate converter registers
+ */
+
+#define ESRC_ADC 0x78
+#define ESRC_DAC1 0x74
+#define ESRC_DAC2 0x70
+#define ESRC_ADC_VOLL 0x6c
+#define ESRC_ADC_VOLR 0x6d
+#define ESRC_DAC1_VOLL 0x7c
+#define ESRC_DAC1_VOLR 0x7d
+#define ESRC_DAC2_VOLL 0x7e
+#define ESRC_DAC2_VOLR 0x7f
+#define ESRC_TRUNC_N 0x00
+#define ESRC_IREGS 0x01
+#define ESRC_ACF 0x02
+#define ESRC_VFF 0x03
+#define ESRC_SET_TRUNC(n) ((n)<<9)
+#define ESRC_SET_N(n) ((n)<<4)
+#define ESRC_SMF 0x8000
+#define ESRC_SET_VFI(n) ((n)<<10)
+#define ESRC_SET_ACI(n) (n)
+#define ESRC_SET_ADC_VOL(n) ((n)<<8)
+#define ESRC_SET_DAC_VOLI(n) ((n)<<12)
+#define ESRC_SET_DAC_VOLF(n) (n)
+#define SRC_MAGIC ((1<15)|(1<<13)|(1<<11)|(1<<9))
+
+#define EAP_SIC 0x20
+#define EAP_P1_S_MB 0x00000001
+#define EAP_P1_S_EB 0x00000002
+#define EAP_P2_S_MB 0x00000004
+#define EAP_P2_S_EB 0x00000008
+#define EAP_R1_S_MB 0x00000010
+#define EAP_R1_S_EB 0x00000020
+#define EAP_P2_DAC_SEN 0x00000040
+#define EAP_P1_SCT_RLD 0x00000080
+#define EAP_P1_INTR_EN 0x00000100
+#define EAP_P2_INTR_EN 0x00000200
+#define EAP_R1_INTR_EN 0x00000400
+#define EAP_P1_PAUSE 0x00000800
+#define EAP_P2_PAUSE 0x00001000
+#define EAP_P1_LOOP_SEL 0x00002000
+#define EAP_P2_LOOP_SEL 0x00004000
+#define EAP_R1_LOOP_SEL 0x00008000
+#define EAP_SET_P2_ST_INC(i) ((i) << 16)
+#define EAP_SET_P2_END_INC(i) ((i) << 19)
+#define EAP_INC_BITS 0x003f0000
+
+#define EAP_DAC1_CSR 0x24
+#define EAP_DAC2_CSR 0x28
+#define EAP_ADC_CSR 0x2c
+#define EAP_GET_CURRSAMP(r) ((r) >> 16)
+
+#define EAP_DAC_PAGE 0xc
+#define EAP_ADC_PAGE 0xd
+#define EAP_UART_PAGE1 0xe
+#define EAP_UART_PAGE2 0xf
+
+#define EAP_DAC1_ADDR 0x30
+#define EAP_DAC1_SIZE 0x34
+#define EAP_DAC2_ADDR 0x38
+#define EAP_DAC2_SIZE 0x3c
+#define EAP_ADC_ADDR 0x30
+#define EAP_ADC_SIZE 0x34
+#define EAP_SET_SIZE(c,s) (((c)<<16) | (s))
+
+#define EAP_READ_TIMEOUT 5000
+#define EAP_WRITE_TIMEOUT 5000
+
+
+#define EAP_XTAL_FREQ 1411200 /* 22.5792 / 16 MHz */
+
+/* AK4531 registers */
+#define AK_MASTER_L 0x00
+#define AK_MASTER_R 0x01
+#define AK_VOICE_L 0x02
+#define AK_VOICE_R 0x03
+#define AK_FM_L 0x04
+#define AK_FM_R 0x05
+#define AK_CD_L 0x06
+#define AK_CD_R 0x07
+#define AK_LINE_L 0x08
+#define AK_LINE_R 0x09
+#define AK_AUX_L 0x0a
+#define AK_AUX_R 0x0b
+#define AK_MONO1 0x0c
+#define AK_MONO2 0x0d
+#define AK_MIC 0x0e
+#define AK_MONO 0x0f
+#define AK_OUT_MIXER1 0x10
+#define AK_M_FM_L 0x40
+#define AK_M_FM_R 0x20
+#define AK_M_LINE_L 0x10
+#define AK_M_LINE_R 0x08
+#define AK_M_CD_L 0x04
+#define AK_M_CD_R 0x02
+#define AK_M_MIC 0x01
+#define AK_OUT_MIXER2 0x11
+#define AK_M_AUX_L 0x20
+#define AK_M_AUX_R 0x10
+#define AK_M_VOICE_L 0x08
+#define AK_M_VOICE_R 0x04
+#define AK_M_MONO2 0x02
+#define AK_M_MONO1 0x01
+#define AK_IN_MIXER1_L 0x12
+#define AK_IN_MIXER1_R 0x13
+#define AK_IN_MIXER2_L 0x14
+#define AK_IN_MIXER2_R 0x15
+#define AK_M_TMIC 0x80
+#define AK_M_TMONO1 0x40
+#define AK_M_TMONO2 0x20
+#define AK_M2_AUX_L 0x10
+#define AK_M2_AUX_R 0x08
+#define AK_M_VOICE 0x04
+#define AK_M2_MONO2 0x02
+#define AK_M2_MONO1 0x01
+#define AK_RESET 0x16
+#define AK_PD 0x02
+#define AK_NRST 0x01
+#define AK_CS 0x17
+#define AK_ADSEL 0x18
+#define AK_MGAIN 0x19
+#define AK_NPORTS 0x20
+
+/* Not sensical for AC97? */
+#define VOL_TO_ATT5(v) (0x1f - ((v) >> 3))
+#define VOL_TO_GAIN5(v) VOL_TO_ATT5(v)
+#define ATT5_TO_VOL(v) ((0x1f - (v)) << 3)
+#define GAIN5_TO_VOL(v) ATT5_TO_VOL(v)
+#define VOL_0DB 200
+
+/* Futzable parms */
+#define EAP_MASTER_VOL 0
+#define EAP_VOICE_VOL 1
+#define EAP_FM_VOL 2
+#define EAP_VIDEO_VOL 2 /* ES1371 */
+#define EAP_CD_VOL 3
+#define EAP_LINE_VOL 4
+#define EAP_AUX_VOL 5
+#define EAP_MIC_VOL 6
+#define EAP_RECORD_SOURCE 7
+#define EAP_OUTPUT_SELECT 8
+#define EAP_MIC_PREAMP 9
+#define EAP_OUTPUT_CLASS 10
+#define EAP_RECORD_CLASS 11
+#define EAP_INPUT_CLASS 12
+
+#define MIDI_BUSY_WAIT 100
+#define MIDI_BUSY_DELAY 100 /* Delay when UART is busy */
+
+#define EAP_EV1938_A 0x00
+#define EAP_ES1371_A 0x02
+#define EAP_CT5880_C 0x02
+#define EAP_CT5880_D 0x03
+#define EAP_ES1373_A 0x04
+#define EAP_ES1373_B 0x06
+#define EAP_CT5880_A 0x07
+#define EAP_ES1373_8 0x08
+#define EAP_ES1371_B 0x09