Update of /cvsroot/alsa/alsa-driver/pci/emu10k1
In directory sc8-pr-cvs1.sourceforge.net:/tmp/cvs-serv31910/pci/emu10k1
Modified Files:
Makefile
Added Files:
emu10k1x.c
Log Message:
- added emu10k1x driver by Francisco Moraes <[EMAIL PROTECTED]>.
--- NEW FILE: emu10k1x.c ---
/*
* Copyright (c) by Francisco Moraes <[EMAIL PROTECTED]>
* Driver EMU10K1X chips
*
* BUGS:
* --
*
* TODO:
* --
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include <sound/driver.h>
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/pci.h>
#include <linux/slab.h>
#include <linux/moduleparam.h>
#include <sound/core.h>
#include <sound/initval.h>
#include <sound/pcm.h>
#include <sound/ac97_codec.h>
MODULE_AUTHOR("Francisco Moraes <[EMAIL PROTECTED]>");
MODULE_DESCRIPTION("EMU10K1X");
MODULE_LICENSE("GPL");
MODULE_CLASSES("{sound}");
MODULE_DEVICES("{{Dell Creative Labs,SB Live!}");
// module parameters (see "Module Parameters")
static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;
static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;
static int enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE_PNP;
static int boot_devs;
module_param_array(index, int, boot_devs, 0444);
MODULE_PARM_DESC(index, "Index value for the EMU10K1X soundcard.");
MODULE_PARM_SYNTAX(index, SNDRV_INDEX_DESC);
module_param_array(id, charp, boot_devs, 0444);
MODULE_PARM_DESC(id, "ID string for the EMU10K1X soundcard.");
MODULE_PARM_SYNTAX(id, SNDRV_ID_DESC);
module_param_array(enable, bool, boot_devs, 0444);
MODULE_PARM_DESC(enable, "Enable the EMU10K1X soundcard.");
MODULE_PARM_SYNTAX(enable, SNDRV_ENABLE_DESC);
// some definitions were borrowed from emu10k1 driver as they seem to be the same
/************************************************************************************************/
/* PCI function 0 registers, address = <val> + PCIBASE0
*/
/************************************************************************************************/
#define PTR 0x00 /* Indexed register set pointer
register */
/* NOTE: The CHANNELNUM and ADDRESS
words can */
/* be modified independently of each
other. */
#define DATA 0x04 /* Indexed register set data register
*/
#define IPR 0x08 /* Global interrupt pending register
*/
/* Clear pending interrupts by writing
a 1 to */
/* the relevant bits and zero to the
other bits */
#define IPR_CH_0_LOOP 0x00000800 /* Channel 0 loop
*/
#define IPR_CH_0_HALF_LOOP 0x00000100 /* Channel 0 half loop
*/
#define INTE 0x0c /* Interrupt enable register
*/
#define INTE_CH_0_LOOP 0x00000800 /* Channel 0 loop
*/
#define INTE_CH_0_HALF_LOOP 0x00000100 /* Channel 0 half loop
*/
#define HCFG 0x14 /* Hardware config register
*/
#define HCFG_LOCKSOUNDCACHE 0x00000008 /* 1 = Cancel bustmaster accesses to
soundcache */
/* NOTE: This should generally never
be used. */
#define HCFG_AUDIOENABLE 0x00000001 /* 0 = CODECs transmit zero-valued
samples */
/* Should be set to 1 when the EMU10K1
is */
/* completely initialized.
*/
#define AC97DATA 0x1c /* AC97 register set data register (16
bit) */
#define AC97ADDRESS 0x1e /* AC97 register set address register
(8 bit) */
#define SPCS0 0x42 /* SPDIF output Channel Status 0
register */
#define SPCS1 0x43 /* SPDIF output Channel Status 1
register */
#define SPCS2 0x44 /* SPDIF output Channel Status 2
register */
#define SPCS_CLKACCYMASK 0x30000000 /* Clock accuracy
*/
#define SPCS_CLKACCY_1000PPM 0x00000000 /* 1000 parts per million
*/
#define SPCS_CLKACCY_50PPM 0x10000000 /* 50 parts per million
*/
#define SPCS_CLKACCY_VARIABLE 0x20000000 /* Variable accuracy
*/
#define SPCS_SAMPLERATEMASK 0x0f000000 /* Sample rate
*/
#define SPCS_SAMPLERATE_44 0x00000000 /* 44.1kHz sample rate
*/
#define SPCS_SAMPLERATE_48 0x02000000 /* 48kHz sample rate
*/
#define SPCS_SAMPLERATE_32 0x03000000 /* 32kHz sample rate
*/
#define SPCS_CHANNELNUMMASK 0x00f00000 /* Channel number
*/
#define SPCS_CHANNELNUM_UNSPEC 0x00000000 /* Unspecified channel number
*/
#define SPCS_CHANNELNUM_LEFT 0x00100000 /* Left channel
*/
#define SPCS_CHANNELNUM_RIGHT 0x00200000 /* Right channel
*/
#define SPCS_SOURCENUMMASK 0x000f0000 /* Source number
*/
#define SPCS_SOURCENUM_UNSPEC 0x00000000 /* Unspecified source number
*/
#define SPCS_GENERATIONSTATUS 0x00008000 /* Originality flag (see IEC-958 spec)
*/
#define SPCS_CATEGORYCODEMASK 0x00007f00 /* Category code (see IEC-958 spec)
*/
#define SPCS_MODEMASK 0x000000c0 /* Mode (see IEC-958 spec)
*/
#define SPCS_EMPHASISMASK 0x00000038 /* Emphasis
*/
#define SPCS_EMPHASIS_NONE 0x00000000 /* No emphasis
*/
#define SPCS_EMPHASIS_50_15 0x00000008 /* 50/15 usec 2 channel
*/
#define SPCS_COPYRIGHT 0x00000004 /* Copyright asserted flag -- do not
modify */
#define SPCS_NOTAUDIODATA 0x00000002 /* 0 = Digital audio, 1 = not audio
*/
#define SPCS_PROFESSIONAL 0x00000001 /* 0 = Consumer (IEC-958), 1 = pro
(AES3-1992) */
#define chip_t emu10k1x_t
typedef struct snd_emu10k1x_voice emu10k1x_voice_t;
typedef struct snd_emu10k1x emu10k1x_t;
typedef struct snd_emu10k1x_pcm emu10k1x_pcm_t;
struct snd_emu10k1x_voice {
emu10k1x_t *emu;
int number;
int use;
void (*interrupt)(emu10k1x_t *emu, emu10k1x_voice_t *pvoice);
emu10k1x_pcm_t *epcm;
};
struct snd_emu10k1x_pcm {
emu10k1x_t *emu;
snd_pcm_substream_t *substream;
emu10k1x_voice_t *voice;
unsigned short running;
};
// definition of the chip-specific record
struct snd_emu10k1x {
snd_card_t *card;
struct pci_dev *pci;
unsigned long port;
struct resource *res_port;
int irq;
unsigned int revision; /* chip revision */
unsigned int serial; /* serial number */
unsigned short model; /* subsystem id */
spinlock_t emu_lock;
spinlock_t voice_lock;
ac97_t *ac97;
snd_pcm_t *pcm;
emu10k1x_voice_t voices[3];
struct snd_dma_device dma_dev;
struct snd_dma_buffer buffer;
};
#define emu10k1x_t_magic 0xa15a4501
#define emu10k1x_pcm_t_magic 0xa15a4502
/* hardware definition */
static snd_pcm_hardware_t snd_emu10k1x_playback_hw = {
.info = (SNDRV_PCM_INFO_MMAP |
SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_BLOCK_TRANSFER |
SNDRV_PCM_INFO_MMAP_VALID),
.formats = SNDRV_PCM_FMTBIT_S16_LE,
.rates = SNDRV_PCM_RATE_48000,
.rate_min = 48000,
.rate_max = 48000,
.channels_min = 1,
.channels_max = 2,
.buffer_bytes_max = (32*1024),
.period_bytes_min = 64,
.period_bytes_max = (32*1024),
.periods_min = 2,
.periods_max = 2,
.fifo_size = 0,
};
static unsigned int snd_emu10k1x_ptr_read(emu10k1x_t * emu,
unsigned int reg,
unsigned int chn)
{
unsigned long flags;
unsigned int regptr, val;
regptr = (reg << 16) | chn;
spin_lock_irqsave(&emu->emu_lock, flags);
outl(regptr, emu->port + PTR);
val = inl(emu->port + DATA);
spin_unlock_irqrestore(&emu->emu_lock, flags);
return val;
}
static void snd_emu10k1x_ptr_write(emu10k1x_t *emu,
unsigned int reg,
unsigned int chn,
unsigned int data)
{
unsigned int regptr;
unsigned long flags;
regptr = (reg << 16) | chn;
spin_lock_irqsave(&emu->emu_lock, flags);
outl(regptr, emu->port + PTR);
outl(data, emu->port + DATA);
spin_unlock_irqrestore(&emu->emu_lock, flags);
}
static void snd_emu10k1x_intr_enable(emu10k1x_t *emu, unsigned int intrenb)
{
unsigned long flags;
unsigned int enable;
spin_lock_irqsave(&emu->emu_lock, flags);
enable = inl(emu->port + INTE) | intrenb;
outl(enable, emu->port + INTE);
spin_unlock_irqrestore(&emu->emu_lock, flags);
}
static int voice_alloc(emu10k1x_t *emu, emu10k1x_voice_t **rvoice)
{
emu10k1x_voice_t *voice;
int idx;
*rvoice = NULL;
for (idx = 0; idx < 3; idx ++) {
voice = &emu->voices[idx];
if (voice->use)
continue;
voice->use = 1;
*rvoice = voice;
return 0;
}
return -ENOMEM;
}
static int snd_emu10k1x_voice_alloc(emu10k1x_t *emu, emu10k1x_voice_t **rvoice)
{
unsigned long flags;
int result;
snd_assert(rvoice != NULL, return -EINVAL);
spin_lock_irqsave(&emu->voice_lock, flags);
result = voice_alloc(emu, rvoice);
spin_unlock_irqrestore(&emu->voice_lock, flags);
return result;
}
static int snd_emu10k1x_voice_free(emu10k1x_t *emu, emu10k1x_voice_t *pvoice)
{
unsigned long flags;
snd_assert(pvoice != NULL, return -EINVAL);
spin_lock_irqsave(&emu->voice_lock, flags);
pvoice->interrupt = NULL;
pvoice->use = 0;
pvoice->epcm = NULL;
spin_unlock_irqrestore(&emu->voice_lock, flags);
return 0;
}
static void snd_emu10k1x_pcm_free_substream(snd_pcm_runtime_t *runtime)
{
emu10k1x_pcm_t *epcm = snd_magic_cast(emu10k1x_pcm_t, runtime->private_data,
return);
if (epcm)
snd_magic_kfree(epcm);
}
static void snd_emu10k1x_pcm_interrupt(emu10k1x_t *emu, emu10k1x_voice_t *voice)
{
emu10k1x_pcm_t *epcm;
if ((epcm = voice->epcm) == NULL)
return;
if (epcm->substream == NULL)
return;
#if 0
printk("IRQ: position = 0x%x, period = 0x%x, size = 0x%x\n",
epcm->substream->runtime->hw->pointer(emu, epcm->substream),
snd_pcm_lib_period_bytes(epcm->substream),
snd_pcm_lib_buffer_bytes(epcm->substream));
#endif
snd_pcm_period_elapsed(epcm->substream);
}
/* open callback */
static int snd_emu10k1x_playback_open(snd_pcm_substream_t *substream)
{
emu10k1x_t *chip = snd_pcm_substream_chip(substream);
emu10k1x_pcm_t *epcm;
snd_pcm_runtime_t *runtime = substream->runtime;
epcm = snd_magic_kcalloc(emu10k1x_pcm_t, 0, GFP_KERNEL);
if (epcm == NULL)
return -ENOMEM;
epcm->emu = chip;
epcm->substream = substream;
runtime->private_data = epcm;
runtime->private_free = snd_emu10k1x_pcm_free_substream;
runtime->hw = snd_emu10k1x_playback_hw;
return 0;
}
/* close callback */
static int snd_emu10k1x_playback_close(snd_pcm_substream_t *substream)
{
return 0;
}
/* hw_params callback */
static int snd_emu10k1x_pcm_hw_params(snd_pcm_substream_t *substream,
snd_pcm_hw_params_t * hw_params)
{
int err;
snd_pcm_runtime_t *runtime = substream->runtime;
emu10k1x_pcm_t *epcm = snd_magic_cast(emu10k1x_pcm_t, runtime->private_data,
return -ENXIO);
if (! epcm->voice) {
if ((err = snd_emu10k1x_voice_alloc(epcm->emu, &epcm->voice)) < 0)
return err;
epcm->voice->epcm = epcm;
epcm->voice->interrupt = snd_emu10k1x_pcm_interrupt;
}
return snd_pcm_lib_malloc_pages(substream,
params_buffer_bytes(hw_params));
}
/* hw_free callback */
static int snd_emu10k1x_pcm_hw_free(snd_pcm_substream_t *substream)
{
snd_pcm_runtime_t *runtime = substream->runtime;
emu10k1x_pcm_t *epcm;
if (runtime->private_data == NULL)
return 0;
epcm = snd_magic_cast(emu10k1x_pcm_t, runtime->private_data, return -ENXIO);
if (epcm->voice) {
snd_emu10k1x_voice_free(epcm->emu, epcm->voice);
epcm->voice = NULL;
}
return snd_pcm_lib_free_pages(substream);
}
/* prepare callback */
static int snd_emu10k1x_pcm_prepare(snd_pcm_substream_t *substream)
{
emu10k1x_t *emu = snd_pcm_substream_chip(substream);
snd_pcm_runtime_t *runtime = substream->runtime;
emu10k1x_pcm_t *epcm = snd_magic_cast(emu10k1x_pcm_t, runtime->private_data,
return -ENXIO);
int voice = epcm->voice->number;
snd_emu10k1x_ptr_write(emu, 0x00, voice, 0);
snd_emu10k1x_ptr_write(emu, 0x01, voice, 0);
snd_emu10k1x_ptr_write(emu, 0x02, voice, 0);
snd_emu10k1x_ptr_write(emu, 0x04, voice, runtime->dma_addr);
snd_emu10k1x_ptr_write(emu, 0x05, voice, frames_to_bytes(runtime,
runtime->buffer_size)<<16); // buffer size in bytes
snd_emu10k1x_ptr_write(emu, 0x06, voice, 0);
snd_emu10k1x_ptr_write(emu, 0x07, voice, 0);
snd_emu10k1x_ptr_write(emu, 0x08, voice, 0);
return 0;
}
/* trigger callback */
static int snd_emu10k1x_pcm_trigger(snd_pcm_substream_t *substream,
int cmd)
{
emu10k1x_t *emu = snd_pcm_substream_chip(substream);
snd_pcm_runtime_t *runtime = substream->runtime;
emu10k1x_pcm_t *epcm = snd_magic_cast(emu10k1x_pcm_t, runtime->private_data,
return -ENXIO);
int channel = epcm->voice->number;
int result = 0;
// printk("trigger - emu10k1x = 0x%x, cmd = %i, pointer = %d\n", (int)emu, cmd,
(int)substream->ops->pointer(substream));
switch (cmd) {
case SNDRV_PCM_TRIGGER_START:
snd_emu10k1x_ptr_write(emu, 0x40, 0, snd_emu10k1x_ptr_read(emu, 0x40,
0)|(1<<channel));
epcm->running = 1;
break;
case SNDRV_PCM_TRIGGER_STOP:
snd_emu10k1x_ptr_write(emu, 0x40, 0, snd_emu10k1x_ptr_read(emu, 0x40,
0) & ~(1<<channel));
epcm->running = 0;
break;
default:
result = -EINVAL;
break;
}
return result;
}
/* pointer callback */
static snd_pcm_uframes_t
snd_emu10k1x_pcm_pointer(snd_pcm_substream_t *substream)
{
emu10k1x_t *emu = snd_pcm_substream_chip(substream);
snd_pcm_runtime_t *runtime = substream->runtime;
emu10k1x_pcm_t *epcm = snd_magic_cast(emu10k1x_pcm_t, runtime->private_data,
return -ENXIO);
unsigned int ptr = 0;
int channel = epcm->voice->number;
if (!epcm->running)
return 0;
// printk("pointer: %08X %08X\n",
// snd_emu10k1x_ptr_read(emu, 0x06, channel),
// snd_emu10k1x_ptr_read(emu, 0x12, channel));
ptr = bytes_to_frames(runtime, snd_emu10k1x_ptr_read(emu, 0x06, channel));
if (ptr >= runtime->buffer_size)
ptr -= runtime->buffer_size;
// printk("ptr = 0x%x, buffer_size = 0x%x, period_size = 0x%x, bits=%d,
rate=%d\n", ptr, (int)runtime->buffer_size, (int)runtime->period_size,
(int)runtime->frame_bits, (int)runtime->rate);
return ptr;
}
/* operators */
static snd_pcm_ops_t snd_emu10k1x_playback_ops = {
.open = snd_emu10k1x_playback_open,
.close = snd_emu10k1x_playback_close,
.ioctl = snd_pcm_lib_ioctl,
.hw_params = snd_emu10k1x_pcm_hw_params,
.hw_free = snd_emu10k1x_pcm_hw_free,
.prepare = snd_emu10k1x_pcm_prepare,
.trigger = snd_emu10k1x_pcm_trigger,
.pointer = snd_emu10k1x_pcm_pointer,
};
static unsigned short snd_emu10k1x_ac97_read(ac97_t *ac97,
unsigned short reg)
{
emu10k1x_t *emu = snd_magic_cast(emu10k1x_t, ac97->private_data, return
-ENXIO);
unsigned long flags;
unsigned short val;
spin_lock_irqsave(&emu->emu_lock, flags);
outb(reg, emu->port + AC97ADDRESS);
val = inw(emu->port + AC97DATA);
spin_unlock_irqrestore(&emu->emu_lock, flags);
return val;
}
static void snd_emu10k1x_ac97_write(ac97_t *ac97,
unsigned short reg, unsigned short val)
{
emu10k1x_t *emu = snd_magic_cast(emu10k1x_t, ac97->private_data, return);
unsigned long flags;
spin_lock_irqsave(&emu->emu_lock, flags);
outb(reg, emu->port + AC97ADDRESS);
outw(val, emu->port + AC97DATA);
spin_unlock_irqrestore(&emu->emu_lock, flags);
}
static int snd_emu10k1x_ac97(emu10k1x_t *chip)
{
ac97_bus_t bus, *pbus;
ac97_t ac97;
int err;
memset(&bus, 0, sizeof(bus));
bus.write = snd_emu10k1x_ac97_write;
bus.read = snd_emu10k1x_ac97_read;
if ((err = snd_ac97_bus(chip->card, &bus, &pbus)) < 0)
return err;
memset(&ac97, 0, sizeof(ac97));
ac97.private_data = chip;
return snd_ac97_mixer(pbus, &ac97, &chip->ac97);
}
static int snd_emu10k1x_free(emu10k1x_t *chip)
{
snd_emu10k1x_ptr_write(chip, 0x40, 0, 0);
// disable interrupts
outl(0, chip->port + INTE);
// disable audio
outl(HCFG_LOCKSOUNDCACHE, chip->port + HCFG);
// release the i/o port
if (chip->res_port) {
release_resource(chip->res_port);
kfree_nocheck(chip->res_port);
}
// release the irq
if (chip->irq >= 0)
free_irq(chip->irq, (void *)chip);
// release the data
snd_magic_kfree(chip);
return 0;
}
static int snd_emu10k1x_dev_free(snd_device_t *device)
{
emu10k1x_t *chip = snd_magic_cast(emu10k1x_t,
device->device_data, return -ENXIO);
return snd_emu10k1x_free(chip);
}
static irqreturn_t snd_emu10k1x_interrupt(int irq, void *dev_id,
struct pt_regs *regs)
{
unsigned int status;
emu10k1x_t *chip = snd_magic_cast(emu10k1x_t, dev_id, return IRQ_NONE);
int i;
int mask;
spin_lock(&chip->emu_lock);
status = inl(chip->port + IPR);
// call updater, unlock before it
spin_unlock(&chip->emu_lock);
if (! status)
return IRQ_NONE;
mask = IPR_CH_0_LOOP|IPR_CH_0_HALF_LOOP;
for(i = 0; i < 3; i++) {
emu10k1x_voice_t *pvoice = chip->voices;
if(status & mask) {
if(pvoice->use && pvoice->interrupt)
pvoice->interrupt(chip, pvoice);
}
pvoice++;
mask <<= 1;
}
spin_lock(&chip->emu_lock);
// acknowledge the interrupt if necessary
outl(status, chip->port+IPR);
spin_unlock(&chip->emu_lock);
// printk(KERN_INFO "interrupt %08x\n", status);
return IRQ_HANDLED;
}
static void snd_emu10k1x_pcm_free(snd_pcm_t *pcm)
{
emu10k1x_t *emu = snd_magic_cast(emu10k1x_t, pcm->private_data, return);
emu->pcm = NULL;
snd_pcm_lib_preallocate_free_for_all(pcm);
}
static int __devinit snd_emu10k1x_pcm(emu10k1x_t *emu, int device, snd_pcm_t **rpcm)
{
snd_pcm_t *pcm;
snd_pcm_substream_t *substream;
int err;
if (rpcm)
*rpcm = NULL;
if ((err = snd_pcm_new(emu->card, "emu10k1x", device, 3, 0, &pcm)) < 0)
return err;
pcm->private_data = emu;
pcm->private_free = snd_emu10k1x_pcm_free;
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_emu10k1x_playback_ops);
// snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_emu10k1x_capture_ops);
pcm->info_flags = 0;
pcm->dev_subclass = SNDRV_PCM_SUBCLASS_GENERIC_MIX;
strcpy(pcm->name, "EMU10K1X");
emu->pcm = pcm;
for(substream = pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream;
substream;
substream = substream->next)
if ((err = snd_pcm_lib_preallocate_pages(substream,
SNDRV_DMA_TYPE_DEV,
snd_dma_pci_data(emu->pci),
32*1024, 32*1024)) < 0)
return err;
/*
for (substream = pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream;
substream;
substream = substream->next)
snd_pcm_lib_preallocate_pages(substream,
SNDRV_DMA_TYPE_DEV,
snd_dma_pci_data(emu->pci),
64*1024, 64*1024);
*/
if (rpcm)
*rpcm = pcm;
return 0;
}
static int __devinit snd_emu10k1x_create(snd_card_t *card,
struct pci_dev *pci,
emu10k1x_t **rchip)
{
emu10k1x_t *chip;
int err;
int ch;
static snd_device_ops_t ops = {
.dev_free = snd_emu10k1x_dev_free,
};
*rchip = NULL;
if ((err = pci_enable_device(pci)) < 0)
return err;
if (pci_set_dma_mask(pci, 0x0fffffff) < 0 ||
pci_set_consistent_dma_mask(pci, 0x0fffffff) < 0) {
printk(KERN_ERR "error to set 28bit mask DMA\n");
return -ENXIO;
}
chip = snd_magic_kcalloc(emu10k1x_t, 0, GFP_KERNEL);
if (chip == NULL)
return -ENOMEM;
chip->card = card;
chip->pci = pci;
chip->irq = -1;
spin_lock_init(&chip->emu_lock);
spin_lock_init(&chip->voice_lock);
chip->port = pci_resource_start(pci, 0);
if ((chip->res_port = request_region(chip->port, 8,
"My Chip")) == NULL) {
snd_emu10k1x_free(chip);
printk(KERN_ERR "cannot allocate the port\n");
return -EBUSY;
}
if (request_irq(pci->irq, snd_emu10k1x_interrupt,
SA_INTERRUPT|SA_SHIRQ, "EMU10K1X",
(void *)chip)) {
snd_emu10k1x_free(chip);
printk(KERN_ERR "cannot grab irq\n");
return -EBUSY;
}
chip->irq = pci->irq;
memset(&chip->dma_dev, 0, sizeof(chip->dma_dev));
chip->dma_dev.type = SNDRV_DMA_TYPE_DEV;
chip->dma_dev.dev = snd_dma_pci_data(pci);
if(snd_dma_alloc_pages(&chip->dma_dev, 32 * 1024, &chip->buffer) < 0) {
snd_emu10k1x_free(chip);
return -ENOMEM;
}
pci_set_master(pci);
/* read revision & serial */
pci_read_config_byte(pci, PCI_REVISION_ID, (char *)&chip->revision);
pci_read_config_dword(pci, PCI_SUBSYSTEM_VENDOR_ID, &chip->serial);
pci_read_config_word(pci, PCI_SUBSYSTEM_ID, &chip->model);
printk(KERN_INFO "Model %04x Rev %08x Serial %08x\n", chip->model,
chip->revision, chip->serial);
outl(0, chip->port + INTE);
for(ch = 0; ch < 3; ch++) {
chip->voices[ch].emu = chip;
chip->voices[ch].number = ch;
}
/*
* Init to 0x02109204 :
* Clock accuracy = 0 (1000ppm)
* Sample Rate = 2 (48kHz)
* Audio Channel = 1 (Left of 2)
* Source Number = 0 (Unspecified)
* Generation Status = 1 (Original for Cat Code 12)
* Cat Code = 12 (Digital Signal Mixer)
* Mode = 0 (Mode 0)
* Emphasis = 0 (None)
* CP = 1 (Copyright unasserted)
* AN = 0 (Audio data)
* P = 0 (Consumer)
*/
snd_emu10k1x_ptr_write(chip, SPCS0, 0,
SPCS_CLKACCY_1000PPM | SPCS_SAMPLERATE_48 |
SPCS_CHANNELNUM_LEFT | SPCS_SOURCENUM_UNSPEC |
SPCS_GENERATIONSTATUS | 0x00001200 |
0x00000000 | SPCS_EMPHASIS_NONE | SPCS_COPYRIGHT);
snd_emu10k1x_ptr_write(chip, SPCS1, 0,
SPCS_CLKACCY_1000PPM | SPCS_SAMPLERATE_48 |
SPCS_CHANNELNUM_LEFT | SPCS_SOURCENUM_UNSPEC |
SPCS_GENERATIONSTATUS | 0x00001200 |
0x00000000 | SPCS_EMPHASIS_NONE | SPCS_COPYRIGHT);
snd_emu10k1x_ptr_write(chip, SPCS2, 0,
SPCS_CLKACCY_1000PPM | SPCS_SAMPLERATE_48 |
SPCS_CHANNELNUM_LEFT | SPCS_SOURCENUM_UNSPEC |
SPCS_GENERATIONSTATUS | 0x00001200 |
0x00000000 | SPCS_EMPHASIS_NONE | SPCS_COPYRIGHT);
snd_emu10k1x_ptr_write(chip, 0x41, 0, 0x70f); // ???
snd_emu10k1x_ptr_write(chip, 0x45, 0, 0);
outl(HCFG_LOCKSOUNDCACHE|HCFG_AUDIOENABLE, chip->port+HCFG);
snd_emu10k1x_intr_enable(chip, (INTE_CH_0_LOOP|INTE_CH_0_HALF_LOOP));
// snd_emu10k1x_intr_enable(chip, (INTE_CH_0_LOOP<<1);
// snd_emu10k1x_intr_enable(chip, INTE_CH_0_LOOP<<2);
if ((err = snd_device_new(card, SNDRV_DEV_LOWLEVEL,
chip, &ops)) < 0) {
snd_emu10k1x_free(chip);
return err;
}
*rchip = chip;
return 0;
}
static void snd_emu10k1x_proc_reg_read(snd_info_entry_t *entry,
snd_info_buffer_t * buffer)
{
emu10k1x_t *emu = snd_magic_cast(emu10k1x_t, entry->private_data, return);
snd_iprintf(buffer, "Registers:\n\n");
unsigned long value,value1,value2;
unsigned long flags;
int i;
for(i = 0; i < 0x40; i+=4) {
spin_lock_irqsave(&emu->emu_lock, flags);
value = inl(emu->port + i);
spin_unlock_irqrestore(&emu->emu_lock, flags);
snd_iprintf(buffer, "Register %02X: %08lX\n", i, value);
}
snd_iprintf(buffer, "\nRegisters\n\n");
for(i = 0; i < 0x50; i++) {
value = snd_emu10k1x_ptr_read(emu, i, 0);
value1 = snd_emu10k1x_ptr_read(emu, i, 1);
value2 = snd_emu10k1x_ptr_read(emu, i, 2);
snd_iprintf(buffer, "%02X: %08lX %08lX %08lX\n", i, value, value,
value2);
}
}
int __devinit snd_emu10k1x_proc_init(emu10k1x_t * emu)
{
snd_info_entry_t *entry;
if(! snd_card_proc_new(emu->card, "emu10k1x_regs", &entry))
snd_info_set_text_ops(entry, emu, 1024, snd_emu10k1x_proc_reg_read);
return 0;
}
static int __devinit snd_emu10k1x_probe(struct pci_dev *pci,
const struct pci_device_id *pci_id)
{
static int dev;
snd_card_t *card;
emu10k1x_t *chip;
int err;
if (dev >= SNDRV_CARDS)
return -ENODEV;
if (!enable[dev]) {
dev++;
return -ENOENT;
}
card = snd_card_new(index[dev], id[dev], THIS_MODULE, 0);
if (card == NULL)
return -ENOMEM;
if ((err = snd_emu10k1x_create(card, pci, &chip)) < 0) {
snd_card_free(card);
return err;
}
if ((err = snd_emu10k1x_pcm(chip, 0, NULL)) < 0) {
snd_card_free(card);
return err;
}
if ((err = snd_emu10k1x_ac97(chip)) < 0) {
snd_card_free(card);
return err;
}
snd_emu10k1x_proc_init(chip);
strcpy(card->driver, "Dell SB Live");
strcpy(card->shortname, "EMU10K1X");
sprintf(card->longname, "%s at 0x%lx irq %i",
card->shortname, chip->port, chip->irq);
if ((err = snd_card_register(card)) < 0) {
snd_card_free(card);
return err;
}
pci_set_drvdata(pci, card);
dev++;
return 0;
}
static void __devexit snd_emu10k1x_remove(struct pci_dev *pci)
{
snd_card_free(pci_get_drvdata(pci));
pci_set_drvdata(pci, NULL);
}
// PCI IDs
static struct pci_device_id snd_emu10k1x_ids[] = {
{ 0x1102, 0x0006, PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0 }, /* Dell OEM version
(EMU10K1) */
{ 0, }
};
MODULE_DEVICE_TABLE(pci, snd_emu10k1x_ids);
// pci_driver definition
static struct pci_driver driver = {
.name = "Emu10k1x",
.id_table = snd_emu10k1x_ids,
.probe = snd_emu10k1x_probe,
.remove = __devexit_p(snd_emu10k1x_remove),
};
// initialization of the module
static int __init alsa_card_emu10k1x_init(void)
{
int err;
if ((err = pci_module_init(&driver)) > 0)
return err;
return 0;
}
// clean up the module
static void __exit alsa_card_emu10k1x_exit(void)
{
pci_unregister_driver(&driver);
}
module_init(alsa_card_emu10k1x_init)
module_exit(alsa_card_emu10k1x_exit)
EXPORT_NO_SYMBOLS; /* for old kernels only */
Index: Makefile
===================================================================
RCS file: /cvsroot/alsa/alsa-driver/pci/emu10k1/Makefile,v
retrieving revision 1.5
retrieving revision 1.6
diff -u -r1.5 -r1.6
--- Makefile 11 Nov 2003 13:12:25 -0000 1.5
+++ Makefile 14 May 2004 13:42:49 -0000 1.6
@@ -5,6 +5,10 @@
include $(SND_TOPDIR)/toplevel.config
include $(SND_TOPDIR)/Makefile.conf
+snd-emu10k1x-objs := emu10k1x.o
+
+obj-$(CONFIG_SND_EMU10K1X) += snd-emu10k1x.o
+
export-objs := emu10k1_main.o
include $(SND_TOPDIR)/alsa-kernel/pci/emu10k1/Makefile
-------------------------------------------------------
This SF.Net email is sponsored by: SourceForge.net Broadband
Sign-up now for SourceForge Broadband and get the fastest
6.0/768 connection for only $19.95/mo for the first 3 months!
http://ads.osdn.com/?ad_id=2562&alloc_id=6184&op=click
_______________________________________________
Alsa-cvslog mailing list
[EMAIL PROTECTED]
https://lists.sourceforge.net/lists/listinfo/alsa-cvslog