Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/isaki-audio2]: src/sys/dev/pci Reimplement emuxki driver.
details:   https://anonhg.NetBSD.org/src/rev/7073727c3129
branches:  isaki-audio2
changeset: 456021:7073727c3129
user:      isaki <isaki%NetBSD.org@localhost>
date:      Wed May 01 06:34:46 2019 +0000
description:
Reimplement emuxki driver.
- Use single voice per playback and per recording.
- Use fixed format, 2ch/48kHz, to simplify.
- Fix several problems in previous driver.
  And now it works even on alpha!
The driver is written by Y.Sugahara.  Thanks!
diffstat:
 sys/dev/pci/emuxki.c    |  2830 +++++++++++++++-------------------------------
 sys/dev/pci/emuxkivar.h |   263 ----
 2 files changed, 929 insertions(+), 2164 deletions(-)
diffs (truncated from 3518 to 300 lines):
diff -r a0df361c31f1 -r 7073727c3129 sys/dev/pci/emuxki.c
--- a/sys/dev/pci/emuxki.c      Wed May 01 06:03:14 2019 +0000
+++ b/sys/dev/pci/emuxki.c      Wed May 01 06:34:46 2019 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: emuxki.c,v 1.67.2.3 2019/05/01 06:03:14 isaki Exp $    */
+/*     $NetBSD: emuxki.c,v 1.67.2.4 2019/05/01 06:34:46 isaki Exp $    */
 
 /*-
  * Copyright (c) 2001, 2007 The NetBSD Foundation, Inc.
@@ -30,147 +30,227 @@
  */
 
 /*
- * Driver for Creative Labs SBLive! series and probably PCI512.
- *
- * Known bugs:
- * - inversed stereo at ac97 codec level
- *   (XXX jdolecek - don't see the problem? maybe because auvia(4) has
- *    it swapped too?)
- * - bass disappear when you plug rear jack-in on Cambridge FPS2000 speakers
- *   (and presumably all speakers that support front and rear jack-in)
- *
- * TODO:
- * - Digital Outputs
- * - (midi/mpu),joystick support
- * - Multiple voices play (problem with /dev/audio architecture)
- * - Multiple sources recording (Pb with audio(4))
- * - Independent modification of each channel's parameters (via mixer ?)
- * - DSP FX patches (to make fx like chipmunk)
+ * EMU10K1 single voice driver
+ * o. only 1 voice playback, 1 recording
+ * o. only s16le 2ch 48k
+ * This makes it simple to control buffers and interrupts
+ * while satisfying playback and recording quality.
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: emuxki.c,v 1.67.2.3 2019/05/01 06:03:14 isaki Exp $");
+__KERNEL_RCSID(0, "$NetBSD: emuxki.c,v 1.67.2.4 2019/05/01 06:34:46 isaki Exp $");
 
 #include <sys/param.h>
 #include <sys/device.h>
 #include <sys/errno.h>
 #include <sys/systm.h>
 #include <sys/audioio.h>
-#include <sys/select.h>
 #include <sys/mutex.h>
 #include <sys/kmem.h>
 #include <sys/malloc.h>
 #include <sys/fcntl.h>
 
+#include <sys/bus.h>
+#include <sys/intr.h>
+
 #include <dev/audio_if.h>
 #include <dev/audiovar.h>
 
 #include <dev/ic/ac97reg.h>
 #include <dev/ic/ac97var.h>
 
+#include <dev/pci/pcidevs.h>
 #include <dev/pci/pcireg.h>
 #include <dev/pci/pcivar.h>
-#include <dev/pci/pcidevs.h>
+
 #include <dev/pci/emuxkireg.h>
-#include <dev/pci/emuxkivar.h>
+
+/* #define EMUXKI_DEBUG 1 */
+#ifdef EMUXKI_DEBUG
+#define emudebug EMUXKI_DEBUG
+# define DPRINTF(fmt...)       do { if (emudebug) printf(fmt); } while (0)
+# define DPRINTFN(n,fmt...)    do { if (emudebug>=(n)) printf(fmt); } while (0)
+#else
+# define DPRINTF(fmt...)       do { } while (0)
+# define DPRINTFN(n,fmt...)    do { } while (0)
+#endif
+
+/*
+ * PCI
+ * Note: emuxki's page table entry uses only 31bit addressing.
+ *       (Maybe, later chip has 32bit mode, but it isn't used now.)
+ */
+
+#define EMU_PCI_CBIO           (0x10)
+#define EMU_SUBSYS_APS         (0x40011102)
+
+#define EMU_PTESIZE            (4096)
+#define EMU_MINPTE             (3)
+/*
+ * Hardware limit of PTE is 4096 entry but it's too big for single voice.
+ * Reasonable candidate is:
+ *  48kHz * 2ch * 2byte * 1sec * 3buf/EMU_PTESIZE = 141
+ * and then round it up to 2^n.
+ */
+#define EMU_MAXPTE             (256)
+#define EMU_NUMCHAN            (64)
+
+/*
+ * Internal recording DMA buffer
+ */
+/* Recommend the same size as EMU_PTESIZE to be symmetrical for play/rec */
+#define EMU_REC_DMABLKSIZE     (4096)
+/* must be EMU_REC_DMABLKSIZE * 2 */
+#define EMU_REC_DMASIZE                (8192)
+/* must be EMU_RECBS_BUFSIZE_(EMU_REC_DMASIZE) */
+#define EMU_REC_BUFSIZE_RECBS  EMU_RECBS_BUFSIZE_8192
+
+/*
+ * DMA memory management
+ */
+
+#define EMU_DMA_ALIGN          (4096)
+#define EMU_DMA_NSEGS          (1)
+
+struct dmamem {
+       bus_dma_tag_t           dmat;
+       bus_size_t              size;
+       bus_size_t              align;
+       bus_size_t              bound;
+       bus_dma_segment_t       *segs;
+       int                     nsegs;
+       int                     rsegs;
+       void *                  kaddr;
+       bus_dmamap_t            map;
+};
+
+#define KERNADDR(ptr)          ((void *)((ptr)->kaddr))
+/*
+ * (ptr)->segs[] is CPU's PA translated by CPU's MMU.
+ * (ptr)->map->dm_segs[] is PCI device's PA translated by PCI's MMU.
+ */
+#define DMASEGADDR(ptr, segno) ((ptr)->map->dm_segs[segno].ds_addr)
+#define DMAADDR(ptr)           DMASEGADDR(ptr, 0)
+#define DMASIZE(ptr)           ((ptr)->size)
 
