Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/arm/sunxi Add support for Allwinner A64 audio codec.



details:   https://anonhg.NetBSD.org/src/rev/7d5fc43b73f0
branches:  trunk
changeset: 361746:7d5fc43b73f0
user:      jmcneill <jmcneill%NetBSD.org@localhost>
date:      Thu May 10 00:00:21 2018 +0000

description:
Add support for Allwinner A64 audio codec.

diffstat:

 sys/arch/arm/sunxi/files.sunxi         |   17 +-
 sys/arch/arm/sunxi/sun50i_a64_acodec.c |  483 +++++++++++++++++++
 sys/arch/arm/sunxi/sun8i_codec.c       |  312 ++++++++++++
 sys/arch/arm/sunxi/sunxi_i2s.c         |  827 +++++++++++++++++++++++++++++++++
 4 files changed, 1638 insertions(+), 1 deletions(-)

diffs (truncated from 1670 to 300 lines):

diff -r 331798f87e7a -r 7d5fc43b73f0 sys/arch/arm/sunxi/files.sunxi
--- a/sys/arch/arm/sunxi/files.sunxi    Wed May 09 23:59:05 2018 +0000
+++ b/sys/arch/arm/sunxi/files.sunxi    Thu May 10 00:00:21 2018 +0000
@@ -1,4 +1,4 @@
-#      $NetBSD: files.sunxi,v 1.54 2018/05/06 10:34:34 jmcneill Exp $
+#      $NetBSD: files.sunxi,v 1.55 2018/05/10 00:00:21 jmcneill Exp $
 #
 # Configuration info for Allwinner sunxi family SoCs
 #
@@ -208,11 +208,26 @@
 file   arch/arm/sunxi/sun4i_a10_codec.c        sunxi_codec
 file   arch/arm/sunxi/sun6i_a31_codec.c        sunxi_codec
 
+# Audio codec (sun8i)
+device sun8icodec
+attach sun8icodec at fdt with sun8i_codec
+file   arch/arm/sunxi/sun8i_codec.c            sun8i_codec
+
 # H3 Audio codec (analog part)
 device h3codec
 attach h3codec at fdt with h3_codec
 file   arch/arm/sunxi/sun8i_h3_codec.c         h3_codec needs-flag
 
+# A64 Audio codec (analog part)
+device a64acodec
+attach a64acodec at fdt with a64_acodec
+file   arch/arm/sunxi/sun50i_a64_acodec.c      a64_acodec
+
+# I2S/PCM controller
+device sunxii2s: auconv, mulaw, aurateconv
+attach sunxii2s at fdt with sunxi_i2s
+file   arch/arm/sunxi/sunxi_i2s.c              sunxi_i2s
+
 # Display controller
 attach genfb at fdt with simplefb
 file   dev/fdt/simplefb.c                      simplefb
