WARNING: sound/oss/nm256_audio.o - Section mismatch: reference to .init.text:nm256_peek_for_sig from .text between 'nm256_install' (at offset 0x3ba4) and 'nm256_probe' WARNING: sound/oss/nm256_audio.o - Section mismatch: reference to .init.text:nm256_peek_for_sig from .text between 'nm256_install' (at offset 0x3bac) and 'nm256_probe' WARNING: sound/oss/nm256_audio.o - Section mismatch: reference to .init.text: from .text between 'nm256_install' (at offset 0x3dcc) and 'nm256_probe' WARNING: sound/oss/nm256_audio.o - Section mismatch: reference to .init.text: from .text between 'nm256_install' (at offset 0x3dd0) and 'nm256_probe' Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
		
			
				
	
	
		
			1663 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1663 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* 
 | |
|  * Audio driver for the NeoMagic 256AV and 256ZX chipsets in native
 | |
|  * mode, with AC97 mixer support.
 | |
|  *
 | |
|  * Overall design and parts of this code stolen from vidc_*.c and
 | |
|  * skeleton.c.
 | |
|  *
 | |
|  * Yeah, there are a lot of magic constants in here.  You tell ME what
 | |
|  * they are.  I just get this stuff psychically, remember? 
 | |
|  *
 | |
|  * This driver was written by someone who wishes to remain anonymous. 
 | |
|  * It is in the public domain, so share and enjoy.  Try to make a profit
 | |
|  * off of it; go on, I dare you.  
 | |
|  *
 | |
|  * Changes:
 | |
|  * 11-10-2000	Bartlomiej Zolnierkiewicz <bkz@linux-ide.org>
 | |
|  *		Added some __init
 | |
|  * 19-04-2001	Marcus Meissner <mm@caldera.de>
 | |
|  *		Ported to 2.4 PCI API.
 | |
|  */
 | |
| 
 | |
| #include <linux/pci.h>
 | |
| #include <linux/init.h>
 | |
| #include <linux/interrupt.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/delay.h>
 | |
| #include <linux/spinlock.h>
 | |
| #include "sound_config.h"
 | |
| 
 | |
| static int nm256_debug;
 | |
| static int force_load;
 | |
| 
 | |
| #include "nm256.h"
 | |
| #include "nm256_coeff.h"
 | |
| 
 | |
| /* 
 | |
|  * The size of the playback reserve.  When the playback buffer has less
 | |
|  * than NM256_PLAY_WMARK_SIZE bytes to output, we request a new
 | |
|  * buffer.
 | |
|  */
 | |
| #define NM256_PLAY_WMARK_SIZE 512
 | |
| 
 | |
| static struct audio_driver nm256_audio_driver;
 | |
| 
 | |
| static int nm256_grabInterrupt (struct nm256_info *card);
 | |
| static int nm256_releaseInterrupt (struct nm256_info *card);
 | |
| static irqreturn_t nm256_interrupt (int irq, void *dev_id, struct pt_regs *dummy);
 | |
| static irqreturn_t nm256_interrupt_zx (int irq, void *dev_id, struct pt_regs *dummy);
 | |
| 
 | |
| /* These belong in linux/pci.h. */
 | |
| #define PCI_DEVICE_ID_NEOMAGIC_NM256AV_AUDIO 0x8005
 | |
| #define PCI_DEVICE_ID_NEOMAGIC_NM256ZX_AUDIO 0x8006
 | |
| #define PCI_DEVICE_ID_NEOMAGIC_NM256XL_PLUS_AUDIO 0x8016
 | |
| 
 | |
| /* List of cards.  */
 | |
| static struct nm256_info *nmcard_list;
 | |
| 
 | |
| /* Release the mapped-in memory for CARD.  */
 | |
| static void
 | |
| nm256_release_ports (struct nm256_info *card)
 | |
