/*      $NetBSD: vsaudio.c,v 1.7 2020/09/12 05:19:16 isaki Exp $        */
/*      $OpenBSD: vsaudio.c,v 1.4 2013/05/15 21:21:11 ratchov Exp $     */

/*
* Copyright (c) 2011 Miodrag Vallat.
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
/*
* Copyright (c) 1995 Rolf Grossmann
* 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.
* 3. All advertising materials mentioning features or use of this software
*    must display the following acknowledgement:
*      This product includes software developed by Rolf Grossmann.
* 4. The name of the author may not be used to endorse or promote products
*    derived from this software without specific prior written permission
*
* 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.
*/

/*
* Audio backend for the VAXstation 4000 AMD79C30 audio chip.
* Currently working in pseudo-DMA mode; DMA operation may be possible and
* needs to be investigated.
*/
/*
* Although he did not claim copyright for his work, this code owes a lot
* to Blaz Antonic <[email protected]> who figured out a working
* interrupt triggering routine in vsaudio_match().
*/
/*
* Ported to NetBSD, from OpenBSD, by Björn Johannesson ([email protected])
* in December 2014
*/

#include "audio.h"
#if NAUDIO > 0

#include <sys/errno.h>
#include <sys/evcnt.h>
#include <sys/intr.h>
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>

#include <machine/cpu.h>
#include <machine/sid.h>
#include <machine/scb.h>
#include <machine/vsbus.h>

#include <sys/audioio.h>
#include <dev/audio/audio_if.h>

#include <dev/ic/am7930reg.h>
#include <dev/ic/am7930var.h>

/* physical addresses of the AM79C30 chip */
#define VSAUDIO_CSR                     0x200d0000
#define VSAUDIO_CSR_KA49                0x26800000

struct vsaudio_softc {
       struct am7930_softc sc_am7930;  /* glue to MI code */

       bus_space_tag_t sc_bt;          /* bus cookie */
       bus_space_handle_t sc_bh;       /* device registers */
};

static int vsaudio_match(struct device *parent, struct cfdata *match, void *);
static void vsaudio_attach(device_t parent, device_t self, void *);

static void vsaudio_hwintr(void *);

CFATTACH_DECL_NEW(vsaudio, sizeof(struct vsaudio_softc), vsaudio_match,
               vsaudio_attach, NULL, NULL);

/*
* Hardware access routines for the MI code
*/
uint8_t vsaudio_codec_dread(struct am7930_softc *, int);
void    vsaudio_codec_dwrite(struct am7930_softc *, int, uint8_t);

struct am7930_glue vsaudio_glue = {
       vsaudio_codec_dread,
       vsaudio_codec_dwrite,
};

/*
* Interface to the MI audio layer.
*/
int     vsaudio_getdev(void *, struct audio_device *);

struct audio_hw_if vsaudio_hw_if = {
       .query_format           = am7930_query_format,
       .set_format             = am7930_set_format,
       .commit_settings        = am7930_commit_settings,
       .trigger_output         = am7930_trigger_output,
       .trigger_input          = am7930_trigger_input,
       .halt_output            = am7930_halt_output,
       .halt_input             = am7930_halt_input,
       .getdev                 = vsaudio_getdev,
       .set_port               = am7930_set_port,
       .get_port               = am7930_get_port,
       .query_devinfo          = am7930_query_devinfo,
       .get_props              = am7930_get_props,
       .get_locks              = am7930_get_locks,
};


struct audio_device vsaudio_device = {
       "am7930",
       "x",
       "vsaudio"
};


static int
vsaudio_match(struct device *parent, struct cfdata *match, void *aux)
{
       struct vsbus_attach_args *va = aux;
       volatile uint32_t *regs;
       int i;

       switch (vax_boardtype) {
#if defined(VAX_BTYP_46) || defined(VAX_BTYP_48)
       case VAX_BTYP_46:
       case VAX_BTYP_48:
               if (va->va_paddr != VSAUDIO_CSR)
                       return 0;
               break;
#endif
#if defined(VAX_BTYP_49)
       case VAX_BTYP_49:
               if (va->va_paddr != VSAUDIO_CSR_KA49)
                       return 0;
               break;
#endif
       default:
               return 0;
       }

       regs = (volatile uint32_t *)va->va_addr;
       regs[AM7930_DREG_CR] = AM7930_IREG_INIT;
       regs[AM7930_DREG_DR] = AM7930_INIT_PMS_ACTIVE | AM7930_INIT_INT_ENABLE;

       regs[AM7930_DREG_CR] = AM7930_IREG_MUX_MCR1;
       regs[AM7930_DREG_DR] = 0;

       regs[AM7930_DREG_CR] = AM7930_IREG_MUX_MCR2;
       regs[AM7930_DREG_DR] = 0;

       regs[AM7930_DREG_CR] = AM7930_IREG_MUX_MCR3;
       regs[AM7930_DREG_DR] = (AM7930_MCRCHAN_BB << 4) | AM7930_MCRCHAN_BA;

       regs[AM7930_DREG_CR] = AM7930_IREG_MUX_MCR4;
       regs[AM7930_DREG_DR] = AM7930_MCR4_INT_ENABLE;

       for (i = 10; i < 20; i++)
               regs[AM7930_DREG_BBTB] = i;
       delay(1000000); /* XXX too large */

       return 1;
}


static void
vsaudio_attach(device_t parent, device_t self, void *aux)
{
       struct vsbus_attach_args *va = aux;
       struct vsaudio_softc *sc = device_private(self);
       struct am7930_softc *amsc = &sc->sc_am7930;

       if (bus_space_map(va->va_memt, va->va_paddr, AM7930_DREG_SIZE << 2, 0,
           &sc->sc_bh) != 0) {
               printf(": can't map registers\n");
               return;
       }
       sc->sc_bt = va->va_memt;
       amsc->sc_dev = self;
       amsc->sc_glue = &vsaudio_glue;
       am7930_init(amsc, AUDIOAMD_POLL_MODE);
       scb_vecalloc(va->va_cvec, vsaudio_hwintr, amsc, SCB_ISTACK,
           &amsc->sc_intrcnt);
       evcnt_attach_dynamic(&amsc->sc_intrcnt, EVCNT_TYPE_INTR, NULL,
           device_xname(self), "intr");

       aprint_normal("\n");
       audio_attach_mi(&vsaudio_hw_if, sc, self);
}

static void
vsaudio_hwintr(void *arg)
{

       am7930_hwintr(arg);
}

/* direct read */
uint8_t
vsaudio_codec_dread(struct am7930_softc *amsc, int reg)
{
       struct vsaudio_softc *sc = (struct vsaudio_softc *)amsc;

       return bus_space_read_1(sc->sc_bt, sc->sc_bh, reg << 2);
}

/* direct write */
void
vsaudio_codec_dwrite(struct am7930_softc *amsc, int reg, uint8_t val)
{
       struct vsaudio_softc *sc = (struct vsaudio_softc *)amsc;

       bus_space_write_1(sc->sc_bt, sc->sc_bh, reg << 2, val);
}

int
vsaudio_getdev(void *addr, struct audio_device *retp)
{

       *retp = vsaudio_device;
       return 0;
}

#endif /* NAUDIO > 0 */