diff -r 331798f87e7a -r 7d5fc43b73f0 sys/arch/arm/sunxi/sun50i_a64_acodec.c
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sys/arch/arm/sunxi/sun50i_a64_acodec.c    Thu May 10 00:00:21 2018 +0000
@@ -0,0 +1,483 @@
+/* $NetBSD: sun50i_a64_acodec.c,v 1.1 2018/05/10 00:00:21 jmcneill Exp $ */
+
+/*-
+ * Copyright (c) 2018 Jared McNeill <jmcneill%invisible.ca@localhost>
+ * 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.
+ *
+ * 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 <sys/cdefs.h>
+__KERNEL_RCSID(0, "$NetBSD: sun50i_a64_acodec.c,v 1.1 2018/05/10 00:00:21 jmcneill Exp $");
+
+#include <sys/param.h>
+#include <sys/bus.h>
+#include <sys/cpu.h>
+#include <sys/device.h>
+#include <sys/kmem.h>
+#include <sys/bitops.h>
+
+#include <dev/audio_dai.h>
+
+#include <dev/fdt/fdtvar.h>
+
+#define        A64_PR_CFG              0x00
+#define         A64_AC_PR_RST          __BIT(28)
+#define         A64_AC_PR_RW           __BIT(24)
+#define         A64_AC_PR_ADDR         __BITS(20,16)
+#define         A64_ACDA_PR_WDAT       __BITS(15,8)
+#define         A64_ACDA_PR_RDAT       __BITS(7,0)
+
+#define        A64_HP_CTRL             0x00
+#define         A64_HPPA_EN            __BIT(6)
+#define         A64_HPVOL              __BITS(5,0)
+#define        A64_OL_MIX_CTRL         0x01
+#define         A64_LMIXMUTE_LDAC      __BIT(1)
+#define        A64_OR_MIX_CTRL         0x02
+#define         A64_RMIXMUTE_RDAC      __BIT(1)
+#define        A64_LINEOUT_CTRL0       0x05
+#define         A64_LINEOUT_LEFT_EN    __BIT(7)
+#define         A64_LINEOUT_RIGHT_EN   __BIT(6)
+#define        A64_LINEOUT_CTRL1       0x06
+#define         A64_LINEOUT_VOL        __BITS(4,0)
+#define        A64_MIC1_CTRL           0x07
+#define         A64_MIC1G              __BITS(6,4)
+#define         A64_MIC1AMPEN          __BIT(3)
+#define         A64_MIC1BOOST          __BITS(2,0)
+#define        A64_MIC2_CTRL           0x08
+#define         A64_MIC2_SEL           __BIT(7)
+#define         A64_MIC2G              __BITS(6,4)
+#define         A64_MIC2AMPEN          __BIT(3)
+#define         A64_MIC2BOOST          __BITS(2,0)
+#define        A64_LINEIN_CTRL         0x09
+#define         A64_LINEING            __BITS(6,4)
+#define        A64_MIX_DAC_CTRL        0x0a
+#define         A64_DACAREN            __BIT(7)
+#define         A64_DACALEN            __BIT(6)
+#define         A64_RMIXEN             __BIT(5)
+#define         A64_LMIXEN             __BIT(4)
+#define         A64_RHPPAMUTE          __BIT(3)
+#define         A64_LHPPAMUTE          __BIT(2)
+#define         A64_RHPIS              __BIT(1)
+#define         A64_LHPIS              __BIT(0)
+#define        A64_L_ADCMIX_SRC        0x0b
+#define        A64_R_ADCMIX_SRC        0x0c
+#define         A64_ADCMIX_SRC_MIC1    __BIT(6)
+#define         A64_ADCMIX_SRC_MIC2    __BIT(5)
+#define         A64_ADCMIX_SRC_LINEIN  __BIT(2)
+#define         A64_ADCMIX_SRC_OMIXER  __BIT(0)
+#define        A64_ADC_CTRL            0x0d
+#define         A64_ADCREN             __BIT(7)
+#define         A64_ADCLEN             __BIT(6)
+#define         A64_ADCG               __BITS(2,0)
+
+struct a64_acodec_softc {
+       device_t                sc_dev;
+       bus_space_tag_t         sc_bst;
+       bus_space_handle_t      sc_bsh;
+       int                     sc_phandle;
+
+       struct audio_dai_device sc_dai;
+};
+
+enum a64_acodec_mixer_ctrl {
+       A64_CODEC_OUTPUT_CLASS,
+       A64_CODEC_INPUT_CLASS,
+       A64_CODEC_RECORD_CLASS,
+
+       A64_CODEC_OUTPUT_MASTER_VOLUME,
+       A64_CODEC_OUTPUT_HP_VOLUME,
+       A64_CODEC_INPUT_DAC_VOLUME,
+       A64_CODEC_INPUT_LINEIN_VOLUME,
+       A64_CODEC_INPUT_MIC1_VOLUME,
+       A64_CODEC_INPUT_MIC2_VOLUME,
+       A64_CODEC_RECORD_AGC_VOLUME,
+       A64_CODEC_RECORD_SOURCE,
+
+       A64_CODEC_MIXER_CTRL_LAST
+};
+
+static const struct a64_acodec_mixer {
+       const char *                    name;
+       enum a64_acodec_mixer_ctrl      mixer_class;
+       u_int                           reg;
+       u_int                           mask;
+} a64_acodec_mixers[A64_CODEC_MIXER_CTRL_LAST] = {
+       [A64_CODEC_OUTPUT_MASTER_VOLUME]        = { AudioNmaster,
+           A64_CODEC_OUTPUT_CLASS, A64_LINEOUT_CTRL1, A64_LINEOUT_VOL },
+       [A64_CODEC_OUTPUT_HP_VOLUME]    = { AudioNheadphone,
+           A64_CODEC_OUTPUT_CLASS, A64_HP_CTRL, A64_HPVOL },
+       [A64_CODEC_INPUT_DAC_VOLUME]    = { AudioNdac,
+           A64_CODEC_INPUT_CLASS, A64_LINEOUT_CTRL1, A64_LINEOUT_VOL },
+       [A64_CODEC_INPUT_LINEIN_VOLUME] = { AudioNline,
+           A64_CODEC_INPUT_CLASS, A64_LINEIN_CTRL, A64_LINEING },
+       [A64_CODEC_INPUT_MIC1_VOLUME]   = { "mic1",
+           A64_CODEC_INPUT_CLASS, A64_MIC1_CTRL, A64_MIC1G },
+       [A64_CODEC_INPUT_MIC2_VOLUME]   = { "mic2",
+           A64_CODEC_INPUT_CLASS, A64_MIC2_CTRL, A64_MIC2G },
+       [A64_CODEC_RECORD_AGC_VOLUME]   = { AudioNagc,
+           A64_CODEC_RECORD_CLASS, A64_ADC_CTRL, A64_ADCG },
+};
+
+#define        RD4(sc, reg)                    \
+       bus_space_read_4((sc)->sc_bst, (sc)->sc_bsh, (reg))
+#define        WR4(sc, reg, val)               \
+       bus_space_write_4((sc)->sc_bst, (sc)->sc_bsh, (reg), (val))
+
+static u_int
+a64_acodec_pr_read(struct a64_acodec_softc *sc, u_int addr)
+{
+       uint32_t val;
+
+       /* Read current value */
+       val = RD4(sc, A64_PR_CFG);
+
+       /* De-assert reset */
+       val |= A64_AC_PR_RST;
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Read mode */
+       val &= ~A64_AC_PR_RW;
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Set address */
+       val &= ~A64_AC_PR_ADDR;
+       val |= __SHIFTIN(addr, A64_AC_PR_ADDR);
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Read data */
+       return __SHIFTOUT(RD4(sc, A64_PR_CFG), A64_ACDA_PR_RDAT);
+}
+
+static void
+a64_acodec_pr_write(struct a64_acodec_softc *sc, u_int addr, u_int data)
+{
+       uint32_t val;
+
+       /* Read current value */
+       val = RD4(sc, A64_PR_CFG);
+
+       /* De-assert reset */
+       val |= A64_AC_PR_RST;
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Set address */
+       val &= ~A64_AC_PR_ADDR;
+       val |= __SHIFTIN(addr, A64_AC_PR_ADDR);
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Write data */
+       val &= ~A64_ACDA_PR_WDAT;
+       val |= __SHIFTIN(data, A64_ACDA_PR_WDAT);
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Write mode */
+       val |= A64_AC_PR_RW;
+       WR4(sc, A64_PR_CFG, val);
+
+       /* Clear write mode */
+       val &= ~A64_AC_PR_RW;
+       WR4(sc, A64_PR_CFG, val);
+}
+
+static void
+a64_acodec_pr_set_clear(struct a64_acodec_softc *sc, u_int addr, u_int set, u_int clr)
+{
+       u_int old, new;
+
+       old = a64_acodec_pr_read(sc, addr);
+       new = set | (old & ~clr);
+       a64_acodec_pr_write(sc, addr, new);
+}
+
+static int
+a64_acodec_trigger_output(void *priv, void *start, void *end, int blksize,
+    void (*intr)(void *), void *intrarg, const audio_params_t *params)
+{
+       struct a64_acodec_softc * const sc = priv;
+
+       /* Enable DAC analog l/r channels and output mixer */
+       a64_acodec_pr_set_clear(sc, A64_MIX_DAC_CTRL,
+           A64_DACAREN | A64_DACALEN | A64_RMIXEN | A64_LMIXEN, 0);
+       /* Unmute DAC l/r channels to output mixer */
+       a64_acodec_pr_set_clear(sc, A64_OL_MIX_CTRL,
+           A64_LMIXMUTE_LDAC, 0);
+       a64_acodec_pr_set_clear(sc, A64_OR_MIX_CTRL,
+           A64_RMIXMUTE_RDAC, 0);
+
+       return 0;
+}
+
+static int
+a64_acodec_trigger_input(void *priv, void *start, void *end, int blksize,
+    void (*intr)(void *), void *intrarg, const audio_params_t *params)
+{
+       struct a64_acodec_softc * const sc = priv;
+
+       /* Enable ADC analog l/r channels */
+       a64_acodec_pr_set_clear(sc, A64_ADC_CTRL,
+           A64_ADCREN | A64_ADCLEN, 0);
+
+       return 0;
+}
+
+static int
+a64_acodec_halt_output(void *priv)
+{
+       struct a64_acodec_softc * const sc = priv;
+
+       /* Mute DAC l/r channels to output mixer */
+       a64_acodec_pr_set_clear(sc, A64_OL_MIX_CTRL,
+           0, A64_LMIXMUTE_LDAC);
+       a64_acodec_pr_set_clear(sc, A64_OR_MIX_CTRL,
+           0, A64_RMIXMUTE_RDAC);
+       /* Disable DAC analog l/r channels and output mixer */
+       a64_acodec_pr_set_clear(sc, A64_MIX_DAC_CTRL,
+           0, A64_DACAREN | A64_DACALEN | A64_RMIXEN | A64_LMIXEN);
+
+       return 0;
+}
+
+static int



Home | Main Index | Thread Index | Old Index