📄 miro.c
字号:
* 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 + -