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

📄 miro.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 3 页
字号:
 *  Init */static int __devinit snd_miro_configure(struct snd_miro *chip){	unsigned char wss_base_bits;	unsigned char irq_bits;	unsigned char dma_bits;	unsigned char mpu_port_bits = 0;	unsigned char mpu_irq_bits;	unsigned long flags;	switch (chip->hardware) {	case OPTi9XX_HW_82C924:		snd_miro_write_mask(chip, OPTi9XX_MC_REG(6), 0x02, 0x02);		snd_miro_write_mask(chip, OPTi9XX_MC_REG(1), 0x80, 0x80);		snd_miro_write_mask(chip, OPTi9XX_MC_REG(2), 0x20, 0x20); /* OPL4 */		snd_miro_write_mask(chip, OPTi9XX_MC_REG(3), 0xf0, 0xff);		snd_miro_write_mask(chip, OPTi9XX_MC_REG(5), 0x02, 0x02);		break;	case OPTi9XX_HW_82C929:		/* untested init commands for OPTi929 */		snd_miro_write_mask(chip, OPTi9XX_MC_REG(1), 0x80, 0x80);		snd_miro_write_mask(chip, OPTi9XX_MC_REG(2), 0x20, 0x20); /* OPL4 */		snd_miro_write_mask(chip, OPTi9XX_MC_REG(4), 0x00, 0x0c);		snd_miro_write_mask(chip, OPTi9XX_MC_REG(5), 0x02, 0x02);		break;	default:		snd_printk(KERN_ERR "chip %d not supported\n", chip->hardware);		return -EINVAL;	}	switch (chip->wss_base) {	case 0x530:		wss_base_bits = 0x00;		break;	case 0x604:		wss_base_bits = 0x03;		break;	case 0xe80:		wss_base_bits = 0x01;		break;	case 0xf40:		wss_base_bits = 0x02;		break;	default:		snd_printk(KERN_ERR "WSS port 0x%lx not valid\n", chip->wss_base);		goto __skip_base;	}	snd_miro_write_mask(chip, OPTi9XX_MC_REG(1), wss_base_bits << 4, 0x30);__skip_base:	switch (chip->irq) {	case 5:		irq_bits = 0x05;		break;	case 7:		irq_bits = 0x01;		break;	case 9:		irq_bits = 0x02;		break;	case 10:		irq_bits = 0x03;		break;	case 11:		irq_bits = 0x04;		break;	default:		snd_printk(KERN_ERR "WSS irq # %d not valid\n", chip->irq);		goto __skip_resources;	}	switch (chip->dma1) {	case 0:		dma_bits = 0x01;		break;	case 1:		dma_bits = 0x02;		break;	case 3:		dma_bits = 0x03;		break;	default:		snd_printk(KERN_ERR "WSS dma1 # %d not valid\n", chip->dma1);		goto __skip_resources;	}	if (chip->dma1 == chip->dma2) {		snd_printk(KERN_ERR "don't want to share dmas\n");		return -EBUSY;	}	switch (chip->dma2) {	case 0:	case 1:		break;	default:		snd_printk(KERN_ERR "WSS dma2 # %d not valid\n", chip->dma2);		goto __skip_resources;	}	dma_bits |= 0x04;	spin_lock_irqsave(&chip->lock, flags);	outb(irq_bits << 3 | dma_bits, chip->wss_base);	spin_unlock_irqrestore(&chip->lock, flags);__skip_resources:	if (chip->hardware > OPTi9XX_HW_82C928) {		switch (chip->mpu_port) {		case 0:		case -1:			break;		case 0x300:			mpu_port_bits = 0x03;			break;		case 0x310:			mpu_port_bits = 0x02;			break;		case 0x320:			mpu_port_bits = 0x01;			break;		case 0x330:			mpu_port_bits = 0x00;			break;		default:			snd_printk(KERN_ERR "MPU-401 port 0x%lx not valid\n",				   chip->mpu_port);			goto __skip_mpu;		}		switch (chip->mpu_irq) {		case 5:			mpu_irq_bits = 0x02;			break;		case 7:			mpu_irq_bits = 0x03;			break;		case 9:			mpu_irq_bits = 0x00;			break;		case 10:			mpu_irq_bits = 0x01;			break;		default:			snd_printk(KERN_ERR "MPU-401 irq # %d not valid\n",				   chip->mpu_irq);			goto __skip_mpu;		}		snd_miro_write_mask(chip, OPTi9XX_MC_REG(6),			(chip->mpu_port <= 0) ? 0x00 :				0x80 | mpu_port_bits << 5 | mpu_irq_bits << 3,			0xf8);	}__skip_mpu:	return 0;}static int __devinit snd_card_miro_detect(struct snd_card *card,					  struct snd_miro *chip){	int i, err;	unsigned char value;	for (i = OPTi9XX_HW_82C929; i <= OPTi9XX_HW_82C924; i++) {		if ((err = snd_miro_init(chip, i)) < 0)			return err;		if ((chip->res_mc_base = request_region(chip->mc_base, chip->mc_base_size, "OPTi9xx MC")) == NULL)			continue;		value = snd_miro_read(chip, OPTi9XX_MC_REG(1));		if ((value != 0xff) && (value != inb(chip->mc_base + 1)))			if (value == snd_miro_read(chip, OPTi9XX_MC_REG(1)))				return 1;		release_and_free_resource(chip->res_mc_base);		chip->res_mc_base = NULL;	}	return -ENODEV;}static int __devinit snd_card_miro_aci_detect(struct snd_card *card,					      struct snd_miro * miro){	unsigned char regval;	int i;	mutex_init(&miro->aci_mutex);	/* get ACI port from OPTi9xx MC 4 */	miro->mc_base = 0xf8c;	regval=inb(miro->mc_base + 4);	miro->aci_port = (regval & 0x10) ? 0x344: 0x354;	if ((miro->res_aci_port = request_region(miro->aci_port, 3, "miro aci")) == NULL) {		snd_printk(KERN_ERR "aci i/o area 0x%lx-0x%lx already used.\n", 			   miro->aci_port, miro->aci_port+2);		return -ENOMEM;	}        /* force ACI into a known state */	for (i = 0; i < 3; i++)		if (aci_cmd(miro, ACI_ERROR_OP, -1, -1) < 0) {			snd_printk(KERN_ERR "can't force aci into known state.\n");			return -ENXIO;		}	if ((miro->aci_vendor=aci_cmd(miro, ACI_READ_IDCODE, -1, -1)) < 0 ||	    (miro->aci_product=aci_cmd(miro, ACI_READ_IDCODE, -1, -1)) < 0) {		snd_printk(KERN_ERR "can't read aci id on 0x%lx.\n", miro->aci_port);		return -ENXIO;	}	if ((miro->aci_version=aci_cmd(miro, ACI_READ_VERSION, -1, -1)) < 0) {		snd_printk(KERN_ERR "can't read aci version on 0x%lx.\n", 			   miro->aci_port);		return -ENXIO;	}	if (aci_cmd(miro, ACI_INIT, -1, -1) < 0 ||	    aci_cmd(miro, ACI_ERROR_OP, ACI_ERROR_OP, ACI_ERROR_OP) < 0 ||	    aci_cmd(miro, ACI_ERROR_OP, ACI_ERROR_OP, ACI_ERROR_OP) < 0) {		snd_printk(KERN_ERR "can't initialize aci.\n"); 		return -ENXIO;	}	return 0;}static void snd_card_miro_free(struct snd_card *card){	struct snd_miro *miro = card->private_data;        	release_and_free_resource(miro->res_aci_port);	release_and_free_resource(miro->res_mc_base);}static int __devinit snd_miro_match(struct device *devptr, unsigned int n){	return 1;}static int __devinit snd_miro_probe(struct device *devptr, unsigned int n){	static long possible_ports[] = {0x530, 0xe80, 0xf40, 0x604, -1};	static long possible_mpu_ports[] = {0x330, 0x300, 0x310, 0x320, -1};	static int possible_irqs[] = {11, 9, 10, 7, -1};	static int possible_mpu_irqs[] = {10, 5, 9, 7, -1};	static int possible_dma1s[] = {3, 1, 0, -1};	static int possible_dma2s[][2] = {{1,-1}, {0,-1}, {-1,-1}, {0,-1}};	int error;	struct snd_miro *miro;	struct snd_cs4231 *codec;	struct snd_timer *timer;	struct snd_card *card;	struct snd_pcm *pcm;	struct snd_rawmidi *rmidi;	if (!(card = snd_card_new(index, id, THIS_MODULE,				  sizeof(struct snd_miro))))		return -ENOMEM;	card->private_free = snd_card_miro_free;	miro = card->private_data;	miro->card = card;	if ((error = snd_card_miro_aci_detect(card, miro)) < 0) {		snd_card_free(card);		snd_printk(KERN_ERR "unable to detect aci chip\n");		return -ENODEV;	}	/* init proc interface */	snd_miro_proc_init(miro);	if ((error = snd_card_miro_detect(card, miro)) < 0) {		snd_card_free(card);		snd_printk(KERN_ERR "unable to detect OPTi9xx chip\n");		return -ENODEV;	}	if (! miro->res_mc_base &&	    (miro->res_mc_base = request_region(miro->mc_base, miro->mc_base_size,						"miro (OPTi9xx MC)")) == NULL) {		snd_card_free(card);		snd_printk(KERN_ERR "request for OPTI9xx MC failed\n");		return -ENOMEM;	}	miro->wss_base = port;	miro->fm_port = fm_port;	miro->mpu_port = mpu_port;	miro->irq = irq;	miro->mpu_irq = mpu_irq;	miro->dma1 = dma1;	miro->dma2 = dma2;	if (miro->wss_base == SNDRV_AUTO_PORT) {		if ((miro->wss_base = snd_legacy_find_free_ioport(possible_ports, 4)) < 0) {			snd_card_free(card);			snd_printk(KERN_ERR "unable to find a free WSS port\n");			return -EBUSY;		}	}	if (miro->mpu_port == SNDRV_AUTO_PORT) {		if ((miro->mpu_port = snd_legacy_find_free_ioport(possible_mpu_ports, 2)) < 0) {			snd_card_free(card);			snd_printk(KERN_ERR "unable to find a free MPU401 port\n");			return -EBUSY;		}	}	if (miro->irq == SNDRV_AUTO_IRQ) {		if ((miro->irq = snd_legacy_find_free_irq(possible_irqs)) < 0) {			snd_card_free(card);			snd_printk(KERN_ERR "unable to find a free IRQ\n");			return -EBUSY;		}	}	if (miro->mpu_irq == SNDRV_AUTO_IRQ) {		if ((miro->mpu_irq = snd_legacy_find_free_irq(possible_mpu_irqs)) < 0) {			snd_card_free(card);			snd_printk(KERN_ERR "unable to find a free MPU401 IRQ\n");			return -EBUSY;		}	}	if (miro->dma1 == SNDRV_AUTO_DMA) {		if ((miro->dma1 = snd_legacy_find_free_dma(possible_dma1s)) < 0) {			snd_card_free(card);			snd_printk(KERN_ERR "unable to find a free DMA1\n");			return -EBUSY;		}	}	if (miro->dma2 == SNDRV_AUTO_DMA) {		if ((miro->dma2 = snd_legacy_find_free_dma(possible_dma2s[miro->dma1 % 4])) < 0) {			snd_card_free(card);			snd_printk(KERN_ERR "unable to find a free DMA2\n");			return -EBUSY;		}	}	if ((error = snd_miro_configure(miro))) {		snd_card_free(card);		return error;	}	if ((error = snd_cs4231_create(card, miro->wss_base + 4, -1,				       miro->irq, miro->dma1, miro->dma2,				       CS4231_HW_AD1845,				       0,				       &codec)) < 0) {		snd_card_free(card);		return error;	}	if ((error = snd_cs4231_pcm(codec, 0, &pcm)) < 0) {		snd_card_free(card);		return error;	}	if ((error = snd_cs4231_mixer(codec)) < 0) {		snd_card_free(card);		return error;	}	if ((error = snd_cs4231_timer(codec, 0, &timer)) < 0) {		snd_card_free(card);		return error;	}	miro->pcm = pcm;	if ((error = snd_miro_mixer(miro)) < 0) {		snd_card_free(card);		return error;	}	if (miro->aci_vendor == 'm') {		/* It looks like a miro sound card. */		switch (miro->aci_product) {		case 'A':			sprintf(card->shortname, 				"miroSOUND PCM1 pro / PCM12");			break;		case 'B':			sprintf(card->shortname, 				"miroSOUND PCM12");			break;		case 'C':			sprintf(card->shortname, 				"miroSOUND PCM20 radio");			break;		default:			sprintf(card->shortname, 				"unknown miro");			snd_printk(KERN_INFO "unknown miro aci id\n");			break;		}	} else {		snd_printk(KERN_INFO "found unsupported aci card\n");		sprintf(card->shortname, "unknown Cardinal Technologies");	}	strcpy(card->driver, "miro");	sprintf(card->longname, "%s: OPTi%s, %s at 0x%lx, irq %d, dma %d&%d",		card->shortname, miro->name, pcm->name, miro->wss_base + 4,		miro->irq, miro->dma1, miro->dma2);	if (miro->mpu_port <= 0 || miro->mpu_port == SNDRV_AUTO_PORT)		rmidi = NULL;	else		if ((error = snd_mpu401_uart_new(card, 0, MPU401_HW_MPU401,				miro->mpu_port, 0, miro->mpu_irq, IRQF_DISABLED,				&rmidi)))			snd_printk(KERN_WARNING "no MPU-401 device at 0x%lx?\n", miro->mpu_port);	if (miro->fm_port > 0 && miro->fm_port != SNDRV_AUTO_PORT) {		struct snd_opl3 *opl3 = NULL;		struct snd_opl4 *opl4;		if (snd_opl4_create(card, miro->fm_port, miro->fm_port - 8, 				    2, &opl3, &opl4) < 0)			snd_printk(KERN_WARNING "no OPL4 device at 0x%lx\n", miro->fm_port);	}	if ((error = snd_set_aci_init_values(miro)) < 0) {		snd_card_free(card);                return error;	}	snd_card_set_dev(card, devptr);	if ((error = snd_card_register(card))) {		snd_card_free(card);		return error;	}	dev_set_drvdata(devptr, card);	return 0;}static int __devexit snd_miro_remove(struct device *devptr, unsigned int dev){	snd_card_free(dev_get_drvdata(devptr));	dev_set_drvdata(devptr, NULL);	return 0;}#define DEV_NAME "miro"static struct isa_driver snd_miro_driver = {	.match		= snd_miro_match,	.probe		= snd_miro_probe,	.remove		= __devexit_p(snd_miro_remove),	/* FIXME: suspend/resume */	.driver		= {		.name	= DEV_NAME	},};static int __init alsa_card_miro_init(void){	return isa_register_driver(&snd_miro_driver, 1);}static void __exit alsa_card_miro_exit(void){	isa_unregister_driver(&snd_miro_driver);}module_init(alsa_card_miro_init)module_exit(alsa_card_miro_exit)

⌨️ 快捷键说明

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