| {
 | |
|     int x;
 | |
| 
 | |
|     for (x = 0; x < 2; x++) {
 | |
| 	if (card->port[x].ptr != NULL) {
 | |
| 	    iounmap (card->port[x].ptr);
 | |
| 	    card->port[x].ptr = NULL;
 | |
| 	}
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Map in the memory ports for CARD, if they aren't already mapped in
 | |
|  * and have been configured.  If successful, a zero value is returned;
 | |
|  * otherwise any previously mapped-in areas are released and a non-zero
 | |
|  * value is returned.
 | |
|  *
 | |
|  * This is invoked twice, once for each port.  Ideally it would only be
 | |
|  * called once, but we now need to map in the second port in order to
 | |
|  * check how much memory the card has on the 256ZX.
 | |
|  */
 | |
| static int
 | |
| nm256_remap_ports (struct nm256_info *card)
 | |
| {
 | |
|     int x;
 | |
| 
 | |
|     for (x = 0; x < 2; x++) {
 | |
| 	if (card->port[x].ptr == NULL && card->port[x].end_offset > 0) {
 | |
| 	    u32 physaddr 
 | |
| 		= card->port[x].physaddr + card->port[x].start_offset;
 | |
| 	    u32 size 
 | |
| 		= card->port[x].end_offset - card->port[x].start_offset;
 | |
| 
 | |
| 	    card->port[x].ptr = ioremap_nocache (physaddr, size);
 | |
| 						  
 | |
| 	    if (card->port[x].ptr == NULL) {
 | |
| 		printk (KERN_ERR "NM256: Unable to remap port %d\n", x + 1);
 | |
| 		nm256_release_ports (card);
 | |
| 		return -1;
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /* Locate the card in our list. */
 | |
| static struct nm256_info *
 | |
| nm256_find_card (int dev)
 | |
| {
 | |
|     struct nm256_info *card;
 | |
| 
 | |
|     for (card = nmcard_list; card != NULL; card = card->next_card)
 | |
| 	if (card->dev[0] == dev || card->dev[1] == dev)
 | |
| 	    return card;
 | |
| 
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Ditto, but find the card struct corresponding to the mixer device DEV 
 | |
|  * instead. 
 | |
|  */
 | |
| static struct nm256_info *
 | |
| nm256_find_card_for_mixer (int dev)
 | |
| {
 | |
|     struct nm256_info *card;
 | |
| 
 | |
|     for (card = nmcard_list; card != NULL; card = card->next_card)
 | |
| 	if (card->mixer_oss_dev == dev)
 | |
| 	    return card;
 | |
| 
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static int usecache;
 | |
| static int buffertop;
 | |
| 
 | |
| /* Check to see if we're using the bank of cached coefficients. */
 | |
| static int
 | |
| nm256_cachedCoefficients (struct nm256_info *card)
 | |
| {
 | |
|     return usecache;
 | |
| }
 | |
| 
 | |
| /* The actual rates supported by the card. */
 | |
| static int samplerates[9] = {
 | |
|     8000, 11025, 16000, 22050, 24000, 32000, 44100, 48000, 99999999
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * Set the card samplerate, word size and stereo mode to correspond to
 | |
|  * the settings in the CARD struct for the specified device in DEV.
 | |
|  * We keep two separate sets of information, one for each device; the
 | |
|  * hardware is not actually configured until a read or write is
 | |
|  * attempted.
 | |
|  */
 | |
| 
 | |
| static int
 | |
| nm256_setInfo (int dev, struct nm256_info *card)
 | |
| {
 | |
|     int x;
 | |
|     int w;
 | |
|     int targetrate;
 | |
| 
 | |
|     if (card->dev[0] == dev)
 | |
| 	w = 0;
 | |
|     else if (card->dev[1] == dev)
 | |
| 	w = 1;
 | |
|     else
 | |
| 	return -ENODEV;
 | |
| 
 | |
|     targetrate = card->sinfo[w].samplerate;
 | |
| 
 | |
|     if ((card->sinfo[w].bits != 8 && card->sinfo[w].bits != 16)
 | |
| 	|| targetrate < samplerates[0]
 | |
| 	|| targetrate > samplerates[7])
 | |
| 	return -EINVAL;
 | |
| 
 | |
|     for (x = 0; x < 8; x++)
 | |
| 	if (targetrate < ((samplerates[x] + samplerates[x + 1]) / 2))
 | |
| 	    break;
 | |
| 
 | |
|     if (x < 8) {
 | |
| 	u8 ratebits = ((x << 4) & NM_RATE_MASK);
 | |
| 	if (card->sinfo[w].bits == 16)
 | |
| 	    ratebits |= NM_RATE_BITS_16;
 | |
| 	if (card->sinfo[w].stereo)
 | |
| 	    ratebits |= NM_RATE_STEREO;
 | |
| 
 | |
| 	card->sinfo[w].samplerate = samplerates[x];
 | |
| 
 | |
| 
 | |
| 	if (card->dev_for_play == dev && card->playing) {
 | |
| 	    if (nm256_debug)
 | |
| 		printk (KERN_DEBUG "Setting play ratebits to 0x%x\n",
 | |
| 			ratebits);
 | |
| 	    nm256_loadCoefficient (card, 0, x);
 | |
| 	    nm256_writePort8 (card, 2,
 | |
| 			      NM_PLAYBACK_REG_OFFSET + NM_RATE_REG_OFFSET,
 | |
| 			      ratebits);
 | |
| 	}
 | |
| 
 | |
| 	if (card->dev_for_record == dev && card->recording) {
 | |
| 	    if (nm256_debug)
 | |
| 		printk (KERN_DEBUG "Setting record ratebits to 0x%x\n",
 | |
| 			ratebits);
 | |
| 	    nm256_loadCoefficient (card, 1, x);
 | |
| 	    nm256_writePort8 (card, 2,
 | |
| 			      NM_RECORD_REG_OFFSET + NM_RATE_REG_OFFSET,
 | |
| 			      ratebits);
 | |
| 	}
 | |
| 	return 0;
 | |
|     }
 | |
|     else
 | |
| 	return -EINVAL;
 | |
| }
 | |
| 
 | |
| /* Start the play process going. */
 | |
| static void
 | |
| startPlay (struct nm256_info *card)
 | |
| {
 | |
|     if (! card->playing) {
 | |
| 	card->playing = 1;
 | |
| 	if (nm256_grabInterrupt (card) == 0) {
 | |
| 	    nm256_setInfo (card->dev_for_play, card);
 | |
| 
 | |
| 	    /* Enable playback engine and interrupts. */
 | |
| 	    nm256_writePort8 (card, 2, NM_PLAYBACK_ENABLE_REG,
 | |
| 			      NM_PLAYBACK_ENABLE_FLAG | NM_PLAYBACK_FREERUN);
 | |
| 
 | |
| 	    /* Enable both channels. */
 | |
| 	    nm256_writePort16 (card, 2, NM_AUDIO_MUTE_REG, 0x0);
 | |
| 	}
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Request one chunk of AMT bytes from the recording device.  When the
 | |
|  * operation is complete, the data will be copied into BUFFER and the
 | |
|  * function DMAbuf_inputintr will be invoked.
 | |
|  */
 | |
| 
 | |
| static void
 | |
| nm256_startRecording (struct nm256_info *card, char *buffer, u32 amt)
 | |
| {
 | |
|     u32 endpos;
 | |
|     int enableEngine = 0;
 | |
|     u32 ringsize = card->recordBufferSize;
 | |
|     unsigned long flags;
 | |
| 
 | |
|     if (amt > (ringsize / 2)) {
 | |
| 	/*
 | |
| 	 * Of course this won't actually work right, because the
 | |
| 	 * caller is going to assume we will give what we got asked
 | |
| 	 * for.
 | |
| 	 */
 | |
| 	printk (KERN_ERR "NM256: Read request too large: %d\n", amt);
 | |
| 	amt = ringsize / 2;
 | |
|     }
 | |
| 
 | |
|     if (amt < 8) {
 | |
| 	printk (KERN_ERR "NM256: Read request too small; %d\n", amt);
 | |
| 	return;
 | |
|     }
 | |
| 
 | |
|     spin_lock_irqsave(&card->lock,flags);
 | |
|     /*
 | |
|      * If we're not currently recording, set up the start and end registers
 | |
|      * for the recording engine.
 | |
|      */
 | |
|     if (! card->recording) {
 | |
| 	card->recording = 1;
 | |
| 	if (nm256_grabInterrupt (card) == 0) {
 | |
| 	    card->curRecPos = 0;
 | |
| 	    nm256_setInfo (card->dev_for_record, card);
 | |
| 	    nm256_writePort32 (card, 2, NM_RBUFFER_START, card->abuf2);
 | |
| 	    nm256_writePort32 (card, 2, NM_RBUFFER_END,
 | |
| 				 card->abuf2 + ringsize);
 | |
| 
 | |
| 	    nm256_writePort32 (card, 2, NM_RBUFFER_CURRP,
 | |
| 				 card->abuf2 + card->curRecPos);
 | |
| 	    enableEngine = 1;
 | |
| 	}
 | |
| 	else {
 | |
| 	    /* Not sure what else to do here.  */
 | |
| 	    spin_unlock_irqrestore(&card->lock,flags);
 | |
| 	    return;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     /* 
 | |
|      * If we happen to go past the end of the buffer a bit (due to a
 | |
|      * delayed interrupt) it's OK.  So might as well set the watermark
 | |
|      * right at the end of the data we want.
 | |
|      */
 | |
|     endpos = card->abuf2 + ((card->curRecPos + amt) % ringsize);
 | |
| 
 | |
|     card->recBuf = buffer;
 | |
|     card->requestedRecAmt = amt;
 | |
|     nm256_writePort32 (card, 2, NM_RBUFFER_WMARK, endpos);
 | |
|     /* Enable recording engine and interrupts. */
 | |
|     if (enableEngine)
 | |
| 	nm256_writePort8 (card, 2, NM_RECORD_ENABLE_REG,
 | |
| 			    NM_RECORD_ENABLE_FLAG | NM_RECORD_FREERUN);
 | |
| 
 | |
|     spin_unlock_irqrestore(&card->lock,flags);
 | |
| }
 | |
| 
 | |
| /* Stop the play engine. */
 | |
| static void
 | |
| stopPlay (struct nm256_info *card)
 | |
| {
 | |
|     /* Shut off sound from both channels. */
 | |
|     nm256_writePort16 (card, 2, NM_AUDIO_MUTE_REG,
 | |
| 		       NM_AUDIO_MUTE_LEFT | NM_AUDIO_MUTE_RIGHT);
 | |
|     /* Disable play engine. */
 | |
|     nm256_writePort8 (card, 2, NM_PLAYBACK_ENABLE_REG, 0);
 | |
|     if (card->playing) {
 | |
| 	nm256_releaseInterrupt (card);
 | |
| 
 | |
| 	/* Reset the relevant state bits. */
 | |
| 	card->playing = 0;
 | |
| 	card->curPlayPos = 0;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Stop recording. */
 | |
| static void
 | |
| stopRecord (struct nm256_info *card)
 | |
| {
 | |
|     /* Disable recording engine. */
 | |
|     nm256_writePort8 (card, 2, NM_RECORD_ENABLE_REG, 0);
 | |
| 
 | |
|     if (card->recording) {
 | |
| 	nm256_releaseInterrupt (card);
 | |
| 
 | |
| 	card->recording = 0;
 | |
| 	card->curRecPos = 0;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Ring buffers, man.  That's where the hip-hop, wild-n-wooly action's at.
 | |
|  * 1972?  (Well, I suppose it was cheep-n-easy to implement.)
 | |
|  *
 | |
|  * Write AMT bytes of BUFFER to the playback ring buffer, and start the
 | |
|  * playback engine running.  It will only accept up to 1/2 of the total
 | |
|  * size of the ring buffer.  No check is made that we're about to overwrite
 | |
|  * the currently-playing sample.
 | |
|  */
 | |
| 
 | |
| static void
 | |
| nm256_write_block (struct nm256_info *card, char *buffer, u32 amt)
 | |
| {
 | |
|     u32 ringsize = card->playbackBufferSize;
 | |
|     u32 endstop;
 | |
|     unsigned long flags;
 | |
| 
 | |
|     if (amt > (ringsize / 2)) {
 | |
| 	printk (KERN_ERR "NM256: Write request too large: %d\n", amt);
 | |
| 	amt = (ringsize / 2);
 | |
|     }
 | |
| 
 | |
|     if (amt < NM256_PLAY_WMARK_SIZE) {
 | |
| 	printk (KERN_ERR "NM256: Write request too small: %d\n", amt);
 | |
| 	return;
 | |
|     }
 | |
| 
 | |
|     card->curPlayPos %= ringsize;
 | |
| 
 | |
|     card->requested_amt = amt;
 | |
| 
 | |
|     spin_lock_irqsave(&card->lock,flags);
 | |
| 
 | |
|     if ((card->curPlayPos + amt) >= ringsize) {
 | |
| 	u32 rem = ringsize - card->curPlayPos;
 | |
| 
 | |
| 	nm256_writeBuffer8 (card, buffer, 1,
 | |
| 			      card->abuf1 + card->curPlayPos,
 | |
| 			      rem);
 | |
| 	if (amt > rem)
 | |
| 	    nm256_writeBuffer8 (card, buffer + rem, 1, card->abuf1,
 | |
| 				  amt - rem);
 | |
|     } 
 | |
|     else
 | |
| 	nm256_writeBuffer8 (card, buffer, 1,
 | |
| 			      card->abuf1 + card->curPlayPos,
 | |
| 			      amt);
 | |
| 
 | |
|     /*
 | |
|      * Setup the start-n-stop-n-limit registers, and start that engine
 | |
|      * goin'. 
 | |
|      *
 | |
|      * Normally we just let it wrap around to avoid the click-click
 | |
|      * action scene.
 | |
|      */
 | |
|     if (! card->playing) {
 | |
| 	/* The PBUFFER_END register in this case points to one sample
 | |
| 	   before the end of the buffer. */
 | |
| 	int w = (card->dev_for_play == card->dev[0] ? 0 : 1);
 | |
| 	int sampsize = (card->sinfo[w].bits == 16 ? 2 : 1);
 | |
| 
 | |
| 	if (card->sinfo[w].stereo)
 | |
| 	    sampsize *= 2;
 | |
| 
 | |
| 	/* Need to set the not-normally-changing-registers up. */
 | |
| 	nm256_writePort32 (card, 2, NM_PBUFFER_START,
 | |
| 			     card->abuf1 + card->curPlayPos);
 | |
| 	nm256_writePort32 (card, 2, NM_PBUFFER_END,
 | |
| 			     card->abuf1 + ringsize - sampsize);
 | |
| 	nm256_writePort32 (card, 2, NM_PBUFFER_CURRP,
 | |
| 			     card->abuf1 + card->curPlayPos);
 | |
|     }
 | |
|     endstop = (card->curPlayPos + amt - NM256_PLAY_WMARK_SIZE) % ringsize;
 | |
|     nm256_writePort32 (card, 2, NM_PBUFFER_WMARK, card->abuf1 + endstop);
 | |
| 
 | |
|     if (! card->playing)
 | |
| 	startPlay (card);
 | |
| 
 | |
|     spin_unlock_irqrestore(&card->lock,flags);
 | |
| }
 | |
| 
 | |
| /*  We just got a card playback interrupt; process it.  */
 | |
| static void
 | |
| nm256_get_new_block (struct nm256_info *card)
 | |
| {
 | |
|     /* Check to see how much got played so far. */
 | |
|     u32 amt = nm256_readPort32 (card, 2, NM_PBUFFER_CURRP) - card->abuf1;
 | |
| 
 | |
|     if (amt >= card->playbackBufferSize) {
 | |
| 	printk (KERN_ERR "NM256: Sound playback pointer invalid!\n");
 | |
| 	amt = 0;
 | |
|     }
 | |
| 
 | |
|     if (amt < card->curPlayPos)
 | |
| 	amt = (card->playbackBufferSize - card->curPlayPos) + amt;
 | |
|     else
 | |
| 	amt -= card->curPlayPos;
 | |
| 
 | |
|     if (card->requested_amt > (amt + NM256_PLAY_WMARK_SIZE)) {
 | |
| 	u32 endstop =
 | |
| 	    card->curPlayPos + card->requested_amt - NM256_PLAY_WMARK_SIZE;
 | |
| 	nm256_writePort32 (card, 2, NM_PBUFFER_WMARK, card->abuf1 + endstop);
 | |
|     } 
 | |
|     else {
 | |
| 	card->curPlayPos += card->requested_amt;
 | |
| 	/* Get a new block to write.  This will eventually invoke
 | |
| 	   nm256_write_block () or stopPlay ().  */
 | |
| 	DMAbuf_outputintr (card->dev_for_play, 1);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Read the last-recorded block from the ring buffer, copy it into the
 | |
|  * saved buffer pointer, and invoke DMAuf_inputintr() with the recording
 | |
|  * device. 
 | |
|  */
 | |
| 
 | |
| static void
 | |
| nm256_read_block (struct nm256_info *card)
 | |
| {
 | |
|     /* Grab the current position of the recording pointer. */
 | |
|     u32 currptr = nm256_readPort32 (card, 2, NM_RBUFFER_CURRP) - card->abuf2;
 | |
|     u32 amtToRead = card->requestedRecAmt;
 | |
|     u32 ringsize = card->recordBufferSize;
 | |
| 
 | |
|     if (currptr >= card->recordBufferSize) {
 | |
| 	printk (KERN_ERR "NM256: Sound buffer record pointer invalid!\n");
 | |
|         currptr = 0;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * This test is probably redundant; we shouldn't be here unless
 | |
|      * it's true.
 | |
|      */
 | |
|     if (card->recording) {
 | |
| 	/* If we wrapped around, copy everything from the start of our
 | |
| 	   recording buffer to the end of the buffer. */
 | |
| 	if (currptr < card->curRecPos) {
 | |
| 	    u32 amt = min (ringsize - card->curRecPos, amtToRead);
 | |
| 
 | |
| 	    nm256_readBuffer8 (card, card->recBuf, 1,
 | |
| 				 card->abuf2 + card->curRecPos,
 | |
| 				 amt);
 | |
| 	    amtToRead -= amt;
 | |
| 	    card->curRecPos += amt;
 | |
| 	    card->recBuf += amt;
 | |
| 	    if (card->curRecPos == ringsize)
 | |
| 		card->curRecPos = 0;
 | |
| 	}
 | |
| 
 | |
| 	if ((card->curRecPos < currptr) && (amtToRead > 0)) {
 | |
| 	    u32 amt = min (currptr - card->curRecPos, amtToRead);
 | |
| 	    nm256_readBuffer8 (card, card->recBuf, 1,
 | |
| 				 card->abuf2 + card->curRecPos, amt);
 | |
| 	    card->curRecPos = ((card->curRecPos + amt) % ringsize);
 | |
| 	}
 | |
| 	card->recBuf = NULL;
 | |
| 	card->requestedRecAmt = 0;
 | |
| 	DMAbuf_inputintr (card->dev_for_record);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Initialize the hardware. 
 | |
|  */
 | |
| static void
 | |
| nm256_initHw (struct nm256_info *card)
 | |
| {
 | |
|     /* Reset everything. */
 | |
|     nm256_writePort8 (card, 2, 0x0, 0x11);
 | |
|     nm256_writePort16 (card, 2, 0x214, 0);
 | |
| 
 | |
|     stopRecord (card);
 | |
|     stopPlay (card);
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Handle a potential interrupt for the device referred to by DEV_ID. 
 | |
|  *
 | |
|  * I don't like the cut-n-paste job here either between the two routines,
 | |
|  * but there are sufficient differences between the two interrupt handlers
 | |
|  * that parameterizing it isn't all that great either.  (Could use a macro,
 | |
|  * I suppose...yucky bleah.)
 | |
|  */
 | |
| 
 | |
| static irqreturn_t
 | |
| nm256_interrupt (int irq, void *dev_id, struct pt_regs *dummy)
 | |
| {
 | |
|     struct nm256_info *card = (struct nm256_info *)dev_id;
 | |
|     u16 status;
 | |
|     static int badintrcount;
 | |
|     int handled = 0;
 | |
| 
 | |
|     if ((card == NULL) || (card->magsig != NM_MAGIC_SIG)) {
 | |
| 	printk (KERN_ERR "NM256: Bad card pointer\n");
 | |
| 	return IRQ_NONE;
 | |
|     }
 | |
| 
 | |
|     status = nm256_readPort16 (card, 2, NM_INT_REG);
 | |
| 
 | |
|     /* Not ours. */
 | |
|     if (status == 0) {
 | |
| 	if (badintrcount++ > 1000) {
 | |
| 	    /*
 | |
| 	     * I'm not sure if the best thing is to stop the card from
 | |
| 	     * playing or just release the interrupt (after all, we're in
 | |
| 	     * a bad situation, so doing fancy stuff may not be such a good
 | |
| 	     * idea).
 | |
| 	     *
 | |
| 	     * I worry about the card engine continuing to play noise
 | |
| 	     * over and over, however--that could become a very
 | |
| 	     * obnoxious problem.  And we know that when this usually
 | |
| 	     * happens things are fairly safe, it just means the user's
 | |
| 	     * inserted a PCMCIA card and someone's spamming us with IRQ 9s.
 | |
| 	     */
 | |
| 
 | |
| 	    handled = 1;
 | |
| 	    if (card->playing)
 | |
| 		stopPlay (card);
 | |
| 	    if (card->recording)
 | |
| 		stopRecord (card);
 | |
| 	    badintrcount = 0;
 | |
| 	}
 | |
| 	return IRQ_RETVAL(handled);
 | |
|     }
 | |
| 
 | |
|     badintrcount = 0;
 | |
| 
 | |
|     /* Rather boring; check for individual interrupts and process them. */
 | |
| 
 | |
|     if (status & NM_PLAYBACK_INT) {
 | |
| 	handled = 1;
 | |
| 	status &= ~NM_PLAYBACK_INT;
 | |
| 	NM_ACK_INT (card, NM_PLAYBACK_INT);
 | |
| 
 | |
| 	if (card->playing)
 | |
| 	    nm256_get_new_block (card);
 | |
|     }
 | |
| 
 | |
|     if (status & NM_RECORD_INT) {
 | |
| 	handled = 1;
 | |
| 	status &= ~NM_RECORD_INT;
 | |
| 	NM_ACK_INT (card, NM_RECORD_INT);
 | |
| 
 | |
| 	if (card->recording)
 | |
| 	    nm256_read_block (card);
 | |
|     }
 | |
| 
 | |
|     if (status & NM_MISC_INT_1) {
 | |
| 	u8 cbyte;
 | |
| 
 | |
| 	handled = 1;
 | |
| 	status &= ~NM_MISC_INT_1;
 | |
| 	printk (KERN_ERR "NM256: Got misc interrupt #1\n");
 | |
| 	NM_ACK_INT (card, NM_MISC_INT_1);
 | |
| 	nm256_writePort16 (card, 2, NM_INT_REG, 0x8000);
 | |
| 	cbyte = nm256_readPort8 (card, 2, 0x400);
 | |
| 	nm256_writePort8 (card, 2, 0x400, cbyte | 2);
 | |
|     }
 | |
| 
 | |
|     if (status & NM_MISC_INT_2) {
 | |
| 	u8 cbyte;
 | |
| 
 | |
| 	handled = 1;
 | |
| 	status &= ~NM_MISC_INT_2;
 | |
| 	printk (KERN_ERR "NM256: Got misc interrupt #2\n");
 | |
| 	NM_ACK_INT (card, NM_MISC_INT_2);
 | |
| 	cbyte = nm256_readPort8 (card, 2, 0x400);
 | |
| 	nm256_writePort8 (card, 2, 0x400, cbyte & ~2);
 | |
|     }
 | |
| 
 | |
|     /* Unknown interrupt. */
 | |
|     if (status) {
 | |
| 	handled = 1;
 | |
| 	printk (KERN_ERR "NM256: Fire in the hole! Unknown status 0x%x\n",
 | |
| 		status);
 | |
| 	/* Pray. */
 | |
| 	NM_ACK_INT (card, status);
 | |
|     }
 | |
|     return IRQ_RETVAL(handled);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Handle a potential interrupt for the device referred to by DEV_ID.
 | |
|  * This handler is for the 256ZX, and is very similar to the non-ZX
 | |
|  * routine.
 | |
|  */
 | |
| 
 | |
| static irqreturn_t
 | |
| nm256_interrupt_zx (int irq, void *dev_id, struct pt_regs *dummy)
 | |
| {
 | |
|     struct nm256_info *card = (struct nm256_info *)dev_id;
 | |
|     u32 status;
 | |
|     static int badintrcount;
 | |
|     int handled = 0;
 | |
| 
 | |
|     if ((card == NULL) || (card->magsig != NM_MAGIC_SIG)) {
 | |
| 	printk (KERN_ERR "NM256: Bad card pointer\n");
 | |
| 	return IRQ_NONE;
 | |
|     }
 | |
| 
 | |
|     status = nm256_readPort32 (card, 2, NM_INT_REG);
 | |
| 
 | |
|     /* Not ours. */
 | |
|     if (status == 0) {
 | |
| 	if (badintrcount++ > 1000) {
 | |
| 	    printk (KERN_ERR "NM256: Releasing interrupt, over 1000 invalid interrupts\n");
 | |
| 	    /*
 | |
| 	     * I'm not sure if the best thing is to stop the card from
 | |
| 	     * playing or just release the interrupt (after all, we're in
 | |
| 	     * a bad situation, so doing fancy stuff may not be such a good
 | |
| 	     * idea).
 | |
| 	     *
 | |
| 	     * I worry about the card engine continuing to play noise
 | |
| 	     * over and over, however--that could become a very
 | |
| 	     * obnoxious problem.  And we know that when this usually
 | |
| 	     * happens things are fairly safe, it just means the user's
 | |
| 	     * inserted a PCMCIA card and someone's spamming us with 
 | |
| 	     * IRQ 9s.
 | |
| 	     */
 | |
| 
 | |
| 	    handled = 1;
 | |
| 	    if (card->playing)
 | |
| 		stopPlay (card);
 | |
| 	    if (card->recording)
 | |
| 		stopRecord (card);
 | |
| 	    badintrcount = 0;
 | |
| 	}
 | |
| 	return IRQ_RETVAL(handled);
 | |
|     }
 | |
| 
 | |
|     badintrcount = 0;
 | |
| 
 | |
|     /* Rather boring; check for individual interrupts and process them. */
 | |
| 
 | |
|     if (status & NM2_PLAYBACK_INT) {
 | |
| 	handled = 1;
 | |
| 	status &= ~NM2_PLAYBACK_INT;
 | |
| 	NM2_ACK_INT (card, NM2_PLAYBACK_INT);
 | |
| 
 | |
| 	if (card->playing)
 | |
| 	    nm256_get_new_block (card);
 | |
|     }
 | |
| 
 | |
|     if (status & NM2_RECORD_INT) {
 | |
| 	handled = 1;
 | |
| 	status &= ~NM2_RECORD_INT;
 | |
| 	NM2_ACK_INT (card, NM2_RECORD_INT);
 | |
| 
 | |
| 	if (card->recording)
 | |
| 	    nm256_read_block (card);
 | |
|     }
 | |
| 
 | |
|     if (status & NM2_MISC_INT_1) {
 | |
| 	u8 cbyte;
 | |
| 
 | |
| 	handled = 1;
 | |
| 	status &= ~NM2_MISC_INT_1;
 | |
| 	printk (KERN_ERR "NM256: Got misc interrupt #1\n");
 | |
| 	NM2_ACK_INT (card, NM2_MISC_INT_1);
 | |
| 	cbyte = nm256_readPort8 (card, 2, 0x400);
 | |
| 	nm256_writePort8 (card, 2, 0x400, cbyte | 2);
 | |
|     }
 | |
| 
 | |
|     if (status & NM2_MISC_INT_2) {
 | |
| 	u8 cbyte;
 | |
| 
 | |
| 	handled = 1;
 | |
| 	status &= ~NM2_MISC_INT_2;
 | |
| 	printk (KERN_ERR "NM256: Got misc interrupt #2\n");
 | |
| 	NM2_ACK_INT (card, NM2_MISC_INT_2);
 | |
| 	cbyte = nm256_readPort8 (card, 2, 0x400);
 | |
| 	nm256_writePort8 (card, 2, 0x400, cbyte & ~2);
 | |
|     }
 | |
| 
 | |
|     /* Unknown interrupt. */
 | |
|     if (status) {
 | |
| 	handled = 1;
 | |
| 	printk (KERN_ERR "NM256: Fire in the hole! Unknown status 0x%x\n",
 | |
| 		status);
 | |
| 	/* Pray. */
 | |
| 	NM2_ACK_INT (card, status);
 | |
|     }
 | |
|     return IRQ_RETVAL(handled);
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Request our interrupt.
 | |
|  */
 | |
| static int
 | |
| nm256_grabInterrupt (struct nm256_info *card)
 | |
| {
 | |
|     if (card->has_irq++ == 0) {
 | |
| 	if (request_irq (card->irq, card->introutine, SA_SHIRQ,
 | |
| 			 "NM256_audio", card) < 0) {
 | |
| 	    printk (KERN_ERR "NM256: can't obtain IRQ %d\n", card->irq);
 | |
| 	    return -1;
 | |
| 	}
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Release our interrupt. 
 | |
|  */
 | |
| static int
 | |
| nm256_releaseInterrupt (struct nm256_info *card)
 | |
| {
 | |
|     if (card->has_irq <= 0) {
 | |
| 	printk (KERN_ERR "nm256: too many calls to releaseInterrupt\n");
 | |
| 	return -1;
 | |
|     }
 | |
|     card->has_irq--;
 | |
|     if (card->has_irq == 0) {
 | |
| 	free_irq (card->irq, card);
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Waits for the mixer to become ready to be written; returns a zero value
 | |
|  * if it timed out.
 | |
|  */
 | |
| 
 | |
| static int
 | |
| nm256_isReady (struct ac97_hwint *dev)
 | |
| {
 | |
|     struct nm256_info *card = (struct nm256_info *)dev->driver_private;
 | |
|     int t2 = 10;
 | |
|     u32 testaddr;
 | |
|     u16 testb;
 | |
|     int done = 0;
 | |
| 
 | |
|     if (card->magsig != NM_MAGIC_SIG) {
 | |
| 	printk (KERN_ERR "NM256: Bad magic signature in isReady!\n");
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
|     testaddr = card->mixer_status_offset;
 | |
|     testb = card->mixer_status_mask;
 | |
| 
 | |
|     /* 
 | |
|      * Loop around waiting for the mixer to become ready. 
 | |
|      */
 | |
|     while (! done && t2-- > 0) {
 | |
| 	if ((nm256_readPort16 (card, 2, testaddr) & testb) == 0)
 | |
| 	    done = 1;
 | |
| 	else
 | |
| 	    udelay (100);
 | |
|     }
 | |
|     return done;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Return the contents of the AC97 mixer register REG.  Returns a positive
 | |
|  * value if successful, or a negative error code.
 | |
|  */
 | |
| static int
 | |
| nm256_readAC97Reg (struct ac97_hwint *dev, u8 reg)
 | |
| {
 | |
|     struct nm256_info *card = (struct nm256_info *)dev->driver_private;
 | |
| 
 | |
|     if (card->magsig != NM_MAGIC_SIG) {
 | |
| 	printk (KERN_ERR "NM256: Bad magic signature in readAC97Reg!\n");
 | |
| 	return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     if (reg < 128) {
 | |
| 	int res;
 | |
| 
 | |
| 	nm256_isReady (dev);
 | |
| 	res = nm256_readPort16 (card, 2, card->mixer + reg);
 | |
| 	/* Magic delay.  Bleah yucky.  */
 | |
|         udelay (1000);
 | |
| 	return res;
 | |
|     }
 | |
|     else
 | |
| 	return -EINVAL;
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Writes VALUE to AC97 mixer register REG.  Returns 0 if successful, or
 | |
|  * a negative error code. 
 | |
|  */
 | |
| static int
 | |
| nm256_writeAC97Reg (struct ac97_hwint *dev, u8 reg, u16 value)
 | |
| {
 | |
|     unsigned long flags;
 | |
|     int tries = 2;
 | |
|     int done = 0;
 | |
|     u32 base;
 | |
| 
 | |
|     struct nm256_info *card = (struct nm256_info *)dev->driver_private;
 | |
| 
 | |
|     if (card->magsig != NM_MAGIC_SIG) {
 | |
| 	printk (KERN_ERR "NM256: Bad magic signature in writeAC97Reg!\n");
 | |
| 	return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     base = card->mixer;
 | |
| 
 | |
|     spin_lock_irqsave(&card->lock,flags);
 | |
| 
 | |
|     nm256_isReady (dev);
 | |
| 
 | |
|     /* Wait for the write to take, too. */
 | |
|     while ((tries-- > 0) && !done) {
 | |
| 	nm256_writePort16 (card, 2, base + reg, value);
 | |
| 	if (nm256_isReady (dev)) {
 | |
| 	    done = 1;
 | |
| 	    break;
 | |
| 	}
 | |
| 
 | |
|     }
 | |
| 
 | |
|     spin_unlock_irqrestore(&card->lock,flags);
 | |
|     udelay (1000);
 | |
| 
 | |
|     return ! done;
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Initial register values to be written to the AC97 mixer.
 | |
|  * While most of these are identical to the reset values, we do this
 | |
|  * so that we have most of the register contents cached--this avoids
 | |
|  * reading from the mixer directly (which seems to be problematic,
 | |
|  * probably due to ignorance).
 | |
|  */
 | |
| struct initialValues 
 | |
| {
 | |
|     unsigned short port;
 | |
|     unsigned short value;
 | |
| };
 | |
| 
 | |
| static struct initialValues nm256_ac97_initial_values[] = 
 | |
| {
 | |
|     { AC97_MASTER_VOL_STEREO, 0x8000 },
 | |
|     { AC97_HEADPHONE_VOL,     0x8000 },
 | |
|     { AC97_MASTER_VOL_MONO,   0x0000 },
 | |
|     { AC97_PCBEEP_VOL,        0x0000 },
 | |
|     { AC97_PHONE_VOL,         0x0008 },
 | |
|     { AC97_MIC_VOL,           0x8000 },
 | |
|     { AC97_LINEIN_VOL,        0x8808 },
 | |
|     { AC97_CD_VOL,            0x8808 },
 | |
|     { AC97_VIDEO_VOL,         0x8808 },
 | |
|     { AC97_AUX_VOL,           0x8808 },
 | |
|     { AC97_PCMOUT_VOL,        0x0808 },
 | |
|     { AC97_RECORD_SELECT,     0x0000 },
 | |
|     { AC97_RECORD_GAIN,       0x0B0B },
 | |
|     { AC97_GENERAL_PURPOSE,   0x0000 },
 | |
|     { 0xffff, 0xffff }
 | |
| };
 | |
| 
 | |
| /* Initialize the AC97 into a known state.  */
 | |
| static int
 | |
| nm256_resetAC97 (struct ac97_hwint *dev)
 | |
| {
 | |
|     struct nm256_info *card = (struct nm256_info *)dev->driver_private;
 | |
|     int x;
 | |
| 
 | |
|     if (card->magsig != NM_MAGIC_SIG) {
 | |
| 	printk (KERN_ERR "NM256: Bad magic signature in resetAC97!\n");
 | |
| 	return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     /* Reset the mixer.  'Tis magic!  */
 | |
|     nm256_writePort8 (card, 2, 0x6c0, 1);
 | |
| //  nm256_writePort8 (card, 2, 0x6cc, 0x87);	/* This crashes Dell latitudes */
 | |
|     nm256_writePort8 (card, 2, 0x6cc, 0x80);
 | |
|     nm256_writePort8 (card, 2, 0x6cc, 0x0);
 | |
| 
 | |
|     if (! card->mixer_values_init) {
 | |
| 	for (x = 0; nm256_ac97_initial_values[x].port != 0xffff; x++) {
 | |
| 	    ac97_put_register (dev,
 | |
| 			       nm256_ac97_initial_values[x].port,
 | |
| 			       nm256_ac97_initial_values[x].value);
 | |
| 	    card->mixer_values_init = 1;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * We don't do anything particularly special here; it just passes the
 | |
|  * mixer ioctl to the AC97 driver.
 | |
|  */
 | |
| static int
 | |
| nm256_default_mixer_ioctl (int dev, unsigned int cmd, void __user *arg)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card_for_mixer (dev);
 | |
|     if (card != NULL)
 | |
| 	return ac97_mixer_ioctl (&(card->mdev), cmd, arg);
 | |
|     else
 | |
| 	return -ENODEV;
 | |
| }
 | |
| 
 | |
| static struct mixer_operations nm256_mixer_operations = {
 | |
| 	.owner	= THIS_MODULE,
 | |
| 	.id	= "NeoMagic",
 | |
| 	.name	= "NM256AC97Mixer",
 | |
| 	.ioctl	= nm256_default_mixer_ioctl
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * Default settings for the OSS mixer.  These are set last, after the
 | |
|  * mixer is initialized.
 | |
|  *
 | |
|  * I "love" C sometimes.  Got braces?
 | |
|  */
 | |
| static struct ac97_mixer_value_list mixer_defaults[] = {
 | |
|     { SOUND_MIXER_VOLUME,  { { 85, 85 } } },
 | |
|     { SOUND_MIXER_SPEAKER, { { 100 } } },
 | |
|     { SOUND_MIXER_PCM,     { { 65, 65 } } },
 | |
|     { SOUND_MIXER_CD,      { { 65, 65 } } },
 | |
|     { -1,                  {  { 0,  0 } } }
 | |
| };
 | |
| 
 | |
| 
 | |
| /* Installs the AC97 mixer into CARD.  */
 | |
| static int __devinit
 | |
| nm256_install_mixer (struct nm256_info *card)
 | |
| {
 | |
|     int mixer;
 | |
| 
 | |
|     card->mdev.reset_device = nm256_resetAC97;
 | |
|     card->mdev.read_reg = nm256_readAC97Reg;
 | |
|     card->mdev.write_reg = nm256_writeAC97Reg;
 | |
|     card->mdev.driver_private = (void *)card;
 | |
| 
 | |
|     if (ac97_init (&(card->mdev)))
 | |
| 	return -1;
 | |
| 
 | |
|     mixer = sound_alloc_mixerdev();
 | |
|     if (num_mixers >= MAX_MIXER_DEV) {
 | |
| 	printk ("NM256 mixer: Unable to alloc mixerdev\n");
 | |
| 	return -1;
 | |
|     }
 | |
| 
 | |
|     mixer_devs[mixer] = &nm256_mixer_operations;
 | |
|     card->mixer_oss_dev = mixer;
 | |
| 
 | |
|     /* Some reasonable default values.  */
 | |
|     ac97_set_values (&(card->mdev), mixer_defaults);
 | |
| 
 | |
|     printk(KERN_INFO "Initialized AC97 mixer\n");
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * See if the signature left by the NM256 BIOS is intact; if so, we use
 | |
|  * the associated address as the end of our audio buffer in the video
 | |
|  * RAM.
 | |
|  */
 | |
| 
 | |
| static void __devinit
 | |
| nm256_peek_for_sig (struct nm256_info *card)
 | |
| {
 | |
|     u32 port1offset 
 | |
| 	= card->port[0].physaddr + card->port[0].end_offset - 0x0400;
 | |
|     /* The signature is located 1K below the end of video RAM.  */
 | |
|     char __iomem *temp = ioremap_nocache (port1offset, 16);
 | |
|     /* Default buffer end is 5120 bytes below the top of RAM.  */
 | |
|     u32 default_value = card->port[0].end_offset - 0x1400;
 | |
|     u32 sig;
 | |
| 
 | |
|     /* Install the default value first, so we don't have to repeatedly
 | |
|        do it if there is a problem.  */
 | |
|     card->port[0].end_offset = default_value;
 | |
| 
 | |
|     if (temp == NULL) {
 | |
| 	printk (KERN_ERR "NM256: Unable to scan for card signature in video RAM\n");
 | |
| 	return;
 | |
|     }
 | |
|     sig = readl (temp);
 | |
|     if ((sig & NM_SIG_MASK) == NM_SIGNATURE) {
 | |
| 	u32 pointer = readl (temp + 4);
 | |
| 
 | |
| 	/*
 | |
| 	 * If it's obviously invalid, don't use it (the port already has a
 | |
| 	 * suitable default value set).
 | |
| 	 */
 | |
| 	if (pointer != 0xffffffff)
 | |
| 	    card->port[0].end_offset = pointer;
 | |
| 
 | |
| 	printk (KERN_INFO "NM256: Found card signature in video RAM: 0x%x\n",
 | |
| 		pointer);
 | |
|     }
 | |
| 
 | |
|     iounmap (temp);
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Install a driver for the PCI device referenced by PCIDEV.
 | |
|  * VERSTR is a human-readable version string.
 | |
|  */
 | |
| 
 | |
| static int __devinit
 | |
| nm256_install(struct pci_dev *pcidev, enum nm256rev rev, char *verstr)
 | |
| {
 | |
|     struct nm256_info *card;
 | |
|     int x;
 | |
| 
 | |
|     if (pci_enable_device(pcidev))
 | |
| 	    return 0;
 | |
| 
 | |
|     card = kmalloc (sizeof (struct nm256_info), GFP_KERNEL);
 | |
|     if (card == NULL) {
 | |
| 	printk (KERN_ERR "NM256: out of memory!\n");
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
|     card->magsig = NM_MAGIC_SIG;
 | |
|     card->playing  = 0;
 | |
|     card->recording = 0;
 | |
|     card->rev = rev;
 | |
|     spin_lock_init(&card->lock);
 | |
| 
 | |
|     /* Init the memory port info.  */
 | |
|     for (x = 0; x < 2; x++) {
 | |
| 	card->port[x].physaddr = pci_resource_start (pcidev, x);
 | |
| 	card->port[x].ptr = NULL;
 | |
| 	card->port[x].start_offset = 0;
 | |
| 	card->port[x].end_offset = 0;
 | |
|     }
 | |
| 
 | |
|     /* Port 2 is easy.  */
 | |
|     card->port[1].start_offset = 0;
 | |
|     card->port[1].end_offset = NM_PORT2_SIZE;
 | |
| 
 | |
|     /* Yuck.  But we have to map in port 2 so we can check how much RAM the
 | |
|        card has.  */
 | |
|     if (nm256_remap_ports (card)) {
 | |
| 	kfree (card);
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
|     /* 
 | |
|      * The NM256 has two memory ports.  The first port is nothing
 | |
|      * more than a chunk of video RAM, which is used as the I/O ring
 | |
|      * buffer.  The second port has the actual juicy stuff (like the
 | |
|      * mixer and the playback engine control registers).
 | |
|      */
 | |
| 
 | |
|     if (card->rev == REV_NM256AV) {
 | |
| 	/* Ok, try to see if this is a non-AC97 version of the hardware. */
 | |
| 	int pval = nm256_readPort16 (card, 2, NM_MIXER_PRESENCE);
 | |
| 	if ((pval & NM_PRESENCE_MASK) != NM_PRESENCE_VALUE) {
 | |
| 	    if (! force_load) {
 | |
| 		printk (KERN_ERR "NM256: This doesn't look to me like the AC97-compatible version.\n");
 | |
| 		printk (KERN_ERR "       You can force the driver to load by passing in the module\n");
 | |
| 		printk (KERN_ERR "       parameter:\n");
 | |
| 		printk (KERN_ERR "              force_load = 1\n");
 | |
| 		printk (KERN_ERR "\n");
 | |
| 		printk (KERN_ERR "       More likely, you should be using the appropriate SB-16 or\n");
 | |
| 		printk (KERN_ERR "       CS4232 driver instead.  (If your BIOS has settings for\n");
 | |
| 		printk (KERN_ERR "       IRQ and/or DMA for the sound card, this is *not* the correct\n");
 | |
| 		printk (KERN_ERR "       driver to use.)\n");
 | |
| 		nm256_release_ports (card);
 | |
| 		kfree (card);
 | |
| 		return 0;
 | |
| 	    }
 | |
| 	    else {
 | |
| 		printk (KERN_INFO "NM256: Forcing driver load as per user request.\n");
 | |
| 	    }
 | |
| 	}
 | |
| 	else {
 | |
| 	 /*   printk (KERN_INFO "NM256: Congratulations. You're not running Eunice.\n")*/;
 | |
| 	}
 | |
| 	card->port[0].end_offset = 2560 * 1024;
 | |
| 	card->introutine = nm256_interrupt;
 | |
| 	card->mixer_status_offset = NM_MIXER_STATUS_OFFSET;
 | |
| 	card->mixer_status_mask = NM_MIXER_READY_MASK;
 | |
|     } 
 | |
|     else {
 | |
| 	/* Not sure if there is any relevant detect for the ZX or not.  */
 | |
| 	if (nm256_readPort8 (card, 2, 0xa0b) != 0)
 | |
| 	    card->port[0].end_offset = 6144 * 1024;
 | |
| 	else
 | |
| 	    card->port[0].end_offset = 4096 * 1024;
 | |
| 
 | |
| 	card->introutine = nm256_interrupt_zx;
 | |
| 	card->mixer_status_offset = NM2_MIXER_STATUS_OFFSET;
 | |
| 	card->mixer_status_mask = NM2_MIXER_READY_MASK;
 | |
|     }
 | |
| 
 | |
|     if (buffertop >= 98304 && buffertop < card->port[0].end_offset)
 | |
| 	card->port[0].end_offset = buffertop;
 | |
|     else
 | |
| 	nm256_peek_for_sig (card);
 | |
| 
 | |
|     card->port[0].start_offset = card->port[0].end_offset - 98304;
 | |
| 
 | |
|     printk (KERN_INFO "NM256: Mapping port 1 from 0x%x - 0x%x\n",
 | |
| 	    card->port[0].start_offset, card->port[0].end_offset);
 | |
| 
 | |
|     if (nm256_remap_ports (card)) {
 | |
| 	kfree (card);
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
|     /* See if we can get the interrupt. */
 | |
| 
 | |
|     card->irq = pcidev->irq;
 | |
|     card->has_irq = 0;
 | |
| 
 | |
|     if (nm256_grabInterrupt (card) != 0) {
 | |
| 	nm256_release_ports (card);
 | |
| 	kfree (card);
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
|     nm256_releaseInterrupt (card);
 | |
| 
 | |
|     /*
 | |
|      *	Init the board.
 | |
|      */
 | |
| 
 | |
|     card->playbackBufferSize = 16384;
 | |
|     card->recordBufferSize = 16384;
 | |
| 
 | |
|     card->coeffBuf = card->port[0].end_offset - NM_MAX_COEFFICIENT;
 | |
|     card->abuf2 = card->coeffBuf - card->recordBufferSize;
 | |
|     card->abuf1 = card->abuf2 - card->playbackBufferSize;
 | |
|     card->allCoeffBuf = card->abuf2 - (NM_TOTAL_COEFF_COUNT * 4);
 | |
| 
 | |
|     /* Fixed setting. */
 | |
|     card->mixer = NM_MIXER_OFFSET;
 | |
|     card->mixer_values_init = 0;
 | |
| 
 | |
|     card->is_open_play = 0;
 | |
|     card->is_open_record = 0;
 | |
| 
 | |
|     card->coeffsCurrent = 0;
 | |
| 
 | |
|     card->opencnt[0] = 0; card->opencnt[1] = 0;
 | |
| 
 | |
|     /* Reasonable default settings, but largely unnecessary. */
 | |
|     for (x = 0; x < 2; x++) {
 | |
| 	card->sinfo[x].bits = 8;
 | |
| 	card->sinfo[x].stereo = 0;
 | |
| 	card->sinfo[x].samplerate = 8000;
 | |
|     }
 | |
| 
 | |
|     nm256_initHw (card);
 | |
| 
 | |
|     for (x = 0; x < 2; x++) {
 | |
| 	if ((card->dev[x] =
 | |
| 	     sound_install_audiodrv(AUDIO_DRIVER_VERSION,
 | |
| 				    "NM256", &nm256_audio_driver,
 | |
| 				    sizeof(struct audio_driver),
 | |
| 				    DMA_NODMA, AFMT_U8 | AFMT_S16_LE,
 | |
| 				    NULL, -1, -1)) >= 0) {
 | |
| 	    /* 1K minimum buffer size. */
 | |
| 	    audio_devs[card->dev[x]]->min_fragment = 10;
 | |
| 	    /* Maximum of 8K buffer size. */
 | |
| 	    audio_devs[card->dev[x]]->max_fragment = 13;
 | |
| 	}
 | |
| 	else {
 | |
| 	    printk(KERN_ERR "NM256: Too many PCM devices available\n");
 | |
| 	    nm256_release_ports (card);
 | |
| 	    kfree (card);
 | |
| 	    return 0;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     pci_set_drvdata(pcidev,card);
 | |
| 
 | |
|     /* Insert the card in the list.  */
 | |
|     card->next_card = nmcard_list;
 | |
|     nmcard_list = card;
 | |
| 
 | |
|     printk(KERN_INFO "Initialized NeoMagic %s audio in PCI native mode\n",
 | |
| 	   verstr);
 | |
| 
 | |
|     /* 
 | |
|      * And our mixer.  (We should allow support for other mixers, maybe.)
 | |
|      */
 | |
| 
 | |
|     nm256_install_mixer (card);
 | |
| 
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int __devinit
 | |
| nm256_probe(struct pci_dev *pcidev,const struct pci_device_id *pciid)
 | |
| {
 | |
|     if (pcidev->device == PCI_DEVICE_ID_NEOMAGIC_NM256AV_AUDIO)
 | |
| 	return nm256_install(pcidev, REV_NM256AV, "256AV");
 | |
|     if (pcidev->device == PCI_DEVICE_ID_NEOMAGIC_NM256ZX_AUDIO)
 | |
| 	return nm256_install(pcidev, REV_NM256ZX, "256ZX");
 | |
|     if (pcidev->device == PCI_DEVICE_ID_NEOMAGIC_NM256XL_PLUS_AUDIO)
 | |
| 	return nm256_install(pcidev, REV_NM256ZX, "256XL+");
 | |
|     return -1; /* should not come here ... */
 | |
| }
 | |
| 
 | |
| static void __devinit
 | |
| nm256_remove(struct pci_dev *pcidev) {
 | |
|     struct nm256_info *xcard = pci_get_drvdata(pcidev);
 | |
|     struct nm256_info *card,*next_card = NULL;
 | |
| 
 | |
|     for (card = nmcard_list; card != NULL; card = next_card) {
 | |
| 	next_card = card->next_card;
 | |
| 	if (card == xcard) {
 | |
| 	    stopPlay (card);
 | |
| 	    stopRecord (card);
 | |
| 	    if (card->has_irq)
 | |
| 		free_irq (card->irq, card);
 | |
| 	    nm256_release_ports (card);
 | |
| 	    sound_unload_mixerdev (card->mixer_oss_dev);
 | |
| 	    sound_unload_audiodev (card->dev[0]);
 | |
| 	    sound_unload_audiodev (card->dev[1]);
 | |
| 	    kfree (card);
 | |
| 	    break;
 | |
| 	}
 | |
|     }
 | |
|     if (nmcard_list == card)
 | |
|     	nmcard_list = next_card;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Open the device
 | |
|  *
 | |
|  * DEV  - device
 | |
|  * MODE - mode to open device (logical OR of OPEN_READ and OPEN_WRITE)
 | |
|  *
 | |
|  * Called when opening the DMAbuf               (dmabuf.c:259)
 | |
|  */
 | |
| static int
 | |
| nm256_audio_open(int dev, int mode)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
|     int w;
 | |
| 	
 | |
|     if (card == NULL)
 | |
| 	return -ENODEV;
 | |
| 
 | |
|     if (card->dev[0] == dev)
 | |
| 	w = 0;
 | |
|     else if (card->dev[1] == dev)
 | |
| 	w = 1;
 | |
|     else
 | |
| 	return -ENODEV;
 | |
| 
 | |
|     if (card->opencnt[w] > 0)
 | |
| 	return -EBUSY;
 | |
| 
 | |
|     /* No bits set? Huh? */
 | |
|     if (! ((mode & OPEN_READ) || (mode & OPEN_WRITE)))
 | |
| 	return -EIO;
 | |
| 
 | |
|     /*
 | |
|      * If it's open for both read and write, and the card's currently
 | |
|      * being read or written to, then do the opposite of what has
 | |
|      * already been done.  Otherwise, don't specify any mode until the
 | |
|      * user actually tries to do I/O.  (Some programs open the device
 | |
|      * for both read and write, but only actually do reading or writing.)
 | |
|      */
 | |
| 
 | |
|     if ((mode & OPEN_WRITE) && (mode & OPEN_READ)) {
 | |
| 	if (card->is_open_play)
 | |
| 	    mode = OPEN_WRITE;
 | |
| 	else if (card->is_open_record)
 | |
| 	    mode = OPEN_READ;
 | |
| 	else mode = 0;
 | |
|     }
 | |
| 	
 | |
|     if (mode & OPEN_WRITE) {
 | |
| 	if (card->is_open_play == 0) {
 | |
| 	    card->dev_for_play = dev;
 | |
| 	    card->is_open_play = 1;
 | |
| 	}
 | |
| 	else
 | |
| 	    return -EBUSY;
 | |
|     }
 | |
| 
 | |
|     if (mode & OPEN_READ) {
 | |
| 	if (card->is_open_record == 0) {
 | |
| 	    card->dev_for_record = dev;
 | |
| 	    card->is_open_record = 1;
 | |
| 	}
 | |
| 	else
 | |
| 	    return -EBUSY;
 | |
|     }
 | |
| 
 | |
|     card->opencnt[w]++;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Close the device
 | |
|  *
 | |
|  * DEV  - device
 | |
|  *
 | |
|  * Called when closing the DMAbuf               (dmabuf.c:477)
 | |
|  *      after halt_xfer
 | |
|  */
 | |
| static void
 | |
| nm256_audio_close(int dev)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 	
 | |
|     if (card != NULL) {
 | |
| 	int w;
 | |
| 
 | |
| 	if (card->dev[0] == dev)
 | |
| 	    w = 0;
 | |
| 	else if (card->dev[1] == dev)
 | |
| 	    w = 1;
 | |
| 	else
 | |
| 	    return;
 | |
| 
 | |
| 	card->opencnt[w]--;
 | |
| 	if (card->opencnt[w] <= 0) {
 | |
| 	    card->opencnt[w] = 0;
 | |
| 
 | |
| 	    if (card->dev_for_play == dev) {
 | |
| 		stopPlay (card);
 | |
| 		card->is_open_play = 0;
 | |
| 		card->dev_for_play = -1;
 | |
| 	    }
 | |
| 
 | |
| 	    if (card->dev_for_record == dev) {
 | |
| 		stopRecord (card);
 | |
| 		card->is_open_record = 0;
 | |
| 		card->dev_for_record = -1;
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Standard ioctl handler. */
 | |
| static int
 | |
| nm256_audio_ioctl(int dev, unsigned int cmd, void __user *arg)
 | |
| {
 | |
|     int ret;
 | |
|     u32 oldinfo;
 | |
|     int w;
 | |
| 
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 
 | |
|     if (card == NULL)
 | |
| 	return -ENODEV;
 | |
| 
 | |
|     if (dev == card->dev[0])
 | |
| 	w = 0;
 | |
|     else
 | |
| 	w = 1;
 | |
| 
 | |
|     /* 
 | |
|      * The code here is messy.  There are probably better ways to do
 | |
|      * it.  (It should be possible to handle it the same way the AC97 mixer 
 | |
|      * is done.)
 | |
|      */
 | |
|     switch (cmd)
 | |
| 	{
 | |
| 	case SOUND_PCM_WRITE_RATE:
 | |
| 	    if (get_user(ret, (int __user *) arg))
 | |
| 		return -EFAULT;
 | |
| 
 | |
| 	    if (ret != 0) {
 | |
| 		oldinfo = card->sinfo[w].samplerate;
 | |
| 		card->sinfo[w].samplerate = ret;
 | |
| 		ret = nm256_setInfo(dev, card);
 | |
| 		if (ret != 0)
 | |
| 		    card->sinfo[w].samplerate = oldinfo;
 | |
| 	    }
 | |
| 	    if (ret == 0)
 | |
| 		ret = card->sinfo[w].samplerate;
 | |
| 	    break;
 | |
| 
 | |
| 	case SOUND_PCM_READ_RATE:
 | |
| 	    ret = card->sinfo[w].samplerate;
 | |
| 	    break;
 | |
| 
 | |
| 	case SNDCTL_DSP_STEREO:
 | |
| 	    if (get_user(ret, (int __user *) arg))
 | |
| 		return -EFAULT;
 | |
| 
 | |
| 	    card->sinfo[w].stereo = ret ? 1 : 0;
 | |
| 	    ret = nm256_setInfo (dev, card);
 | |
| 	    if (ret == 0)
 | |
| 		ret = card->sinfo[w].stereo;
 | |
| 
 | |
| 	    break;
 | |
| 
 | |
| 	case SOUND_PCM_WRITE_CHANNELS:
 | |
| 	    if (get_user(ret, (int __user *) arg))
 | |
| 		return -EFAULT;
 | |
| 
 | |
| 	    if (ret < 1 || ret > 3)
 | |
| 		ret = card->sinfo[w].stereo + 1;
 | |
| 	    else {
 | |
| 		card->sinfo[w].stereo = ret - 1;
 | |
| 		ret = nm256_setInfo (dev, card);
 | |
| 		if (ret == 0)
 | |
| 		    ret = card->sinfo[w].stereo + 1;
 | |
| 	    }
 | |
| 	    break;
 | |
| 
 | |
| 	case SOUND_PCM_READ_CHANNELS:
 | |
| 	    ret = card->sinfo[w].stereo + 1;
 | |
| 	    break;
 | |
| 
 | |
| 	case SNDCTL_DSP_SETFMT:
 | |
| 	    if (get_user(ret, (int __user *) arg))
 | |
| 		return -EFAULT;
 | |
| 
 | |
| 	    if (ret != 0) {
 | |
| 		oldinfo = card->sinfo[w].bits;
 | |
| 		card->sinfo[w].bits = ret;
 | |
| 		ret = nm256_setInfo (dev, card);
 | |
| 		if (ret != 0)
 | |
| 		    card->sinfo[w].bits = oldinfo;
 | |
| 	    }
 | |
| 	    if (ret == 0)
 | |
| 		ret = card->sinfo[w].bits;
 | |
| 	    break;
 | |
| 
 | |
| 	case SOUND_PCM_READ_BITS:
 | |
| 	    ret = card->sinfo[w].bits;
 | |
| 	    break;
 | |
| 
 | |
| 	default:
 | |
| 	    return -EINVAL;
 | |
| 	}
 | |
|     return put_user(ret, (int __user *) arg);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Given the sound device DEV and an associated physical buffer PHYSBUF, 
 | |
|  * return a pointer to the actual buffer in kernel space. 
 | |
|  *
 | |
|  * This routine should exist as part of the soundcore routines.
 | |
|  */
 | |
| 
 | |
| static char *
 | |
| nm256_getDMAbuffer (int dev, unsigned long physbuf)
 | |
| {
 | |
|     struct audio_operations *adev = audio_devs[dev];
 | |
|     struct dma_buffparms *dmap = adev->dmap_out;
 | |
|     char *dma_start =
 | |
| 	(char *)(physbuf - (unsigned long)dmap->raw_buf_phys 
 | |
| 		 + (unsigned long)dmap->raw_buf);
 | |
| 
 | |
|     return dma_start;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Output a block to sound device
 | |
|  *
 | |
|  * dev          - device number
 | |
|  * buf          - physical address of buffer
 | |
|  * total_count  - total byte count in buffer
 | |
|  * intrflag     - set if this has been called from an interrupt 
 | |
|  *				  (via DMAbuf_outputintr)
 | |
|  * restart_dma  - set if engine needs to be re-initialised
 | |
|  *
 | |
|  * Called when:
 | |
|  *  1. Starting output                                  (dmabuf.c:1327)
 | |
|  *  2.                                                  (dmabuf.c:1504)
 | |
|  *  3. A new buffer needs to be sent to the device      (dmabuf.c:1579)
 | |
|  */
 | |
| static void
 | |
| nm256_audio_output_block(int dev, unsigned long physbuf,
 | |
| 				       int total_count, int intrflag)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 
 | |
|     if (card != NULL) {
 | |
| 	char *dma_buf = nm256_getDMAbuffer (dev, physbuf);
 | |
| 	card->is_open_play = 1;
 | |
| 	card->dev_for_play = dev;
 | |
| 	nm256_write_block (card, dma_buf, total_count);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Ditto, but do recording instead.  */
 | |
| static void
 | |
| nm256_audio_start_input(int dev, unsigned long physbuf, int count,
 | |
| 			int intrflag)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 
 | |
|     if (card != NULL) {
 | |
| 	char *dma_buf = nm256_getDMAbuffer (dev, physbuf);
 | |
| 	card->is_open_record = 1;
 | |
| 	card->dev_for_record = dev;
 | |
| 	nm256_startRecording (card, dma_buf, count);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * Prepare for inputting samples to DEV. 
 | |
|  * Each requested buffer will be BSIZE byes long, with a total of
 | |
|  * BCOUNT buffers. 
 | |
|  */
 | |
| 
 | |
| static int
 | |
| nm256_audio_prepare_for_input(int dev, int bsize, int bcount)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 
 | |
|     if (card == NULL) 
 | |
| 	return -ENODEV;
 | |
| 
 | |
|     if (card->is_open_record && card->dev_for_record != dev)
 | |
| 	return -EBUSY;
 | |
| 
 | |
|     audio_devs[dev]->dmap_in->flags |= DMA_NODMA;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Prepare for outputting samples to `dev'
 | |
|  *
 | |
|  * Each buffer that will be passed will be `bsize' bytes long,
 | |
|  * with a total of `bcount' buffers.
 | |
|  *
 | |
|  * Called when:
 | |
|  *  1. A trigger enables audio output                   (dmabuf.c:978)
 | |
|  *  2. We get a write buffer without dma_mode setup     (dmabuf.c:1152)
 | |
|  *  3. We restart a transfer                            (dmabuf.c:1324)
 | |
|  */
 | |
| 
 | |
| static int
 | |
| nm256_audio_prepare_for_output(int dev, int bsize, int bcount)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 
 | |
|     if (card == NULL)
 | |
| 	return -ENODEV;
 | |
| 
 | |
|     if (card->is_open_play && card->dev_for_play != dev)
 | |
| 	return -EBUSY;
 | |
| 
 | |
|     audio_devs[dev]->dmap_out->flags |= DMA_NODMA;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /* Stop the current operations associated with DEV.  */
 | |
| static void
 | |
| nm256_audio_reset(int dev)
 | |
| {
 | |
|     struct nm256_info *card = nm256_find_card (dev);
 | |
| 
 | |
|     if (card != NULL) {
 | |
| 	if (card->dev_for_play == dev)
 | |
| 	    stopPlay (card);
 | |
| 	if (card->dev_for_record == dev)
 | |
| 	    stopRecord (card);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static int
 | |
| nm256_audio_local_qlen(int dev)
 | |
| {
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static struct audio_driver nm256_audio_driver =
 | |
| {
 | |
| 	.owner			= THIS_MODULE,
 | |
| 	.open			= nm256_audio_open,
 | |
| 	.close			= nm256_audio_close,
 | |
| 	.output_block		= nm256_audio_output_block,
 | |
| 	.start_input		= nm256_audio_start_input,
 | |
| 	.ioctl			= nm256_audio_ioctl,
 | |
| 	.prepare_for_input	= nm256_audio_prepare_for_input,
 | |
| 	.prepare_for_output	= nm256_audio_prepare_for_output,
 | |
| 	.halt_io		= nm256_audio_reset,
 | |
| 	.local_qlen		= nm256_audio_local_qlen,
 | |
| };
 | |
| 
 | |
| static struct pci_device_id nm256_pci_tbl[] = {
 | |
| 	{PCI_VENDOR_ID_NEOMAGIC, PCI_DEVICE_ID_NEOMAGIC_NM256AV_AUDIO,
 | |
| 	PCI_ANY_ID, PCI_ANY_ID, 0, 0},
 | |
| 	{PCI_VENDOR_ID_NEOMAGIC, PCI_DEVICE_ID_NEOMAGIC_NM256ZX_AUDIO,
 | |
| 	PCI_ANY_ID, PCI_ANY_ID, 0, 0},
 | |
| 	{PCI_VENDOR_ID_NEOMAGIC, PCI_DEVICE_ID_NEOMAGIC_NM256XL_PLUS_AUDIO,
 | |
| 	PCI_ANY_ID, PCI_ANY_ID, 0, 0},
 | |
| 	{0,}
 | |
| };
 | |
| MODULE_DEVICE_TABLE(pci, nm256_pci_tbl);
 | |
| MODULE_LICENSE("GPL");
 | |
| 
 | |
| 
 | |
| static struct pci_driver nm256_pci_driver = {
 | |
| 	.name		= "nm256_audio",
 | |
| 	.id_table	= nm256_pci_tbl,
 | |
| 	.probe		= nm256_probe,
 | |
| 	.remove		= nm256_remove,
 | |
| };
 | |
| 
 | |
| module_param(usecache, bool, 0);
 | |
| module_param(buffertop, int, 0);
 | |
| module_param(nm256_debug, bool, 0644);
 | |
| module_param(force_load, bool, 0);
 | |
| 
 | |
| static int __init do_init_nm256(void)
 | |
| {
 | |
|     printk (KERN_INFO "NeoMagic 256AV/256ZX audio driver, version 1.1p\n");
 | |
|     return pci_register_driver(&nm256_pci_driver);
 | |
| }
 | |
| 
 | |
| static void __exit cleanup_nm256 (void)
 | |
| {
 | |
|     pci_unregister_driver(&nm256_pci_driver);
 | |
| }
 | |
| 
 | |
| module_init(do_init_nm256);
 | |
| module_exit(cleanup_nm256);
 | |
| 
 | |
| /*
 | |
|  * Local variables:
 | |
|  *  c-basic-offset: 4
 | |
|  * End:
 | |
|  */
 |