-/* autoconf goo */
+struct emuxki_softc {
+       device_t                sc_dev;
+       device_t                sc_audev;
+       enum {
+               EMUXKI_SBLIVE = 0x00,
+               EMUXKI_AUDIGY = 0x01,
+               EMUXKI_AUDIGY2 = 0x02,
+               EMUXKI_LIVE_5_1 = 0x04,
+               EMUXKI_APS = 0x08
+       } sc_type;
+       audio_device_t          sc_audv;        /* for GETDEV */
+
+       /* Autoconfig parameters */
+       bus_space_tag_t         sc_iot;
+       bus_space_handle_t      sc_ioh;
+       bus_addr_t              sc_iob;
+       bus_size_t              sc_ios;
+       pci_chipset_tag_t       sc_pc;          /* PCI tag */
+       bus_dma_tag_t           sc_dmat;
+       void                    *sc_ih;         /* interrupt handler */
+       kmutex_t                sc_intr_lock;
+       kmutex_t                sc_lock;
+       kmutex_t                sc_index_lock;
+
+       /* register parameters */
+       struct dmamem           *ptb;           /* page table */
+
+       struct dmamem           *pmem;          /* play memory */
+       void                    (*pintr)(void *);
+       void                    *pintrarg;
+       audio_params_t          play;
+       int                     pframesize;
+       int                     pblksize;
+       int                     plength;
+       int                     poffset;
+
+       struct dmamem           *rmem;          /* rec internal memory */
+       void                    (*rintr)(void *);
+       void                    *rintrarg;
+       audio_params_t          rec;
+       void                    *rptr;          /* rec MI ptr */
+       int                     rcurrent;       /* rec software trans count */
+       int                     rframesize;
+       int                     rblksize;
+       int                     rlength;
+       int                     roffset;
+
+       /* others */
+
+       struct ac97_host_if     hostif;
+       struct ac97_codec_if    *codecif;
+};
+
+/* blackmagic */
+#define X1(x)          ((sc->sc_type & EMUXKI_AUDIGY) ? EMU_A_##x : EMU_##x)
+#define X2(x, y)       ((sc->sc_type & EMUXKI_AUDIGY) \
+    ? EMU_A_##x(EMU_A_##y) : EMU_##x(EMU_##y))
+#define EMU_A_DSP_FX           EMU_DSP_FX
+#define EMU_A_DSP_IN_AC97      EMU_DSP_IN_AC97
+
+/* prototypes */
+static struct dmamem *dmamem_alloc(struct emuxki_softc *, size_t);
+static void    dmamem_free(struct dmamem *);
+static void    dmamem_sync(struct dmamem *, int);
+static uint8_t emuxki_readio_1(struct emuxki_softc *, int) __unused;
+static uint16_t        emuxki_readio_2(struct emuxki_softc *, int);
+static uint32_t        emuxki_readio_4(struct emuxki_softc *, int);
+static void    emuxki_writeio_1(struct emuxki_softc *, int, uint8_t);
+static void    emuxki_writeio_2(struct emuxki_softc *, int, uint16_t);
+static void    emuxki_writeio_4(struct emuxki_softc *, int, uint32_t);
+static uint32_t        emuxki_readptr(struct emuxki_softc *, int, int, int);
+static void    emuxki_writeptr(struct emuxki_softc *, int, int, int, uint32_t);
+static uint32_t        emuxki_read(struct emuxki_softc *, int, int);
+static void    emuxki_write(struct emuxki_softc *, int, int, uint32_t);
 static int     emuxki_match(device_t, cfdata_t, void *);
 static void    emuxki_attach(device_t, device_t, void *);
 static int     emuxki_detach(device_t, int);
