/* * * AD1816 lowlevel sound driver for Linux 2.6.0 and above * * Copyright (C) 1998-2003 by Thorsten Knabe <linux@thorsten-knabe.de> * * Based on the CS4232/AD1848 driver Copyright (C) by Hannu Savolainen 1993-1996 * * * version: 1.5 * status: beta * date: 2003/07/15 * * Changes: * Oleg Drokin: Some cleanup of load/unload functions. 1998/11/24 * * Thorsten Knabe: attach and unload rewritten, * some argument checks added 1998/11/30 * * Thorsten Knabe: Buggy isa bridge workaround added 1999/01/16 * * David Moews/Thorsten Knabe: Introduced options * parameter. Added slightly modified patch from * David Moews to disable dsp audio sources by setting * bit 0 of options parameter. This seems to be * required by some Aztech/Newcom SC-16 cards. 1999/04/18 * * Christoph Hellwig: Adapted to module_init/module_exit. 2000/03/03 * * Christoph Hellwig: Added isapnp support 2000/03/15 * * Arnaldo Carvalho de Melo: get rid of check_region 2001/10/07 * * Thorsten Knabe: Compiling with CONFIG_PNP enabled * works again. It is now possible to use more than one * AD1816 sound card. Sample rate now may be changed during * playback/capture. printk() uses log levels everywhere. * SMP fixes. DMA handling fixes. * Other minor code cleanup. 2003/07/15 * */ #include <linux/config.h> #include <linux/module.h> #include <linux/init.h> #include <linux/interrupt.h> #include <linux/isapnp.h> #include <linux/stddef.h> #include <linux/spinlock.h> #include "sound_config.h" #define DEBUGNOISE(x) #define CHECK_FOR_POWER { int timeout=100; \ while (timeout > 0 && (inb(devc->base)&0x80)!= 0x80) {\ timeout--; \ } \ if (timeout==0) {\ printk(KERN_WARNING "ad1816: Check for power failed in %s line: %d\n",__FILE__,__LINE__); \ } \ } /* structure to hold device specific information */ typedef struct { int base; /* set in attach */ int irq; int dma_playback; int dma_capture; int opened; /* open */ int speed; int channels; int audio_format; int audio_mode; int recmask; /* setup */ unsigned char format_bits; int supported_devices; int supported_rec_devices; unsigned short levels[SOUND_MIXER_NRDEVICES]; /* misc */ struct pnp_dev *pnpdev; /* configured via pnp */ int dev_no; /* this is the # in audio_devs and NOT in ad1816_info */ spinlock_t lock; } ad1816_info; static int nr_ad1816_devs; static int ad1816_clockfreq = 33000; static int options; /* supported audio formats */ static int ad_format_mask = AFMT_U8 | AFMT_S16_LE | AFMT_S16_BE | AFMT_MU_LAW | AFMT_A_LAW; /* array of device info structures */ static ad1816_info dev_info[MAX_AUDIO_DEV]; /* ------------------------------------------------------------------- */ /* functions for easier access to inderect registers */ static int ad_read (ad1816_info * devc, int reg) { int result; CHECK_FOR_POWER; outb ((unsigned char) (reg & 0x3f), devc->base+0); result = inb(devc->base+2); result+= inb(devc->base+3)<<8; return (result); } static void ad_write (ad1816_info * devc, int reg, int data) { CHECK_FOR_POWER; outb ((unsigned char) (reg & 0xff), devc->base+0); outb ((unsigned char) (data & 0xff),devc->base+2); outb ((unsigned char) ((data>>8)&0xff),devc->base+3); } /* ------------------------------------------------------------------- */ /* function interface required by struct audio_driver */ static void ad1816_halt_input (int dev) { unsigned long flags; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; unsigned char buffer; DEBUGNOISE(printk(KERN_DEBUG "ad1816: halt_input called\n")); spin_lock_irqsave(&devc->lock,flags); if(!isa_dma_bridge_buggy) { disable_dma(audio_devs[dev]->dmap_in->dma); } buffer=inb(devc->base+9); if (buffer & 0x01) { /* disable capture */ outb(buffer & ~0x01,devc->base+9); } if(!isa_dma_bridge_buggy) { enable_dma(audio_devs[dev]->dmap_in->dma); } /* Clear interrupt status */ outb (~0x40, devc->base+1); devc->audio_mode &= ~PCM_ENABLE_INPUT; spin_unlock_irqrestore(&devc->lock,flags); } static void ad1816_halt_output (int dev) { unsigned long flags; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; unsigned char buffer; DEBUGNOISE(printk(KERN_DEBUG "ad1816: halt_output called!\n")); spin_lock_irqsave(&devc->lock,flags); /* Mute pcm output */ ad_write(devc, 4, ad_read(devc,4)|0x8080); if(!isa_dma_bridge_buggy) { disable_dma(audio_devs[dev]->dmap_out->dma); } buffer=inb(devc->base+8); if (buffer & 0x01) { /* disable capture */ outb(buffer & ~0x01,devc->base+8); } if(!isa_dma_bridge_buggy) { enable_dma(audio_devs[dev]->dmap_out->dma); } /* Clear interrupt status */ outb ((unsigned char)~0x80, devc->base+1); devc->audio_mode &= ~PCM_ENABLE_OUTPUT; spin_unlock_irqrestore(&devc->lock,flags); } static void ad1816_output_block (int dev, unsigned long buf, int count, int intrflag) { unsigned long flags; unsigned long cnt; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; DEBUGNOISE(printk(KERN_DEBUG "ad1816: output_block called buf=%ld count=%d flags=%d\n",buf,count,intrflag)); cnt = count/4 - 1; spin_lock_irqsave(&devc->lock,flags); /* set transfer count */ ad_write (devc, 8, cnt & 0xffff); devc->audio_mode |= PCM_ENABLE_OUTPUT; spin_unlock_irqrestore(&devc->lock,flags); } static void ad1816_start_input (int dev, unsigned long buf, int count, int intrflag) { unsigned long flags; unsigned long cnt; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; DEBUGNOISE(printk(KERN_DEBUG "ad1816: start_input called buf=%ld count=%d flags=%d\n",buf,count,intrflag)); cnt = count/4 - 1; spin_lock_irqsave(&devc->lock,flags); /* set transfer count */ ad_write (devc, 10, cnt & 0xffff); devc->audio_mode |= PCM_ENABLE_INPUT; spin_unlock_irqrestore(&devc->lock,flags); } static int ad1816_prepare_for_input (int dev, int bsize, int bcount) { unsigned long flags; unsigned int freq; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; unsigned char fmt_bits; DEBUGNOISE(printk(KERN_DEBUG "ad1816: prepare_for_input called: bsize=%d bcount=%d\n",bsize,bcount)); spin_lock_irqsave(&devc->lock,flags); fmt_bits= (devc->format_bits&0x7)<<3; /* set mono/stereo mode */ if (devc->channels > 1) { fmt_bits |=0x4; } /* set Mono/Stereo in playback/capture register */ outb( (inb(devc->base+8) & ~0x3C)|fmt_bits, devc->base+8); outb( (inb(devc->base+9) & ~0x3C)|fmt_bits, devc->base+9); freq=((unsigned int)devc->speed*33000)/ad1816_clockfreq; /* write playback/capture speeds */ ad_write (devc, 2, freq & 0xffff); ad_write (devc, 3, freq & 0xffff); spin_unlock_irqrestore(&devc->lock,flags); ad1816_halt_input(dev); return 0; } static int ad1816_prepare_for_output (int dev, int bsize, int bcount) { unsigned long flags; unsigned int freq; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; unsigned char fmt_bits; DEBUGNOISE(printk(KERN_DEBUG "ad1816: prepare_for_output called: bsize=%d bcount=%d\n",bsize,bcount)); spin_lock_irqsave(&devc->lock,flags); fmt_bits= (devc->format_bits&0x7)<<3; /* set mono/stereo mode */ if (devc->channels > 1) { fmt_bits |=0x4; } /* write format bits to playback/capture registers */ outb( (inb(devc->base+8) & ~0x3C)|fmt_bits, devc->base+8); outb( (inb(devc->base+9) & ~0x3C)|fmt_bits, devc->base+9); freq=((unsigned int)devc->speed*33000)/ad1816_clockfreq; /* write playback/capture speeds */ ad_write (devc, 2, freq & 0xffff); ad_write (devc, 3, freq & 0xffff); spin_unlock_irqrestore(&devc->lock,flags); ad1816_halt_output(dev); return 0; } static void ad1816_trigger (int dev, int state) { unsigned long flags; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; DEBUGNOISE(printk(KERN_DEBUG "ad1816: trigger called! (devc=%d,devc->base=%d\n", devc, devc->base)); /* mode may have changed */ spin_lock_irqsave(&devc->lock,flags); /* mask out modes not specified on open call */ state &= devc->audio_mode; /* setup soundchip to new io-mode */ if (state & PCM_ENABLE_INPUT) { /* enable capture */ outb(inb(devc->base+9)|0x01, devc->base+9); } else { /* disable capture */ outb(inb(devc->base+9)&~0x01, devc->base+9); } if (state & PCM_ENABLE_OUTPUT) { /* enable playback */ outb(inb(devc->base+8)|0x01, devc->base+8); /* unmute pcm output */ ad_write(devc, 4, ad_read(devc,4)&~0x8080); } else { /* mute pcm output */ ad_write(devc, 4, ad_read(devc,4)|0x8080); /* disable capture */ outb(inb(devc->base+8)&~0x01, devc->base+8); } spin_unlock_irqrestore(&devc->lock,flags); } /* halt input & output */ static void ad1816_halt (int dev) { ad1816_halt_input(dev); ad1816_halt_output(dev); } static void ad1816_reset (int dev) { ad1816_halt (dev); } /* set playback speed */ static int ad1816_set_speed (int dev, int arg) { unsigned long flags; unsigned int freq; int ret; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; spin_lock_irqsave(&devc->lock, flags); if (arg == 0) { ret = devc->speed; spin_unlock_irqrestore(&devc->lock, flags); return ret; } /* range checking */ if (arg < 4000) { arg = 4000; } if (arg > 55000) { arg = 55000; } devc->speed = arg; /* change speed during playback */ freq=((unsigned int)devc->speed*33000)/ad1816_clockfreq; /* write playback/capture speeds */ ad_write (devc, 2, freq & 0xffff); ad_write (devc, 3, freq & 0xffff); ret = devc->speed; spin_unlock_irqrestore(&devc->lock, flags); return ret; } static unsigned int ad1816_set_bits (int dev, unsigned int arg) { unsigned long flags; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; static struct format_tbl { int format; unsigned char bits; } format2bits[] = { { 0, 0 }, { AFMT_MU_LAW, 1 }, { AFMT_A_LAW, 3 }, { AFMT_IMA_ADPCM, 0 }, { AFMT_U8, 0 }, { AFMT_S16_LE, 2 }, { AFMT_S16_BE, 6 }, { AFMT_S8, 0 }, { AFMT_U16_LE, 0 }, { AFMT_U16_BE, 0 } }; int i, n = sizeof (format2bits) / sizeof (struct format_tbl); spin_lock_irqsave(&devc->lock, flags); /* return current format */ if (arg == 0) { arg = devc->audio_format; spin_unlock_irqrestore(&devc->lock, flags); return arg; } devc->audio_format = arg; /* search matching format bits */ for (i = 0; i < n; i++) if (format2bits[i].format == arg) { devc->format_bits = format2bits[i].bits; devc->audio_format = arg; spin_unlock_irqrestore(&devc->lock, flags); return arg; } /* Still hanging here. Something must be terribly wrong */ devc->format_bits = 0; devc->audio_format = AFMT_U8; spin_unlock_irqrestore(&devc->lock, flags); return(AFMT_U8); } static short ad1816_set_channels (int dev, short arg) { ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; if (arg != 1 && arg != 2) return devc->channels; devc->channels = arg; return arg; } /* open device */ static int ad1816_open (int dev, int mode) { ad1816_info *devc = NULL; unsigned long flags; /* is device number valid ? */ if (dev < 0 || dev >= num_audiodevs) return -(ENXIO); /* get device info of this dev */ devc = (ad1816_info *) audio_devs[dev]->devc; /* make check if device already open atomic */ spin_lock_irqsave(&devc->lock,flags); if (devc->opened) { spin_unlock_irqrestore(&devc->lock,flags); return -(EBUSY); } /* mark device as open */ devc->opened = 1; devc->audio_mode = 0; devc->speed = 8000; devc->audio_format=AFMT_U8; devc->channels=1; spin_unlock_irqrestore(&devc->lock,flags); ad1816_reset(devc->dev_no); /* halt all pending output */ return 0; } static void ad1816_close (int dev) /* close device */ { unsigned long flags; ad1816_info *devc = (ad1816_info *) audio_devs[dev]->devc; /* halt all pending output */ ad1816_reset(devc->dev_no); spin_lock_irqsave(&devc->lock,flags); devc->opened = 0; devc->audio_mode = 0; devc->speed = 8000; devc->audio_format=AFMT_U8; devc->format_bits = 0; spin_unlock_irqrestore(&devc->lock,flags); } /* ------------------------------------------------------------------- */ /* Audio driver structure */ static struct audio_driver ad1816_audio_driver = { .owner = THIS_MODULE, .open = ad1816_open, .close = ad1816_close, .output_block = ad1816_output_block, .start_input = ad1816_start_input, .prepare_for_input = ad1816_prepare_for_input, .prepare_for_output = ad1816_prepare_for_output, .halt_io = ad1816_halt, .halt_input = ad1816_halt_input, .halt_output = ad1816_halt_output, .trigger = ad1816_trigger, .set_speed = ad1816_set_speed, .set_bits = ad1816_set_bits, .set_channels = ad1816_set_channels, }; /* ------------------------------------------------------------------- */ /* Interrupt handler */ static irqreturn_t ad1816_interrupt (int irq, void *dev_id, struct pt_regs *dummy) { unsigned char status; ad1816_info *devc = (ad1816_info *)dev_id; if (irq < 0 || irq > 15) { printk(KERN_WARNING "ad1816: Got bogus interrupt %d\n", irq); return IRQ_NONE; } spin_lock(&devc->lock); /* read interrupt register */ status = inb (devc->base+1); /* Clear all interrupt */ outb (~status, devc->base+1); DEBUGNOISE(printk(KERN_DEBUG "ad1816: Got interrupt subclass %d\n",status)); if (status == 0) { DEBUGNOISE(printk(KERN_DEBUG "ad1816: interrupt: Got interrupt, but no source.\n")); spin_unlock(&devc->lock); return IRQ_NONE; } if (devc->opened && (devc->audio_mode & PCM_ENABLE_INPUT) && (status&64)) DMAbuf_inputintr (devc->dev_no); if (devc->opened && (devc->audio_mode & PCM_ENABLE_OUTPUT) && (status & 128)) DMAbuf_outputintr (devc->dev_no, 1); spin_unlock(&devc->lock); return IRQ_HANDLED; } /* ------------------------------------------------------------------- */ /* Mixer stuff */ struct mixer_def { unsigned int regno: 7; unsigned int polarity:1; /* 0=normal, 1=reversed */ unsigned int bitpos:4; unsigned int nbits:4; }; static char mix_cvt[101] = { 0, 0, 3, 7,10,13,16,19,21,23,26,28,30,32,34,35,37,39,40,42, 43,45,46,47,49,50,51,52,53,55,56,57,58,59,60,61,62,63,64,65, 65,66,67,68,69,70,70,71,72,73,73,74,75,75,76,77,77,78,79,79, 80,81,81,82,82,83,84,84,85,85,86,86,87,87,88,88,89,89,90,90, 91,91,92,92,93,93,94,94,95,95,96,96,96,97,97,98,98,98,99,99, 100 }; typedef struct mixer_def mixer_ent; /* * Most of the mixer entries work in backwards. Setting the polarity field * makes them to work correctly. * * The channel numbering used by individual soundcards is not fixed. Some * cards have assigned different meanings for the AUX1, AUX2 and LINE inputs. * The current version doesn't try to compensate this. */ #define MIX_ENT(name, reg_l, pola_l, pos_l, len_l, reg_r, pola_r, pos_r, len_r) \ {{reg_l, pola_l, pos_l, len_l}, {reg_r, pola_r, pos_r, len_r}} mixer_ent mix_devices[SOUND_MIXER_NRDEVICES][2] = { MIX_ENT(SOUND_MIXER_VOLUME, 14, 1, 8, 5, 14, 1, 0, 5), MIX_ENT(SOUND_MIXER_BASS, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_TREBLE, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_SYNTH, 5, 1, 8, 6, 5, 1, 0, 6), MIX_ENT(SOUND_MIXER_PCM, 4, 1, 8, 6, 4, 1, 0, 6), MIX_ENT(SOUND_MIXER_SPEAKER, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_LINE, 18, 1, 8, 5, 18, 1, 0, 5), MIX_ENT(SOUND_MIXER_MIC, 19, 1, 8, 5, 19, 1, 0, 5), MIX_ENT(SOUND_MIXER_CD, 15, 1, 8, 5, 15, 1, 0, 5), MIX_ENT(SOUND_MIXER_IMIX, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_ALTPCM, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_RECLEV, 20, 0, 8, 4, 20, 0, 0, 4), MIX_ENT(SOUND_MIXER_IGAIN, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_OGAIN, 0, 0, 0, 0, 0, 0, 0, 0), MIX_ENT(SOUND_MIXER_LINE1, 17, 1, 8, 5, 17, 1, 0, 5), MIX_ENT(SOUND_MIXER_LINE2, 16, 1, 8, 5, 16, 1, 0, 5), MIX_ENT(SOUND_MIXER_LINE3, 39, 0, 9, 4, 39, 1, 0, 5) }; static unsigned short default_mixer_levels[SOUND_MIXER_NRDEVICES] = { 0x4343, /* Master Volume */ 0x3232, /* Bass */ 0x3232, /* Treble */ 0x0000, /* FM */ 0x4343, /* PCM */ 0x0000, /* PC Speaker */ 0x0000, /* Ext Line */ 0x0000, /* Mic */ 0x0000, /* CD */ 0x0000, /* Recording monitor */ 0x0000, /* SB PCM */ 0x0000, /* Recording level */ 0x0000, /* Input gain */ 0x0000, /* Output gain */ 0x0000, /* Line1 */ 0x0000, /* Line2 */ 0x0000 /* Line3 (usually line in)*/ }; #define LEFT_CHN 0 #define RIGHT_CHN 1 static int ad1816_set_recmask (ad1816_info * devc, int mask) { unsigned long flags; unsigned char recdev; int i, n; spin_lock_irqsave(&devc->lock, flags); mask &= devc->supported_rec_devices; n = 0; /* Count selected device bits */ for (i = 0; i < 32; i++) if (mask & (1 << i)) n++; if (n == 0) mask = SOUND_MASK_MIC; else if (n != 1) { /* Too many devices selected */ /* Filter out active settings */ mask &= ~devc->recmask; n = 0; /* Count selected device bits */ for (i = 0; i < 32; i++) if (mask & (1 << i)) n++; if (n != 1) mask = SOUND_MASK_MIC; } switch (mask) { case SOUND_MASK_MIC: recdev = 5; break; case SOUND_MASK_LINE: recdev = 0; break; case SOUND_MASK_CD: recdev = 2; break; case SOUND_MASK_LINE1: recdev = 4; break; case SOUND_MASK_LINE2: recdev = 3; break; case SOUND_MASK_VOLUME: recdev = 1; break; default: mask = SOUND_MASK_MIC; recdev = 5; } recdev <<= 4; ad_write (devc, 20, (ad_read (devc, 20) & 0x8f8f) | recdev | (recdev<<8)); devc->recmask = mask; spin_unlock_irqrestore(&devc->lock, flags); return mask; } static void change_bits (int *regval, int dev, int chn, int newval) { unsigned char mask; int shift; /* Reverse polarity*/ if (mix_devices[dev][chn].polarity == 1) newval = 100 - newval; mask = (1 << mix_devices[dev][chn].nbits) - 1; shift = mix_devices[dev][chn].bitpos; /* Scale it */ newval = (int) ((newval * mask) + 50) / 100; /* Clear bits */ *regval &= ~(mask << shift); /* Set new value */ *regval |= (newval & mask) << shift; } static int ad1816_mixer_get (ad1816_info * devc, int dev) { DEBUGNOISE(printk(KERN_DEBUG "ad1816: mixer_get called!\n")); /* range check + supported mixer check */ if (dev < 0 || dev >= SOUND_MIXER_NRDEVICES ) return (-(EINVAL)); if (!((1 << dev) & devc->supported_devices)) return -(EINVAL); return devc->levels[dev]; } static int ad1816_mixer_set (ad1816_info * devc, int dev, int value) { int left = value & 0x000000ff; int right = (value & 0x0000ff00) >> 8; int retvol; int regoffs; int val; int valmute; unsigned long flags; DEBUGNOISE(printk(KERN_DEBUG "ad1816: mixer_set called!\n")); if (dev < 0 || dev >= SOUND_MIXER_NRDEVICES ) return -(EINVAL); if (left > 100) left = 100; if (left < 0) left = 0; if (right > 100) right = 100; if (right < 0) right = 0; /* Mono control */ if (mix_devices[dev][RIGHT_CHN].nbits == 0) right = left; retvol = left | (right << 8); /* Scale it */ left = mix_cvt[left]; right = mix_cvt[right]; /* reject all mixers that are not supported */ if (!(devc->supported_devices & (1 << dev))) return -(EINVAL); /* sanity check */ if (mix_devices[dev][LEFT_CHN].nbits == 0) return -(EINVAL); spin_lock_irqsave(&devc->lock, flags); /* keep precise volume internal */ devc->levels[dev] = retvol; /* Set the left channel */ regoffs = mix_devices[dev][LEFT_CHN].regno; val = ad_read (devc, regoffs); change_bits (&val, dev, LEFT_CHN, left); valmute=val; /* Mute bit masking on some registers */ if ( regoffs==5 || regoffs==14 || regoffs==15 || regoffs==16 || regoffs==17 || regoffs==18 || regoffs==19 || regoffs==39) { if (left==0) valmute |= 0x8000; else valmute &= ~0x8000; } ad_write (devc, regoffs, valmute); /* mute */ /* * Set the right channel */ /* Was just a mono channel */ if (mix_devices[dev][RIGHT_CHN].nbits == 0) { spin_unlock_irqrestore(&devc->lock, flags); return retvol; } regoffs = mix_devices[dev][RIGHT_CHN].regno; val = ad_read (devc, regoffs); change_bits (&val, dev, RIGHT_CHN, right); valmute=val; if ( regoffs==5 || regoffs==14 || regoffs==15 || regoffs==16 || regoffs==17 || regoffs==18 || regoffs==19 || regoffs==39) { if (right==0) valmute |= 0x80; else valmute &= ~0x80; } ad_write (devc, regoffs, valmute); /* mute */ spin_unlock_irqrestore(&devc->lock, flags); return retvol; } #define MIXER_DEVICES ( SOUND_MASK_VOLUME | \ SOUND_MASK_SYNTH | \ SOUND_MASK_PCM | \ SOUND_MASK_LINE | \ SOUND_MASK_LINE1 | \ SOUND_MASK_LINE2 | \ SOUND_MASK_LINE3 | \ SOUND_MASK_MIC | \ SOUND_MASK_CD | \ SOUND_MASK_RECLEV \ ) #define REC_DEVICES ( SOUND_MASK_LINE2 |\ SOUND_MASK_LINE |\ SOUND_MASK_LINE1 |\ SOUND_MASK_MIC |\ SOUND_MASK_CD |\ SOUND_MASK_VOLUME \ ) static void ad1816_mixer_reset (ad1816_info * devc) { int i; devc->supported_devices = MIXER_DEVICES; devc->supported_rec_devices = REC_DEVICES; for (i = 0; i < SOUND_MIXER_NRDEVICES; i++) if (devc->supported_devices & (1 << i)) ad1816_mixer_set (devc, i, default_mixer_levels[i]); ad1816_set_recmask (devc, SOUND_MASK_MIC); } static int ad1816_mixer_ioctl (int dev, unsigned int cmd, void __user * arg) { ad1816_info *devc = mixer_devs[dev]->devc; int val; int __user *p = arg; DEBUGNOISE(printk(KERN_DEBUG "ad1816: mixer_ioctl called!\n")); /* Mixer ioctl */ if (((cmd >> 8) & 0xff) == 'M') { /* set ioctl */ if (_SIOC_DIR (cmd) & _SIOC_WRITE) { switch (cmd & 0xff){ case SOUND_MIXER_RECSRC: if (get_user(val, p)) return -EFAULT; val=ad1816_set_recmask (devc, val); return put_user(val, p); break; default: if (get_user(val, p)) return -EFAULT; if ((val=ad1816_mixer_set (devc, cmd & 0xff, val))<0) return val; else return put_user(val, p); } } else { /* read ioctl */ switch (cmd & 0xff) { case SOUND_MIXER_RECSRC: val=devc->recmask; return put_user(val, p); break; case SOUND_MIXER_DEVMASK: val=devc->supported_devices; return put_user(val, p); break; case SOUND_MIXER_STEREODEVS: val=devc->supported_devices & ~(SOUND_MASK_SPEAKER | SOUND_MASK_IMIX); return put_user(val, p); break; case SOUND_MIXER_RECMASK: val=devc->supported_rec_devices; return put_user(val, p); break; case SOUND_MIXER_CAPS: val=SOUND_CAP_EXCL_INPUT; return put_user(val, p); break; default: if ((val=ad1816_mixer_get (devc, cmd & 0xff))<0) return val; else return put_user(val, p); } } } else /* not for mixer */ return -(EINVAL); } /* ------------------------------------------------------------------- */ /* Mixer structure */ static struct mixer_operations ad1816_mixer_operations = { .owner = THIS_MODULE, .id = "AD1816", .name = "AD1816 Mixer", .ioctl = ad1816_mixer_ioctl }; /* ------------------------------------------------------------------- */ /* stuff for card recognition, init and unloading PNP ...*/ /* check if AD1816 present at specified hw_config and register device with OS * return 1 if initialization was successful, 0 otherwise */ static int __init ad1816_init_card (struct address_info *hw_config, struct pnp_dev *pnp) { ad1816_info *devc = NULL; int tmp; int oss_devno = -1; printk(KERN_INFO "ad1816: initializing card: io=0x%x, irq=%d, dma=%d, " "dma2=%d, clockfreq=%d, options=%d isadmabug=%d " "%s\n", hw_config->io_base, hw_config->irq, hw_config->dma, hw_config->dma2, ad1816_clockfreq, options, isa_dma_bridge_buggy, pnp?"(PNP)":""); /* ad1816_info structure remaining ? */ if (nr_ad1816_devs >= MAX_AUDIO_DEV) { printk(KERN_WARNING "ad1816: no more ad1816_info structures " "left\n"); goto out; } devc = &dev_info[nr_ad1816_devs]; devc->base = hw_config->io_base; devc->irq = hw_config->irq; devc->dma_playback=hw_config->dma; devc->dma_capture=hw_config->dma2; devc->opened = 0; devc->pnpdev = pnp; spin_lock_init(&devc->lock); if (!request_region(devc->base, 16, "AD1816 Sound")) { printk(KERN_WARNING "ad1816: I/O port 0x%03x not free\n", devc->base); goto out; } printk(KERN_INFO "ad1816: Examining AD1816 at address 0x%03x.\n", devc->base); /* tests for ad1816 */ /* base+0: bit 1 must be set but not 255 */ tmp=inb(devc->base); if ( (tmp&0x80)==0 || tmp==255 ) { printk (KERN_INFO "ad1816: Chip is not an AD1816 or chip " "is not active (Test 0)\n"); goto out_release_region; } /* writes to ireg 8 are copied to ireg 9 */ ad_write(devc,8,12345); if (ad_read(devc,9)!=12345) { printk(KERN_INFO "ad1816: Chip is not an AD1816 (Test 1)\n"); goto out_release_region; } /* writes to ireg 8 are copied to ireg 9 */ ad_write(devc,8,54321); if (ad_read(devc,9)!=54321) { printk(KERN_INFO "ad1816: Chip is not an AD1816 (Test 2)\n"); goto out_release_region; } /* writes to ireg 10 are copied to ireg 11 */ ad_write(devc,10,54321); if (ad_read(devc,11)!=54321) { printk (KERN_INFO "ad1816: Chip is not an AD1816 (Test 3)\n"); goto out_release_region; } /* writes to ireg 10 are copied to ireg 11 */ ad_write(devc,10,12345); if (ad_read(devc,11)!=12345) { printk (KERN_INFO "ad1816: Chip is not an AD1816 (Test 4)\n"); goto out_release_region; } /* bit in base +1 cannot be set to 1 */ tmp=inb(devc->base+1); outb(0xff,devc->base+1); if (inb(devc->base+1)!=tmp) { printk(KERN_INFO "ad1816: Chip is not an AD1816 (Test 5)\n"); goto out_release_region; } printk(KERN_INFO "ad1816: AD1816 (version %d) successfully detected!\n", ad_read(devc,45)); /* disable all interrupts */ ad_write(devc,1,0); /* Clear pending interrupts */ outb (0, devc->base+1); /* allocate irq */ if (devc->irq < 0 || devc->irq > 15) goto out_release_region; if (request_irq(devc->irq, ad1816_interrupt,0, "SoundPort", devc) < 0) { printk(KERN_WARNING "ad1816: IRQ in use\n"); goto out_release_region; } /* DMA stuff */ if (sound_alloc_dma (devc->dma_playback, "Sound System")) { printk(KERN_WARNING "ad1816: Can't allocate DMA%d\n", devc->dma_playback); goto out_free_irq; } if ( devc->dma_capture >= 0 && devc->dma_capture != devc->dma_playback) { if (sound_alloc_dma(devc->dma_capture, "Sound System (capture)")) { printk(KERN_WARNING "ad1816: Can't allocate DMA%d\n", devc->dma_capture); goto out_free_dma; } devc->audio_mode=DMA_AUTOMODE|DMA_DUPLEX; } else { printk(KERN_WARNING "ad1816: Only one DMA channel " "available/configured. No duplex operation possible\n"); devc->audio_mode=DMA_AUTOMODE; } conf_printf2 ("AD1816 audio driver", devc->base, devc->irq, devc->dma_playback, devc->dma_capture); /* register device */ if ((oss_devno = sound_install_audiodrv (AUDIO_DRIVER_VERSION, "AD1816 audio driver", &ad1816_audio_driver, sizeof (struct audio_driver), devc->audio_mode, ad_format_mask, devc, devc->dma_playback, devc->dma_capture)) < 0) { printk(KERN_WARNING "ad1816: Can't install sound driver\n"); goto out_free_dma_2; } ad_write(devc,32,0x80f0); /* sound system mode */ if (options&1) { ad_write(devc,33,0); /* disable all audiosources for dsp */ } else { ad_write(devc,33,0x03f8); /* enable all audiosources for dsp */ } ad_write(devc,4,0x8080); /* default values for volumes (muted)*/ ad_write(devc,5,0x8080); ad_write(devc,6,0x8080); ad_write(devc,7,0x8080); ad_write(devc,15,0x8888); ad_write(devc,16,0x8888); ad_write(devc,17,0x8888); ad_write(devc,18,0x8888); ad_write(devc,19,0xc888); /* +20db mic active */ ad_write(devc,14,0x0000); /* Master volume unmuted */ ad_write(devc,39,0x009f); /* 3D effect on 0% phone out muted */ ad_write(devc,44,0x0080); /* everything on power, 3d enabled for d/a */ outb(0x10,devc->base+8); /* set dma mode */ outb(0x10,devc->base+9); /* enable capture + playback interrupt */ ad_write(devc,1,0xc000); /* set mixer defaults */ ad1816_mixer_reset (devc); /* register mixer */ if ((audio_devs[oss_devno]->mixer_dev=sound_install_mixer( MIXER_DRIVER_VERSION, "AD1816 audio driver", &ad1816_mixer_operations, sizeof (struct mixer_operations), devc)) < 0) { printk(KERN_WARNING "Can't install mixer\n"); } /* make ad1816_info active */ nr_ad1816_devs++; printk(KERN_INFO "ad1816: card successfully installed!\n"); return 1; /* error handling */ out_free_dma_2: if (devc->dma_capture >= 0 && devc->dma_capture != devc->dma_playback) sound_free_dma(devc->dma_capture); out_free_dma: sound_free_dma(devc->dma_playback); out_free_irq: free_irq(devc->irq, devc); out_release_region: release_region(devc->base, 16); out: return 0; } static void __exit unload_card(ad1816_info *devc) { int mixer, dev = 0; if (devc != NULL) { printk("ad1816: Unloading card at address 0x%03x\n",devc->base); dev = devc->dev_no; mixer = audio_devs[dev]->mixer_dev; /* unreg mixer*/ if(mixer>=0) { sound_unload_mixerdev(mixer); } /* unreg audiodev */ sound_unload_audiodev(dev); /* free dma channels */ if (devc->dma_capture>=0 && devc->dma_capture != devc->dma_playback) { sound_free_dma(devc->dma_capture); } sound_free_dma (devc->dma_playback); /* free irq */ free_irq(devc->irq, devc); /* free io */ release_region (devc->base, 16); #ifdef __ISAPNP__ if (devc->pnpdev) { pnp_disable_dev(devc->pnpdev); pnp_device_detach(devc->pnpdev); } #endif } else printk(KERN_WARNING "ad1816: no device/card specified\n"); } static int __initdata io = -1; static int __initdata irq = -1; static int __initdata dma = -1; static int __initdata dma2 = -1; #ifdef __ISAPNP__ /* use isapnp for configuration */ static int isapnp = 1; static int isapnpjump; module_param(isapnp, bool, 0); module_param(isapnpjump, int, 0); #endif module_param(io, int, 0); module_param(irq, int, 0); module_param(dma, int, 0); module_param(dma2, int, 0); module_param(ad1816_clockfreq, int, 0); module_param(options, int, 0); #ifdef __ISAPNP__ static struct { unsigned short card_vendor, card_device; unsigned short vendor; unsigned short function; struct ad1816_data *data; } isapnp_ad1816_list[] __initdata = { { ISAPNP_ANY_ID, ISAPNP_ANY_ID, ISAPNP_VENDOR('A','D','S'), ISAPNP_FUNCTION(0x7150), NULL }, { ISAPNP_ANY_ID, ISAPNP_ANY_ID, ISAPNP_VENDOR('A','D','S'), ISAPNP_FUNCTION(0x7180), NULL }, {0} }; MODULE_DEVICE_TABLE(isapnp, isapnp_ad1816_list); static void __init ad1816_config_pnp_card(struct pnp_card *card, unsigned short vendor, unsigned short function) { struct address_info cfg; struct pnp_dev *card_dev = pnp_find_dev(card, vendor, function, NULL); if (!card_dev) return; if (pnp_device_attach(card_dev) < 0) { printk(KERN_WARNING "ad1816: Failed to attach PnP device\n"); return; } if (pnp_activate_dev(card_dev) < 0) { printk(KERN_WARNING "ad1816: Failed to activate PnP device\n"); pnp_device_detach(card_dev); return; } cfg.io_base = pnp_port_start(card_dev, 2); cfg.irq = pnp_irq(card_dev, 0); cfg.dma = pnp_irq(card_dev, 0); cfg.dma2 = pnp_irq(card_dev, 1); if (!ad1816_init_card(&cfg, card_dev)) { pnp_disable_dev(card_dev); pnp_device_detach(card_dev); } } static void __init ad1816_config_pnp_cards(void) { int nr_pnp_cfg; int i; /* Count entries in isapnp_ad1816_list */ for (nr_pnp_cfg = 0; isapnp_ad1816_list[nr_pnp_cfg].card_vendor != 0; nr_pnp_cfg++); /* Check and adjust isapnpjump */ if( isapnpjump < 0 || isapnpjump >= nr_pnp_cfg) { printk(KERN_WARNING "ad1816: Valid range for isapnpjump is 0-%d. " "Adjusted to 0.\n", nr_pnp_cfg-1); isapnpjump = 0; } for (i = isapnpjump; isapnp_ad1816_list[i].card_vendor != 0; i++) { struct pnp_card *card = NULL; /* iterate over all pnp cards */ while ((card = pnp_find_card(isapnp_ad1816_list[i].card_vendor, isapnp_ad1816_list[i].card_device, card))) ad1816_config_pnp_card(card, isapnp_ad1816_list[i].vendor, isapnp_ad1816_list[i].function); } } #endif /* module initialization */ static int __init init_ad1816(void) { printk(KERN_INFO "ad1816: AD1816 sounddriver " "Copyright (C) 1998-2003 by Thorsten Knabe and " "others\n"); #ifdef AD1816_CLOCK /* set ad1816_clockfreq if set during compilation */ ad1816_clockfreq=AD1816_CLOCK; #endif if (ad1816_clockfreq<5000 || ad1816_clockfreq>100000) { ad1816_clockfreq=33000; } #ifdef __ISAPNP__ /* configure PnP cards */ if(isapnp) ad1816_config_pnp_cards(); #endif /* configure card by module params */ if (io != -1 && irq != -1 && dma != -1) { struct address_info cfg; cfg.io_base = io; cfg.irq = irq; cfg.dma = dma; cfg.dma2 = dma2; ad1816_init_card(&cfg, NULL); } if (nr_ad1816_devs <= 0) return -ENODEV; return 0; } /* module cleanup */ static void __exit cleanup_ad1816 (void) { int i; ad1816_info *devc = NULL; /* remove any soundcard */ for (i = 0; i < nr_ad1816_devs; i++) { devc = &dev_info[i]; unload_card(devc); } nr_ad1816_devs=0; printk(KERN_INFO "ad1816: driver unloaded!\n"); } module_init(init_ad1816); module_exit(cleanup_ad1816); #ifndef MODULE /* kernel command line parameter evaluation */ static int __init setup_ad1816(char *str) { /* io, irq, dma, dma2 */ int ints[5]; str = get_options(str, ARRAY_SIZE(ints), ints); io = ints[1]; irq = ints[2]; dma = ints[3]; dma2 = ints[4]; return 1; } __setup("ad1816=", setup_ad1816); #endif MODULE_LICENSE("GPL");