⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 fm801.c

📁 Linux Kernel 2.6.9 for OMAP1710
💻 C
📖 第 1 页 / 共 3 页
字号:
/* *  The driver for the ForteMedia FM801 based soundcards *  Copyright (c) by Jaroslav Kysela <perex@suse.cz> * * *   This program is free software; you can redistribute it and/or modify *   it under the terms of the GNU General Public License as published by *   the Free Software Foundation; either version 2 of the License, or *   (at your option) any later version. * *   This program is distributed in the hope that it will be useful, *   but WITHOUT ANY WARRANTY; without even the implied warranty of *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the *   GNU General Public License for more details. * *   You should have received a copy of the GNU General Public License *   along with this program; if not, write to the Free Software *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA * */#include <sound/driver.h>#include <linux/delay.h>#include <linux/init.h>#include <linux/interrupt.h>#include <linux/pci.h>#include <linux/slab.h>#include <linux/moduleparam.h>#include <sound/core.h>#include <sound/pcm.h>#include <sound/ac97_codec.h>#include <sound/mpu401.h>#include <sound/opl3.h>#include <sound/initval.h>#include <asm/io.h>#if (defined(CONFIG_SND_FM801_TEA575X) || defined(CONFIG_SND_FM801_TEA575X_MODULE)) && (defined(CONFIG_VIDEO_DEV) || defined(CONFIG_VIDEO_DEV_MODULE))#include <sound/tea575x-tuner.h>#define TEA575X_RADIO 1#endifMODULE_AUTHOR("Jaroslav Kysela <perex@suse.cz>");MODULE_DESCRIPTION("ForteMedia FM801");MODULE_LICENSE("GPL");MODULE_SUPPORTED_DEVICE("{{ForteMedia,FM801},"		"{Genius,SoundMaker Live 5.1}}");static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;	/* Index 0-MAX */static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;	/* ID for this card */static int enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE_PNP;	/* Enable this card *//* *  Enable TEA575x tuner *    1 = MediaForte 256-PCS *    2 = MediaForte 256-PCPR *    3 = MediaForte 64-PCR *  High 16-bits are video (radio) device number + 1 */static int tea575x_tuner[SNDRV_CARDS] = { [0 ... (SNDRV_CARDS-1)] = 0 };static int boot_devs;module_param_array(index, int, boot_devs, 0444);MODULE_PARM_DESC(index, "Index value for the FM801 soundcard.");module_param_array(id, charp, boot_devs, 0444);MODULE_PARM_DESC(id, "ID string for the FM801 soundcard.");module_param_array(enable, bool, boot_devs, 0444);MODULE_PARM_DESC(enable, "Enable FM801 soundcard.");module_param_array(tea575x_tuner, bool, boot_devs, 0444);MODULE_PARM_DESC(tea575x_tuner, "Enable TEA575x tuner.");/* *  Direct registers */#define FM801_REG(chip, reg)	(chip->port + FM801_##reg)#define FM801_PCM_VOL		0x00	/* PCM Output Volume */#define FM801_FM_VOL		0x02	/* FM Output Volume */#define FM801_I2S_VOL		0x04	/* I2S Volume */#define FM801_REC_SRC		0x06	/* Record Source */#define FM801_PLY_CTRL		0x08	/* Playback Control */#define FM801_PLY_COUNT		0x0a	/* Playback Count */#define FM801_PLY_BUF1		0x0c	/* Playback Bufer I */#define FM801_PLY_BUF2		0x10	/* Playback Buffer II */#define FM801_CAP_CTRL		0x14	/* Capture Control */#define FM801_CAP_COUNT		0x16	/* Capture Count */#define FM801_CAP_BUF1		0x18	/* Capture Buffer I */#define FM801_CAP_BUF2		0x1c	/* Capture Buffer II */#define FM801_CODEC_CTRL	0x22	/* Codec Control */#define FM801_I2S_MODE		0x24	/* I2S Mode Control */#define FM801_VOLUME		0x26	/* Volume Up/Down/Mute Status */#define FM801_I2C_CTRL		0x29	/* I2C Control */#define FM801_AC97_CMD		0x2a	/* AC'97 Command */#define FM801_AC97_DATA		0x2c	/* AC'97 Data */#define FM801_MPU401_DATA	0x30	/* MPU401 Data */#define FM801_MPU401_CMD	0x31	/* MPU401 Command */#define FM801_GPIO_CTRL		0x52	/* General Purpose I/O Control */#define FM801_GEN_CTRL		0x54	/* General Control */#define FM801_IRQ_MASK		0x56	/* Interrupt Mask */#define FM801_IRQ_STATUS	0x5a	/* Interrupt Status */#define FM801_OPL3_BANK0	0x68	/* OPL3 Status Read / Bank 0 Write */#define FM801_OPL3_DATA0	0x69	/* OPL3 Data 0 Write */#define FM801_OPL3_BANK1	0x6a	/* OPL3 Bank 1 Write */#define FM801_OPL3_DATA1	0x6b	/* OPL3 Bank 1 Write */#define FM801_POWERDOWN		0x70	/* Blocks Power Down Control */#define FM801_AC97_ADDR_SHIFT	10/* playback and record control register bits */#define FM801_BUF1_LAST		(1<<1)#define FM801_BUF2_LAST		(1<<2)#define FM801_START		(1<<5)#define FM801_PAUSE		(1<<6)#define FM801_IMMED_STOP	(1<<7)#define FM801_RATE_SHIFT	8#define FM801_RATE_MASK		(15 << FM801_RATE_SHIFT)#define FM801_CHANNELS_4	(1<<12)	/* playback only */#define FM801_CHANNELS_6	(2<<12)	/* playback only */#define FM801_CHANNELS_6MS	(3<<12)	/* playback only */#define FM801_CHANNELS_MASK	(3<<12)#define FM801_16BIT		(1<<14)#define FM801_STEREO		(1<<15)/* IRQ status bits */#define FM801_IRQ_PLAYBACK	(1<<8)#define FM801_IRQ_CAPTURE	(1<<9)#define FM801_IRQ_VOLUME	(1<<14)#define FM801_IRQ_MPU		(1<<15)/* GPIO control register */#define FM801_GPIO_GP0		(1<<0)	/* read/write */#define FM801_GPIO_GP1		(1<<1)#define FM801_GPIO_GP2		(1<<2)#define FM801_GPIO_GP3		(1<<3)#define FM801_GPIO_GP(x)	(1<<(0+(x)))#define FM801_GPIO_GD0		(1<<8)	/* directions: 1 = input, 0 = output*/#define FM801_GPIO_GD1		(1<<9)#define FM801_GPIO_GD2		(1<<10)#define FM801_GPIO_GD3		(1<<11)#define FM801_GPIO_GD(x)	(1<<(8+(x)))#define FM801_GPIO_GS0		(1<<12)	/* function select: */#define FM801_GPIO_GS1		(1<<13)	/*    1 = GPIO */#define FM801_GPIO_GS2		(1<<14)	/*    0 = other (S/PDIF, VOL) */#define FM801_GPIO_GS3		(1<<15)#define FM801_GPIO_GS(x)	(1<<(12+(x)))	/* */typedef struct _snd_fm801 fm801_t;struct _snd_fm801 {	int irq;	unsigned long port;	/* I/O port number */	unsigned int multichannel: 1,	/* multichannel support */		     secondary: 1;	/* secondary codec */	unsigned char secondary_addr;	/* address of the secondary codec */	unsigned short ply_ctrl; /* playback control */	unsigned short cap_ctrl; /* capture control */	unsigned long ply_buffer;	unsigned int ply_buf;	unsigned int ply_count;	unsigned int ply_size;	unsigned int ply_pos;	unsigned long cap_buffer;	unsigned int cap_buf;	unsigned int cap_count;	unsigned int cap_size;	unsigned int cap_pos;	ac97_bus_t *ac97_bus;	ac97_t *ac97;	ac97_t *ac97_sec;	struct pci_dev *pci;	snd_card_t *card;	snd_pcm_t *pcm;	snd_rawmidi_t *rmidi;	snd_pcm_substream_t *playback_substream;	snd_pcm_substream_t *capture_substream;	unsigned int p_dma_size;	unsigned int c_dma_size;	spinlock_t reg_lock;	snd_info_entry_t *proc_entry;#ifdef TEA575X_RADIO	tea575x_t tea;#endif};static struct pci_device_id snd_fm801_ids[] = {	{ 0x1319, 0x0801, PCI_ANY_ID, PCI_ANY_ID, PCI_CLASS_MULTIMEDIA_AUDIO << 8, 0xffff00, 0, },   /* FM801 */	{ 0, }};MODULE_DEVICE_TABLE(pci, snd_fm801_ids);/* *  common I/O routines */static int snd_fm801_update_bits(fm801_t *chip, unsigned short reg,				 unsigned short mask, unsigned short value){	int change;	unsigned short old, new;	spin_lock(&chip->reg_lock);	old = inw(chip->port + reg);	new = (old & ~mask) | value;	change = old != new;	if (change)		outw(new, chip->port + reg);	spin_unlock(&chip->reg_lock);	return change;}static void snd_fm801_codec_write(ac97_t *ac97,				  unsigned short reg,				  unsigned short val){	fm801_t *chip = ac97->private_data;	int idx;	/*	 *  Wait until the codec interface is not ready..	 */	for (idx = 0; idx < 100; idx++) {		if (!(inw(FM801_REG(chip, AC97_CMD)) & (1<<9)))			goto ok1;		udelay(10);	}	snd_printk("AC'97 interface is busy (1)\n");	return; ok1:	/* write data and address */	outw(val, FM801_REG(chip, AC97_DATA));	outw(reg | (ac97->addr << FM801_AC97_ADDR_SHIFT), FM801_REG(chip, AC97_CMD));	/*	 *  Wait until the write command is not completed..         */	for (idx = 0; idx < 1000; idx++) {		if (!(inw(FM801_REG(chip, AC97_CMD)) & (1<<9)))			return;		udelay(10);	}	snd_printk("AC'97 interface #%d is busy (2)\n", ac97->num);}static unsigned short snd_fm801_codec_read(ac97_t *ac97, unsigned short reg){	fm801_t *chip = ac97->private_data;	int idx;	/*	 *  Wait until the codec interface is not ready..	 */	for (idx = 0; idx < 100; idx++) {		if (!(inw(FM801_REG(chip, AC97_CMD)) & (1<<9)))			goto ok1;		udelay(10);	}	snd_printk("AC'97 interface is busy (1)\n");	return 0; ok1:	/* read command */	outw(reg | (ac97->addr << FM801_AC97_ADDR_SHIFT) | (1<<7), FM801_REG(chip, AC97_CMD));	for (idx = 0; idx < 100; idx++) {		if (!(inw(FM801_REG(chip, AC97_CMD)) & (1<<9)))			goto ok2;		udelay(10);	}	snd_printk("AC'97 interface #%d is busy (2)\n", ac97->num);	return 0; ok2:	for (idx = 0; idx < 1000; idx++) {		if (inw(FM801_REG(chip, AC97_CMD)) & (1<<8))			goto ok3;		udelay(10);	}	snd_printk("AC'97 interface #%d is not valid (2)\n", ac97->num);	return 0; ok3:	return inw(FM801_REG(chip, AC97_DATA));}static unsigned int rates[] = {  5500,  8000,  9600, 11025,  16000, 19200, 22050, 32000,  38400, 44100, 48000};static snd_pcm_hw_constraint_list_t hw_constraints_rates = {	.count = ARRAY_SIZE(rates),	.list = rates,	.mask = 0,};static unsigned int channels[] = {  2, 4, 6};#define CHANNELS sizeof(channels) / sizeof(channels[0])static snd_pcm_hw_constraint_list_t hw_constraints_channels = {	.count = CHANNELS,	.list = channels,	.mask = 0,};/* *  Sample rate routines */static unsigned short snd_fm801_rate_bits(unsigned int rate){	unsigned int idx;	for (idx = 0; idx < ARRAY_SIZE(rates); idx++)		if (rates[idx] == rate)			return idx;	snd_BUG();	return ARRAY_SIZE(rates) - 1;}/* *  PCM part */static int snd_fm801_playback_trigger(snd_pcm_substream_t * substream,				      int cmd){	fm801_t *chip = snd_pcm_substream_chip(substream);	spin_lock(&chip->reg_lock);	switch (cmd) {	case SNDRV_PCM_TRIGGER_START:		chip->ply_ctrl &= ~(FM801_BUF1_LAST |				     FM801_BUF2_LAST |				     FM801_PAUSE);		chip->ply_ctrl |= FM801_START |				   FM801_IMMED_STOP;		break;	case SNDRV_PCM_TRIGGER_STOP:		chip->ply_ctrl &= ~(FM801_START | FM801_PAUSE);		break;	case SNDRV_PCM_TRIGGER_PAUSE_PUSH:		chip->ply_ctrl |= FM801_PAUSE;		break;	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:		chip->ply_ctrl &= ~FM801_PAUSE;		break;	default:		spin_unlock(&chip->reg_lock);		snd_BUG();		return -EINVAL;	}	outw(chip->ply_ctrl, FM801_REG(chip, PLY_CTRL));	spin_unlock(&chip->reg_lock);	return 0;}static int snd_fm801_capture_trigger(snd_pcm_substream_t * substream,				     int cmd){	fm801_t *chip = snd_pcm_substream_chip(substream);	spin_lock(&chip->reg_lock);	switch (cmd) {	case SNDRV_PCM_TRIGGER_START:		chip->cap_ctrl &= ~(FM801_BUF1_LAST |				     FM801_BUF2_LAST |				     FM801_PAUSE);		chip->cap_ctrl |= FM801_START |				   FM801_IMMED_STOP;		break;	case SNDRV_PCM_TRIGGER_STOP:		chip->cap_ctrl &= ~(FM801_START | FM801_PAUSE);		break;	case SNDRV_PCM_TRIGGER_PAUSE_PUSH:		chip->cap_ctrl |= FM801_PAUSE;		break;	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:		chip->cap_ctrl &= ~FM801_PAUSE;		break;	default:		spin_unlock(&chip->reg_lock);		snd_BUG();		return -EINVAL;	}	outw(chip->cap_ctrl, FM801_REG(chip, CAP_CTRL));	spin_unlock(&chip->reg_lock);	return 0;}static int snd_fm801_hw_params(snd_pcm_substream_t * substream,			       snd_pcm_hw_params_t * hw_params){	return snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params));}static int snd_fm801_hw_free(snd_pcm_substream_t * substream){	return snd_pcm_lib_free_pages(substream);}static int snd_fm801_playback_prepare(snd_pcm_substream_t * substream){	unsigned long flags;	fm801_t *chip = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime = substream->runtime;	chip->ply_size = snd_pcm_lib_buffer_bytes(substream);	chip->ply_count = snd_pcm_lib_period_bytes(substream);	spin_lock_irqsave(&chip->reg_lock, flags);	chip->ply_ctrl &= ~(FM801_START | FM801_16BIT |			     FM801_STEREO | FM801_RATE_MASK |			     FM801_CHANNELS_MASK);	if (snd_pcm_format_width(runtime->format) == 16)		chip->ply_ctrl |= FM801_16BIT;	if (runtime->channels > 1) {		chip->ply_ctrl |= FM801_STEREO;		if (runtime->channels == 4)			chip->ply_ctrl |= FM801_CHANNELS_4;		else if (runtime->channels == 6)			chip->ply_ctrl |= FM801_CHANNELS_6;	}	chip->ply_ctrl |= snd_fm801_rate_bits(runtime->rate) << FM801_RATE_SHIFT;	chip->ply_buf = 0;	outw(chip->ply_ctrl, FM801_REG(chip, PLY_CTRL));	outw(chip->ply_count - 1, FM801_REG(chip, PLY_COUNT));	chip->ply_buffer = runtime->dma_addr;	chip->ply_pos = 0;	outl(chip->ply_buffer, FM801_REG(chip, PLY_BUF1));	outl(chip->ply_buffer + (chip->ply_count % chip->ply_size), FM801_REG(chip, PLY_BUF2));	spin_unlock_irqrestore(&chip->reg_lock, flags);	return 0;}static int snd_fm801_capture_prepare(snd_pcm_substream_t * substream){	unsigned long flags;	fm801_t *chip = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime = substream->runtime;	chip->cap_size = snd_pcm_lib_buffer_bytes(substream);	chip->cap_count = snd_pcm_lib_period_bytes(substream);	spin_lock_irqsave(&chip->reg_lock, flags);	chip->cap_ctrl &= ~(FM801_START | FM801_16BIT |			     FM801_STEREO | FM801_RATE_MASK);	if (snd_pcm_format_width(runtime->format) == 16)		chip->cap_ctrl |= FM801_16BIT;	if (runtime->channels > 1)		chip->cap_ctrl |= FM801_STEREO;	chip->cap_ctrl |= snd_fm801_rate_bits(runtime->rate) << FM801_RATE_SHIFT;	chip->cap_buf = 0;	outw(chip->cap_ctrl, FM801_REG(chip, CAP_CTRL));	outw(chip->cap_count - 1, FM801_REG(chip, CAP_COUNT));	chip->cap_buffer = runtime->dma_addr;	chip->cap_pos = 0;	outl(chip->cap_buffer, FM801_REG(chip, CAP_BUF1));	outl(chip->cap_buffer + (chip->cap_count % chip->cap_size), FM801_REG(chip, CAP_BUF2));	spin_unlock_irqrestore(&chip->reg_lock, flags);	return 0;}static snd_pcm_uframes_t snd_fm801_playback_pointer(snd_pcm_substream_t * substream){	fm801_t *chip = snd_pcm_substream_chip(substream);	size_t ptr;	if (!(chip->ply_ctrl & FM801_START))		return 0;	spin_lock(&chip->reg_lock);	ptr = chip->ply_pos + (chip->ply_count - 1) - inw(FM801_REG(chip, PLY_COUNT));	if (inw(FM801_REG(chip, IRQ_STATUS)) & FM801_IRQ_PLAYBACK) {		ptr += chip->ply_count;		ptr %= chip->ply_size;	}	spin_unlock(&chip->reg_lock);	return bytes_to_frames(substream->runtime, ptr);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -