cs4231.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 2,248 行 · 第 1/5 页

C
2,248
字号
		val2 = mask - val2;	}	val1 <<= shift_left;	val2 <<= shift_right;	spin_lock_irqsave(&chip->lock, flags);	val1 = (chip->image[left_reg] & ~(mask << shift_left)) | val1;	val2 = (chip->image[right_reg] & ~(mask << shift_right)) | val2;	change = val1 != chip->image[left_reg] || val2 != chip->image[right_reg];	snd_cs4231_out(chip, left_reg, val1);	snd_cs4231_out(chip, right_reg, val2);	spin_unlock_irqrestore(&chip->lock, flags);	return change;}#define CS4231_SINGLE(xname, xindex, reg, shift, mask, invert) \{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = xindex, \  .info = snd_cs4231_info_single, \  .get = snd_cs4231_get_single, .put = snd_cs4231_put_single, \  .private_value = reg | (shift << 8) | (mask << 16) | (invert << 24) }#define CS4231_DOUBLE(xname, xindex, left_reg, right_reg, shift_left, shift_right, mask, invert) \{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, .index = xindex, \  .info = snd_cs4231_info_double, \  .get = snd_cs4231_get_double, .put = snd_cs4231_put_double, \  .private_value = left_reg | (right_reg << 8) | (shift_left << 16) | (shift_right << 19) | (mask << 24) | (invert << 22) }static snd_kcontrol_new_t snd_cs4231_controls[] = {CS4231_DOUBLE("PCM Playback Switch", 0, CS4231_LEFT_OUTPUT, CS4231_RIGHT_OUTPUT, 7, 7, 1, 1),CS4231_DOUBLE("PCM Playback Volume", 0, CS4231_LEFT_OUTPUT, CS4231_RIGHT_OUTPUT, 0, 0, 63, 1),CS4231_DOUBLE("Line Playback Switch", 0, CS4231_LEFT_LINE_IN, CS4231_RIGHT_LINE_IN, 7, 7, 1, 1),CS4231_DOUBLE("Line Playback Volume", 0, CS4231_LEFT_LINE_IN, CS4231_RIGHT_LINE_IN, 0, 0, 31, 1),CS4231_DOUBLE("Aux Playback Switch", 0, CS4231_AUX1_LEFT_INPUT, CS4231_AUX1_RIGHT_INPUT, 7, 7, 1, 1),CS4231_DOUBLE("Aux Playback Volume", 0, CS4231_AUX1_LEFT_INPUT, CS4231_AUX1_RIGHT_INPUT, 0, 0, 31, 1),CS4231_DOUBLE("Aux Playback Switch", 1, CS4231_AUX2_LEFT_INPUT, CS4231_AUX2_RIGHT_INPUT, 7, 7, 1, 1),CS4231_DOUBLE("Aux Playback Volume", 1, CS4231_AUX2_LEFT_INPUT, CS4231_AUX2_RIGHT_INPUT, 0, 0, 31, 1),CS4231_SINGLE("Mono Playback Switch", 0, CS4231_MONO_CTRL, 7, 1, 1),CS4231_SINGLE("Mono Playback Volume", 0, CS4231_MONO_CTRL, 0, 15, 1),CS4231_SINGLE("Mono Output Playback Switch", 0, CS4231_MONO_CTRL, 6, 1, 1),CS4231_SINGLE("Mono Output Playback Bypass", 0, CS4231_MONO_CTRL, 5, 1, 0),CS4231_DOUBLE("Capture Volume", 0, CS4231_LEFT_INPUT, CS4231_RIGHT_INPUT, 0, 0, 15, 0),{	.iface	= SNDRV_CTL_ELEM_IFACE_MIXER,	.name	= "Capture Source",	.info	= snd_cs4231_info_mux,	.get	= snd_cs4231_get_mux,	.put	= snd_cs4231_put_mux,},CS4231_DOUBLE("Mic Boost", 0, CS4231_LEFT_INPUT, CS4231_RIGHT_INPUT, 5, 5, 1, 0),CS4231_SINGLE("Loopback Capture Switch", 0, CS4231_LOOPBACK, 0, 1, 0),CS4231_SINGLE("Loopback Capture Volume", 0, CS4231_LOOPBACK, 2, 63, 1),/* SPARC specific uses of XCTL{0,1} general purpose outputs.  */CS4231_SINGLE("Line Out Switch", 0, CS4231_PIN_CTRL, 6, 1, 1),CS4231_SINGLE("Headphone Out Switch", 0, CS4231_PIN_CTRL, 7, 1, 1)};                                        int snd_cs4231_mixer(cs4231_t *chip){	snd_card_t *card;	int err, idx;	snd_assert(chip != NULL && chip->pcm != NULL, return -EINVAL);	card = chip->card;	strcpy(card->mixername, chip->pcm->name);	for (idx = 0; idx < ARRAY_SIZE(snd_cs4231_controls); idx++) {		if ((err = snd_ctl_add(card,				       snd_ctl_new1(&snd_cs4231_controls[idx],						    chip))) < 0)			return err;	}	return 0;}static int dev;static int cs4231_attach_begin(snd_card_t **rcard){	snd_card_t *card;	*rcard = NULL;	if (dev >= SNDRV_CARDS)		return -ENODEV;	if (!enable[dev]) {		dev++;		return -ENOENT;	}	card = snd_card_new(index[dev], id[dev], THIS_MODULE, 0);	if (card == NULL)		return -ENOMEM;	strcpy(card->driver, "CS4231");	strcpy(card->shortname, "Sun CS4231");	*rcard = card;	return 0;}static int cs4231_attach_finish(snd_card_t *card, cs4231_t *chip){	int err;	if ((err = snd_cs4231_pcm(chip)) < 0)		goto out_err;	if ((err = snd_cs4231_mixer(chip)) < 0)		goto out_err;	if ((err = snd_cs4231_timer(chip)) < 0)		goto out_err;	if ((err = snd_card_register(card)) < 0)		goto out_err;	chip->next = cs4231_list;	cs4231_list = chip;	dev++;	return 0;out_err:	snd_card_free(card);	return err;}#ifdef SBUS_SUPPORTstatic int snd_cs4231_sbus_free(cs4231_t *chip){	if (chip->irq[0])		free_irq(chip->irq[0], chip);	if (chip->port)		sbus_iounmap(chip->port, chip->regs_size);	if (chip->timer)		snd_device_free(chip->card, chip->timer);	kfree(chip);	return 0;}static int snd_cs4231_sbus_dev_free(snd_device_t *device){	cs4231_t *cp = device->device_data;	return snd_cs4231_sbus_free(cp);}static snd_device_ops_t snd_cs4231_sbus_dev_ops = {	.dev_free	=	snd_cs4231_sbus_dev_free,};static int __init snd_cs4231_sbus_create(snd_card_t *card,					 struct sbus_dev *sdev,					 int dev,					 cs4231_t **rchip){	cs4231_t *chip;	int err;	*rchip = NULL;	chip = kcalloc(1, sizeof(*chip), GFP_KERNEL);	if (chip == NULL)		return -ENOMEM;	spin_lock_init(&chip->lock);	init_MUTEX(&chip->mce_mutex);	init_MUTEX(&chip->open_mutex);	chip->card = card;	chip->dev_u.sdev = sdev;	chip->regs_size = sdev->reg_addrs[0].reg_size;	memcpy(&chip->image, &snd_cs4231_original_image,	       sizeof(snd_cs4231_original_image));	chip->port = sbus_ioremap(&sdev->resource[0], 0,				  chip->regs_size, "cs4231");	if (!chip->port) {		snd_printk("cs4231-%d: Unable to map chip registers.\n", dev);		return -EIO;	}	if (request_irq(sdev->irqs[0], snd_cs4231_sbus_interrupt,			SA_SHIRQ, "cs4231", chip)) {		snd_printk("cs4231-%d: Unable to grab SBUS IRQ %s\n",			   dev,			   __irq_itoa(sdev->irqs[0]));		snd_cs4231_sbus_free(chip);		return -EBUSY;	}	chip->irq[0] = sdev->irqs[0];	if (snd_cs4231_probe(chip) < 0) {		snd_cs4231_sbus_free(chip);		return -ENODEV;	}	snd_cs4231_init(chip);	if ((err = snd_device_new(card, SNDRV_DEV_LOWLEVEL,				  chip, &snd_cs4231_sbus_dev_ops)) < 0) {		snd_cs4231_sbus_free(chip);		return err;	}	*rchip = chip;	return 0;}static int cs4231_sbus_attach(struct sbus_dev *sdev){	struct resource *rp = &sdev->resource[0];	cs4231_t *cp;	snd_card_t *card;	int err;	err = cs4231_attach_begin(&card);	if (err)		return err;	sprintf(card->longname, "%s at 0x%02lx:0x%08lx, irq %s",		card->shortname,		rp->flags & 0xffL,		rp->start,		__irq_itoa(sdev->irqs[0]));	if ((err = snd_cs4231_sbus_create(card, sdev, dev, &cp)) < 0) {		snd_card_free(card);		return err;	}	return cs4231_attach_finish(card, cp);}#endif#ifdef EBUS_SUPPORTstatic int snd_cs4231_ebus_free(cs4231_t *chip){	if (chip->eb2c.regs) {		ebus_dma_unregister(&chip->eb2c);		iounmap(chip->eb2c.regs);	}	if (chip->eb2p.regs) {		ebus_dma_unregister(&chip->eb2p);		iounmap(chip->eb2p.regs);	}	if (chip->port)		iounmap(chip->port);	if (chip->timer)		snd_device_free(chip->card, chip->timer);	kfree(chip);	return 0;}static int snd_cs4231_ebus_dev_free(snd_device_t *device){	cs4231_t *cp = device->device_data;	return snd_cs4231_ebus_free(cp);}static snd_device_ops_t snd_cs4231_ebus_dev_ops = {	.dev_free	=	snd_cs4231_ebus_dev_free,};static int __init snd_cs4231_ebus_create(snd_card_t *card,					 struct linux_ebus_device *edev,					 int dev,					 cs4231_t **rchip){	cs4231_t *chip;	int err;	*rchip = NULL;	chip = kcalloc(1, sizeof(*chip), GFP_KERNEL);	if (chip == NULL)		return -ENOMEM;	spin_lock_init(&chip->lock);	spin_lock_init(&chip->eb2c.lock);	spin_lock_init(&chip->eb2p.lock);	init_MUTEX(&chip->mce_mutex);	init_MUTEX(&chip->open_mutex);	chip->flags |= CS4231_FLAG_EBUS;	chip->card = card;	chip->dev_u.pdev = edev->bus->self;	memcpy(&chip->image, &snd_cs4231_original_image,	       sizeof(snd_cs4231_original_image));	strcpy(chip->eb2c.name, "cs4231(capture)");	chip->eb2c.flags = EBUS_DMA_FLAG_USE_EBDMA_HANDLER;	chip->eb2c.callback = snd_cs4231_ebus_capture_callback;	chip->eb2c.client_cookie = chip;	chip->eb2c.irq = edev->irqs[0];	strcpy(chip->eb2p.name, "cs4231(play)");	chip->eb2p.flags = EBUS_DMA_FLAG_USE_EBDMA_HANDLER;	chip->eb2p.callback = snd_cs4231_ebus_play_callback;	chip->eb2p.client_cookie = chip;	chip->eb2p.irq = edev->irqs[1];	chip->port = ioremap(edev->resource[0].start, 0x10);	chip->eb2p.regs = ioremap(edev->resource[1].start, 0x10);	chip->eb2c.regs = ioremap(edev->resource[2].start, 0x10);	if (!chip->port || !chip->eb2p.regs || !chip->eb2c.regs) {		snd_cs4231_ebus_free(chip);		snd_printk("cs4231-%d: Unable to map chip registers.\n", dev);		return -EIO;	}	if (ebus_dma_register(&chip->eb2c)) {		snd_cs4231_ebus_free(chip);		snd_printk("cs4231-%d: Unable to register EBUS capture DMA\n", dev);		return -EBUSY;	}	if (ebus_dma_irq_enable(&chip->eb2c, 1)) {		snd_cs4231_ebus_free(chip);		snd_printk("cs4231-%d: Unable to enable EBUS capture IRQ\n", dev);		return -EBUSY;	}	if (ebus_dma_register(&chip->eb2p)) {		snd_cs4231_ebus_free(chip);		snd_printk("cs4231-%d: Unable to register EBUS play DMA\n", dev);		return -EBUSY;	}	if (ebus_dma_irq_enable(&chip->eb2p, 1)) {		snd_cs4231_ebus_free(chip);		snd_printk("cs4231-%d: Unable to enable EBUS play IRQ\n", dev);		return -EBUSY;	}	if (snd_cs4231_probe(chip) < 0) {		snd_cs4231_ebus_free(chip);		return -ENODEV;	}	snd_cs4231_init(chip);	if ((err = snd_device_new(card, SNDRV_DEV_LOWLEVEL,				  chip, &snd_cs4231_ebus_dev_ops)) < 0) {		snd_cs4231_ebus_free(chip);		return err;	}	*rchip = chip;	return 0;}static int cs4231_ebus_attach(struct linux_ebus_device *edev){	snd_card_t *card;	cs4231_t *chip;	int err;	err = cs4231_attach_begin(&card);	if (err)		return err;	sprintf(card->longname, "%s at 0x%lx, irq %s",		card->shortname,		edev->resource[0].start,		__irq_itoa(edev->irqs[0]));	if ((err = snd_cs4231_ebus_create(card, edev, dev, &chip)) < 0) {		snd_card_free(card);		return err;	}	return cs4231_attach_finish(card, chip);}#endifstatic int __init cs4231_init(void){#ifdef SBUS_SUPPORT	struct sbus_bus *sbus;	struct sbus_dev *sdev;#endif#ifdef EBUS_SUPPORT	struct linux_ebus *ebus;	struct linux_ebus_device *edev;#endif	int found;	found = 0;#ifdef SBUS_SUPPORT	for_all_sbusdev(sdev, sbus) {		if (!strcmp(sdev->prom_name, "SUNW,CS4231")) {			if (cs4231_sbus_attach(sdev) == 0)				found++;		}	}#endif#ifdef EBUS_SUPPORT	for_each_ebus(ebus) {		for_each_ebusdev(edev, ebus) {			int match = 0;			if (!strcmp(edev->prom_name, "SUNW,CS4231")) {				match = 1;			} else if (!strcmp(edev->prom_name, "audio")) {				char compat[16];				prom_getstring(edev->prom_node, "compatible",					       compat, sizeof(compat));				compat[15] = '\0';				if (!strcmp(compat, "SUNW,CS4231"))					match = 1;			}			if (match &&			    cs4231_ebus_attach(edev) == 0)				found++;		}	}#endif	return (found > 0) ? 0 : -EIO;}static void __exit cs4231_exit(void){	cs4231_t *p = cs4231_list;	while (p != NULL) {		cs4231_t *next = p->next;		snd_card_free(p->card);		p = next;	}	cs4231_list = NULL;}module_init(cs4231_init);module_exit(cs4231_exit);

⌨️ 快捷键说明

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