freebsd-skq/sys/i386/isa/sound/sound_pnp.c
Peter Wemm a0f70ac053 Part 2 of pcvt/voxware revival. I hope I have not clobbered any other
deltas, but it is possible since I had a few merge conflicts over the last
few days while this has been sitting ready to go.

(Part 1 was committed to the config files, but cvs aborted grrr..)

Approved by:    core
1999-01-01 08:18:13 +00:00

187 lines
4.9 KiB
C

/*
* sound/sound_pnp.c
*
* PnP soundcard support (Linux spesific)
*
* Copyright by Hannu Savolainen 1995
*
* 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 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 AUTHOR 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.
*
*/
#include <i386/isa/sound/sound_config.h>
/*
* XXX check what to use in place of CONFIG_PNP
*/
#if (NSND > 0) && defined(CONFIG_PNP)
#include <linux/pnp.h>
static struct pnp_sounddev *pnp_devs[20] = { NULL };
static int max_pnpdevs = 20;
static int nr_pnpdevs = 0;
static int pnp_sig = 0;
void
install_pnp_sounddrv(struct pnp_sounddev * drv)
{
if (nr_pnpdevs < max_pnpdevs) {
pnp_devs[nr_pnpdevs++] = drv;
} else
printf("Sound: More than 20 PnP drivers defined\n");
}
void
cs4232_pnp(void *parm)
{
struct pnp_dev *dev = (struct pnp_dev *) parm;
char *name;
int portmask = 0x00, irqmask = 0x01, dmamask = 0x03;
int opl3_driver, wss_driver;
printf("CS4232 driver waking up\n");
if (dev->card && dev->card->name)
name = dev->card->name;
else
name = "PnP WSS";
if ((wss_driver = sndtable_identify_card("AD1848")))
portmask |= 0x01; /* MSS */
else
printf("Sound: MSS/WSS device detected but no driver enabled\n");
if ((opl3_driver = sndtable_identify_card("OPL3")))
portmask |= 0x02; /* OPL3 */
else
printf("Sound: OPL3/4 device detected but no driver enabled\n");
printf("WSS driver %d, OPL3 driver %d\n", wss_driver, opl3_driver);
if (!portmask) /* No drivers available */
return;
if (!pnp_allocate_device(pnp_sig, dev, portmask, irqmask, dmamask, 0x00))
printf("Device activation failed\n");
else {
struct address_info hw_config;
int wss_base, opl3_base;
int irq;
int dma1, dma2;
printf("Device activation OK\n");
wss_base = pnp_get_port(dev, 0);
opl3_base = pnp_get_port(dev, 1);
irq = pnp_get_irq(dev, 0);
dma1 = pnp_get_dma(dev, 0);
dma2 = pnp_get_dma(dev, 1);
printf("I/O0 %03x\n", wss_base);
printf("I/O1 %03x\n", opl3_base);
printf("IRQ %d\n", irq);
printf("DMA0 %d\n", dma1);
printf("DMA1 %d\n", dma2);
if (opl3_base && opl3_driver) {
hw_config.io_base = opl3_base;
hw_config.irq = 0;
hw_config.dma = -1;
hw_config.dma2 = -1;
hw_config.always_detect = 0;
hw_config.name = "";
hw_config.osp = NULL;
hw_config.card_subtype = 0;
if (sndtable_probe(opl3_driver, &hw_config))
sndtable_init_card(opl3_driver, &hw_config);
}
if (wss_base && wss_driver) {
hw_config.io_base = wss_base;
hw_config.irq = irq;
hw_config.dma = dma1;
hw_config.dma2 = (dma2 == NO_DMA) ? dma1 : dma2;
hw_config.always_detect = 0;
hw_config.name = name;
hw_config.osp = NULL;
hw_config.card_subtype = 0;
if (sndtable_probe(wss_driver, &hw_config))
sndtable_init_card(wss_driver, &hw_config);
}
}
}
static int
pnp_activate(int id, struct pnp_dev * dev)
{
int i;
for (i = 0; i < nr_pnpdevs; i++)
if (pnp_devs[i]->id == id) {
printf("PnP dev: %08x, %s\n", id, pnp_devid2asc(id));
pnp_devs[i]->setup((void *) dev);
return 1;
}
return 0;
}
void
sound_pnp_init(void)
{
static struct pnp_sounddev cs4232_dev =
{PNP_DEVID('C', 'S', 'C', 0x0000), cs4232_pnp, "CS4232"};
struct pnp_dev *dev;
install_pnp_sounddrv(&cs4232_dev);
dev = NULL;
if ((pnp_sig = pnp_connect("sound")) == -1) {
printf("Sound: Can't connect to kernel PnP services.\n");
return;
}
while ((dev = pnp_get_next_device(pnp_sig, dev)) != NULL) {
if (!pnp_activate(dev->key, dev)) {
/* Scan all compatible devices */
int i;
for (i = 0; i < dev->ncompat; i++)
if (pnp_activate(dev->compat_keys[i], dev))
break;
}
}
}
void
sound_pnp_disconnect(void)
{
pnp_disconnect(pnp_sig);
}
#endif