|  | /***************************************************************************** | 
|  | * | 
|  | *      ESS Maestro/Maestro-2/Maestro-2E driver for Linux 2.[23].x | 
|  | * | 
|  | *      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., 675 Mass Ave, Cambridge, MA 02139, USA. | 
|  | * | 
|  | *	(c) Copyright 1999	 Alan Cox <alan.cox@linux.org> | 
|  | * | 
|  | *	Based heavily on SonicVibes.c: | 
|  | *      Copyright (C) 1998-1999  Thomas Sailer (sailer@ife.ee.ethz.ch) | 
|  | * | 
|  | *	Heavily modified by Zach Brown <zab@zabbo.net> based on lunch | 
|  | *	with ESS engineers.  Many thanks to Howard Kim for providing | 
|  | *	contacts and hardware.  Honorable mention goes to Eric | 
|  | *	Brombaugh for all sorts of things.  Best regards to the | 
|  | *	proprietors of Hack Central for fine lodging. | 
|  | * | 
|  | *  Supported devices: | 
|  | *  /dev/dsp0-3    standard /dev/dsp device, (mostly) OSS compatible | 
|  | *  /dev/mixer  standard /dev/mixer device, (mostly) OSS compatible | 
|  | * | 
|  | *  Hardware Description | 
|  | * | 
|  | *	A working Maestro setup contains the Maestro chip wired to a | 
|  | *	codec or 2.  In the Maestro we have the APUs, the ASSP, and the | 
|  | *	Wavecache.  The APUs can be though of as virtual audio routing | 
|  | *	channels.  They can take data from a number of sources and perform | 
|  | *	basic encodings of the data.  The wavecache is a storehouse for | 
|  | *	PCM data.  Typically it deals with PCI and interracts with the | 
|  | *	APUs.  The ASSP is a wacky DSP like device that ESS is loth | 
|  | *	to release docs on.  Thankfully it isn't required on the Maestro | 
|  | *	until you start doing insane things like FM emulation and surround | 
|  | *	encoding.  The codecs are almost always AC-97 compliant codecs, | 
|  | *	but it appears that early Maestros may have had PT101 (an ESS | 
|  | *	part?) wired to them.  The only real difference in the Maestro | 
|  | *	families is external goop like docking capability, memory for | 
|  | *	the ASSP, and initialization differences. | 
|  | * | 
|  | *  Driver Operation | 
|  | * | 
|  | *	We only drive the APU/Wavecache as typical DACs and drive the | 
|  | *	mixers in the codecs.  There are 64 APUs.  We assign 6 to each | 
|  | *	/dev/dsp? device.  2 channels for output, and 4 channels for | 
|  | *	input. | 
|  | * | 
|  | *	Each APU can do a number of things, but we only really use | 
|  | *	3 basic functions.  For playback we use them to convert PCM | 
|  | *	data fetched over PCI by the wavecahche into analog data that | 
|  | *	is handed to the codec.  One APU for mono, and a pair for stereo. | 
|  | *	When in stereo, the combination of smarts in the APU and Wavecache | 
|  | *	decide which wavecache gets the left or right channel. | 
|  | * | 
|  | *	For record we still use the old overly mono system.  For each in | 
|  | *	coming channel the data comes in from the codec, through a 'input' | 
|  | *	APU, through another rate converter APU, and then into memory via | 
|  | *	the wavecache and PCI.  If its stereo, we mash it back into LRLR in | 
|  | *	software.  The pass between the 2 APUs is supposedly what requires us | 
|  | *	to have a 512 byte buffer sitting around in wavecache/memory. | 
|  | * | 
|  | *	The wavecache makes our life even more fun.  First off, it can | 
|  | *	only address the first 28 bits of PCI address space, making it | 
|  | *	useless on quite a few architectures.  Secondly, its insane. | 
|  | *	It claims to fetch from 4 regions of PCI space, each 4 meg in length. | 
|  | *	But that doesn't really work.  You can only use 1 region.  So all our | 
|  | *	allocations have to be in 4meg of each other.  Booo.  Hiss. | 
|  | *	So we have a module parameter, dsps_order, that is the order of | 
|  | *	the number of dsps to provide.  All their buffer space is allocated | 
|  | *	on open time.  The sonicvibes OSS routines we inherited really want | 
|  | *	power of 2 buffers, so we have all those next to each other, then | 
|  | *	512 byte regions for the recording wavecaches.  This ends up | 
|  | *	wasting quite a bit of memory.  The only fixes I can see would be | 
|  | *	getting a kernel allocator that could work in zones, or figuring out | 
|  | *	just how to coerce the WP into doing what we want. | 
|  | * | 
|  | *	The indirection of the various registers means we have to spinlock | 
|  | *	nearly all register accesses.  We have the main register indirection | 
|  | *	like the wave cache, maestro registers, etc.  Then we have beasts | 
|  | *	like the APU interface that is indirect registers gotten at through | 
|  | *	the main maestro indirection.  Ouch.  We spinlock around the actual | 
|  | *	ports on a per card basis.  This means spinlock activity at each IO | 
|  | *	operation, but the only IO operation clusters are in non critical | 
|  | *	paths and it makes the code far easier to follow.  Interrupts are | 
|  | *	blocked while holding the locks because the int handler has to | 
|  | *	get at some of them :(.  The mixer interface doesn't, however. | 
|  | *	We also have an OSS state lock that is thrown around in a few | 
|  | *	places. | 
|  | * | 
|  | *	This driver has brute force APM suspend support.  We catch suspend | 
|  | *	notifications and stop all work being done on the chip.  Any people | 
|  | *	that try between this shutdown and the real suspend operation will | 
|  | *	be put to sleep.  When we resume we restore our software state on | 
|  | *	the chip and wake up the people that were using it.  The code thats | 
|  | *	being used now is quite dirty and assumes we're on a uni-processor | 
|  | *	machine.  Much of it will need to be cleaned up for SMP ACPI or | 
|  | *	similar. | 
|  | * | 
|  | *	We also pay attention to PCI power management now.  The driver | 
|  | *	will power down units of the chip that it knows aren't needed. | 
|  | *	The WaveProcessor and company are only powered on when people | 
|  | *	have /dev/dsp*s open.  On removal the driver will | 
|  | *	power down the maestro entirely.  There could still be | 
|  | *	trouble with BIOSen that magically change power states | 
|  | *	themselves, but we'll see. | 
|  | * | 
|  | * History | 
|  | *  v0.15 - May 21 2001 - Marcus Meissner <mm@caldera.de> | 
|  | *      Ported to Linux 2.4 PCI API. Some clean ups, global devs list | 
|  | *      removed (now using pci device driver data). | 
|  | *      PM needs to be polished still. Bumped version. | 
|  | *  (still kind of v0.14) May 13 2001 - Ben Pfaff <pfaffben@msu.edu> | 
|  | *      Add support for 978 docking and basic hardware volume control | 
|  | *  (still kind of v0.14) Nov 23 - Alan Cox <alan@redhat.com> | 
|  | *	Add clocking= for people with seriously warped hardware | 
|  | *  (still v0.14) Nov 10 2000 - Bartlomiej Zolnierkiewicz <bkz@linux-ide.org> | 
|  | *	add __init to maestro_ac97_init() and maestro_install() | 
|  | *  (still based on v0.14) Mar 29 2000 - Zach Brown <zab@redhat.com> | 
|  | *	move to 2.3 power management interface, which | 
|  | *		required hacking some suspend/resume/check paths | 
|  | *	make static compilation work | 
|  | *  v0.14 - Jan 28 2000 - Zach Brown <zab@redhat.com> | 
|  | *	add PCI power management through ACPI regs. | 
|  | *	we now shut down on machine reboot/halt | 
|  | *	leave scary PCI config items alone (isa stuff, mostly) | 
|  | *	enable 1921s, it seems only mine was broke. | 
|  | *	fix swapped left/right pcm dac.  har har. | 
|  | *	up bob freq, increase buffers, fix pointers at underflow | 
|  | *	silly compilation problems | 
|  | *  v0.13 - Nov 18 1999 - Zach Brown <zab@redhat.com> | 
|  | *	fix nec Versas?  man would that be cool. | 
|  | *  v0.12 - Nov 12 1999 - Zach Brown <zab@redhat.com> | 
|  | *	brown bag volume max fix.. | 
|  | *  v0.11 - Nov 11 1999 - Zach Brown <zab@redhat.com> | 
|  | *	use proper stereo apu decoding, mmap/write should work. | 
|  | *	make volume sliders more useful, tweak rate calculation. | 
|  | *	fix lame 8bit format reporting bug.  duh. apm apu saving buglet also | 
|  | *	fix maestro 1 clock freq "bug", remove pt101 support | 
|  | *  v0.10 - Oct 28 1999 - Zach Brown <zab@redhat.com> | 
|  | *	aha, so, sometimes the WP writes a status word to offset 0 | 
|  | *	  from one of the PCMBARs.  rearrange allocation accordingly.. | 
|  | *	  cheers again to Eric for being a good hacker in investigating this. | 
|  | *	Jeroen Hoogervorst submits 7500 fix out of nowhere.  yay.  :) | 
|  | *  v0.09 - Oct 23 1999 - Zach Brown <zab@redhat.com> | 
|  | *	added APM support. | 
|  | *	re-order something such that some 2Es now work.  Magic! | 
|  | *	new codec reset routine.  made some codecs come to life. | 
|  | *	fix clear_advance, sync some control with ESS. | 
|  | *	now write to all base regs to be paranoid. | 
|  | *  v0.08 - Oct 20 1999 - Zach Brown <zab@redhat.com> | 
|  | *	Fix initial buflen bug.  I am so smart.  also smp compiling.. | 
|  | *	I owe Eric yet another beer: fixed recmask, igain, | 
|  | *	  muting, and adc sync consistency.  Go Team. | 
|  | *  v0.07 - Oct 4 1999 - Zach Brown <zab@redhat.com> | 
|  | *	tweak adc/dac, formating, and stuff to allow full duplex | 
|  | *	allocate dsps memory at open() so we can fit in the wavecache window | 
|  | *	fix wavecache braindamage.  again.  no more scribbling? | 
|  | *	fix ess 1921 codec bug on some laptops. | 
|  | *	fix dumb pci scanning bug | 
|  | *	started 2.3 cleanup, redid spinlocks, little cleanups | 
|  | *  v0.06 - Sep 20 1999 - Zach Brown <zab@redhat.com> | 
|  | *	fix wavecache thinkos.  limit to 1 /dev/dsp. | 
|  | *	eric is wearing his thinking toque this week. | 
|  | *		spotted apu mode bugs and gain ramping problem | 
|  | *	don't touch weird mixer regs, make recmask optional | 
|  | *	fixed igain inversion, defaults for mixers, clean up rec_start | 
|  | *	make mono recording work. | 
|  | *	report subsystem stuff, please send reports. | 
|  | *	littles: parallel out, amp now | 
|  | *  v0.05 - Sep 17 1999 - Zach Brown <zab@redhat.com> | 
|  | *	merged and fixed up Eric's initial recording code | 
|  | *	munged format handling to catch misuse, needs rewrite. | 
|  | *	revert ring bus init, fixup shared int, add pci busmaster setting | 
|  | *	fix mixer oss interface, fix mic mute and recmask | 
|  | *	mask off unsupported mixers, reset with all 1s, modularize defaults | 
|  | *	make sure bob is running while we need it | 
|  | *	got rid of device limit, initial minimal apm hooks | 
|  | *	pull out dead code/includes, only allow multimedia/audio maestros | 
|  | *  v0.04 - Sep 01 1999 - Zach Brown <zab@redhat.com> | 
|  | *	copied memory leak fix from sonicvibes driver | 
|  | *	different ac97 reset, play with 2.0 ac97, simplify ring bus setup | 
|  | *	bob freq code, region sanity, jitter sync fix; all from Eric | 
|  | * | 
|  | * TODO | 
|  | *	fix bob frequency | 
|  | *	endianness | 
|  | *	do smart things with ac97 2.0 bits. | 
|  | *	dual codecs | 
|  | *	leave 54->61 open | 
|  | * | 
|  | *	it also would be fun to have a mode that would not use pci dma at all | 
|  | *	but would copy into the wavecache on board memory and use that | 
|  | *	on architectures that don't like the maestro's pci dma ickiness. | 
|  | */ | 
|  |  | 
|  | /*****************************************************************************/ | 
|  |  | 
|  | #include <linux/module.h> | 
|  | #include <linux/sched.h> | 
|  | #include <linux/smp_lock.h> | 
|  | #include <linux/string.h> | 
|  | #include <linux/ctype.h> | 
|  | #include <linux/ioport.h> | 
|  | #include <linux/delay.h> | 
|  | #include <linux/sound.h> | 
|  | #include <linux/slab.h> | 
|  | #include <linux/soundcard.h> | 
|  | #include <linux/pci.h> | 
|  | #include <linux/spinlock.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/interrupt.h> | 
|  | #include <linux/poll.h> | 
|  | #include <linux/reboot.h> | 
|  | #include <linux/bitops.h> | 
|  | #include <linux/wait.h> | 
|  |  | 
|  | #include <asm/current.h> | 
|  | #include <asm/dma.h> | 
|  | #include <asm/io.h> | 
|  | #include <asm/page.h> | 
|  | #include <asm/uaccess.h> | 
|  |  | 
|  | #include <linux/pm.h> | 
|  | static int maestro_pm_callback(struct pm_dev *dev, pm_request_t rqst, void *d); | 
|  |  | 
|  | #include "maestro.h" | 
|  |  | 
|  | static struct pci_driver maestro_pci_driver; | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | #define M_DEBUG 1 | 
|  |  | 
|  | #ifdef M_DEBUG | 
|  | static int debug; | 
|  | #define M_printk(args...) {if (debug) printk(args);} | 
|  | #else | 
|  | #define M_printk(x) | 
|  | #endif | 
|  |  | 
|  | /* we try to setup 2^(dsps_order) /dev/dsp devices */ | 
|  | static int dsps_order; | 
|  | /* whether or not we mess around with power management */ | 
|  | static int use_pm=2; /* set to 1 for force */ | 
|  | /* clocking for broken hardware - a few laptops seem to use a 50Khz clock | 
|  | ie insmod with clocking=50000 or so */ | 
|  |  | 
|  | static int clocking=48000; | 
|  |  | 
|  | MODULE_AUTHOR("Zach Brown <zab@zabbo.net>, Alan Cox <alan@redhat.com>"); | 
|  | MODULE_DESCRIPTION("ESS Maestro Driver"); | 
|  | MODULE_LICENSE("GPL"); | 
|  |  | 
|  | #ifdef M_DEBUG | 
|  | module_param(debug, bool, 0644); | 
|  | #endif | 
|  | module_param(dsps_order, int, 0); | 
|  | module_param(use_pm, int, 0); | 
|  | module_param(clocking, int, 0); | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  | #define DRIVER_VERSION "0.15" | 
|  |  | 
|  | #ifndef PCI_VENDOR_ESS | 
|  | #define PCI_VENDOR_ESS			0x125D | 
|  | #define PCI_DEVICE_ID_ESS_ESS1968	0x1968		/* Maestro 2	*/ | 
|  | #define PCI_DEVICE_ID_ESS_ESS1978      	0x1978		/* Maestro 2E	*/ | 
|  |  | 
|  | #define PCI_VENDOR_ESS_OLD		0x1285		/* Platform Tech, | 
|  | the people the maestro | 
|  | was bought from */ | 
|  | #define PCI_DEVICE_ID_ESS_ESS0100	0x0100		/* maestro 1 */ | 
|  | #endif /* PCI_VENDOR_ESS */ | 
|  |  | 
|  | #define ESS_CHAN_HARD		0x100 | 
|  |  | 
|  | /* NEC Versas ? */ | 
|  | #define NEC_VERSA_SUBID1	0x80581033 | 
|  | #define NEC_VERSA_SUBID2	0x803c1033 | 
|  |  | 
|  |  | 
|  | /* changed so that I could actually find all the | 
|  | references and fix them up.  it's a little more readable now. */ | 
|  | #define ESS_FMT_STEREO	0x01 | 
|  | #define ESS_FMT_16BIT	0x02 | 
|  | #define ESS_FMT_MASK	0x03 | 
|  | #define ESS_DAC_SHIFT	0 | 
|  | #define ESS_ADC_SHIFT	4 | 
|  |  | 
|  | #define ESS_STATE_MAGIC		0x125D1968 | 
|  | #define ESS_CARD_MAGIC		0x19283746 | 
|  |  | 
|  | #define DAC_RUNNING		1 | 
|  | #define ADC_RUNNING		2 | 
|  |  | 
|  | #define MAX_DSP_ORDER	2 | 
|  | #define MAX_DSPS	(1<<MAX_DSP_ORDER) | 
|  | #define NR_DSPS		(1<<dsps_order) | 
|  | #define NR_IDRS		32 | 
|  |  | 
|  | #define NR_APUS		64 | 
|  | #define NR_APU_REGS	16 | 
|  |  | 
|  | /* acpi states */ | 
|  | enum { | 
|  | ACPI_D0=0, | 
|  | ACPI_D1, | 
|  | ACPI_D2, | 
|  | ACPI_D3 | 
|  | }; | 
|  |  | 
|  | /* bits in the acpi masks */ | 
|  | #define ACPI_12MHZ	( 1 << 15) | 
|  | #define ACPI_24MHZ	( 1 << 14) | 
|  | #define ACPI_978	( 1 << 13) | 
|  | #define ACPI_SPDIF	( 1 << 12) | 
|  | #define ACPI_GLUE	( 1 << 11) | 
|  | #define ACPI__10	( 1 << 10) /* reserved */ | 
|  | #define ACPI_PCIINT	( 1 << 9) | 
|  | #define ACPI_HV		( 1 << 8) /* hardware volume */ | 
|  | #define ACPI_GPIO	( 1 << 7) | 
|  | #define ACPI_ASSP	( 1 << 6) | 
|  | #define ACPI_SB		( 1 << 5) /* sb emul */ | 
|  | #define ACPI_FM		( 1 << 4) /* fm emul */ | 
|  | #define ACPI_RB		( 1 << 3) /* ringbus / aclink */ | 
|  | #define ACPI_MIDI	( 1 << 2) | 
|  | #define ACPI_GP		( 1 << 1) /* game port */ | 
|  | #define ACPI_WP		( 1 << 0) /* wave processor */ | 
|  |  | 
|  | #define ACPI_ALL	(0xffff) | 
|  | #define ACPI_SLEEP	(~(ACPI_SPDIF|ACPI_ASSP|ACPI_SB|ACPI_FM| \ | 
|  | ACPI_MIDI|ACPI_GP|ACPI_WP)) | 
|  | #define ACPI_NONE	(ACPI__10) | 
|  |  | 
|  | /* these masks indicate which units we care about at | 
|  | which states */ | 
|  | static u16 acpi_state_mask[] = { | 
|  | [ACPI_D0] = ACPI_ALL, | 
|  | [ACPI_D1] = ACPI_SLEEP, | 
|  | [ACPI_D2] = ACPI_SLEEP, | 
|  | [ACPI_D3] = ACPI_NONE | 
|  | }; | 
|  |  | 
|  | static char version[] __devinitdata = | 
|  | KERN_INFO "maestro: version " DRIVER_VERSION " time " __TIME__ " " __DATE__ "\n"; | 
|  |  | 
|  |  | 
|  |  | 
|  | static const unsigned sample_size[] = { 1, 2, 2, 4 }; | 
|  | static const unsigned sample_shift[] = { 0, 1, 1, 2 }; | 
|  |  | 
|  | enum card_types_t { | 
|  | TYPE_MAESTRO, | 
|  | TYPE_MAESTRO2, | 
|  | TYPE_MAESTRO2E | 
|  | }; | 
|  |  | 
|  | static const char *card_names[]={ | 
|  | [TYPE_MAESTRO] = "ESS Maestro", | 
|  | [TYPE_MAESTRO2] = "ESS Maestro 2", | 
|  | [TYPE_MAESTRO2E] = "ESS Maestro 2E" | 
|  | }; | 
|  |  | 
|  | static int clock_freq[]={ | 
|  | [TYPE_MAESTRO] = (49152000L / 1024L), | 
|  | [TYPE_MAESTRO2] = (50000000L / 1024L), | 
|  | [TYPE_MAESTRO2E] = (50000000L / 1024L) | 
|  | }; | 
|  |  | 
|  | static int maestro_notifier(struct notifier_block *nb, unsigned long event, void *buf); | 
|  |  | 
|  | static struct notifier_block maestro_nb = {maestro_notifier, NULL, 0}; | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | struct ess_state { | 
|  | unsigned int magic; | 
|  | /* FIXME: we probably want submixers in here, but only one record pair */ | 
|  | u8 apu[6];		/* l/r output, l/r intput converters, l/r input apus */ | 
|  | u8 apu_mode[6];		/* Running mode for this APU */ | 
|  | u8 apu_pan[6];		/* Panning setup for this APU */ | 
|  | u32 apu_base[6];	/* base address for this apu */ | 
|  | struct ess_card *card;	/* Card info */ | 
|  | /* wave stuff */ | 
|  | unsigned int rateadc, ratedac; | 
|  | unsigned char fmt, enable; | 
|  |  | 
|  | int index; | 
|  |  | 
|  | /* this locks around the oss state in the driver */ | 
|  | spinlock_t lock; | 
|  | /* only let 1 be opening at a time */ | 
|  | struct semaphore open_sem; | 
|  | wait_queue_head_t open_wait; | 
|  | mode_t open_mode; | 
|  |  | 
|  | /* soundcore stuff */ | 
|  | int dev_audio; | 
|  |  | 
|  | struct dmabuf { | 
|  | void *rawbuf; | 
|  | unsigned buforder; | 
|  | unsigned numfrag; | 
|  | unsigned fragshift; | 
|  | /* XXX zab - swptr only in here so that it can be referenced by | 
|  | clear_advance, as far as I can tell :( */ | 
|  | unsigned hwptr, swptr; | 
|  | unsigned total_bytes; | 
|  | int count; | 
|  | unsigned error; /* over/underrun */ | 
|  | wait_queue_head_t wait; | 
|  | /* redundant, but makes calculations easier */ | 
|  | unsigned fragsize; | 
|  | unsigned dmasize; | 
|  | unsigned fragsamples; | 
|  | /* OSS stuff */ | 
|  | unsigned mapped:1; | 
|  | unsigned ready:1;	/* our oss buffers are ready to go */ | 
|  | unsigned endcleared:1; | 
|  | unsigned ossfragshift; | 
|  | int ossmaxfrags; | 
|  | unsigned subdivision; | 
|  | u16 base;		/* Offset for ptr */ | 
|  | } dma_dac, dma_adc; | 
|  |  | 
|  | /* pointer to each dsp?s piece of the apu->src buffer page */ | 
|  | void *mixbuf; | 
|  |  | 
|  | }; | 
|  |  | 
|  | struct ess_card { | 
|  | unsigned int magic; | 
|  |  | 
|  | /* We keep maestro cards in a linked list */ | 
|  | struct ess_card *next; | 
|  |  | 
|  | int dev_mixer; | 
|  |  | 
|  | int card_type; | 
|  |  | 
|  | /* as most of this is static, | 
|  | perhaps it should be a pointer to a global struct */ | 
|  | struct mixer_goo { | 
|  | int modcnt; | 
|  | int supported_mixers; | 
|  | int stereo_mixers; | 
|  | int record_sources; | 
|  | /* the caller must guarantee arg sanity before calling these */ | 
|  | /*		int (*read_mixer)(struct ess_card *card, int index);*/ | 
|  | void (*write_mixer)(struct ess_card *card,int mixer, unsigned int left,unsigned int right); | 
|  | int (*recmask_io)(struct ess_card *card,int rw,int mask); | 
|  | unsigned int mixer_state[SOUND_MIXER_NRDEVICES]; | 
|  | } mix; | 
|  |  | 
|  | int power_regs; | 
|  |  | 
|  | int in_suspend; | 
|  | wait_queue_head_t suspend_queue; | 
|  |  | 
|  | struct ess_state channels[MAX_DSPS]; | 
|  | u16 maestro_map[NR_IDRS];	/* Register map */ | 
|  | /* we have to store this junk so that we can come back from a | 
|  | suspend */ | 
|  | u16 apu_map[NR_APUS][NR_APU_REGS];	/* contents of apu regs */ | 
|  |  | 
|  | /* this locks around the physical registers on the card */ | 
|  | spinlock_t lock; | 
|  |  | 
|  | /* memory for this card.. wavecache limited :(*/ | 
|  | void *dmapages; | 
|  | int dmaorder; | 
|  |  | 
|  | /* hardware resources */ | 
|  | struct pci_dev *pcidev; | 
|  | u32 iobase; | 
|  | u32 irq; | 
|  |  | 
|  | int bob_freq; | 
|  | char dsps_open; | 
|  |  | 
|  | int dock_mute_vol; | 
|  | }; | 
|  |  | 
|  | static void set_mixer(struct ess_card *card,unsigned int mixer, unsigned int val ); | 
|  |  | 
|  | static unsigned | 
|  | ld2(unsigned int x) | 
|  | { | 
|  | unsigned r = 0; | 
|  |  | 
|  | if (x >= 0x10000) { | 
|  | x >>= 16; | 
|  | r += 16; | 
|  | } | 
|  | if (x >= 0x100) { | 
|  | x >>= 8; | 
|  | r += 8; | 
|  | } | 
|  | if (x >= 0x10) { | 
|  | x >>= 4; | 
|  | r += 4; | 
|  | } | 
|  | if (x >= 4) { | 
|  | x >>= 2; | 
|  | r += 2; | 
|  | } | 
|  | if (x >= 2) | 
|  | r++; | 
|  | return r; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | static void check_suspend(struct ess_card *card); | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  |  | 
|  | /* | 
|  | *	ESS Maestro AC97 codec programming interface. | 
|  | */ | 
|  |  | 
|  | static void maestro_ac97_set(struct ess_card *card, u8 cmd, u16 val) | 
|  | { | 
|  | int io = card->iobase; | 
|  | int i; | 
|  | /* | 
|  | *	Wait for the codec bus to be free | 
|  | */ | 
|  |  | 
|  | check_suspend(card); | 
|  |  | 
|  | for(i=0;i<10000;i++) | 
|  | { | 
|  | if(!(inb(io+ESS_AC97_INDEX)&1)) | 
|  | break; | 
|  | } | 
|  | /* | 
|  | *	Write the bus | 
|  | */ | 
|  | outw(val, io+ESS_AC97_DATA); | 
|  | mdelay(1); | 
|  | outb(cmd, io+ESS_AC97_INDEX); | 
|  | mdelay(1); | 
|  | } | 
|  |  | 
|  | static u16 maestro_ac97_get(struct ess_card *card, u8 cmd) | 
|  | { | 
|  | int io = card->iobase; | 
|  | int sanity=10000; | 
|  | u16 data; | 
|  | int i; | 
|  |  | 
|  | check_suspend(card); | 
|  | /* | 
|  | *	Wait for the codec bus to be free | 
|  | */ | 
|  |  | 
|  | for(i=0;i<10000;i++) | 
|  | { | 
|  | if(!(inb(io+ESS_AC97_INDEX)&1)) | 
|  | break; | 
|  | } | 
|  |  | 
|  | outb(cmd|0x80, io+ESS_AC97_INDEX); | 
|  | mdelay(1); | 
|  |  | 
|  | while(inb(io+ESS_AC97_INDEX)&1) | 
|  | { | 
|  | sanity--; | 
|  | if(!sanity) | 
|  | { | 
|  | printk(KERN_ERR "maestro: ac97 codec timeout reading 0x%x.\n",cmd); | 
|  | return 0; | 
|  | } | 
|  | } | 
|  | data=inw(io+ESS_AC97_DATA); | 
|  | mdelay(1); | 
|  | return data; | 
|  | } | 
|  |  | 
|  | /* OSS interface to the ac97s.. */ | 
|  |  | 
|  | #define AC97_STEREO_MASK (SOUND_MASK_VOLUME|\ | 
|  | SOUND_MASK_PCM|SOUND_MASK_LINE|SOUND_MASK_CD|\ | 
|  | SOUND_MASK_VIDEO|SOUND_MASK_LINE1|SOUND_MASK_IGAIN) | 
|  |  | 
|  | #define AC97_SUPPORTED_MASK (AC97_STEREO_MASK | \ | 
|  | SOUND_MASK_BASS|SOUND_MASK_TREBLE|SOUND_MASK_MIC|\ | 
|  | SOUND_MASK_SPEAKER) | 
|  |  | 
|  | #define AC97_RECORD_MASK (SOUND_MASK_MIC|\ | 
|  | SOUND_MASK_CD| SOUND_MASK_VIDEO| SOUND_MASK_LINE1| SOUND_MASK_LINE|\ | 
|  | SOUND_MASK_PHONEIN) | 
|  |  | 
|  | #define supported_mixer(CARD,FOO) ( CARD->mix.supported_mixers & (1<<FOO) ) | 
|  |  | 
|  | /* this table has default mixer values for all OSS mixers. | 
|  | be sure to fill it in if you add oss mixers | 
|  | to anyone's supported mixer defines */ | 
|  |  | 
|  | static unsigned int mixer_defaults[SOUND_MIXER_NRDEVICES] = { | 
|  | [SOUND_MIXER_VOLUME] =          0x3232, | 
|  | [SOUND_MIXER_BASS] =            0x3232, | 
|  | [SOUND_MIXER_TREBLE] =          0x3232, | 
|  | [SOUND_MIXER_SPEAKER] =         0x3232, | 
|  | [SOUND_MIXER_MIC] =     0x8000, /* annoying */ | 
|  | [SOUND_MIXER_LINE] =    0x3232, | 
|  | [SOUND_MIXER_CD] =      0x3232, | 
|  | [SOUND_MIXER_VIDEO] =   0x3232, | 
|  | [SOUND_MIXER_LINE1] =   0x3232, | 
|  | [SOUND_MIXER_PCM] =             0x3232, | 
|  | [SOUND_MIXER_IGAIN] =           0x3232 | 
|  | }; | 
|  |  | 
|  | static struct ac97_mixer_hw { | 
|  | unsigned char offset; | 
|  | int scale; | 
|  | } ac97_hw[SOUND_MIXER_NRDEVICES]= { | 
|  | [SOUND_MIXER_VOLUME]	=	{0x02,63}, | 
|  | [SOUND_MIXER_BASS]	=	{0x08,15}, | 
|  | [SOUND_MIXER_TREBLE]	=	{0x08,15}, | 
|  | [SOUND_MIXER_SPEAKER]	=	{0x0a,15}, | 
|  | [SOUND_MIXER_MIC]	=	{0x0e,31}, | 
|  | [SOUND_MIXER_LINE]	=	{0x10,31}, | 
|  | [SOUND_MIXER_CD]	=	{0x12,31}, | 
|  | [SOUND_MIXER_VIDEO]	=	{0x14,31}, | 
|  | [SOUND_MIXER_LINE1]	=	{0x16,31}, | 
|  | [SOUND_MIXER_PCM]	=	{0x18,31}, | 
|  | [SOUND_MIXER_IGAIN]	=	{0x1c,15} | 
|  | }; | 
|  |  | 
|  | #if 0 /* *shrug* removed simply because we never used it. | 
|  | feel free to implement again if needed */ | 
|  |  | 
|  | /* reads the given OSS mixer from the ac97 | 
|  | the caller must have insured that the ac97 knows | 
|  | about that given mixer, and should be holding a | 
|  | spinlock for the card */ | 
|  | static int ac97_read_mixer(struct ess_card *card, int mixer) | 
|  | { | 
|  | u16 val; | 
|  | int ret=0; | 
|  | struct ac97_mixer_hw *mh = &ac97_hw[mixer]; | 
|  |  | 
|  | val = maestro_ac97_get(card, mh->offset); | 
|  |  | 
|  | if(AC97_STEREO_MASK & (1<<mixer)) { | 
|  | /* nice stereo mixers .. */ | 
|  | int left,right; | 
|  |  | 
|  | left = (val >> 8)  & 0x7f; | 
|  | right = val  & 0x7f; | 
|  |  | 
|  | if (mixer == SOUND_MIXER_IGAIN) { | 
|  | right = (right * 100) / mh->scale; | 
|  | left = (left * 100) / mh->scale; | 
|  | } else { | 
|  | right = 100 - ((right * 100) / mh->scale); | 
|  | left = 100 - ((left * 100) / mh->scale); | 
|  | } | 
|  |  | 
|  | ret = left | (right << 8); | 
|  | } else if (mixer == SOUND_MIXER_SPEAKER) { | 
|  | ret = 100 - ((((val & 0x1e)>>1) * 100) / mh->scale); | 
|  | } else if (mixer == SOUND_MIXER_MIC) { | 
|  | ret = 100 - (((val & 0x1f) * 100) / mh->scale); | 
|  | /*  the low bit is optional in the tone sliders and masking | 
|  | it lets is avoid the 0xf 'bypass'.. */ | 
|  | } else if (mixer == SOUND_MIXER_BASS) { | 
|  | ret = 100 - ((((val >> 8) & 0xe) * 100) / mh->scale); | 
|  | } else if (mixer == SOUND_MIXER_TREBLE) { | 
|  | ret = 100 - (((val & 0xe) * 100) / mh->scale); | 
|  | } | 
|  |  | 
|  | M_printk("read mixer %d (0x%x) %x -> %x\n",mixer,mh->offset,val,ret); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /* write the OSS encoded volume to the given OSS encoded mixer, | 
|  | again caller's job to make sure all is well in arg land, | 
|  | call with spinlock held */ | 
|  |  | 
|  | /* linear scale -> log */ | 
|  | static unsigned char lin2log[101] = | 
|  | { | 
|  | 0, 0 , 15 , 23 , 30 , 34 , 38 , 42 , 45 , 47 , | 
|  | 50 , 52 , 53 , 55 , 57 , 58 , 60 , 61 , 62 , | 
|  | 63 , 65 , 66 , 67 , 68 , 69 , 69 , 70 , 71 , | 
|  | 72 , 73 , 73 , 74 , 75 , 75 , 76 , 77 , 77 , | 
|  | 78 , 78 , 79 , 80 , 80 , 81 , 81 , 82 , 82 , | 
|  | 83 , 83 , 84 , 84 , 84 , 85 , 85 , 86 , 86 , | 
|  | 87 , 87 , 87 , 88 , 88 , 88 , 89 , 89 , 89 , | 
|  | 90 , 90 , 90 , 91 , 91 , 91 , 92 , 92 , 92 , | 
|  | 93 , 93 , 93 , 94 , 94 , 94 , 94 , 95 , 95 , | 
|  | 95 , 95 , 96 , 96 , 96 , 96 , 97 , 97 , 97 , | 
|  | 97 , 98 , 98 , 98 , 98 , 99 , 99 , 99 , 99 , 99 | 
|  | }; | 
|  |  | 
|  | static void ac97_write_mixer(struct ess_card *card,int mixer, unsigned int left, unsigned int right) | 
|  | { | 
|  | u16 val=0; | 
|  | struct ac97_mixer_hw *mh = &ac97_hw[mixer]; | 
|  |  | 
|  | M_printk("wrote mixer %d (0x%x) %d,%d",mixer,mh->offset,left,right); | 
|  |  | 
|  | if(AC97_STEREO_MASK & (1<<mixer)) { | 
|  | /* stereo mixers, mute them if we can */ | 
|  |  | 
|  | if (mixer == SOUND_MIXER_IGAIN) { | 
|  | /* igain's slider is reversed.. */ | 
|  | right = (right * mh->scale) / 100; | 
|  | left = (left * mh->scale) / 100; | 
|  | if ((left == 0) && (right == 0)) | 
|  | val |= 0x8000; | 
|  | } else if (mixer == SOUND_MIXER_PCM || mixer == SOUND_MIXER_CD) { | 
|  | /* log conversion seems bad for them */ | 
|  | if ((left == 0) && (right == 0)) | 
|  | val = 0x8000; | 
|  | right = ((100 - right) * mh->scale) / 100; | 
|  | left = ((100 - left) * mh->scale) / 100; | 
|  | } else { | 
|  | /* log conversion for the stereo controls */ | 
|  | if((left == 0) && (right == 0)) | 
|  | val = 0x8000; | 
|  | right = ((100 - lin2log[right]) * mh->scale) / 100; | 
|  | left = ((100 - lin2log[left]) * mh->scale) / 100; | 
|  | } | 
|  |  | 
|  | val |= (left << 8) | right; | 
|  |  | 
|  | } else if (mixer == SOUND_MIXER_SPEAKER) { | 
|  | val = (((100 - left) * mh->scale) / 100) << 1; | 
|  | } else if (mixer == SOUND_MIXER_MIC) { | 
|  | val = maestro_ac97_get(card, mh->offset) & ~0x801f; | 
|  | val |= (((100 - left) * mh->scale) / 100); | 
|  | /*  the low bit is optional in the tone sliders and masking | 
|  | it lets is avoid the 0xf 'bypass'.. */ | 
|  | } else if (mixer == SOUND_MIXER_BASS) { | 
|  | val = maestro_ac97_get(card , mh->offset) & ~0x0f00; | 
|  | val |= ((((100 - left) * mh->scale) / 100) << 8) & 0x0e00; | 
|  | } else if (mixer == SOUND_MIXER_TREBLE)  { | 
|  | val = maestro_ac97_get(card , mh->offset) & ~0x000f; | 
|  | val |= (((100 - left) * mh->scale) / 100) & 0x000e; | 
|  | } | 
|  |  | 
|  | maestro_ac97_set(card , mh->offset, val); | 
|  |  | 
|  | M_printk(" -> %x\n",val); | 
|  | } | 
|  |  | 
|  | /* the following tables allow us to go from | 
|  | OSS <-> ac97 quickly. */ | 
|  |  | 
|  | enum ac97_recsettings { | 
|  | AC97_REC_MIC=0, | 
|  | AC97_REC_CD, | 
|  | AC97_REC_VIDEO, | 
|  | AC97_REC_AUX, | 
|  | AC97_REC_LINE, | 
|  | AC97_REC_STEREO, /* combination of all enabled outputs..  */ | 
|  | AC97_REC_MONO,        /*.. or the mono equivalent */ | 
|  | AC97_REC_PHONE | 
|  | }; | 
|  |  | 
|  | static unsigned int ac97_oss_mask[] = { | 
|  | [AC97_REC_MIC] = SOUND_MASK_MIC, | 
|  | [AC97_REC_CD] = SOUND_MASK_CD, | 
|  | [AC97_REC_VIDEO] = SOUND_MASK_VIDEO, | 
|  | [AC97_REC_AUX] = SOUND_MASK_LINE1, | 
|  | [AC97_REC_LINE] = SOUND_MASK_LINE, | 
|  | [AC97_REC_PHONE] = SOUND_MASK_PHONEIN | 
|  | }; | 
|  |  | 
|  | /* indexed by bit position */ | 
|  | static unsigned int ac97_oss_rm[] = { | 
|  | [SOUND_MIXER_MIC] = AC97_REC_MIC, | 
|  | [SOUND_MIXER_CD] = AC97_REC_CD, | 
|  | [SOUND_MIXER_VIDEO] = AC97_REC_VIDEO, | 
|  | [SOUND_MIXER_LINE1] = AC97_REC_AUX, | 
|  | [SOUND_MIXER_LINE] = AC97_REC_LINE, | 
|  | [SOUND_MIXER_PHONEIN] = AC97_REC_PHONE | 
|  | }; | 
|  |  | 
|  | /* read or write the recmask | 
|  | the ac97 can really have left and right recording | 
|  | inputs independently set, but OSS doesn't seem to | 
|  | want us to express that to the user. | 
|  | the caller guarantees that we have a supported bit set, | 
|  | and they must be holding the card's spinlock */ | 
|  | static int | 
|  | ac97_recmask_io(struct ess_card *card, int read, int mask) | 
|  | { | 
|  | unsigned int val = ac97_oss_mask[ maestro_ac97_get(card, 0x1a) & 0x7 ]; | 
|  |  | 
|  | if (read) return val; | 
|  |  | 
|  | /* oss can have many inputs, maestro can't.  try | 
|  | to pick the 'new' one */ | 
|  |  | 
|  | if (mask != val) mask &= ~val; | 
|  |  | 
|  | val = ffs(mask) - 1; | 
|  | val = ac97_oss_rm[val]; | 
|  | val |= val << 8;  /* set both channels */ | 
|  |  | 
|  | M_printk("maestro: setting ac97 recmask to 0x%x\n",val); | 
|  |  | 
|  | maestro_ac97_set(card,0x1a,val); | 
|  |  | 
|  | return 0; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | *	The Maestro can be wired to a standard AC97 compliant codec | 
|  | *	(see www.intel.com for the pdf's on this), or to a PT101 codec | 
|  | *	which appears to be the ES1918 (data sheet on the esstech.com.tw site) | 
|  | * | 
|  | *	The PT101 setup is untested. | 
|  | */ | 
|  |  | 
|  | static u16 __init maestro_ac97_init(struct ess_card *card) | 
|  | { | 
|  | u16 vend1, vend2, caps; | 
|  |  | 
|  | card->mix.supported_mixers = AC97_SUPPORTED_MASK; | 
|  | card->mix.stereo_mixers = AC97_STEREO_MASK; | 
|  | card->mix.record_sources = AC97_RECORD_MASK; | 
|  | /*	card->mix.read_mixer = ac97_read_mixer;*/ | 
|  | card->mix.write_mixer = ac97_write_mixer; | 
|  | card->mix.recmask_io = ac97_recmask_io; | 
|  |  | 
|  | vend1 = maestro_ac97_get(card, 0x7c); | 
|  | vend2 = maestro_ac97_get(card, 0x7e); | 
|  |  | 
|  | caps = maestro_ac97_get(card, 0x00); | 
|  |  | 
|  | printk(KERN_INFO "maestro: AC97 Codec detected: v: 0x%2x%2x caps: 0x%x pwr: 0x%x\n", | 
|  | vend1,vend2,caps,maestro_ac97_get(card,0x26) & 0xf); | 
|  |  | 
|  | if (! (caps & 0x4) ) { | 
|  | /* no bass/treble nobs */ | 
|  | card->mix.supported_mixers &= ~(SOUND_MASK_BASS|SOUND_MASK_TREBLE); | 
|  | } | 
|  |  | 
|  | /* XXX endianness, dork head. */ | 
|  | /* vendor specifc bits.. */ | 
|  | switch ((long)(vend1 << 16) | vend2) { | 
|  | case 0x545200ff:	/* TriTech */ | 
|  | /* no idea what this does */ | 
|  | maestro_ac97_set(card,0x2a,0x0001); | 
|  | maestro_ac97_set(card,0x2c,0x0000); | 
|  | maestro_ac97_set(card,0x2c,0xffff); | 
|  | break; | 
|  | #if 0	/* i thought the problems I was seeing were with | 
|  | the 1921, but apparently they were with the pci board | 
|  | it was on, so this code is commented out. | 
|  | lets see if this holds true. */ | 
|  | case 0x83847609:	/* ESS 1921 */ | 
|  | /* writing to 0xe (mic) or 0x1a (recmask) seems | 
|  | to hang this codec */ | 
|  | card->mix.supported_mixers &= ~(SOUND_MASK_MIC); | 
|  | card->mix.record_sources = 0; | 
|  | card->mix.recmask_io = NULL; | 
|  | #if 0	/* don't ask.  I have yet to see what these actually do. */ | 
|  | maestro_ac97_set(card,0x76,0xABBA); /* o/~ Take a chance on me o/~ */ | 
|  | udelay(20); | 
|  | maestro_ac97_set(card,0x78,0x3002); | 
|  | udelay(20); | 
|  | maestro_ac97_set(card,0x78,0x3802); | 
|  | udelay(20); | 
|  | #endif | 
|  | break; | 
|  | #endif | 
|  | default: break; | 
|  | } | 
|  |  | 
|  | maestro_ac97_set(card, 0x1E, 0x0404); | 
|  | /* null misc stuff */ | 
|  | maestro_ac97_set(card, 0x20, 0x0000); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #if 0  /* there has been 1 person on the planet with a pt101 that we | 
|  | know of.  If they care, they can put this back in :) */ | 
|  | static u16 maestro_pt101_init(struct ess_card *card,int iobase) | 
|  | { | 
|  | printk(KERN_INFO "maestro: PT101 Codec detected, initializing but _not_ installing mixer device.\n"); | 
|  | /* who knows.. */ | 
|  | maestro_ac97_set(iobase, 0x2A, 0x0001); | 
|  | maestro_ac97_set(iobase, 0x2C, 0x0000); | 
|  | maestro_ac97_set(iobase, 0x2C, 0xFFFF); | 
|  | maestro_ac97_set(iobase, 0x10, 0x9F1F); | 
|  | maestro_ac97_set(iobase, 0x12, 0x0808); | 
|  | maestro_ac97_set(iobase, 0x14, 0x9F1F); | 
|  | maestro_ac97_set(iobase, 0x16, 0x9F1F); | 
|  | maestro_ac97_set(iobase, 0x18, 0x0404); | 
|  | maestro_ac97_set(iobase, 0x1A, 0x0000); | 
|  | maestro_ac97_set(iobase, 0x1C, 0x0000); | 
|  | maestro_ac97_set(iobase, 0x02, 0x0404); | 
|  | maestro_ac97_set(iobase, 0x04, 0x0808); | 
|  | maestro_ac97_set(iobase, 0x0C, 0x801F); | 
|  | maestro_ac97_set(iobase, 0x0E, 0x801F); | 
|  | return 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /* this is very magic, and very slow.. */ | 
|  | static void | 
|  | maestro_ac97_reset(int ioaddr, struct pci_dev *pcidev) | 
|  | { | 
|  | u16 save_68; | 
|  | u16 w; | 
|  | u32 vend; | 
|  |  | 
|  | outw( inw(ioaddr + 0x38) & 0xfffc, ioaddr + 0x38); | 
|  | outw( inw(ioaddr + 0x3a) & 0xfffc, ioaddr + 0x3a); | 
|  | outw( inw(ioaddr + 0x3c) & 0xfffc, ioaddr + 0x3c); | 
|  |  | 
|  | /* reset the first codec */ | 
|  | outw(0x0000,  ioaddr+0x36); | 
|  | save_68 = inw(ioaddr+0x68); | 
|  | pci_read_config_word(pcidev, 0x58, &w);	/* something magical with gpio and bus arb. */ | 
|  | pci_read_config_dword(pcidev, PCI_SUBSYSTEM_VENDOR_ID, &vend); | 
|  | if( w & 0x1) | 
|  | save_68 |= 0x10; | 
|  | outw(0xfffe, ioaddr + 0x64);	/* tickly gpio 0.. */ | 
|  | outw(0x0001, ioaddr + 0x68); | 
|  | outw(0x0000, ioaddr + 0x60); | 
|  | udelay(20); | 
|  | outw(0x0001, ioaddr + 0x60); | 
|  | mdelay(20); | 
|  |  | 
|  | outw(save_68 | 0x1, ioaddr + 0x68);	/* now restore .. */ | 
|  | outw( (inw(ioaddr + 0x38) & 0xfffc)|0x1, ioaddr + 0x38); | 
|  | outw( (inw(ioaddr + 0x3a) & 0xfffc)|0x1, ioaddr + 0x3a); | 
|  | outw( (inw(ioaddr + 0x3c) & 0xfffc)|0x1, ioaddr + 0x3c); | 
|  |  | 
|  | /* now the second codec */ | 
|  | outw(0x0000,  ioaddr+0x36); | 
|  | outw(0xfff7, ioaddr + 0x64); | 
|  | save_68 = inw(ioaddr+0x68); | 
|  | outw(0x0009, ioaddr + 0x68); | 
|  | outw(0x0001, ioaddr + 0x60); | 
|  | udelay(20); | 
|  | outw(0x0009, ioaddr + 0x60); | 
|  | mdelay(500);	/* .. ouch.. */ | 
|  | outw( inw(ioaddr + 0x38) & 0xfffc, ioaddr + 0x38); | 
|  | outw( inw(ioaddr + 0x3a) & 0xfffc, ioaddr + 0x3a); | 
|  | outw( inw(ioaddr + 0x3c) & 0xfffc, ioaddr + 0x3c); | 
|  |  | 
|  | #if 0 /* the loop here needs to be much better if we want it.. */ | 
|  | M_printk("trying software reset\n"); | 
|  | /* try and do a software reset */ | 
|  | outb(0x80|0x7c, ioaddr + 0x30); | 
|  | for (w=0; ; w++) { | 
|  | if ((inw(ioaddr+ 0x30) & 1) == 0) { | 
|  | if(inb(ioaddr + 0x32) !=0) break; | 
|  |  | 
|  | outb(0x80|0x7d, ioaddr + 0x30); | 
|  | if (((inw(ioaddr+ 0x30) & 1) == 0) && (inb(ioaddr + 0x32) !=0)) break; | 
|  | outb(0x80|0x7f, ioaddr + 0x30); | 
|  | if (((inw(ioaddr+ 0x30) & 1) == 0) && (inb(ioaddr + 0x32) !=0)) break; | 
|  | } | 
|  |  | 
|  | if( w > 10000) { | 
|  | outb( inb(ioaddr + 0x37) | 0x08, ioaddr + 0x37);  /* do a software reset */ | 
|  | mdelay(500); /* oh my.. */ | 
|  | outb( inb(ioaddr + 0x37) & ~0x08, ioaddr + 0x37); | 
|  | udelay(1); | 
|  | outw( 0x80, ioaddr+0x30); | 
|  | for(w = 0 ; w < 10000; w++) { | 
|  | if((inw(ioaddr + 0x30) & 1) ==0) break; | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif | 
|  | if ( vend == NEC_VERSA_SUBID1 || vend == NEC_VERSA_SUBID2) { | 
|  | /* turn on external amp? */ | 
|  | outw(0xf9ff, ioaddr + 0x64); | 
|  | outw(inw(ioaddr+0x68) | 0x600, ioaddr + 0x68); | 
|  | outw(0x0209, ioaddr + 0x60); | 
|  | } | 
|  |  | 
|  | /* Turn on the 978 docking chip. | 
|  | First frob the "master output enable" bit, | 
|  | then set most of the playback volume control registers to max. */ | 
|  | outb(inb(ioaddr+0xc0)|(1<<5), ioaddr+0xc0); | 
|  | outb(0xff, ioaddr+0xc3); | 
|  | outb(0xff, ioaddr+0xc4); | 
|  | outb(0xff, ioaddr+0xc6); | 
|  | outb(0xff, ioaddr+0xc8); | 
|  | outb(0x3f, ioaddr+0xcf); | 
|  | outb(0x3f, ioaddr+0xd0); | 
|  | } | 
|  | /* | 
|  | *	Indirect register access. Not all registers are readable so we | 
|  | *	need to keep register state ourselves | 
|  | */ | 
|  |  | 
|  | #define WRITEABLE_MAP	0xEFFFFF | 
|  | #define READABLE_MAP	0x64003F | 
|  |  | 
|  | /* | 
|  | *	The Maestro engineers were a little indirection happy. These indirected | 
|  | *	registers themselves include indirect registers at another layer | 
|  | */ | 
|  |  | 
|  | static void __maestro_write(struct ess_card *card, u16 reg, u16 data) | 
|  | { | 
|  | long ioaddr = card->iobase; | 
|  |  | 
|  | outw(reg, ioaddr+0x02); | 
|  | outw(data, ioaddr+0x00); | 
|  | if( reg >= NR_IDRS) printk("maestro: IDR %d out of bounds!\n",reg); | 
|  | else card->maestro_map[reg]=data; | 
|  |  | 
|  | } | 
|  |  | 
|  | static void maestro_write(struct ess_state *s, u16 reg, u16 data) | 
|  | { | 
|  | unsigned long flags; | 
|  |  | 
|  | check_suspend(s->card); | 
|  | spin_lock_irqsave(&s->card->lock,flags); | 
|  |  | 
|  | __maestro_write(s->card,reg,data); | 
|  |  | 
|  | spin_unlock_irqrestore(&s->card->lock,flags); | 
|  | } | 
|  |  | 
|  | static u16 __maestro_read(struct ess_card *card, u16 reg) | 
|  | { | 
|  | long ioaddr = card->iobase; | 
|  |  | 
|  | outw(reg, ioaddr+0x02); | 
|  | return card->maestro_map[reg]=inw(ioaddr+0x00); | 
|  | } | 
|  |  | 
|  | static u16 maestro_read(struct ess_state *s, u16 reg) | 
|  | { | 
|  | if(READABLE_MAP & (1<<reg)) | 
|  | { | 
|  | unsigned long flags; | 
|  | check_suspend(s->card); | 
|  | spin_lock_irqsave(&s->card->lock,flags); | 
|  |  | 
|  | __maestro_read(s->card,reg); | 
|  |  | 
|  | spin_unlock_irqrestore(&s->card->lock,flags); | 
|  | } | 
|  | return s->card->maestro_map[reg]; | 
|  | } | 
|  |  | 
|  | /* | 
|  | *	These routines handle accessing the second level indirections to the | 
|  | *	wave ram. | 
|  | */ | 
|  |  | 
|  | /* | 
|  | *	The register names are the ones ESS uses (see 104T31.ZIP) | 
|  | */ | 
|  |  | 
|  | #define IDR0_DATA_PORT		0x00 | 
|  | #define IDR1_CRAM_POINTER	0x01 | 
|  | #define IDR2_CRAM_DATA		0x02 | 
|  | #define IDR3_WAVE_DATA		0x03 | 
|  | #define IDR4_WAVE_PTR_LOW	0x04 | 
|  | #define IDR5_WAVE_PTR_HI	0x05 | 
|  | #define IDR6_TIMER_CTRL		0x06 | 
|  | #define IDR7_WAVE_ROMRAM	0x07 | 
|  |  | 
|  | static void apu_index_set(struct ess_card *card, u16 index) | 
|  | { | 
|  | int i; | 
|  | __maestro_write(card, IDR1_CRAM_POINTER, index); | 
|  | for(i=0;i<1000;i++) | 
|  | if(__maestro_read(card, IDR1_CRAM_POINTER)==index) | 
|  | return; | 
|  | printk(KERN_WARNING "maestro: APU register select failed.\n"); | 
|  | } | 
|  |  | 
|  | static void apu_data_set(struct ess_card *card, u16 data) | 
|  | { | 
|  | int i; | 
|  | for(i=0;i<1000;i++) | 
|  | { | 
|  | if(__maestro_read(card, IDR0_DATA_PORT)==data) | 
|  | return; | 
|  | __maestro_write(card, IDR0_DATA_PORT, data); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | *	This is the public interface for APU manipulation. It handles the | 
|  | *	interlock to avoid two APU writes in parallel etc. Don't diddle | 
|  | *	directly with the stuff above. | 
|  | */ | 
|  |  | 
|  | static void apu_set_register(struct ess_state *s, u16 channel, u8 reg, u16 data) | 
|  | { | 
|  | unsigned long flags; | 
|  |  | 
|  | check_suspend(s->card); | 
|  |  | 
|  | if(channel&ESS_CHAN_HARD) | 
|  | channel&=~ESS_CHAN_HARD; | 
|  | else | 
|  | { | 
|  | if(channel>5) | 
|  | printk("BAD CHANNEL %d.\n",channel); | 
|  | else | 
|  | channel = s->apu[channel]; | 
|  | /* store based on real hardware apu/reg */ | 
|  | s->card->apu_map[channel][reg]=data; | 
|  | } | 
|  | reg|=(channel<<4); | 
|  |  | 
|  | /* hooray for double indirection!! */ | 
|  | spin_lock_irqsave(&s->card->lock,flags); | 
|  |  | 
|  | apu_index_set(s->card, reg); | 
|  | apu_data_set(s->card, data); | 
|  |  | 
|  | spin_unlock_irqrestore(&s->card->lock,flags); | 
|  | } | 
|  |  | 
|  | static u16 apu_get_register(struct ess_state *s, u16 channel, u8 reg) | 
|  | { | 
|  | unsigned long flags; | 
|  | u16 v; | 
|  |  | 
|  | check_suspend(s->card); | 
|  |  | 
|  | if(channel&ESS_CHAN_HARD) | 
|  | channel&=~ESS_CHAN_HARD; | 
|  | else | 
|  | channel = s->apu[channel]; | 
|  |  | 
|  | reg|=(channel<<4); | 
|  |  | 
|  | spin_lock_irqsave(&s->card->lock,flags); | 
|  |  | 
|  | apu_index_set(s->card, reg); | 
|  | v=__maestro_read(s->card, IDR0_DATA_PORT); | 
|  |  | 
|  | spin_unlock_irqrestore(&s->card->lock,flags); | 
|  | return v; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* | 
|  | *	The wavecache buffers between the APUs and | 
|  | *	pci bus mastering | 
|  | */ | 
|  |  | 
|  | static void wave_set_register(struct ess_state *s, u16 reg, u16 value) | 
|  | { | 
|  | long ioaddr = s->card->iobase; | 
|  | unsigned long flags; | 
|  | check_suspend(s->card); | 
|  |  | 
|  | spin_lock_irqsave(&s->card->lock,flags); | 
|  |  | 
|  | outw(reg, ioaddr+0x10); | 
|  | outw(value, ioaddr+0x12); | 
|  |  | 
|  | spin_unlock_irqrestore(&s->card->lock,flags); | 
|  | } | 
|  |  | 
|  | static u16 wave_get_register(struct ess_state *s, u16 reg) | 
|  | { | 
|  | long ioaddr = s->card->iobase; | 
|  | unsigned long flags; | 
|  | u16 value; | 
|  | check_suspend(s->card); | 
|  |  | 
|  | spin_lock_irqsave(&s->card->lock,flags); | 
|  | outw(reg, ioaddr+0x10); | 
|  | value=inw(ioaddr+0x12); | 
|  | spin_unlock_irqrestore(&s->card->lock,flags); | 
|  |  | 
|  | return value; | 
|  | } | 
|  |  | 
|  | static void sound_reset(int ioaddr) | 
|  | { | 
|  | outw(0x2000, 0x18+ioaddr); | 
|  | udelay(1); | 
|  | outw(0x0000, 0x18+ioaddr); | 
|  | udelay(1); | 
|  | } | 
|  |  | 
|  | /* sets the play formats of these apus, should be passed the already shifted format */ | 
|  | static void set_apu_fmt(struct ess_state *s, int apu, int mode) | 
|  | { | 
|  | int apu_fmt = 0x10; | 
|  |  | 
|  | if(!(mode&ESS_FMT_16BIT)) apu_fmt+=0x20; | 
|  | if((mode&ESS_FMT_STEREO)) apu_fmt+=0x10; | 
|  | s->apu_mode[apu]   = apu_fmt; | 
|  | s->apu_mode[apu+1] = apu_fmt; | 
|  | } | 
|  |  | 
|  | /* this only fixes the output apu mode to be later set by start_dac and | 
|  | company.  output apu modes are set in ess_rec_setup */ | 
|  | static void set_fmt(struct ess_state *s, unsigned char mask, unsigned char data) | 
|  | { | 
|  | s->fmt = (s->fmt & mask) | data; | 
|  | set_apu_fmt(s, 0, (s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_MASK); | 
|  | } | 
|  |  | 
|  | /* this is off by a little bit.. */ | 
|  | static u32 compute_rate(struct ess_state *s, u32 freq) | 
|  | { | 
|  | u32 clock = clock_freq[s->card->card_type]; | 
|  |  | 
|  | freq = (freq * clocking)/48000; | 
|  |  | 
|  | if (freq == 48000) | 
|  | return 0x10000; | 
|  |  | 
|  | return ((freq / clock) <<16 )+ | 
|  | (((freq % clock) << 16) / clock); | 
|  | } | 
|  |  | 
|  | static void set_dac_rate(struct ess_state *s, unsigned int rate) | 
|  | { | 
|  | u32 freq; | 
|  | int fmt = (s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_MASK; | 
|  |  | 
|  | if (rate > 48000) | 
|  | rate = 48000; | 
|  | if (rate < 4000) | 
|  | rate = 4000; | 
|  |  | 
|  | s->ratedac = rate; | 
|  |  | 
|  | if(! (fmt & ESS_FMT_16BIT) && !(fmt & ESS_FMT_STEREO)) | 
|  | rate >>= 1; | 
|  |  | 
|  | /*	M_printk("computing dac rate %d with mode %d\n",rate,s->fmt);*/ | 
|  |  | 
|  | freq = compute_rate(s, rate); | 
|  |  | 
|  | /* Load the frequency, turn on 6dB */ | 
|  | apu_set_register(s, 0, 2,(apu_get_register(s, 0, 2)&0x00FF)| | 
|  | ( ((freq&0xFF)<<8)|0x10 )); | 
|  | apu_set_register(s, 0, 3, freq>>8); | 
|  | apu_set_register(s, 1, 2,(apu_get_register(s, 1, 2)&0x00FF)| | 
|  | ( ((freq&0xFF)<<8)|0x10 )); | 
|  | apu_set_register(s, 1, 3, freq>>8); | 
|  | } | 
|  |  | 
|  | static void set_adc_rate(struct ess_state *s, unsigned rate) | 
|  | { | 
|  | u32 freq; | 
|  |  | 
|  | /* Sample Rate conversion APUs don't like 0x10000 for their rate */ | 
|  | if (rate > 47999) | 
|  | rate = 47999; | 
|  | if (rate < 4000) | 
|  | rate = 4000; | 
|  |  | 
|  | s->rateadc = rate; | 
|  |  | 
|  | freq = compute_rate(s, rate); | 
|  |  | 
|  | /* Load the frequency, turn on 6dB */ | 
|  | apu_set_register(s, 2, 2,(apu_get_register(s, 2, 2)&0x00FF)| | 
|  | ( ((freq&0xFF)<<8)|0x10 )); | 
|  | apu_set_register(s, 2, 3, freq>>8); | 
|  | apu_set_register(s, 3, 2,(apu_get_register(s, 3, 2)&0x00FF)| | 
|  | ( ((freq&0xFF)<<8)|0x10 )); | 
|  | apu_set_register(s, 3, 3, freq>>8); | 
|  |  | 
|  | /* fix mixer rate at 48khz.  and its _must_ be 0x10000. */ | 
|  | freq = 0x10000; | 
|  |  | 
|  | apu_set_register(s, 4, 2,(apu_get_register(s, 4, 2)&0x00FF)| | 
|  | ( ((freq&0xFF)<<8)|0x10 )); | 
|  | apu_set_register(s, 4, 3, freq>>8); | 
|  | apu_set_register(s, 5, 2,(apu_get_register(s, 5, 2)&0x00FF)| | 
|  | ( ((freq&0xFF)<<8)|0x10 )); | 
|  | apu_set_register(s, 5, 3, freq>>8); | 
|  | } | 
|  |  | 
|  | /* Stop our host of recording apus */ | 
|  | static inline void stop_adc(struct ess_state *s) | 
|  | { | 
|  | /* XXX lets hope we don't have to lock around this */ | 
|  | if (! (s->enable & ADC_RUNNING)) return; | 
|  |  | 
|  | s->enable &= ~ADC_RUNNING; | 
|  | apu_set_register(s, 2, 0, apu_get_register(s, 2, 0)&0xFF0F); | 
|  | apu_set_register(s, 3, 0, apu_get_register(s, 3, 0)&0xFF0F); | 
|  | apu_set_register(s, 4, 0, apu_get_register(s, 2, 0)&0xFF0F); | 
|  | apu_set_register(s, 5, 0, apu_get_register(s, 3, 0)&0xFF0F); | 
|  | } | 
|  |  | 
|  | /* stop output apus */ | 
|  | static void stop_dac(struct ess_state *s) | 
|  | { | 
|  | /* XXX have to lock around this? */ | 
|  | if (! (s->enable & DAC_RUNNING)) return; | 
|  |  | 
|  | s->enable &= ~DAC_RUNNING; | 
|  | apu_set_register(s, 0, 0, apu_get_register(s, 0, 0)&0xFF0F); | 
|  | apu_set_register(s, 1, 0, apu_get_register(s, 1, 0)&0xFF0F); | 
|  | } | 
|  |  | 
|  | static void start_dac(struct ess_state *s) | 
|  | { | 
|  | /* XXX locks? */ | 
|  | if (	(s->dma_dac.mapped || s->dma_dac.count > 0) && | 
|  | s->dma_dac.ready && | 
|  | (! (s->enable & DAC_RUNNING)) ) { | 
|  |  | 
|  | s->enable |= DAC_RUNNING; | 
|  |  | 
|  | apu_set_register(s, 0, 0, | 
|  | (apu_get_register(s, 0, 0)&0xFF0F)|s->apu_mode[0]); | 
|  |  | 
|  | if((s->fmt >> ESS_DAC_SHIFT)  & ESS_FMT_STEREO) | 
|  | apu_set_register(s, 1, 0, | 
|  | (apu_get_register(s, 1, 0)&0xFF0F)|s->apu_mode[1]); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void start_adc(struct ess_state *s) | 
|  | { | 
|  | /* XXX locks? */ | 
|  | if ((s->dma_adc.mapped || s->dma_adc.count < (signed)(s->dma_adc.dmasize - 2*s->dma_adc.fragsize)) | 
|  | && s->dma_adc.ready && (! (s->enable & ADC_RUNNING)) ) { | 
|  |  | 
|  | s->enable |= ADC_RUNNING; | 
|  | apu_set_register(s, 2, 0, | 
|  | (apu_get_register(s, 2, 0)&0xFF0F)|s->apu_mode[2]); | 
|  | apu_set_register(s, 4, 0, | 
|  | (apu_get_register(s, 4, 0)&0xFF0F)|s->apu_mode[4]); | 
|  |  | 
|  | if( s->fmt & (ESS_FMT_STEREO << ESS_ADC_SHIFT)) { | 
|  | apu_set_register(s, 3, 0, | 
|  | (apu_get_register(s, 3, 0)&0xFF0F)|s->apu_mode[3]); | 
|  | apu_set_register(s, 5, 0, | 
|  | (apu_get_register(s, 5, 0)&0xFF0F)|s->apu_mode[5]); | 
|  | } | 
|  |  | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | /* | 
|  | *	Native play back driver | 
|  | */ | 
|  |  | 
|  | /* the mode passed should be already shifted and masked */ | 
|  | static void | 
|  | ess_play_setup(struct ess_state *ess, int mode, u32 rate, void *buffer, int size) | 
|  | { | 
|  | u32 pa; | 
|  | u32 tmpval; | 
|  | int high_apu = 0; | 
|  | int channel; | 
|  |  | 
|  | M_printk("mode=%d rate=%d buf=%p len=%d.\n", | 
|  | mode, rate, buffer, size); | 
|  |  | 
|  | /* all maestro sizes are in 16bit words */ | 
|  | size >>=1; | 
|  |  | 
|  | if(mode&ESS_FMT_STEREO) { | 
|  | high_apu++; | 
|  | /* only 16/stereo gets size divided */ | 
|  | if(mode&ESS_FMT_16BIT) | 
|  | size>>=1; | 
|  | } | 
|  |  | 
|  | for(channel=0; channel <= high_apu; channel++) | 
|  | { | 
|  | pa = virt_to_bus(buffer); | 
|  |  | 
|  | /* set the wavecache control reg */ | 
|  | tmpval = (pa - 0x10) & 0xFFF8; | 
|  | if(!(mode & ESS_FMT_16BIT)) tmpval |= 4; | 
|  | if(mode & ESS_FMT_STEREO) tmpval |= 2; | 
|  | ess->apu_base[channel]=tmpval; | 
|  | wave_set_register(ess, ess->apu[channel]<<3, tmpval); | 
|  |  | 
|  | pa -= virt_to_bus(ess->card->dmapages); | 
|  | pa>>=1; /* words */ | 
|  |  | 
|  | /* base offset of dma calcs when reading the pointer | 
|  | on the left one */ | 
|  | if(!channel) ess->dma_dac.base = pa&0xFFFF; | 
|  |  | 
|  | pa|=0x00400000;			/* System RAM */ | 
|  |  | 
|  | /* XXX the 16bit here might not be needed.. */ | 
|  | if((mode & ESS_FMT_STEREO) && (mode & ESS_FMT_16BIT)) { | 
|  | if(channel) | 
|  | pa|=0x00800000;			/* Stereo */ | 
|  | pa>>=1; | 
|  | } | 
|  |  | 
|  | /* XXX think about endianess when writing these registers */ | 
|  | M_printk("maestro: ess_play_setup: APU[%d] pa = 0x%x\n", ess->apu[channel], pa); | 
|  | /* start of sample */ | 
|  | apu_set_register(ess, channel, 4, ((pa>>16)&0xFF)<<8); | 
|  | apu_set_register(ess, channel, 5, pa&0xFFFF); | 
|  | /* sample end */ | 
|  | apu_set_register(ess, channel, 6, (pa+size)&0xFFFF); | 
|  | /* setting loop len == sample len */ | 
|  | apu_set_register(ess, channel, 7, size); | 
|  |  | 
|  | /* clear effects/env.. */ | 
|  | apu_set_register(ess, channel, 8, 0x0000); | 
|  | /* set amp now to 0xd0 (?), low byte is 'amplitude dest'? */ | 
|  | apu_set_register(ess, channel, 9, 0xD000); | 
|  |  | 
|  | /* clear routing stuff */ | 
|  | apu_set_register(ess, channel, 11, 0x0000); | 
|  | /* dma on, no envelopes, filter to all 1s) */ | 
|  | apu_set_register(ess, channel, 0, 0x400F); | 
|  |  | 
|  | if(mode&ESS_FMT_16BIT) | 
|  | ess->apu_mode[channel]=0x10; | 
|  | else | 
|  | ess->apu_mode[channel]=0x30; | 
|  |  | 
|  | if(mode&ESS_FMT_STEREO) { | 
|  | /* set panning: left or right */ | 
|  | apu_set_register(ess, channel, 10, 0x8F00 | (channel ? 0 : 0x10)); | 
|  | ess->apu_mode[channel] += 0x10; | 
|  | } else | 
|  | apu_set_register(ess, channel, 10, 0x8F08); | 
|  | } | 
|  |  | 
|  | /* clear WP interrupts */ | 
|  | outw(1, ess->card->iobase+0x04); | 
|  | /* enable WP ints */ | 
|  | outw(inw(ess->card->iobase+0x18)|4, ess->card->iobase+0x18); | 
|  |  | 
|  | /* go team! */ | 
|  | set_dac_rate(ess,rate); | 
|  | start_dac(ess); | 
|  | } | 
|  |  | 
|  | /* | 
|  | *	Native record driver | 
|  | */ | 
|  |  | 
|  | /* again, passed mode is alrady shifted/masked */ | 
|  | static void | 
|  | ess_rec_setup(struct ess_state *ess, int mode, u32 rate, void *buffer, int size) | 
|  | { | 
|  | int apu_step = 2; | 
|  | int channel; | 
|  |  | 
|  | M_printk("maestro: ess_rec_setup: mode=%d rate=%d buf=0x%p len=%d.\n", | 
|  | mode, rate, buffer, size); | 
|  |  | 
|  | /* all maestro sizes are in 16bit words */ | 
|  | size >>=1; | 
|  |  | 
|  | /* we're given the full size of the buffer, but | 
|  | in stereo each channel will only use its half */ | 
|  | if(mode&ESS_FMT_STEREO) { | 
|  | size >>=1; | 
|  | apu_step = 1; | 
|  | } | 
|  |  | 
|  | /* APU assignments: 2 = mono/left SRC | 
|  | 3 = right SRC | 
|  | 4 = mono/left Input Mixer | 
|  | 5 = right Input Mixer */ | 
|  | for(channel=2;channel<6;channel+=apu_step) | 
|  | { | 
|  | int i; | 
|  | int bsize, route; | 
|  | u32 pa; | 
|  | u32 tmpval; | 
|  |  | 
|  | /* data seems to flow from the codec, through an apu into | 
|  | the 'mixbuf' bit of page, then through the SRC apu | 
|  | and out to the real 'buffer'.  ok.  sure.  */ | 
|  |  | 
|  | if(channel & 0x04) { | 
|  | /* ok, we're an input mixer going from adc | 
|  | through the mixbuf to the other apus */ | 
|  |  | 
|  | if(!(channel & 0x01)) { | 
|  | pa = virt_to_bus(ess->mixbuf); | 
|  | } else { | 
|  | pa = virt_to_bus(ess->mixbuf + (PAGE_SIZE >> 4)); | 
|  | } | 
|  |  | 
|  | /* we source from a 'magic' apu */ | 
|  | bsize = PAGE_SIZE >> 5;	/* half of this channels alloc, in words */ | 
|  | route = 0x14 + (channel - 4); /* parallel in crap, see maestro reg 0xC [8-11] */ | 
|  | ess->apu_mode[channel] = 0x90;  /* Input Mixer */ | 
|  |  | 
|  | } else { | 
|  | /* we're a rate converter taking | 
|  | input from the input apus and outputing it to | 
|  | system memory */ | 
|  | if(!(channel & 0x01))  { | 
|  | pa = virt_to_bus(buffer); | 
|  | } else { | 
|  | /* right channel records its split half. | 
|  | *2 accommodates for rampant shifting earlier */ | 
|  | pa = virt_to_bus(buffer + size*2); | 
|  | } | 
|  |  | 
|  | ess->apu_mode[channel] = 0xB0;  /* Sample Rate Converter */ | 
|  |  | 
|  | bsize = size; | 
|  | /* get input from inputing apu */ | 
|  | route = channel + 2; | 
|  | } | 
|  |  | 
|  | M_printk("maestro: ess_rec_setup: getting pa 0x%x from %d\n",pa,channel); | 
|  |  | 
|  | /* set the wavecache control reg */ | 
|  | tmpval = (pa - 0x10) & 0xFFF8; | 
|  | ess->apu_base[channel]=tmpval; | 
|  | wave_set_register(ess, ess->apu[channel]<<3, tmpval); | 
|  |  | 
|  | pa -= virt_to_bus(ess->card->dmapages); | 
|  | pa>>=1; /* words */ | 
|  |  | 
|  | /* base offset of dma calcs when reading the pointer | 
|  | on this left one */ | 
|  | if(channel==2) ess->dma_adc.base = pa&0xFFFF; | 
|  |  | 
|  | pa|=0x00400000;			/* bit 22 -> System RAM */ | 
|  |  | 
|  | M_printk("maestro: ess_rec_setup: APU[%d] pa = 0x%x size = 0x%x route = 0x%x\n", | 
|  | ess->apu[channel], pa, bsize, route); | 
|  |  | 
|  | /* Begin loading the APU */ | 
|  | for(i=0;i<15;i++)		/* clear all PBRs */ | 
|  | apu_set_register(ess, channel, i, 0x0000); | 
|  |  | 
|  | apu_set_register(ess, channel, 0, 0x400F); | 
|  |  | 
|  | /* need to enable subgroups.. and we should probably | 
|  | have different groups for different /dev/dsps..  */ | 
|  | apu_set_register(ess, channel, 2, 0x8); | 
|  |  | 
|  | /* Load the buffer into the wave engine */ | 
|  | apu_set_register(ess, channel, 4, ((pa>>16)&0xFF)<<8); | 
|  | /* XXX reg is little endian.. */ | 
|  | apu_set_register(ess, channel, 5, pa&0xFFFF); | 
|  | apu_set_register(ess, channel, 6, (pa+bsize)&0xFFFF); | 
|  | apu_set_register(ess, channel, 7, bsize); | 
|  |  | 
|  | /* clear effects/env.. */ | 
|  | apu_set_register(ess, channel, 8, 0x00F0); | 
|  |  | 
|  | /* amplitude now?  sure.  why not.  */ | 
|  | apu_set_register(ess, channel, 9, 0x0000); | 
|  |  | 
|  | /* set filter tune, radius, polar pan */ | 
|  | apu_set_register(ess, channel, 10, 0x8F08); | 
|  |  | 
|  | /* route input */ | 
|  | apu_set_register(ess, channel, 11, route); | 
|  | } | 
|  |  | 
|  | /* clear WP interrupts */ | 
|  | outw(1, ess->card->iobase+0x04); | 
|  | /* enable WP ints */ | 
|  | outw(inw(ess->card->iobase+0x18)|4, ess->card->iobase+0x18); | 
|  |  | 
|  | /* let 'er rip */ | 
|  | set_adc_rate(ess,rate); | 
|  | start_adc(ess); | 
|  | } | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | static void set_dmaa(struct ess_state *s, unsigned int addr, unsigned int count) | 
|  | { | 
|  | M_printk("set_dmaa??\n"); | 
|  | } | 
|  |  | 
|  | static void set_dmac(struct ess_state *s, unsigned int addr, unsigned int count) | 
|  | { | 
|  | M_printk("set_dmac??\n"); | 
|  | } | 
|  |  | 
|  | /* Playback pointer */ | 
|  | static inline unsigned get_dmaa(struct ess_state *s) | 
|  | { | 
|  | int offset; | 
|  |  | 
|  | offset = apu_get_register(s,0,5); | 
|  |  | 
|  | /*	M_printk("dmaa: offset: %d, base: %d\n",offset,s->dma_dac.base); */ | 
|  |  | 
|  | offset-=s->dma_dac.base; | 
|  |  | 
|  | return (offset&0xFFFE)<<1; /* hardware is in words */ | 
|  | } | 
|  |  | 
|  | /* Record pointer */ | 
|  | static inline unsigned get_dmac(struct ess_state *s) | 
|  | { | 
|  | int offset; | 
|  |  | 
|  | offset = apu_get_register(s,2,5); | 
|  |  | 
|  | /*	M_printk("dmac: offset: %d, base: %d\n",offset,s->dma_adc.base); */ | 
|  |  | 
|  | /* The offset is an address not a position relative to base */ | 
|  | offset-=s->dma_adc.base; | 
|  |  | 
|  | return (offset&0xFFFE)<<1; /* hardware is in words */ | 
|  | } | 
|  |  | 
|  | /* | 
|  | *	Meet Bob, the timer... | 
|  | */ | 
|  |  | 
|  | static irqreturn_t ess_interrupt(int irq, void *dev_id, struct pt_regs *regs); | 
|  |  | 
|  | static void stop_bob(struct ess_state *s) | 
|  | { | 
|  | /* Mask IDR 11,17 */ | 
|  | maestro_write(s,  0x11, maestro_read(s, 0x11)&~1); | 
|  | maestro_write(s,  0x17, maestro_read(s, 0x17)&~1); | 
|  | } | 
|  |  | 
|  | /* eventually we could be clever and limit bob ints | 
|  | to the frequency at which our smallest duration | 
|  | chunks may expire */ | 
|  | #define ESS_SYSCLK	50000000 | 
|  | static void start_bob(struct ess_state *s) | 
|  | { | 
|  | int prescale; | 
|  | int divide; | 
|  |  | 
|  | /* XXX make freq selector much smarter, see calc_bob_rate */ | 
|  | int freq = 200; | 
|  |  | 
|  | /* compute ideal interrupt frequency for buffer size & play rate */ | 
|  | /* first, find best prescaler value to match freq */ | 
|  | for(prescale=5;prescale<12;prescale++) | 
|  | if(freq > (ESS_SYSCLK>>(prescale+9))) | 
|  | break; | 
|  |  | 
|  | /* next, back off prescaler whilst getting divider into optimum range */ | 
|  | divide=1; | 
|  | while((prescale > 5) && (divide<32)) | 
|  | { | 
|  | prescale--; | 
|  | divide <<=1; | 
|  | } | 
|  | divide>>=1; | 
|  |  | 
|  | /* now fine-tune the divider for best match */ | 
|  | for(;divide<31;divide++) | 
|  | if(freq >= ((ESS_SYSCLK>>(prescale+9))/(divide+1))) | 
|  | break; | 
|  |  | 
|  | /* divide = 0 is illegal, but don't let prescale = 4! */ | 
|  | if(divide == 0) | 
|  | { | 
|  | divide++; | 
|  | if(prescale>5) | 
|  | prescale--; | 
|  | } | 
|  |  | 
|  | maestro_write(s, 6, 0x9000 | (prescale<<5) | divide); /* set reg */ | 
|  |  | 
|  | /* Now set IDR 11/17 */ | 
|  | maestro_write(s, 0x11, maestro_read(s, 0x11)|1); | 
|  | maestro_write(s, 0x17, maestro_read(s, 0x17)|1); | 
|  | } | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | /* this quickly calculates the frequency needed for bob | 
|  | and sets it if its different than what bob is | 
|  | currently running at.  its called often so | 
|  | needs to be fairly quick. */ | 
|  | #define BOB_MIN 50 | 
|  | #define BOB_MAX 400 | 
|  | static void calc_bob_rate(struct ess_state *s) { | 
|  | #if 0 /* this thing tries to set the frequency of bob such that | 
|  | there are 2 interrupts / buffer walked by the dac/adc.  That | 
|  | is probably very wrong for people who actually care about | 
|  | mid buffer positioning.  it should be calculated as bytes/interrupt | 
|  | and that needs to be decided :)  so for now just use the static 150 | 
|  | in start_bob.*/ | 
|  |  | 
|  | unsigned int dac_rate=2,adc_rate=1,newrate; | 
|  | static int israte=-1; | 
|  |  | 
|  | if (s->dma_dac.fragsize == 0) dac_rate = BOB_MIN; | 
|  | else  { | 
|  | dac_rate =	(2 * s->ratedac * sample_size[(s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_MASK]) / | 
|  | (s->dma_dac.fragsize) ; | 
|  | } | 
|  |  | 
|  | if (s->dma_adc.fragsize == 0) adc_rate = BOB_MIN; | 
|  | else { | 
|  | adc_rate =	(2 * s->rateadc * sample_size[(s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_MASK]) / | 
|  | (s->dma_adc.fragsize) ; | 
|  | } | 
|  |  | 
|  | if(dac_rate > adc_rate) newrate = adc_rate; | 
|  | else newrate=dac_rate; | 
|  |  | 
|  | if(newrate > BOB_MAX) newrate = BOB_MAX; | 
|  | else { | 
|  | if(newrate < BOB_MIN) | 
|  | newrate = BOB_MIN; | 
|  | } | 
|  |  | 
|  | if( israte != newrate) { | 
|  | printk("dac: %d  adc: %d rate: %d\n",dac_rate,adc_rate,israte); | 
|  | israte=newrate; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | } | 
|  |  | 
|  | static int | 
|  | prog_dmabuf(struct ess_state *s, unsigned rec) | 
|  | { | 
|  | struct dmabuf *db = rec ? &s->dma_adc : &s->dma_dac; | 
|  | unsigned rate = rec ? s->rateadc : s->ratedac; | 
|  | unsigned bytepersec; | 
|  | unsigned bufs; | 
|  | unsigned char fmt; | 
|  | unsigned long flags; | 
|  |  | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | fmt = s->fmt; | 
|  | if (rec) { | 
|  | stop_adc(s); | 
|  | fmt >>= ESS_ADC_SHIFT; | 
|  | } else { | 
|  | stop_dac(s); | 
|  | fmt >>= ESS_DAC_SHIFT; | 
|  | } | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | fmt &= ESS_FMT_MASK; | 
|  |  | 
|  | db->hwptr = db->swptr = db->total_bytes = db->count = db->error = db->endcleared = 0; | 
|  |  | 
|  | /* this algorithm is a little nuts.. where did /1000 come from? */ | 
|  | bytepersec = rate << sample_shift[fmt]; | 
|  | bufs = PAGE_SIZE << db->buforder; | 
|  | if (db->ossfragshift) { | 
|  | if ((1000 << db->ossfragshift) < bytepersec) | 
|  | db->fragshift = ld2(bytepersec/1000); | 
|  | else | 
|  | db->fragshift = db->ossfragshift; | 
|  | } else { | 
|  | db->fragshift = ld2(bytepersec/100/(db->subdivision ? db->subdivision : 1)); | 
|  | if (db->fragshift < 3) | 
|  | db->fragshift = 3; | 
|  | } | 
|  | db->numfrag = bufs >> db->fragshift; | 
|  | while (db->numfrag < 4 && db->fragshift > 3) { | 
|  | db->fragshift--; | 
|  | db->numfrag = bufs >> db->fragshift; | 
|  | } | 
|  | db->fragsize = 1 << db->fragshift; | 
|  | if (db->ossmaxfrags >= 4 && db->ossmaxfrags < db->numfrag) | 
|  | db->numfrag = db->ossmaxfrags; | 
|  | db->fragsamples = db->fragsize >> sample_shift[fmt]; | 
|  | db->dmasize = db->numfrag << db->fragshift; | 
|  |  | 
|  | M_printk("maestro: setup oss: numfrag: %d fragsize: %d dmasize: %d\n",db->numfrag,db->fragsize,db->dmasize); | 
|  |  | 
|  | memset(db->rawbuf, (fmt & ESS_FMT_16BIT) ? 0 : 0x80, db->dmasize); | 
|  |  | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | if (rec) | 
|  | ess_rec_setup(s, fmt, s->rateadc, db->rawbuf, db->dmasize); | 
|  | else | 
|  | ess_play_setup(s, fmt, s->ratedac, db->rawbuf, db->dmasize); | 
|  |  | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | db->ready = 1; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static __inline__ void | 
|  | clear_advance(struct ess_state *s) | 
|  | { | 
|  | unsigned char c = ((s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_16BIT) ? 0 : 0x80; | 
|  |  | 
|  | unsigned char *buf = s->dma_dac.rawbuf; | 
|  | unsigned bsize = s->dma_dac.dmasize; | 
|  | unsigned bptr = s->dma_dac.swptr; | 
|  | unsigned len = s->dma_dac.fragsize; | 
|  |  | 
|  | if (bptr + len > bsize) { | 
|  | unsigned x = bsize - bptr; | 
|  | memset(buf + bptr, c, x); | 
|  | /* account for wrapping? */ | 
|  | bptr = 0; | 
|  | len -= x; | 
|  | } | 
|  | memset(buf + bptr, c, len); | 
|  | } | 
|  |  | 
|  | /* call with spinlock held! */ | 
|  | static void | 
|  | ess_update_ptr(struct ess_state *s) | 
|  | { | 
|  | unsigned hwptr; | 
|  | int diff; | 
|  |  | 
|  | /* update ADC pointer */ | 
|  | if (s->dma_adc.ready) { | 
|  | /* oh boy should this all be re-written.  everything in the current code paths think | 
|  | that the various counters/pointers are expressed in bytes to the user but we have | 
|  | two apus doing stereo stuff so we fix it up here.. it propagates to all the various | 
|  | counters from here.  */ | 
|  | if ( s->fmt & (ESS_FMT_STEREO << ESS_ADC_SHIFT)) { | 
|  | hwptr = (get_dmac(s)*2) % s->dma_adc.dmasize; | 
|  | } else { | 
|  | hwptr = get_dmac(s) % s->dma_adc.dmasize; | 
|  | } | 
|  | diff = (s->dma_adc.dmasize + hwptr - s->dma_adc.hwptr) % s->dma_adc.dmasize; | 
|  | s->dma_adc.hwptr = hwptr; | 
|  | s->dma_adc.total_bytes += diff; | 
|  | s->dma_adc.count += diff; | 
|  | if (s->dma_adc.count >= (signed)s->dma_adc.fragsize) | 
|  | wake_up(&s->dma_adc.wait); | 
|  | if (!s->dma_adc.mapped) { | 
|  | if (s->dma_adc.count > (signed)(s->dma_adc.dmasize - ((3 * s->dma_adc.fragsize) >> 1))) { | 
|  | /* FILL ME | 
|  | wrindir(s, SV_CIENABLE, s->enable); */ | 
|  | stop_adc(s); | 
|  | /* brute force everyone back in sync, sigh */ | 
|  | s->dma_adc.count = 0; | 
|  | s->dma_adc.swptr = 0; | 
|  | s->dma_adc.hwptr = 0; | 
|  | s->dma_adc.error++; | 
|  | } | 
|  | } | 
|  | } | 
|  | /* update DAC pointer */ | 
|  | if (s->dma_dac.ready) { | 
|  | hwptr = get_dmaa(s) % s->dma_dac.dmasize; | 
|  | /* the apu only reports the length it has seen, not the | 
|  | length of the memory that has been used (the WP | 
|  | knows that) */ | 
|  | if ( ((s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_MASK) == (ESS_FMT_STEREO|ESS_FMT_16BIT)) | 
|  | hwptr<<=1; | 
|  |  | 
|  | diff = (s->dma_dac.dmasize + hwptr - s->dma_dac.hwptr) % s->dma_dac.dmasize; | 
|  | /*		M_printk("updating dac: hwptr: %d diff: %d\n",hwptr,diff);*/ | 
|  | s->dma_dac.hwptr = hwptr; | 
|  | s->dma_dac.total_bytes += diff; | 
|  | if (s->dma_dac.mapped) { | 
|  | s->dma_dac.count += diff; | 
|  | if (s->dma_dac.count >= (signed)s->dma_dac.fragsize) { | 
|  | wake_up(&s->dma_dac.wait); | 
|  | } | 
|  | } else { | 
|  | s->dma_dac.count -= diff; | 
|  | /*			M_printk("maestro: ess_update_ptr: diff: %d, count: %d\n", diff, s->dma_dac.count); */ | 
|  | if (s->dma_dac.count <= 0) { | 
|  | M_printk("underflow! diff: %d count: %d hw: %d sw: %d\n", diff, s->dma_dac.count, | 
|  | hwptr, s->dma_dac.swptr); | 
|  | /* FILL ME | 
|  | wrindir(s, SV_CIENABLE, s->enable); */ | 
|  | /* XXX how on earth can calling this with the lock held work.. */ | 
|  | stop_dac(s); | 
|  | /* brute force everyone back in sync, sigh */ | 
|  | s->dma_dac.count = 0; | 
|  | s->dma_dac.swptr = hwptr; | 
|  | s->dma_dac.error++; | 
|  | } else if (s->dma_dac.count <= (signed)s->dma_dac.fragsize && !s->dma_dac.endcleared) { | 
|  | clear_advance(s); | 
|  | s->dma_dac.endcleared = 1; | 
|  | } | 
|  | if (s->dma_dac.count + (signed)s->dma_dac.fragsize <= (signed)s->dma_dac.dmasize) { | 
|  | wake_up(&s->dma_dac.wait); | 
|  | /*				printk("waking up DAC count: %d sw: %d hw: %d\n",s->dma_dac.count, s->dma_dac.swptr, | 
|  | hwptr);*/ | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static irqreturn_t | 
|  | ess_interrupt(int irq, void *dev_id, struct pt_regs *regs) | 
|  | { | 
|  | struct ess_state *s; | 
|  | struct ess_card *c = (struct ess_card *)dev_id; | 
|  | int i; | 
|  | u32 event; | 
|  |  | 
|  | if ( ! (event = inb(c->iobase+0x1A)) ) | 
|  | return IRQ_NONE; | 
|  |  | 
|  | outw(inw(c->iobase+4)&1, c->iobase+4); | 
|  |  | 
|  | /*	M_printk("maestro int: %x\n",event);*/ | 
|  | if(event&(1<<6)) | 
|  | { | 
|  | int x; | 
|  | enum {UP_EVT, DOWN_EVT, MUTE_EVT} vol_evt; | 
|  | int volume; | 
|  |  | 
|  | /* Figure out which volume control button was pushed, | 
|  | based on differences from the default register | 
|  | values. */ | 
|  | x = inb(c->iobase+0x1c); | 
|  | if (x&1) vol_evt = MUTE_EVT; | 
|  | else if (((x>>1)&7) > 4) vol_evt = UP_EVT; | 
|  | else vol_evt = DOWN_EVT; | 
|  |  | 
|  | /* Reset the volume control registers. */ | 
|  | outb(0x88, c->iobase+0x1c); | 
|  | outb(0x88, c->iobase+0x1d); | 
|  | outb(0x88, c->iobase+0x1e); | 
|  | outb(0x88, c->iobase+0x1f); | 
|  |  | 
|  | /* Deal with the button press in a hammer-handed | 
|  | manner by adjusting the master mixer volume. */ | 
|  | volume = c->mix.mixer_state[0] & 0xff; | 
|  | if (vol_evt == UP_EVT) { | 
|  | volume += 5; | 
|  | if (volume > 100) | 
|  | volume = 100; | 
|  | } | 
|  | else if (vol_evt == DOWN_EVT) { | 
|  | volume -= 5; | 
|  | if (volume < 0) | 
|  | volume = 0; | 
|  | } else { | 
|  | /* vol_evt == MUTE_EVT */ | 
|  | if (volume == 0) | 
|  | volume = c->dock_mute_vol; | 
|  | else { | 
|  | c->dock_mute_vol = volume; | 
|  | volume = 0; | 
|  | } | 
|  | } | 
|  | set_mixer (c, 0, (volume << 8) | volume); | 
|  | } | 
|  |  | 
|  | /* Ack all the interrupts. */ | 
|  | outb(0xFF, c->iobase+0x1A); | 
|  |  | 
|  | /* | 
|  | *	Update the pointers for all APU's we are running. | 
|  | */ | 
|  | for(i=0;i<NR_DSPS;i++) | 
|  | { | 
|  | s=&c->channels[i]; | 
|  | if(s->dev_audio == -1) | 
|  | break; | 
|  | spin_lock(&s->lock); | 
|  | ess_update_ptr(s); | 
|  | spin_unlock(&s->lock); | 
|  | } | 
|  | return IRQ_HANDLED; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | static const char invalid_magic[] = KERN_CRIT "maestro: invalid magic value in %s\n"; | 
|  |  | 
|  | #define VALIDATE_MAGIC(FOO,MAG)                         \ | 
|  | ({                                                \ | 
|  | if (!(FOO) || (FOO)->magic != MAG) { \ | 
|  | printk(invalid_magic,__FUNCTION__);            \ | 
|  | return -ENXIO;                    \ | 
|  | }                                         \ | 
|  | }) | 
|  |  | 
|  | #define VALIDATE_STATE(a) VALIDATE_MAGIC(a,ESS_STATE_MAGIC) | 
|  | #define VALIDATE_CARD(a) VALIDATE_MAGIC(a,ESS_CARD_MAGIC) | 
|  |  | 
|  | static void set_mixer(struct ess_card *card,unsigned int mixer, unsigned int val ) | 
|  | { | 
|  | unsigned int left,right; | 
|  | /* cleanse input a little */ | 
|  | right = ((val >> 8)  & 0xff) ; | 
|  | left = (val  & 0xff) ; | 
|  |  | 
|  | if(right > 100) right = 100; | 
|  | if(left > 100) left = 100; | 
|  |  | 
|  | card->mix.mixer_state[mixer]=(right << 8) | left; | 
|  | card->mix.write_mixer(card,mixer,left,right); | 
|  | } | 
|  |  | 
|  | static void | 
|  | mixer_push_state(struct ess_card *card) | 
|  | { | 
|  | int i; | 
|  | for(i = 0 ; i < SOUND_MIXER_NRDEVICES ; i++) { | 
|  | if( ! supported_mixer(card,i)) continue; | 
|  |  | 
|  | set_mixer(card,i,card->mix.mixer_state[i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | static int mixer_ioctl(struct ess_card *card, unsigned int cmd, unsigned long arg) | 
|  | { | 
|  | int i, val=0; | 
|  | unsigned long flags; | 
|  | void __user *argp = (void __user *)arg; | 
|  | int __user *p = argp; | 
|  |  | 
|  | VALIDATE_CARD(card); | 
|  | if (cmd == SOUND_MIXER_INFO) { | 
|  | mixer_info info; | 
|  | memset(&info, 0, sizeof(info)); | 
|  | strlcpy(info.id, card_names[card->card_type], sizeof(info.id)); | 
|  | strlcpy(info.name, card_names[card->card_type], sizeof(info.name)); | 
|  | info.modify_counter = card->mix.modcnt; | 
|  | if (copy_to_user(argp, &info, sizeof(info))) | 
|  | return -EFAULT; | 
|  | return 0; | 
|  | } | 
|  | if (cmd == SOUND_OLD_MIXER_INFO) { | 
|  | _old_mixer_info info; | 
|  | memset(&info, 0, sizeof(info)); | 
|  | strlcpy(info.id, card_names[card->card_type], sizeof(info.id)); | 
|  | strlcpy(info.name, card_names[card->card_type], sizeof(info.name)); | 
|  | if (copy_to_user(argp, &info, sizeof(info))) | 
|  | return -EFAULT; | 
|  | return 0; | 
|  | } | 
|  | if (cmd == OSS_GETVERSION) | 
|  | return put_user(SOUND_VERSION, p); | 
|  |  | 
|  | if (_IOC_TYPE(cmd) != 'M' || _IOC_SIZE(cmd) != sizeof(int)) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (_IOC_DIR(cmd) == _IOC_READ) { | 
|  | switch (_IOC_NR(cmd)) { | 
|  | case SOUND_MIXER_RECSRC: /* give them the current record source */ | 
|  |  | 
|  | if(!card->mix.recmask_io) { | 
|  | val = 0; | 
|  | } else { | 
|  | spin_lock_irqsave(&card->lock, flags); | 
|  | val = card->mix.recmask_io(card,1,0); | 
|  | spin_unlock_irqrestore(&card->lock, flags); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case SOUND_MIXER_DEVMASK: /* give them the supported mixers */ | 
|  | val = card->mix.supported_mixers; | 
|  | break; | 
|  |  | 
|  | case SOUND_MIXER_RECMASK: /* Arg contains a bit for each supported recording source */ | 
|  | val = card->mix.record_sources; | 
|  | break; | 
|  |  | 
|  | case SOUND_MIXER_STEREODEVS: /* Mixer channels supporting stereo */ | 
|  | val = card->mix.stereo_mixers; | 
|  | break; | 
|  |  | 
|  | case SOUND_MIXER_CAPS: | 
|  | val = SOUND_CAP_EXCL_INPUT; | 
|  | break; | 
|  |  | 
|  | default: /* read a specific mixer */ | 
|  | i = _IOC_NR(cmd); | 
|  |  | 
|  | if ( ! supported_mixer(card,i)) | 
|  | return -EINVAL; | 
|  |  | 
|  | /* do we ever want to touch the hardware? */ | 
|  | /*                     spin_lock_irqsave(&card->lock, flags); | 
|  | val = card->mix.read_mixer(card,i); | 
|  | spin_unlock_irqrestore(&card->lock, flags);*/ | 
|  |  | 
|  | val = card->mix.mixer_state[i]; | 
|  | /*			M_printk("returned 0x%x for mixer %d\n",val,i);*/ | 
|  |  | 
|  | break; | 
|  | } | 
|  | return put_user(val, p); | 
|  | } | 
|  |  | 
|  | if (_IOC_DIR(cmd) != (_IOC_WRITE|_IOC_READ)) | 
|  | return -EINVAL; | 
|  |  | 
|  | card->mix.modcnt++; | 
|  |  | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  |  | 
|  | switch (_IOC_NR(cmd)) { | 
|  | case SOUND_MIXER_RECSRC: /* Arg contains a bit for each recording source */ | 
|  |  | 
|  | if (!card->mix.recmask_io) return -EINVAL; | 
|  | if(!val) return 0; | 
|  | if(! (val &= card->mix.record_sources)) return -EINVAL; | 
|  |  | 
|  | spin_lock_irqsave(&card->lock, flags); | 
|  | card->mix.recmask_io(card,0,val); | 
|  | spin_unlock_irqrestore(&card->lock, flags); | 
|  | return 0; | 
|  |  | 
|  | default: | 
|  | i = _IOC_NR(cmd); | 
|  |  | 
|  | if ( ! supported_mixer(card,i)) | 
|  | return -EINVAL; | 
|  |  | 
|  | spin_lock_irqsave(&card->lock, flags); | 
|  | set_mixer(card,i,val); | 
|  | spin_unlock_irqrestore(&card->lock, flags); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  | static int ess_open_mixdev(struct inode *inode, struct file *file) | 
|  | { | 
|  | unsigned int minor = iminor(inode); | 
|  | struct ess_card *card = NULL; | 
|  | struct pci_dev *pdev = NULL; | 
|  | struct pci_driver *drvr; | 
|  |  | 
|  | while ((pdev = pci_find_device(PCI_ANY_ID, PCI_ANY_ID, pdev)) != NULL) { | 
|  | drvr = pci_dev_driver (pdev); | 
|  | if (drvr == &maestro_pci_driver) { | 
|  | card = (struct ess_card*)pci_get_drvdata (pdev); | 
|  | if (!card) | 
|  | continue; | 
|  | if (card->dev_mixer == minor) | 
|  | break; | 
|  | } | 
|  | } | 
|  | if (!card) | 
|  | return -ENODEV; | 
|  | file->private_data = card; | 
|  | return nonseekable_open(inode, file); | 
|  | } | 
|  |  | 
|  | static int ess_release_mixdev(struct inode *inode, struct file *file) | 
|  | { | 
|  | struct ess_card *card = (struct ess_card *)file->private_data; | 
|  |  | 
|  | VALIDATE_CARD(card); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int ess_ioctl_mixdev(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg) | 
|  | { | 
|  | struct ess_card *card = (struct ess_card *)file->private_data; | 
|  |  | 
|  | VALIDATE_CARD(card); | 
|  |  | 
|  | return mixer_ioctl(card, cmd, arg); | 
|  | } | 
|  |  | 
|  | static /*const*/ struct file_operations ess_mixer_fops = { | 
|  | .owner		= THIS_MODULE, | 
|  | .llseek		= no_llseek, | 
|  | .ioctl		= ess_ioctl_mixdev, | 
|  | .open		= ess_open_mixdev, | 
|  | .release	= ess_release_mixdev, | 
|  | }; | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | static int drain_dac(struct ess_state *s, int nonblock) | 
|  | { | 
|  | DECLARE_WAITQUEUE(wait,current); | 
|  | unsigned long flags; | 
|  | int count; | 
|  | signed long tmo; | 
|  |  | 
|  | if (s->dma_dac.mapped || !s->dma_dac.ready) | 
|  | return 0; | 
|  | current->state = TASK_INTERRUPTIBLE; | 
|  | add_wait_queue(&s->dma_dac.wait, &wait); | 
|  | for (;;) { | 
|  | /* XXX uhm.. questionable locking*/ | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | count = s->dma_dac.count; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | if (count <= 0) | 
|  | break; | 
|  | if (signal_pending(current)) | 
|  | break; | 
|  | if (nonblock) { | 
|  | remove_wait_queue(&s->dma_dac.wait, &wait); | 
|  | current->state = TASK_RUNNING; | 
|  | return -EBUSY; | 
|  | } | 
|  | tmo = (count * HZ) / s->ratedac; | 
|  | tmo >>= sample_shift[(s->fmt >> ESS_DAC_SHIFT) & ESS_FMT_MASK]; | 
|  | /* XXX this is just broken.  someone is waking us up alot, or schedule_timeout is broken. | 
|  | or something.  who cares. - zach */ | 
|  | if (!schedule_timeout(tmo ? tmo : 1) && tmo) | 
|  | M_printk(KERN_DEBUG "maestro: dma timed out?? %ld\n",jiffies); | 
|  | } | 
|  | remove_wait_queue(&s->dma_dac.wait, &wait); | 
|  | current->state = TASK_RUNNING; | 
|  | if (signal_pending(current)) | 
|  | return -ERESTARTSYS; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  | /* Zach sez: "god this is gross.." */ | 
|  | static int | 
|  | comb_stereo(unsigned char *real_buffer,unsigned char  *tmp_buffer, int offset, | 
|  | int count, int bufsize) | 
|  | { | 
|  | /* No such thing as stereo recording, so we | 
|  | use dual input mixers.  which means we have to | 
|  | combine mono to stereo buffer.  yuck. | 
|  |  | 
|  | but we don't have to be able to work a byte at a time..*/ | 
|  |  | 
|  | unsigned char *so,*left,*right; | 
|  | int i; | 
|  |  | 
|  | so = tmp_buffer; | 
|  | left = real_buffer + offset; | 
|  | right = real_buffer + bufsize/2 + offset; | 
|  |  | 
|  | /*	M_printk("comb_stereo writing %d to %p from %p and %p, offset: %d size: %d\n",count/2, tmp_buffer,left,right,offset,bufsize);*/ | 
|  |  | 
|  | for(i=count/4; i ; i--) { | 
|  | (*(so+2)) = *(right++); | 
|  | (*(so+3)) = *(right++); | 
|  | (*so) = *(left++); | 
|  | (*(so+1)) = *(left++); | 
|  | so+=4; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* in this loop, dma_adc.count signifies the amount of data thats waiting | 
|  | to be copied to the user's buffer.  it is filled by the interrupt | 
|  | handler and drained by this loop. */ | 
|  | static ssize_t | 
|  | ess_read(struct file *file, char __user *buffer, size_t count, loff_t *ppos) | 
|  | { | 
|  | struct ess_state *s = (struct ess_state *)file->private_data; | 
|  | ssize_t ret; | 
|  | unsigned long flags; | 
|  | unsigned swptr; | 
|  | int cnt; | 
|  | unsigned char *combbuf = NULL; | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  | if (s->dma_adc.mapped) | 
|  | return -ENXIO; | 
|  | if (!s->dma_adc.ready && (ret = prog_dmabuf(s, 1))) | 
|  | return ret; | 
|  | if (!access_ok(VERIFY_WRITE, buffer, count)) | 
|  | return -EFAULT; | 
|  | if(!(combbuf = kmalloc(count,GFP_KERNEL))) | 
|  | return -ENOMEM; | 
|  | ret = 0; | 
|  |  | 
|  | calc_bob_rate(s); | 
|  |  | 
|  | while (count > 0) { | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | /* remember, all these things are expressed in bytes to be | 
|  | sent to the user.. hence the evil / 2 down below */ | 
|  | swptr = s->dma_adc.swptr; | 
|  | cnt = s->dma_adc.dmasize-swptr; | 
|  | if (s->dma_adc.count < cnt) | 
|  | cnt = s->dma_adc.count; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  |  | 
|  | if (cnt > count) | 
|  | cnt = count; | 
|  |  | 
|  | if ( cnt > 0 ) cnt &= ~3; | 
|  |  | 
|  | if (cnt <= 0) { | 
|  | start_adc(s); | 
|  | if (file->f_flags & O_NONBLOCK) | 
|  | { | 
|  | ret = ret ? ret : -EAGAIN; | 
|  | goto rec_return_free; | 
|  | } | 
|  | if (!interruptible_sleep_on_timeout(&s->dma_adc.wait, HZ)) { | 
|  | if(! s->card->in_suspend) printk(KERN_DEBUG "maestro: read: chip lockup? dmasz %u fragsz %u count %i hwptr %u swptr %u\n", | 
|  | s->dma_adc.dmasize, s->dma_adc.fragsize, s->dma_adc.count, | 
|  | s->dma_adc.hwptr, s->dma_adc.swptr); | 
|  | stop_adc(s); | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | set_dmac(s, virt_to_bus(s->dma_adc.rawbuf), s->dma_adc.numfrag << s->dma_adc.fragshift); | 
|  | /* program enhanced mode registers */ | 
|  | /* FILL ME */ | 
|  | /*				wrindir(s, SV_CIDMACBASECOUNT1, (s->dma_adc.fragsamples-1) >> 8); | 
|  | wrindir(s, SV_CIDMACBASECOUNT0, s->dma_adc.fragsamples-1); */ | 
|  | s->dma_adc.count = s->dma_adc.hwptr = s->dma_adc.swptr = 0; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | } | 
|  | if (signal_pending(current)) | 
|  | { | 
|  | ret = ret ? ret : -ERESTARTSYS; | 
|  | goto rec_return_free; | 
|  | } | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if(s->fmt & (ESS_FMT_STEREO << ESS_ADC_SHIFT)) { | 
|  | /* swptr/2 so that we know the real offset in each apu's buffer */ | 
|  | comb_stereo(s->dma_adc.rawbuf,combbuf,swptr/2,cnt,s->dma_adc.dmasize); | 
|  | if (copy_to_user(buffer, combbuf, cnt)) { | 
|  | ret = ret ? ret : -EFAULT; | 
|  | goto rec_return_free; | 
|  | } | 
|  | } else  { | 
|  | if (copy_to_user(buffer, s->dma_adc.rawbuf + swptr, cnt)) { | 
|  | ret = ret ? ret : -EFAULT; | 
|  | goto rec_return_free; | 
|  | } | 
|  | } | 
|  |  | 
|  | swptr = (swptr + cnt) % s->dma_adc.dmasize; | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | s->dma_adc.swptr = swptr; | 
|  | s->dma_adc.count -= cnt; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | count -= cnt; | 
|  | buffer += cnt; | 
|  | ret += cnt; | 
|  | start_adc(s); | 
|  | } | 
|  |  | 
|  | rec_return_free: | 
|  | kfree(combbuf); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static ssize_t | 
|  | ess_write(struct file *file, const char __user *buffer, size_t count, loff_t *ppos) | 
|  | { | 
|  | struct ess_state *s = (struct ess_state *)file->private_data; | 
|  | ssize_t ret; | 
|  | unsigned long flags; | 
|  | unsigned swptr; | 
|  | int cnt; | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  | if (s->dma_dac.mapped) | 
|  | return -ENXIO; | 
|  | if (!s->dma_dac.ready && (ret = prog_dmabuf(s, 0))) | 
|  | return ret; | 
|  | if (!access_ok(VERIFY_READ, buffer, count)) | 
|  | return -EFAULT; | 
|  | ret = 0; | 
|  |  | 
|  | calc_bob_rate(s); | 
|  |  | 
|  | while (count > 0) { | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  |  | 
|  | if (s->dma_dac.count < 0) { | 
|  | s->dma_dac.count = 0; | 
|  | s->dma_dac.swptr = s->dma_dac.hwptr; | 
|  | } | 
|  | swptr = s->dma_dac.swptr; | 
|  |  | 
|  | cnt = s->dma_dac.dmasize-swptr; | 
|  |  | 
|  | if (s->dma_dac.count + cnt > s->dma_dac.dmasize) | 
|  | cnt = s->dma_dac.dmasize - s->dma_dac.count; | 
|  |  | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  |  | 
|  | if (cnt > count) | 
|  | cnt = count; | 
|  |  | 
|  | if (cnt <= 0) { | 
|  | start_dac(s); | 
|  | if (file->f_flags & O_NONBLOCK) { | 
|  | if(!ret) ret = -EAGAIN; | 
|  | goto return_free; | 
|  | } | 
|  | if (!interruptible_sleep_on_timeout(&s->dma_dac.wait, HZ)) { | 
|  | if(! s->card->in_suspend) printk(KERN_DEBUG "maestro: write: chip lockup? dmasz %u fragsz %u count %i hwptr %u swptr %u\n", | 
|  | s->dma_dac.dmasize, s->dma_dac.fragsize, s->dma_dac.count, | 
|  | s->dma_dac.hwptr, s->dma_dac.swptr); | 
|  | stop_dac(s); | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | set_dmaa(s, virt_to_bus(s->dma_dac.rawbuf), s->dma_dac.numfrag << s->dma_dac.fragshift); | 
|  | /* program enhanced mode registers */ | 
|  | /*				wrindir(s, SV_CIDMAABASECOUNT1, (s->dma_dac.fragsamples-1) >> 8); | 
|  | wrindir(s, SV_CIDMAABASECOUNT0, s->dma_dac.fragsamples-1); */ | 
|  | /* FILL ME */ | 
|  | s->dma_dac.count = s->dma_dac.hwptr = s->dma_dac.swptr = 0; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | } | 
|  | if (signal_pending(current)) { | 
|  | if (!ret) ret = -ERESTARTSYS; | 
|  | goto return_free; | 
|  | } | 
|  | continue; | 
|  | } | 
|  | if (copy_from_user(s->dma_dac.rawbuf + swptr, buffer, cnt)) { | 
|  | if (!ret) ret = -EFAULT; | 
|  | goto return_free; | 
|  | } | 
|  | /*		printk("wrote %d bytes at sw: %d cnt: %d while hw: %d\n",cnt, swptr, s->dma_dac.count, s->dma_dac.hwptr);*/ | 
|  |  | 
|  | swptr = (swptr + cnt) % s->dma_dac.dmasize; | 
|  |  | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | s->dma_dac.swptr = swptr; | 
|  | s->dma_dac.count += cnt; | 
|  | s->dma_dac.endcleared = 0; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | count -= cnt; | 
|  | buffer += cnt; | 
|  | ret += cnt; | 
|  | start_dac(s); | 
|  | } | 
|  | return_free: | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /* No kernel lock - we have our own spinlock */ | 
|  | static unsigned int ess_poll(struct file *file, struct poll_table_struct *wait) | 
|  | { | 
|  | struct ess_state *s = (struct ess_state *)file->private_data; | 
|  | unsigned long flags; | 
|  | unsigned int mask = 0; | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  |  | 
|  | /* In 0.14 prog_dmabuf always returns success anyway ... */ | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | if (!s->dma_dac.ready && prog_dmabuf(s, 0)) | 
|  | return 0; | 
|  | } | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | if (!s->dma_adc.ready && prog_dmabuf(s, 1)) | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (file->f_mode & FMODE_WRITE) | 
|  | poll_wait(file, &s->dma_dac.wait, wait); | 
|  | if (file->f_mode & FMODE_READ) | 
|  | poll_wait(file, &s->dma_adc.wait, wait); | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | ess_update_ptr(s); | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | if (s->dma_adc.count >= (signed)s->dma_adc.fragsize) | 
|  | mask |= POLLIN | POLLRDNORM; | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | if (s->dma_dac.mapped) { | 
|  | if (s->dma_dac.count >= (signed)s->dma_dac.fragsize) | 
|  | mask |= POLLOUT | POLLWRNORM; | 
|  | } else { | 
|  | if ((signed)s->dma_dac.dmasize >= s->dma_dac.count + (signed)s->dma_dac.fragsize) | 
|  | mask |= POLLOUT | POLLWRNORM; | 
|  | } | 
|  | } | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | return mask; | 
|  | } | 
|  |  | 
|  | static int ess_mmap(struct file *file, struct vm_area_struct *vma) | 
|  | { | 
|  | struct ess_state *s = (struct ess_state *)file->private_data; | 
|  | struct dmabuf *db; | 
|  | int ret = -EINVAL; | 
|  | unsigned long size; | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  | lock_kernel(); | 
|  | if (vma->vm_flags & VM_WRITE) { | 
|  | if ((ret = prog_dmabuf(s, 1)) != 0) | 
|  | goto out; | 
|  | db = &s->dma_dac; | 
|  | } else | 
|  | #if 0 | 
|  | /* if we can have the wp/wc do the combining | 
|  | we can turn this back on.  */ | 
|  | if (vma->vm_flags & VM_READ) { | 
|  | if ((ret = prog_dmabuf(s, 0)) != 0) | 
|  | goto out; | 
|  | db = &s->dma_adc; | 
|  | } else | 
|  | #endif | 
|  | goto out; | 
|  | ret = -EINVAL; | 
|  | if (vma->vm_pgoff != 0) | 
|  | goto out; | 
|  | size = vma->vm_end - vma->vm_start; | 
|  | if (size > (PAGE_SIZE << db->buforder)) | 
|  | goto out; | 
|  | ret = -EAGAIN; | 
|  | if (remap_pfn_range(vma, vma->vm_start, | 
|  | virt_to_phys(db->rawbuf) >> PAGE_SHIFT, | 
|  | size, vma->vm_page_prot)) | 
|  | goto out; | 
|  | db->mapped = 1; | 
|  | ret = 0; | 
|  | out: | 
|  | unlock_kernel(); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int ess_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg) | 
|  | { | 
|  | struct ess_state *s = (struct ess_state *)file->private_data; | 
|  | unsigned long flags; | 
|  | audio_buf_info abinfo; | 
|  | count_info cinfo; | 
|  | int val, mapped, ret; | 
|  | unsigned char fmtm, fmtd; | 
|  | void __user *argp = (void __user *)arg; | 
|  | int __user *p = argp; | 
|  |  | 
|  | /*	printk("maestro: ess_ioctl: cmd %d\n", cmd);*/ | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  | mapped = ((file->f_mode & FMODE_WRITE) && s->dma_dac.mapped) || | 
|  | ((file->f_mode & FMODE_READ) && s->dma_adc.mapped); | 
|  | switch (cmd) { | 
|  | case OSS_GETVERSION: | 
|  | return put_user(SOUND_VERSION, p); | 
|  |  | 
|  | case SNDCTL_DSP_SYNC: | 
|  | if (file->f_mode & FMODE_WRITE) | 
|  | return drain_dac(s, file->f_flags & O_NONBLOCK); | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_SETDUPLEX: | 
|  | /* XXX fix */ | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETCAPS: | 
|  | return put_user(DSP_CAP_DUPLEX | DSP_CAP_REALTIME | DSP_CAP_TRIGGER | DSP_CAP_MMAP, p); | 
|  |  | 
|  | case SNDCTL_DSP_RESET: | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | stop_dac(s); | 
|  | synchronize_irq(s->card->pcidev->irq); | 
|  | s->dma_dac.swptr = s->dma_dac.hwptr = s->dma_dac.count = s->dma_dac.total_bytes = 0; | 
|  | } | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | stop_adc(s); | 
|  | synchronize_irq(s->card->pcidev->irq); | 
|  | s->dma_adc.swptr = s->dma_adc.hwptr = s->dma_adc.count = s->dma_adc.total_bytes = 0; | 
|  | } | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_SPEED: | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | if (val >= 0) { | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | stop_adc(s); | 
|  | s->dma_adc.ready = 0; | 
|  | set_adc_rate(s, val); | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | stop_dac(s); | 
|  | s->dma_dac.ready = 0; | 
|  | set_dac_rate(s, val); | 
|  | } | 
|  | } | 
|  | return put_user((file->f_mode & FMODE_READ) ? s->rateadc : s->ratedac, p); | 
|  |  | 
|  | case SNDCTL_DSP_STEREO: | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | fmtd = 0; | 
|  | fmtm = ~0; | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | stop_adc(s); | 
|  | s->dma_adc.ready = 0; | 
|  | if (val) | 
|  | fmtd |= ESS_FMT_STEREO << ESS_ADC_SHIFT; | 
|  | else | 
|  | fmtm &= ~(ESS_FMT_STEREO << ESS_ADC_SHIFT); | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | stop_dac(s); | 
|  | s->dma_dac.ready = 0; | 
|  | if (val) | 
|  | fmtd |= ESS_FMT_STEREO << ESS_DAC_SHIFT; | 
|  | else | 
|  | fmtm &= ~(ESS_FMT_STEREO << ESS_DAC_SHIFT); | 
|  | } | 
|  | set_fmt(s, fmtm, fmtd); | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_CHANNELS: | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | if (val != 0) { | 
|  | fmtd = 0; | 
|  | fmtm = ~0; | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | stop_adc(s); | 
|  | s->dma_adc.ready = 0; | 
|  | if (val >= 2) | 
|  | fmtd |= ESS_FMT_STEREO << ESS_ADC_SHIFT; | 
|  | else | 
|  | fmtm &= ~(ESS_FMT_STEREO << ESS_ADC_SHIFT); | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | stop_dac(s); | 
|  | s->dma_dac.ready = 0; | 
|  | if (val >= 2) | 
|  | fmtd |= ESS_FMT_STEREO << ESS_DAC_SHIFT; | 
|  | else | 
|  | fmtm &= ~(ESS_FMT_STEREO << ESS_DAC_SHIFT); | 
|  | } | 
|  | set_fmt(s, fmtm, fmtd); | 
|  | } | 
|  | return put_user((s->fmt & ((file->f_mode & FMODE_READ) ? (ESS_FMT_STEREO << ESS_ADC_SHIFT) | 
|  | : (ESS_FMT_STEREO << ESS_DAC_SHIFT))) ? 2 : 1, p); | 
|  |  | 
|  | case SNDCTL_DSP_GETFMTS: /* Returns a mask */ | 
|  | return put_user(AFMT_U8|AFMT_S16_LE, p); | 
|  |  | 
|  | case SNDCTL_DSP_SETFMT: /* Selects ONE fmt*/ | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | if (val != AFMT_QUERY) { | 
|  | fmtd = 0; | 
|  | fmtm = ~0; | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | stop_adc(s); | 
|  | s->dma_adc.ready = 0; | 
|  | /* fixed at 16bit for now */ | 
|  | fmtd |= ESS_FMT_16BIT << ESS_ADC_SHIFT; | 
|  | #if 0 | 
|  | if (val == AFMT_S16_LE) | 
|  | fmtd |= ESS_FMT_16BIT << ESS_ADC_SHIFT; | 
|  | else | 
|  | fmtm &= ~(ESS_FMT_16BIT << ESS_ADC_SHIFT); | 
|  | #endif | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | stop_dac(s); | 
|  | s->dma_dac.ready = 0; | 
|  | if (val == AFMT_S16_LE) | 
|  | fmtd |= ESS_FMT_16BIT << ESS_DAC_SHIFT; | 
|  | else | 
|  | fmtm &= ~(ESS_FMT_16BIT << ESS_DAC_SHIFT); | 
|  | } | 
|  | set_fmt(s, fmtm, fmtd); | 
|  | } | 
|  | return put_user((s->fmt & ((file->f_mode & FMODE_READ) ? | 
|  | (ESS_FMT_16BIT << ESS_ADC_SHIFT) | 
|  | : (ESS_FMT_16BIT << ESS_DAC_SHIFT))) ? | 
|  | AFMT_S16_LE : | 
|  | AFMT_U8, | 
|  | p); | 
|  |  | 
|  | case SNDCTL_DSP_POST: | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETTRIGGER: | 
|  | val = 0; | 
|  | if ((file->f_mode & FMODE_READ) && (s->enable & ADC_RUNNING)) | 
|  | val |= PCM_ENABLE_INPUT; | 
|  | if ((file->f_mode & FMODE_WRITE) && (s->enable & DAC_RUNNING)) | 
|  | val |= PCM_ENABLE_OUTPUT; | 
|  | return put_user(val, p); | 
|  |  | 
|  | case SNDCTL_DSP_SETTRIGGER: | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | if (val & PCM_ENABLE_INPUT) { | 
|  | if (!s->dma_adc.ready && (ret =  prog_dmabuf(s, 1))) | 
|  | return ret; | 
|  | start_adc(s); | 
|  | } else | 
|  | stop_adc(s); | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | if (val & PCM_ENABLE_OUTPUT) { | 
|  | if (!s->dma_dac.ready && (ret = prog_dmabuf(s, 0))) | 
|  | return ret; | 
|  | start_dac(s); | 
|  | } else | 
|  | stop_dac(s); | 
|  | } | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETOSPACE: | 
|  | if (!(file->f_mode & FMODE_WRITE)) | 
|  | return -EINVAL; | 
|  | if (!s->dma_dac.ready && (ret = prog_dmabuf(s, 0))) | 
|  | return ret; | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | ess_update_ptr(s); | 
|  | abinfo.fragsize = s->dma_dac.fragsize; | 
|  | abinfo.bytes = s->dma_dac.dmasize - s->dma_dac.count; | 
|  | abinfo.fragstotal = s->dma_dac.numfrag; | 
|  | abinfo.fragments = abinfo.bytes >> s->dma_dac.fragshift; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | return copy_to_user(argp, &abinfo, sizeof(abinfo)) ? -EFAULT : 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETISPACE: | 
|  | if (!(file->f_mode & FMODE_READ)) | 
|  | return -EINVAL; | 
|  | if (!s->dma_adc.ready && (ret =  prog_dmabuf(s, 1))) | 
|  | return ret; | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | ess_update_ptr(s); | 
|  | abinfo.fragsize = s->dma_adc.fragsize; | 
|  | abinfo.bytes = s->dma_adc.count; | 
|  | abinfo.fragstotal = s->dma_adc.numfrag; | 
|  | abinfo.fragments = abinfo.bytes >> s->dma_adc.fragshift; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | return copy_to_user(argp, &abinfo, sizeof(abinfo)) ? -EFAULT : 0; | 
|  |  | 
|  | case SNDCTL_DSP_NONBLOCK: | 
|  | file->f_flags |= O_NONBLOCK; | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETODELAY: | 
|  | if (!(file->f_mode & FMODE_WRITE)) | 
|  | return -EINVAL; | 
|  | if (!s->dma_dac.ready && (ret = prog_dmabuf(s, 0))) | 
|  | return ret; | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | ess_update_ptr(s); | 
|  | val = s->dma_dac.count; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | return put_user(val, p); | 
|  |  | 
|  | case SNDCTL_DSP_GETIPTR: | 
|  | if (!(file->f_mode & FMODE_READ)) | 
|  | return -EINVAL; | 
|  | if (!s->dma_adc.ready && (ret =  prog_dmabuf(s, 1))) | 
|  | return ret; | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | ess_update_ptr(s); | 
|  | cinfo.bytes = s->dma_adc.total_bytes; | 
|  | cinfo.blocks = s->dma_adc.count >> s->dma_adc.fragshift; | 
|  | cinfo.ptr = s->dma_adc.hwptr; | 
|  | if (s->dma_adc.mapped) | 
|  | s->dma_adc.count &= s->dma_adc.fragsize-1; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | if (copy_to_user(argp, &cinfo, sizeof(cinfo))) | 
|  | return -EFAULT; | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETOPTR: | 
|  | if (!(file->f_mode & FMODE_WRITE)) | 
|  | return -EINVAL; | 
|  | if (!s->dma_dac.ready && (ret = prog_dmabuf(s, 0))) | 
|  | return ret; | 
|  | spin_lock_irqsave(&s->lock, flags); | 
|  | ess_update_ptr(s); | 
|  | cinfo.bytes = s->dma_dac.total_bytes; | 
|  | cinfo.blocks = s->dma_dac.count >> s->dma_dac.fragshift; | 
|  | cinfo.ptr = s->dma_dac.hwptr; | 
|  | if (s->dma_dac.mapped) | 
|  | s->dma_dac.count &= s->dma_dac.fragsize-1; | 
|  | spin_unlock_irqrestore(&s->lock, flags); | 
|  | if (copy_to_user(argp, &cinfo, sizeof(cinfo))) | 
|  | return -EFAULT; | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_GETBLKSIZE: | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | if ((val = prog_dmabuf(s, 0))) | 
|  | return val; | 
|  | return put_user(s->dma_dac.fragsize, p); | 
|  | } | 
|  | if ((val = prog_dmabuf(s, 1))) | 
|  | return val; | 
|  | return put_user(s->dma_adc.fragsize, p); | 
|  |  | 
|  | case SNDCTL_DSP_SETFRAGMENT: | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | M_printk("maestro: SETFRAGMENT: %0x\n",val); | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | s->dma_adc.ossfragshift = val & 0xffff; | 
|  | s->dma_adc.ossmaxfrags = (val >> 16) & 0xffff; | 
|  | if (s->dma_adc.ossfragshift < 4) | 
|  | s->dma_adc.ossfragshift = 4; | 
|  | if (s->dma_adc.ossfragshift > 15) | 
|  | s->dma_adc.ossfragshift = 15; | 
|  | if (s->dma_adc.ossmaxfrags < 4) | 
|  | s->dma_adc.ossmaxfrags = 4; | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | s->dma_dac.ossfragshift = val & 0xffff; | 
|  | s->dma_dac.ossmaxfrags = (val >> 16) & 0xffff; | 
|  | if (s->dma_dac.ossfragshift < 4) | 
|  | s->dma_dac.ossfragshift = 4; | 
|  | if (s->dma_dac.ossfragshift > 15) | 
|  | s->dma_dac.ossfragshift = 15; | 
|  | if (s->dma_dac.ossmaxfrags < 4) | 
|  | s->dma_dac.ossmaxfrags = 4; | 
|  | } | 
|  | return 0; | 
|  |  | 
|  | case SNDCTL_DSP_SUBDIVIDE: | 
|  | if ((file->f_mode & FMODE_READ && s->dma_adc.subdivision) || | 
|  | (file->f_mode & FMODE_WRITE && s->dma_dac.subdivision)) | 
|  | return -EINVAL; | 
|  | if (get_user(val, p)) | 
|  | return -EFAULT; | 
|  | if (val != 1 && val != 2 && val != 4) | 
|  | return -EINVAL; | 
|  | if (file->f_mode & FMODE_READ) | 
|  | s->dma_adc.subdivision = val; | 
|  | if (file->f_mode & FMODE_WRITE) | 
|  | s->dma_dac.subdivision = val; | 
|  | return 0; | 
|  |  | 
|  | case SOUND_PCM_READ_RATE: | 
|  | return put_user((file->f_mode & FMODE_READ) ? s->rateadc : s->ratedac, p); | 
|  |  | 
|  | case SOUND_PCM_READ_CHANNELS: | 
|  | return put_user((s->fmt & ((file->f_mode & FMODE_READ) ? (ESS_FMT_STEREO << ESS_ADC_SHIFT) | 
|  | : (ESS_FMT_STEREO << ESS_DAC_SHIFT))) ? 2 : 1, p); | 
|  |  | 
|  | case SOUND_PCM_READ_BITS: | 
|  | return put_user((s->fmt & ((file->f_mode & FMODE_READ) ? (ESS_FMT_16BIT << ESS_ADC_SHIFT) | 
|  | : (ESS_FMT_16BIT << ESS_DAC_SHIFT))) ? 16 : 8, p); | 
|  |  | 
|  | case SOUND_PCM_WRITE_FILTER: | 
|  | case SNDCTL_DSP_SETSYNCRO: | 
|  | case SOUND_PCM_READ_FILTER: | 
|  | return -EINVAL; | 
|  |  | 
|  | } | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | static void | 
|  | set_base_registers(struct ess_state *s,void *vaddr) | 
|  | { | 
|  | unsigned long packed_phys = virt_to_bus(vaddr)>>12; | 
|  | wave_set_register(s, 0x01FC , packed_phys); | 
|  | wave_set_register(s, 0x01FD , packed_phys); | 
|  | wave_set_register(s, 0x01FE , packed_phys); | 
|  | wave_set_register(s, 0x01FF , packed_phys); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * this guy makes sure we're in the right power | 
|  | * state for what we want to be doing | 
|  | */ | 
|  | static void maestro_power(struct ess_card *card, int tostate) | 
|  | { | 
|  | u16 active_mask = acpi_state_mask[tostate]; | 
|  | u8 state; | 
|  |  | 
|  | if(!use_pm) return; | 
|  |  | 
|  | pci_read_config_byte(card->pcidev, card->power_regs+0x4, &state); | 
|  | state&=3; | 
|  |  | 
|  | /* make sure we're in the right state */ | 
|  | if(state != tostate) { | 
|  | M_printk(KERN_WARNING "maestro: dev %02x:%02x.%x switching from D%d to D%d\n", | 
|  | card->pcidev->bus->number, | 
|  | PCI_SLOT(card->pcidev->devfn), | 
|  | PCI_FUNC(card->pcidev->devfn), | 
|  | state,tostate); | 
|  | pci_write_config_byte(card->pcidev, card->power_regs+0x4, tostate); | 
|  | } | 
|  |  | 
|  | /* and make sure the units we care about are on | 
|  | XXX we might want to do this before state flipping? */ | 
|  | pci_write_config_word(card->pcidev, 0x54, ~ active_mask); | 
|  | pci_write_config_word(card->pcidev, 0x56, ~ active_mask); | 
|  | } | 
|  |  | 
|  | /* we allocate a large power of two for all our memory. | 
|  | this is cut up into (not to scale :): | 
|  | |silly fifo word	| 512byte mixbuf per adc	| dac/adc * channels | | 
|  | */ | 
|  | static int | 
|  | allocate_buffers(struct ess_state *s) | 
|  | { | 
|  | void *rawbuf=NULL; | 
|  | int order,i; | 
|  | struct page *page, *pend; | 
|  |  | 
|  | /* alloc as big a chunk as we can */ | 
|  | for (order = (dsps_order + (16-PAGE_SHIFT) + 1); order >= (dsps_order + 2 + 1); order--) | 
|  | if((rawbuf = (void *)__get_free_pages(GFP_KERNEL|GFP_DMA, order))) | 
|  | break; | 
|  |  | 
|  | if (!rawbuf) | 
|  | return 1; | 
|  |  | 
|  | M_printk("maestro: allocated %ld (%d) bytes at %p\n",PAGE_SIZE<<order,order, rawbuf); | 
|  |  | 
|  | if ((virt_to_bus(rawbuf) + (PAGE_SIZE << order) - 1) & ~((1<<28)-1))  { | 
|  | printk(KERN_ERR "maestro: DMA buffer beyond 256MB! busaddr 0x%lx  size %ld\n", | 
|  | virt_to_bus(rawbuf), PAGE_SIZE << order); | 
|  | kfree(rawbuf); | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | s->card->dmapages = rawbuf; | 
|  | s->card->dmaorder = order; | 
|  |  | 
|  | for(i=0;i<NR_DSPS;i++) { | 
|  | struct ess_state *ess = &s->card->channels[i]; | 
|  |  | 
|  | if(ess->dev_audio == -1) | 
|  | continue; | 
|  |  | 
|  | ess->dma_dac.ready = s->dma_dac.mapped = 0; | 
|  | ess->dma_adc.ready = s->dma_adc.mapped = 0; | 
|  | ess->dma_adc.buforder = ess->dma_dac.buforder = order - 1 - dsps_order - 1; | 
|  |  | 
|  | /* offset dac and adc buffers starting half way through and then at each [da][ad]c's | 
|  | order's intervals.. */ | 
|  | ess->dma_dac.rawbuf = rawbuf + (PAGE_SIZE<<(order-1)) + (i * ( PAGE_SIZE << (ess->dma_dac.buforder + 1 ))); | 
|  | ess->dma_adc.rawbuf = ess->dma_dac.rawbuf + ( PAGE_SIZE << ess->dma_dac.buforder); | 
|  | /* offset mixbuf by a mixbuf so that the lame status fifo can | 
|  | happily scribble away.. */ | 
|  | ess->mixbuf = rawbuf + (512 * (i+1)); | 
|  |  | 
|  | M_printk("maestro: setup apu %d: dac: %p adc: %p mix: %p\n",i,ess->dma_dac.rawbuf, | 
|  | ess->dma_adc.rawbuf, ess->mixbuf); | 
|  |  | 
|  | } | 
|  |  | 
|  | /* now mark the pages as reserved; otherwise remap_pfn_range doesn't do what we want */ | 
|  | pend = virt_to_page(rawbuf + (PAGE_SIZE << order) - 1); | 
|  | for (page = virt_to_page(rawbuf); page <= pend; page++) | 
|  | SetPageReserved(page); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | static void | 
|  | free_buffers(struct ess_state *s) | 
|  | { | 
|  | struct page *page, *pend; | 
|  |  | 
|  | s->dma_dac.rawbuf = s->dma_adc.rawbuf = NULL; | 
|  | s->dma_dac.mapped = s->dma_adc.mapped = 0; | 
|  | s->dma_dac.ready = s->dma_adc.ready = 0; | 
|  |  | 
|  | M_printk("maestro: freeing %p\n",s->card->dmapages); | 
|  | /* undo marking the pages as reserved */ | 
|  |  | 
|  | pend = virt_to_page(s->card->dmapages + (PAGE_SIZE << s->card->dmaorder) - 1); | 
|  | for (page = virt_to_page(s->card->dmapages); page <= pend; page++) | 
|  | ClearPageReserved(page); | 
|  |  | 
|  | free_pages((unsigned long)s->card->dmapages,s->card->dmaorder); | 
|  | s->card->dmapages = NULL; | 
|  | } | 
|  |  | 
|  | static int | 
|  | ess_open(struct inode *inode, struct file *file) | 
|  | { | 
|  | unsigned int minor = iminor(inode); | 
|  | struct ess_state *s = NULL; | 
|  | unsigned char fmtm = ~0, fmts = 0; | 
|  | struct pci_dev *pdev = NULL; | 
|  | /* | 
|  | *	Scan the cards and find the channel. We only | 
|  | *	do this at open time so it is ok | 
|  | */ | 
|  |  | 
|  | while ((pdev = pci_find_device(PCI_ANY_ID, PCI_ANY_ID, pdev)) != NULL) { | 
|  | struct ess_card *c; | 
|  | struct pci_driver *drvr; | 
|  |  | 
|  | drvr = pci_dev_driver (pdev); | 
|  | if (drvr == &maestro_pci_driver) { | 
|  | int i; | 
|  | struct ess_state *sp; | 
|  |  | 
|  | c = (struct ess_card*)pci_get_drvdata (pdev); | 
|  | if (!c) | 
|  | continue; | 
|  | for(i=0;i<NR_DSPS;i++) | 
|  | { | 
|  | sp=&c->channels[i]; | 
|  | if(sp->dev_audio < 0) | 
|  | continue; | 
|  | if((sp->dev_audio ^ minor) & ~0xf) | 
|  | continue; | 
|  | s=sp; | 
|  | } | 
|  | } | 
|  | } | 
|  | if (!s) | 
|  | return -ENODEV; | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  | file->private_data = s; | 
|  | /* wait for device to become free */ | 
|  | down(&s->open_sem); | 
|  | while (s->open_mode & file->f_mode) { | 
|  | if (file->f_flags & O_NONBLOCK) { | 
|  | up(&s->open_sem); | 
|  | return -EWOULDBLOCK; | 
|  | } | 
|  | up(&s->open_sem); | 
|  | interruptible_sleep_on(&s->open_wait); | 
|  | if (signal_pending(current)) | 
|  | return -ERESTARTSYS; | 
|  | down(&s->open_sem); | 
|  | } | 
|  |  | 
|  | /* under semaphore.. */ | 
|  | if ((s->card->dmapages==NULL) && allocate_buffers(s)) { | 
|  | up(&s->open_sem); | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | /* we're covered by the open_sem */ | 
|  | if( ! s->card->dsps_open )  { | 
|  | maestro_power(s->card,ACPI_D0); | 
|  | start_bob(s); | 
|  | } | 
|  | s->card->dsps_open++; | 
|  | M_printk("maestro: open, %d bobs now\n",s->card->dsps_open); | 
|  |  | 
|  | /* ok, lets write WC base regs now that we've | 
|  | powered up the chip */ | 
|  | M_printk("maestro: writing 0x%lx (bus 0x%lx) to the wp\n",virt_to_bus(s->card->dmapages), | 
|  | ((virt_to_bus(s->card->dmapages))&0xFFE00000)>>12); | 
|  | set_base_registers(s,s->card->dmapages); | 
|  |  | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | /* | 
|  | fmtm &= ~((ESS_FMT_STEREO | ESS_FMT_16BIT) << ESS_ADC_SHIFT); | 
|  | if ((minor & 0xf) == SND_DEV_DSP16) | 
|  | fmts |= ESS_FMT_16BIT << ESS_ADC_SHIFT; */ | 
|  |  | 
|  | fmtm &= ~((ESS_FMT_STEREO|ESS_FMT_16BIT) << ESS_ADC_SHIFT); | 
|  | fmts = (ESS_FMT_STEREO|ESS_FMT_16BIT) << ESS_ADC_SHIFT; | 
|  |  | 
|  | s->dma_adc.ossfragshift = s->dma_adc.ossmaxfrags = s->dma_adc.subdivision = 0; | 
|  | set_adc_rate(s, 8000); | 
|  | } | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | fmtm &= ~((ESS_FMT_STEREO | ESS_FMT_16BIT) << ESS_DAC_SHIFT); | 
|  | if ((minor & 0xf) == SND_DEV_DSP16) | 
|  | fmts |= ESS_FMT_16BIT << ESS_DAC_SHIFT; | 
|  |  | 
|  | s->dma_dac.ossfragshift = s->dma_dac.ossmaxfrags = s->dma_dac.subdivision = 0; | 
|  | set_dac_rate(s, 8000); | 
|  | } | 
|  | set_fmt(s, fmtm, fmts); | 
|  | s->open_mode |= file->f_mode & (FMODE_READ | FMODE_WRITE); | 
|  |  | 
|  | up(&s->open_sem); | 
|  | return nonseekable_open(inode, file); | 
|  | } | 
|  |  | 
|  | static int | 
|  | ess_release(struct inode *inode, struct file *file) | 
|  | { | 
|  | struct ess_state *s = (struct ess_state *)file->private_data; | 
|  |  | 
|  | VALIDATE_STATE(s); | 
|  | lock_kernel(); | 
|  | if (file->f_mode & FMODE_WRITE) | 
|  | drain_dac(s, file->f_flags & O_NONBLOCK); | 
|  | down(&s->open_sem); | 
|  | if (file->f_mode & FMODE_WRITE) { | 
|  | stop_dac(s); | 
|  | } | 
|  | if (file->f_mode & FMODE_READ) { | 
|  | stop_adc(s); | 
|  | } | 
|  |  | 
|  | s->open_mode &= (~file->f_mode) & (FMODE_READ|FMODE_WRITE); | 
|  | /* we're covered by the open_sem */ | 
|  | M_printk("maestro: %d dsps now alive\n",s->card->dsps_open-1); | 
|  | if( --s->card->dsps_open <= 0) { | 
|  | s->card->dsps_open = 0; | 
|  | stop_bob(s); | 
|  | free_buffers(s); | 
|  | maestro_power(s->card,ACPI_D2); | 
|  | } | 
|  | up(&s->open_sem); | 
|  | wake_up(&s->open_wait); | 
|  | unlock_kernel(); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct file_operations ess_audio_fops = { | 
|  | .owner		= THIS_MODULE, | 
|  | .llseek		= no_llseek, | 
|  | .read		= ess_read, | 
|  | .write		= ess_write, | 
|  | .poll		= ess_poll, | 
|  | .ioctl		= ess_ioctl, | 
|  | .mmap		= ess_mmap, | 
|  | .open		= ess_open, | 
|  | .release	= ess_release, | 
|  | }; | 
|  |  | 
|  | static int | 
|  | maestro_config(struct ess_card *card) | 
|  | { | 
|  | struct pci_dev *pcidev = card->pcidev; | 
|  | struct ess_state *ess = &card->channels[0]; | 
|  | int apu,iobase  = card->iobase; | 
|  | u16 w; | 
|  | u32 n; | 
|  |  | 
|  | /* We used to muck around with pci config space that | 
|  | * we had no business messing with.  We don't know enough | 
|  | * about the machine to know which DMA mode is appropriate, | 
|  | * etc.  We were guessing wrong on some machines and making | 
|  | * them unhappy.  We now trust in the BIOS to do things right, | 
|  | * which almost certainly means a new host of problems will | 
|  | * arise with broken BIOS implementations.  screw 'em. | 
|  | * We're already intolerant of machines that don't assign | 
|  | * IRQs. | 
|  | */ | 
|  |  | 
|  | /* do config work at full power */ | 
|  | maestro_power(card,ACPI_D0); | 
|  |  | 
|  | pci_read_config_word(pcidev, 0x50, &w); | 
|  |  | 
|  | w&=~(1<<5);			/* Don't swap left/right (undoc)*/ | 
|  |  | 
|  | pci_write_config_word(pcidev, 0x50, w); | 
|  |  | 
|  | pci_read_config_word(pcidev, 0x52, &w); | 
|  | w&=~(1<<15);		/* Turn off internal clock multiplier */ | 
|  | /* XXX how do we know which to use? */ | 
|  | w&=~(1<<14);		/* External clock */ | 
|  |  | 
|  | w|= (1<<7);		/* Hardware volume control on */ | 
|  | w|= (1<<6);		/* Debounce off: easier to push the HWV buttons. */ | 
|  | w&=~(1<<5);		/* GPIO 4:5 */ | 
|  | w|= (1<<4);             /* Disconnect from the CHI.  Enabling this made a dell 7500 work. */ | 
|  | w&=~(1<<2);		/* MIDI fix off (undoc) */ | 
|  | w&=~(1<<1);		/* reserved, always write 0 */ | 
|  | pci_write_config_word(pcidev, 0x52, w); | 
|  |  | 
|  | /* | 
|  | *	Legacy mode | 
|  | */ | 
|  |  | 
|  | pci_read_config_word(pcidev, 0x40, &w); | 
|  | w|=(1<<15);	/* legacy decode off */ | 
|  | w&=~(1<<14);	/* Disable SIRQ */ | 
|  | w&=~(0x1f);	/* disable mpu irq/io, game port, fm, SB */ | 
|  |  | 
|  | pci_write_config_word(pcidev, 0x40, w); | 
|  |  | 
|  | /* Set up 978 docking control chip. */ | 
|  | pci_read_config_word(pcidev, 0x58, &w); | 
|  | w|=1<<2;	/* Enable 978. */ | 
|  | w|=1<<3;	/* Turn on 978 hardware volume control. */ | 
|  | w&=~(1<<11);	/* Turn on 978 mixer volume control. */ | 
|  | pci_write_config_word(pcidev, 0x58, w); | 
|  |  | 
|  | sound_reset(iobase); | 
|  |  | 
|  | /* | 
|  | *	Ring Bus Setup | 
|  | */ | 
|  |  | 
|  | /* setup usual 0x34 stuff.. 0x36 may be chip specific */ | 
|  | outw(0xC090, iobase+0x34); /* direct sound, stereo */ | 
|  | udelay(20); | 
|  | outw(0x3000, iobase+0x36); /* direct sound, stereo */ | 
|  | udelay(20); | 
|  |  | 
|  |  | 
|  | /* | 
|  | *	Reset the CODEC | 
|  | */ | 
|  |  | 
|  | maestro_ac97_reset(iobase,pcidev); | 
|  |  | 
|  | /* | 
|  | *	Ring Bus Setup | 
|  | */ | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n&=~0xF000; | 
|  | n|=12<<12;		/* Direct Sound, Stereo */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n&=~0x0F00;		/* Modem off */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n&=~0x00F0; | 
|  | n|=9<<4;		/* DAC, Stereo */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n&=~0x000F;		/* ASSP off */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n|=(1<<29);		/* Enable ring bus */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n|=(1<<28);		/* Enable serial bus */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n&=~0x00F00000;		/* MIC off */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  | n=inl(iobase+0x34); | 
|  | n&=~0x000F0000;		/* I2S off */ | 
|  | outl(n, iobase+0x34); | 
|  |  | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w&=~(1<<7);		/* ClkRun off */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w&=~(1<<6);		/* Hardware volume control interrupt off... for now. */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w&=~(1<<4);		/* ASSP irq off */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w&=~(1<<3);		/* ISDN irq off */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w|=(1<<2);		/* Direct Sound IRQ on */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w&=~(1<<1);		/* MPU401 IRQ off */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | w=inw(iobase+0x18); | 
|  | w|=(1<<0);		/* SB IRQ on */ | 
|  | outw(w, iobase+0x18); | 
|  |  | 
|  | /* Set hardware volume control registers to midpoints. | 
|  | We can tell which button was pushed based on how they change. */ | 
|  | outb(0x88, iobase+0x1c); | 
|  | outb(0x88, iobase+0x1d); | 
|  | outb(0x88, iobase+0x1e); | 
|  | outb(0x88, iobase+0x1f); | 
|  |  | 
|  | /* it appears some maestros (dell 7500) only work if these are set, | 
|  | regardless of whether we use the assp or not. */ | 
|  |  | 
|  | outb(0, iobase+0xA4); | 
|  | outb(3, iobase+0xA2); | 
|  | outb(0, iobase+0xA6); | 
|  |  | 
|  | for(apu=0;apu<16;apu++) | 
|  | { | 
|  | /* Write 0 into the buffer area 0x1E0->1EF */ | 
|  | outw(0x01E0+apu, 0x10+iobase); | 
|  | outw(0x0000, 0x12+iobase); | 
|  |  | 
|  | /* | 
|  | * The 1.10 test program seem to write 0 into the buffer area | 
|  | * 0x1D0-0x1DF too. | 
|  | */ | 
|  | outw(0x01D0+apu, 0x10+iobase); | 
|  | outw(0x0000, 0x12+iobase); | 
|  | } | 
|  |  | 
|  | #if 1 | 
|  | wave_set_register(ess, IDR7_WAVE_ROMRAM, | 
|  | (wave_get_register(ess, IDR7_WAVE_ROMRAM)&0xFF00)); | 
|  | wave_set_register(ess, IDR7_WAVE_ROMRAM, | 
|  | wave_get_register(ess, IDR7_WAVE_ROMRAM)|0x100); | 
|  | wave_set_register(ess, IDR7_WAVE_ROMRAM, | 
|  | wave_get_register(ess, IDR7_WAVE_ROMRAM)&~0x200); | 
|  | wave_set_register(ess, IDR7_WAVE_ROMRAM, | 
|  | wave_get_register(ess, IDR7_WAVE_ROMRAM)|~0x400); | 
|  | #else | 
|  | maestro_write(ess, IDR7_WAVE_ROMRAM, | 
|  | (maestro_read(ess, IDR7_WAVE_ROMRAM)&0xFF00)); | 
|  | maestro_write(ess, IDR7_WAVE_ROMRAM, | 
|  | maestro_read(ess, IDR7_WAVE_ROMRAM)|0x100); | 
|  | maestro_write(ess, IDR7_WAVE_ROMRAM, | 
|  | maestro_read(ess, IDR7_WAVE_ROMRAM)&~0x200); | 
|  | maestro_write(ess, IDR7_WAVE_ROMRAM, | 
|  | maestro_read(ess, IDR7_WAVE_ROMRAM)|0x400); | 
|  | #endif | 
|  |  | 
|  | maestro_write(ess, IDR2_CRAM_DATA, 0x0000); | 
|  | maestro_write(ess, 0x08, 0xB004); | 
|  | /* Now back to the DirectSound stuff */ | 
|  | maestro_write(ess, 0x09, 0x001B); | 
|  | maestro_write(ess, 0x0A, 0x8000); | 
|  | maestro_write(ess, 0x0B, 0x3F37); | 
|  | maestro_write(ess, 0x0C, 0x0098); | 
|  |  | 
|  | /* parallel out ?? */ | 
|  | maestro_write(ess, 0x0C, | 
|  | (maestro_read(ess, 0x0C)&~0xF000)|0x8000); | 
|  | /* parallel in, has something to do with recording :) */ | 
|  | maestro_write(ess, 0x0C, | 
|  | (maestro_read(ess, 0x0C)&~0x0F00)|0x0500); | 
|  |  | 
|  | maestro_write(ess, 0x0D, 0x7632); | 
|  |  | 
|  | /* Wave cache control on - test off, sg off, | 
|  | enable, enable extra chans 1Mb */ | 
|  |  | 
|  | outw(inw(0x14+iobase)|(1<<8),0x14+iobase); | 
|  | outw(inw(0x14+iobase)&0xFE03,0x14+iobase); | 
|  | outw((inw(0x14+iobase)&0xFFFC), 0x14+iobase); | 
|  | outw(inw(0x14+iobase)|(1<<7),0x14+iobase); | 
|  |  | 
|  | outw(0xA1A0, 0x14+iobase);      /* 0300 ? */ | 
|  |  | 
|  | /* Now clear the APU control ram */ | 
|  | for(apu=0;apu<NR_APUS;apu++) | 
|  | { | 
|  | for(w=0;w<NR_APU_REGS;w++) | 
|  | apu_set_register(ess, apu|ESS_CHAN_HARD, w, 0); | 
|  |  | 
|  | } | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | } | 
|  |  | 
|  | /* this guy tries to find the pci power management | 
|  | * register bank.  this should really be in core | 
|  | * code somewhere.  1 on success. */ | 
|  | static int | 
|  | parse_power(struct ess_card *card, struct pci_dev *pcidev) | 
|  | { | 
|  | u32 n; | 
|  | u16 w; | 
|  | u8 next; | 
|  | int max = 64;  /* an a 8bit guy pointing to 32bit guys | 
|  | can only express so much. */ | 
|  |  | 
|  | card->power_regs = 0; | 
|  |  | 
|  | /* check to see if we have a capabilities list in | 
|  | the config register */ | 
|  | pci_read_config_word(pcidev, PCI_STATUS, &w); | 
|  | if(!(w & PCI_STATUS_CAP_LIST)) return 0; | 
|  |  | 
|  | /* walk the list, starting at the head. */ | 
|  | pci_read_config_byte(pcidev,PCI_CAPABILITY_LIST,&next); | 
|  |  | 
|  | while(next && max--) { | 
|  | pci_read_config_dword(pcidev, next & ~3, &n); | 
|  | if((n & 0xff) == PCI_CAP_ID_PM) { | 
|  | card->power_regs = next; | 
|  | break; | 
|  | } | 
|  | next = ((n>>8) & 0xff); | 
|  | } | 
|  |  | 
|  | return card->power_regs ? 1 : 0; | 
|  | } | 
|  |  | 
|  | static int __init | 
|  | maestro_probe(struct pci_dev *pcidev,const struct pci_device_id *pdid) | 
|  | { | 
|  | int card_type = pdid->driver_data; | 
|  | u32 n; | 
|  | int iobase; | 
|  | int i, ret; | 
|  | struct ess_card *card; | 
|  | struct ess_state *ess; | 
|  | struct pm_dev *pmdev; | 
|  | int num = 0; | 
|  |  | 
|  | /* when built into the kernel, we only print version if device is found */ | 
|  | #ifndef MODULE | 
|  | static int printed_version; | 
|  | if (!printed_version++) | 
|  | printk(version); | 
|  | #endif | 
|  |  | 
|  | /* don't pick up weird modem maestros */ | 
|  | if(((pcidev->class >> 8) & 0xffff) != PCI_CLASS_MULTIMEDIA_AUDIO) | 
|  | return -ENODEV; | 
|  |  | 
|  |  | 
|  | if ((ret=pci_enable_device(pcidev))) | 
|  | return ret; | 
|  |  | 
|  | iobase = pci_resource_start(pcidev,0); | 
|  | if (!iobase || !(pci_resource_flags(pcidev, 0 ) & IORESOURCE_IO)) | 
|  | return -ENODEV; | 
|  |  | 
|  | if(pcidev->irq == 0) | 
|  | return -ENODEV; | 
|  |  | 
|  | /* stake our claim on the iospace */ | 
|  | if( request_region(iobase, 256, card_names[card_type]) == NULL ) | 
|  | { | 
|  | printk(KERN_WARNING "maestro: can't allocate 256 bytes I/O at 0x%4.4x\n", iobase); | 
|  | return -EBUSY; | 
|  | } | 
|  |  | 
|  | /* just to be sure */ | 
|  | pci_set_master(pcidev); | 
|  |  | 
|  | card = kmalloc(sizeof(struct ess_card), GFP_KERNEL); | 
|  | if(card == NULL) | 
|  | { | 
|  | printk(KERN_WARNING "maestro: out of memory\n"); | 
|  | release_region(iobase, 256); | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | memset(card, 0, sizeof(*card)); | 
|  | card->pcidev = pcidev; | 
|  |  | 
|  | pmdev = pm_register(PM_PCI_DEV, PM_PCI_ID(pcidev), | 
|  | maestro_pm_callback); | 
|  | if (pmdev) | 
|  | pmdev->data = card; | 
|  |  | 
|  | card->iobase = iobase; | 
|  | card->card_type = card_type; | 
|  | card->irq = pcidev->irq; | 
|  | card->magic = ESS_CARD_MAGIC; | 
|  | spin_lock_init(&card->lock); | 
|  | init_waitqueue_head(&card->suspend_queue); | 
|  |  | 
|  | card->dock_mute_vol = 50; | 
|  |  | 
|  | /* init our groups of 6 apus */ | 
|  | for(i=0;i<NR_DSPS;i++) | 
|  | { | 
|  | struct ess_state *s=&card->channels[i]; | 
|  |  | 
|  | s->index = i; | 
|  |  | 
|  | s->card = card; | 
|  | init_waitqueue_head(&s->dma_adc.wait); | 
|  | init_waitqueue_head(&s->dma_dac.wait); | 
|  | init_waitqueue_head(&s->open_wait); | 
|  | spin_lock_init(&s->lock); | 
|  | init_MUTEX(&s->open_sem); | 
|  | s->magic = ESS_STATE_MAGIC; | 
|  |  | 
|  | s->apu[0] = 6*i; | 
|  | s->apu[1] = (6*i)+1; | 
|  | s->apu[2] = (6*i)+2; | 
|  | s->apu[3] = (6*i)+3; | 
|  | s->apu[4] = (6*i)+4; | 
|  | s->apu[5] = (6*i)+5; | 
|  |  | 
|  | if(s->dma_adc.ready || s->dma_dac.ready || s->dma_adc.rawbuf) | 
|  | printk("maestro: BOTCH!\n"); | 
|  | /* register devices */ | 
|  | if ((s->dev_audio = register_sound_dsp(&ess_audio_fops, -1)) < 0) | 
|  | break; | 
|  | } | 
|  |  | 
|  | num = i; | 
|  |  | 
|  | /* clear the rest if we ran out of slots to register */ | 
|  | for(;i<NR_DSPS;i++) | 
|  | { | 
|  | struct ess_state *s=&card->channels[i]; | 
|  | s->dev_audio = -1; | 
|  | } | 
|  |  | 
|  | ess = &card->channels[0]; | 
|  |  | 
|  | /* | 
|  | *	Ok card ready. Begin setup proper | 
|  | */ | 
|  |  | 
|  | printk(KERN_INFO "maestro: Configuring %s found at IO 0x%04X IRQ %d\n", | 
|  | card_names[card_type],iobase,card->irq); | 
|  | pci_read_config_dword(pcidev, PCI_SUBSYSTEM_VENDOR_ID, &n); | 
|  | printk(KERN_INFO "maestro:  subvendor id: 0x%08x\n",n); | 
|  |  | 
|  | /* turn off power management unless: | 
|  | *	- the user explicitly asks for it | 
|  | * 		or | 
|  | *		- we're not a 2e, lesser chipps seem to have problems. | 
|  | *		- we're not on our _very_ small whitelist.  some implemenetations | 
|  | *			really don't like the pm code, others require it. | 
|  | *			feel free to expand this as required. | 
|  | */ | 
|  | #define SUBSYSTEM_VENDOR(x) (x&0xffff) | 
|  | if(	(use_pm != 1) && | 
|  | ((card_type != TYPE_MAESTRO2E)	|| (SUBSYSTEM_VENDOR(n) != 0x1028))) | 
|  | use_pm = 0; | 
|  |  | 
|  | if(!use_pm) | 
|  | printk(KERN_INFO "maestro: not attempting power management.\n"); | 
|  | else { | 
|  | if(!parse_power(card,pcidev)) | 
|  | printk(KERN_INFO "maestro: no PCI power management interface found.\n"); | 
|  | else { | 
|  | pci_read_config_dword(pcidev, card->power_regs, &n); | 
|  | printk(KERN_INFO "maestro: PCI power management capability: 0x%x\n",n>>16); | 
|  | } | 
|  | } | 
|  |  | 
|  | maestro_config(card); | 
|  |  | 
|  | if(maestro_ac97_get(card, 0x00)==0x0080) { | 
|  | printk(KERN_ERR "maestro: my goodness!  you seem to have a pt101 codec, which is quite rare.\n" | 
|  | "\tyou should tell someone about this.\n"); | 
|  | } else { | 
|  | maestro_ac97_init(card); | 
|  | } | 
|  |  | 
|  | if ((card->dev_mixer = register_sound_mixer(&ess_mixer_fops, -1)) < 0) { | 
|  | printk("maestro: couldn't register mixer!\n"); | 
|  | } else { | 
|  | memcpy(card->mix.mixer_state,mixer_defaults,sizeof(card->mix.mixer_state)); | 
|  | mixer_push_state(card); | 
|  | } | 
|  |  | 
|  | if((ret=request_irq(card->irq, ess_interrupt, SA_SHIRQ, card_names[card_type], card))) | 
|  | { | 
|  | printk(KERN_ERR "maestro: unable to allocate irq %d,\n", card->irq); | 
|  | unregister_sound_mixer(card->dev_mixer); | 
|  | for(i=0;i<NR_DSPS;i++) | 
|  | { | 
|  | struct ess_state *s = &card->channels[i]; | 
|  | if(s->dev_audio != -1) | 
|  | unregister_sound_dsp(s->dev_audio); | 
|  | } | 
|  | release_region(card->iobase, 256); | 
|  | unregister_reboot_notifier(&maestro_nb); | 
|  | kfree(card); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /* Turn on hardware volume control interrupt. | 
|  | This has to come after we grab the IRQ above, | 
|  | or a crash will result on installation if a button has been pressed, | 
|  | because in that case we'll get an immediate interrupt. */ | 
|  | n = inw(iobase+0x18); | 
|  | n|=(1<<6); | 
|  | outw(n, iobase+0x18); | 
|  |  | 
|  | pci_set_drvdata(pcidev,card); | 
|  | /* now go to sleep 'till something interesting happens */ | 
|  | maestro_power(card,ACPI_D2); | 
|  |  | 
|  | printk(KERN_INFO "maestro: %d channels configured.\n", num); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void maestro_remove(struct pci_dev *pcidev) { | 
|  | struct ess_card *card = pci_get_drvdata(pcidev); | 
|  | int i; | 
|  | u32 n; | 
|  |  | 
|  | /* XXX maybe should force stop bob, but should be all | 
|  | stopped by _release by now */ | 
|  |  | 
|  | /* Turn off hardware volume control interrupt. | 
|  | This has to come before we leave the IRQ below, | 
|  | or a crash results if a button is pressed ! */ | 
|  | n = inw(card->iobase+0x18); | 
|  | n&=~(1<<6); | 
|  | outw(n, card->iobase+0x18); | 
|  |  | 
|  | free_irq(card->irq, card); | 
|  | unregister_sound_mixer(card->dev_mixer); | 
|  | for(i=0;i<NR_DSPS;i++) | 
|  | { | 
|  | struct ess_state *ess = &card->channels[i]; | 
|  | if(ess->dev_audio != -1) | 
|  | unregister_sound_dsp(ess->dev_audio); | 
|  | } | 
|  | /* Goodbye, Mr. Bond. */ | 
|  | maestro_power(card,ACPI_D3); | 
|  | release_region(card->iobase, 256); | 
|  | kfree(card); | 
|  | pci_set_drvdata(pcidev,NULL); | 
|  | } | 
|  |  | 
|  | static struct pci_device_id maestro_pci_tbl[] = { | 
|  | {PCI_VENDOR_ESS, PCI_DEVICE_ID_ESS_ESS1968, PCI_ANY_ID, PCI_ANY_ID, 0, 0, TYPE_MAESTRO2}, | 
|  | {PCI_VENDOR_ESS, PCI_DEVICE_ID_ESS_ESS1978, PCI_ANY_ID, PCI_ANY_ID, 0, 0, TYPE_MAESTRO2E}, | 
|  | {PCI_VENDOR_ESS_OLD, PCI_DEVICE_ID_ESS_ESS0100, PCI_ANY_ID, PCI_ANY_ID, 0, 0, TYPE_MAESTRO}, | 
|  | {0,} | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(pci, maestro_pci_tbl); | 
|  |  | 
|  | static struct pci_driver maestro_pci_driver = { | 
|  | .name	  = "maestro", | 
|  | .id_table = maestro_pci_tbl, | 
|  | .probe	  = maestro_probe, | 
|  | .remove	  = maestro_remove, | 
|  | }; | 
|  |  | 
|  | static int __init init_maestro(void) | 
|  | { | 
|  | int rc; | 
|  |  | 
|  | rc = pci_module_init(&maestro_pci_driver); | 
|  | if (rc < 0) | 
|  | return rc; | 
|  |  | 
|  | if (register_reboot_notifier(&maestro_nb)) | 
|  | printk(KERN_WARNING "maestro: reboot notifier registration failed; may not reboot properly.\n"); | 
|  | #ifdef MODULE | 
|  | printk(version); | 
|  | #endif | 
|  | if (dsps_order < 0)   { | 
|  | dsps_order = 1; | 
|  | printk(KERN_WARNING "maestro: clipping dsps_order to %d\n",dsps_order); | 
|  | } | 
|  | else if (dsps_order > MAX_DSP_ORDER)  { | 
|  | dsps_order = MAX_DSP_ORDER; | 
|  | printk(KERN_WARNING "maestro: clipping dsps_order to %d\n",dsps_order); | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int maestro_notifier(struct notifier_block *nb, unsigned long event, void *buf) | 
|  | { | 
|  | /* this notifier is called when the kernel is really shut down. */ | 
|  | M_printk("maestro: shutting down\n"); | 
|  | /* this will remove all card instances too */ | 
|  | pci_unregister_driver(&maestro_pci_driver); | 
|  | /* XXX dunno about power management */ | 
|  | return NOTIFY_OK; | 
|  | } | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  |  | 
|  | static void cleanup_maestro(void) { | 
|  | M_printk("maestro: unloading\n"); | 
|  | pci_unregister_driver(&maestro_pci_driver); | 
|  | pm_unregister_all(maestro_pm_callback); | 
|  | unregister_reboot_notifier(&maestro_nb); | 
|  | } | 
|  |  | 
|  | /* --------------------------------------------------------------------- */ | 
|  |  | 
|  | void | 
|  | check_suspend(struct ess_card *card) | 
|  | { | 
|  | DECLARE_WAITQUEUE(wait, current); | 
|  |  | 
|  | if(!card->in_suspend) return; | 
|  |  | 
|  | card->in_suspend++; | 
|  | add_wait_queue(&(card->suspend_queue), &wait); | 
|  | current->state = TASK_UNINTERRUPTIBLE; | 
|  | schedule(); | 
|  | remove_wait_queue(&(card->suspend_queue), &wait); | 
|  | current->state = TASK_RUNNING; | 
|  | } | 
|  |  | 
|  | static int | 
|  | maestro_suspend(struct ess_card *card) | 
|  | { | 
|  | unsigned long flags; | 
|  | int i,j; | 
|  |  | 
|  | spin_lock_irqsave(&card->lock,flags); /* over-kill */ | 
|  |  | 
|  | M_printk("maestro: apm in dev %p\n",card); | 
|  |  | 
|  | /* we have to read from the apu regs, need | 
|  | to power it up */ | 
|  | maestro_power(card,ACPI_D0); | 
|  |  | 
|  | for(i=0;i<NR_DSPS;i++) { | 
|  | struct ess_state *s = &card->channels[i]; | 
|  |  | 
|  | if(s->dev_audio == -1) | 
|  | continue; | 
|  |  | 
|  | M_printk("maestro: stopping apus for device %d\n",i); | 
|  | stop_dac(s); | 
|  | stop_adc(s); | 
|  | for(j=0;j<6;j++) | 
|  | card->apu_map[s->apu[j]][5]=apu_get_register(s,j,5); | 
|  |  | 
|  | } | 
|  |  | 
|  | /* get rid of interrupts? */ | 
|  | if( card->dsps_open > 0) | 
|  | stop_bob(&card->channels[0]); | 
|  |  | 
|  | card->in_suspend++; | 
|  |  | 
|  | spin_unlock_irqrestore(&card->lock,flags); | 
|  |  | 
|  | /* we trust in the bios to power down the chip on suspend. | 
|  | * XXX I'm also not sure that in_suspend will protect | 
|  | * against all reg accesses from here on out. | 
|  | */ | 
|  | return 0; | 
|  | } | 
|  | static int | 
|  | maestro_resume(struct ess_card *card) | 
|  | { | 
|  | unsigned long flags; | 
|  | int i; | 
|  |  | 
|  | spin_lock_irqsave(&card->lock,flags); /* over-kill */ | 
|  |  | 
|  | card->in_suspend = 0; | 
|  |  | 
|  | M_printk("maestro: resuming card at %p\n",card); | 
|  |  | 
|  | /* restore all our config */ | 
|  | maestro_config(card); | 
|  | /* need to restore the base pointers.. */ | 
|  | if(card->dmapages) | 
|  | set_base_registers(&card->channels[0],card->dmapages); | 
|  |  | 
|  | mixer_push_state(card); | 
|  |  | 
|  | /* set each channels' apu control registers before | 
|  | * restoring audio | 
|  | */ | 
|  | for(i=0;i<NR_DSPS;i++) { | 
|  | struct ess_state *s = &card->channels[i]; | 
|  | int chan,reg; | 
|  |  | 
|  | if(s->dev_audio == -1) | 
|  | continue; | 
|  |  | 
|  | for(chan = 0 ; chan < 6 ; chan++) { | 
|  | wave_set_register(s,s->apu[chan]<<3,s->apu_base[chan]); | 
|  | for(reg = 1 ; reg < NR_APU_REGS ; reg++) | 
|  | apu_set_register(s,chan,reg,s->card->apu_map[s->apu[chan]][reg]); | 
|  | } | 
|  | for(chan = 0 ; chan < 6 ; chan++) | 
|  | apu_set_register(s,chan,0,s->card->apu_map[s->apu[chan]][0] & 0xFF0F); | 
|  | } | 
|  |  | 
|  | /* now we flip on the music */ | 
|  |  | 
|  | if( card->dsps_open <= 0) { | 
|  | /* this card's idle */ | 
|  | maestro_power(card,ACPI_D2); | 
|  | } else { | 
|  | /* ok, we're actually playing things on | 
|  | this card */ | 
|  | maestro_power(card,ACPI_D0); | 
|  | start_bob(&card->channels[0]); | 
|  | for(i=0;i<NR_DSPS;i++) { | 
|  | struct ess_state *s = &card->channels[i]; | 
|  |  | 
|  | /* these use the apu_mode, and can handle | 
|  | spurious calls */ | 
|  | start_dac(s); | 
|  | start_adc(s); | 
|  | } | 
|  | } | 
|  |  | 
|  | spin_unlock_irqrestore(&card->lock,flags); | 
|  |  | 
|  | /* all right, we think things are ready, | 
|  | wake up people who were using the device | 
|  | when we suspended */ | 
|  | wake_up(&(card->suspend_queue)); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int | 
|  | maestro_pm_callback(struct pm_dev *dev, pm_request_t rqst, void *data) | 
|  | { | 
|  | struct ess_card *card = (struct ess_card*) dev->data; | 
|  |  | 
|  | if ( ! card ) goto out; | 
|  |  | 
|  | M_printk("maestro: pm event 0x%x received for card %p\n", rqst, card); | 
|  |  | 
|  | switch (rqst) { | 
|  | case PM_SUSPEND: | 
|  | maestro_suspend(card); | 
|  | break; | 
|  | case PM_RESUME: | 
|  | maestro_resume(card); | 
|  | break; | 
|  | /* | 
|  | * we'd also like to find out about | 
|  | * power level changes because some biosen | 
|  | * do mean things to the maestro when they | 
|  | * change their power state. | 
|  | */ | 
|  | } | 
|  | out: | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | module_init(init_maestro); | 
|  | module_exit(cleanup_maestro); |