-
-/* DMA mem mgmt */
-static struct dmamem *dmamem_alloc(bus_dma_tag_t, size_t, bus_size_t,
-               int);
-static void    dmamem_free(struct dmamem *);
-
-/* Emu10k1 init & shutdown */
 static int     emuxki_init(struct emuxki_softc *);
-static void    emuxki_shutdown(struct emuxki_softc *);
-
-/* Emu10k1 mem mgmt */
-static void    *emuxki_pmem_alloc(struct emuxki_softc *, size_t);
-static void    *emuxki_rmem_alloc(struct emuxki_softc *, size_t);
-
-/*
- * Emu10k1 channels funcs : There is no direct access to channels, everything
- * is done through voices I will at least provide channel based fx params
- * modification, later...
- */
-
-/* Emu10k1 voice mgmt */
-static struct emuxki_voice *emuxki_voice_new(struct emuxki_softc *,
-               uint8_t);
-static void    emuxki_voice_delete(struct emuxki_voice *);
-static int     emuxki_voice_set_audioparms(struct emuxki_softc *,
-                                           struct emuxki_voice *, uint8_t,
-                                           uint8_t, uint32_t);
-/* emuxki_voice_set_fxparms will come later, it'll need channel distinction */
-static int     emuxki_voice_set_bufparms(struct emuxki_voice *,
-               void *, uint32_t, uint16_t);
-static void    emuxki_voice_commit_parms(struct emuxki_voice *);
-static int     emuxki_voice_adc_rate(struct emuxki_voice *);
-static uint32_t        emuxki_voice_curaddr(struct emuxki_voice *);
-static void    emuxki_voice_start(struct emuxki_voice *,
-               void (*) (void *), void *);
-static void    emuxki_voice_halt(struct emuxki_voice *);
-
-/*
- * Emu10k1 stream mgmt : not done yet
- */
-#if 0
-static struct emuxki_stream *emuxki_stream_new(struct emu10k1 *);
-static void    emuxki_stream_delete(struct emuxki_stream *);
-static int     emuxki_stream_set_audio_params(struct emuxki_stream *,
-               uint8_t, uint8_t, uint8_t, uint16_t);
-static void    emuxki_stream_start(struct emuxki_stream *);
-static void    emuxki_stream_halt(struct emuxki_stream *);
-#endif
-
-/* audio interface callbacks */
+static void    emuxki_dsp_addop(struct emuxki_softc *, uint16_t *, uint8_t,
+                   uint16_t, uint16_t, uint16_t, uint16_t);
+static void    emuxki_initfx(struct emuxki_softc *);
+static void    emuxki_play_start(struct emuxki_softc *, int, uint32_t,
+                   uint32_t);
+static void    emuxki_play_stop(struct emuxki_softc *, int);
 
 static int     emuxki_open(void *, int);
 static void    emuxki_close(void *);
-
 static int     emuxki_query_format(void *, audio_format_query_t *);
 static int     emuxki_set_format(void *, int,
                    const audio_params_t *, const audio_params_t *,
                    audio_filter_reg_t *, audio_filter_reg_t *);
-
-static int     emuxki_round_blocksize(void *, int, int, const audio_params_t *);
-static size_t  emuxki_round_buffersize(void *, int, size_t);
-
-static int     emuxki_trigger_output(void *, void *, void *, int,
-               void (*)(void *), void *, const audio_params_t *);
-static int     emuxki_trigger_input(void *, void *, void *, int,
-               void (*) (void *), void *, const audio_params_t *);
 static int     emuxki_halt_output(void *);
 static int     emuxki_halt_input(void *);
-
+static int     emuxki_intr(void *);
 static int     emuxki_getdev(void *, struct audio_device *);
 static int     emuxki_set_port(void *, mixer_ctrl_t *);
 static int     emuxki_get_port(void *, mixer_ctrl_t *);
 static int     emuxki_query_devinfo(void *, mixer_devinfo_t *);
-
-static void    *emuxki_allocm(void *, int, size_t);
+static void    *emuxki_allocm(void *, int, size_t);
 static void    emuxki_freem(void *, void *, size_t);
-
+static int     emuxki_round_blocksize(void *, int, int,
Home |
Main Index |
Thread Index |
Old Index