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 + -
显示快捷键?