]> Pileus Git - ~andy/linux/blobdiff - sound/isa/wss/wss_lib.c
Merge branch 'master' into next
[~andy/linux] / sound / isa / wss / wss_lib.c
index 57d1e8ee6bbbf67364fabd4528272ed060b9262c..3d6c5f2838af41156d7cd8d07f480e58f7dd6e04 100644 (file)
@@ -282,7 +282,7 @@ static void snd_wss_debug(struct snd_wss *chip)
        printk(KERN_DEBUG
                "CS4231 REGS:      INDEX = 0x%02x  "
                "                 STATUS = 0x%02x\n",
-                                       wss_inb(chip, CS4231P(REGSEL),
+                                       wss_inb(chip, CS4231P(REGSEL)),
                                        wss_inb(chip, CS4231P(STATUS)));
        printk(KERN_DEBUG
                "  0x00: left input      = 0x%02x  "
@@ -574,7 +574,7 @@ static void snd_wss_calibrate_mute(struct snd_wss *chip, int mute)
 {
        unsigned long flags;
 
-       mute = mute ? 1 : 0;
+       mute = mute ? 0x80 : 0;
        spin_lock_irqsave(&chip->reg_lock, flags);
        if (chip->calibrate_mute == mute) {
                spin_unlock_irqrestore(&chip->reg_lock, flags);
@@ -589,34 +589,34 @@ static void snd_wss_calibrate_mute(struct snd_wss *chip, int mute)
                             chip->image[CS4231_LOOPBACK]);
        }
        snd_wss_dout(chip, CS4231_AUX1_LEFT_INPUT,
-                    mute ? 0x80 : chip->image[CS4231_AUX1_LEFT_INPUT]);
+                    mute | chip->image[CS4231_AUX1_LEFT_INPUT]);
        snd_wss_dout(chip, CS4231_AUX1_RIGHT_INPUT,
-                    mute ? 0x80 : chip->image[CS4231_AUX1_RIGHT_INPUT]);
+                    mute | chip->image[CS4231_AUX1_RIGHT_INPUT]);
        snd_wss_dout(chip, CS4231_AUX2_LEFT_INPUT,
-                    mute ? 0x80 : chip->image[CS4231_AUX2_LEFT_INPUT]);
+                    mute | chip->image[CS4231_AUX2_LEFT_INPUT]);
        snd_wss_dout(chip, CS4231_AUX2_RIGHT_INPUT,
-                    mute ? 0x80 : chip->image[CS4231_AUX2_RIGHT_INPUT]);
+                    mute | chip->image[CS4231_AUX2_RIGHT_INPUT]);
        snd_wss_dout(chip, CS4231_LEFT_OUTPUT,
-                    mute ? 0x80 : chip->image[CS4231_LEFT_OUTPUT]);
+                    mute | chip->image[CS4231_LEFT_OUTPUT]);
        snd_wss_dout(chip, CS4231_RIGHT_OUTPUT,
-                    mute ? 0x80 : chip->image[CS4231_RIGHT_OUTPUT]);
+                    mute | chip->image[CS4231_RIGHT_OUTPUT]);
        if (!(chip->hardware & WSS_HW_AD1848_MASK)) {
                snd_wss_dout(chip, CS4231_LEFT_LINE_IN,
-                            mute ? 0x80 : chip->image[CS4231_LEFT_LINE_IN]);
+                            mute | chip->image[CS4231_LEFT_LINE_IN]);
                snd_wss_dout(chip, CS4231_RIGHT_LINE_IN,
-                            mute ? 0x80 : chip->image[CS4231_RIGHT_LINE_IN]);
+                            mute | chip->image[CS4231_RIGHT_LINE_IN]);
                snd_wss_dout(chip, CS4231_MONO_CTRL,
                             mute ? 0xc0 : chip->image[CS4231_MONO_CTRL]);
        }
        if (chip->hardware == WSS_HW_INTERWAVE) {
                snd_wss_dout(chip, CS4231_LEFT_MIC_INPUT,
-                            mute ? 0x80 : chip->image[CS4231_LEFT_MIC_INPUT]);
+                            mute | chip->image[CS4231_LEFT_MIC_INPUT]);
                snd_wss_dout(chip, CS4231_RIGHT_MIC_INPUT,
-                            mute ? 0x80 : chip->image[CS4231_RIGHT_MIC_INPUT]);
+                            mute | chip->image[CS4231_RIGHT_MIC_INPUT]);
                snd_wss_dout(chip, CS4231_LINE_LEFT_OUTPUT,
-                       mute ? 0x80 : chip->image[CS4231_LINE_LEFT_OUTPUT]);
+                            mute | chip->image[CS4231_LINE_LEFT_OUTPUT]);
                snd_wss_dout(chip, CS4231_LINE_RIGHT_OUTPUT,
-                       mute ? 0x80 : chip->image[CS4231_LINE_RIGHT_OUTPUT]);
+                            mute | chip->image[CS4231_LINE_RIGHT_OUTPUT]);
        }
        chip->calibrate_mute = mute;
        spin_unlock_irqrestore(&chip->reg_lock, flags);
@@ -1073,7 +1073,11 @@ irqreturn_t snd_wss_interrupt(int irq, void *dev_id)
        struct snd_wss *chip = dev_id;
        unsigned char status;
 
-       status = snd_wss_in(chip, CS4231_IRQ_STATUS);
+       if (chip->hardware & WSS_HW_AD1848_MASK)
+               /* pretend it was the only possible irq for AD1848 */
+               status = CS4231_PLAYBACK_IRQ;
+       else
+               status = snd_wss_in(chip, CS4231_IRQ_STATUS);
        if (status & CS4231_TIMER_IRQ) {
                if (chip->timer)
                        snd_timer_interrupt(chip->timer, chip->timer->sticks);
@@ -1105,7 +1109,11 @@ irqreturn_t snd_wss_interrupt(int irq, void *dev_id)
        }
 
        spin_lock(&chip->reg_lock);
-       snd_wss_outm(chip, CS4231_IRQ_STATUS, ~CS4231_ALL_IRQS | ~status, 0);
+       status = ~CS4231_ALL_IRQS | ~status;
+       if (chip->hardware & WSS_HW_AD1848_MASK)
+               wss_outb(chip, CS4231P(STATUS), 0);
+       else
+               snd_wss_outm(chip, CS4231_IRQ_STATUS, status, 0);
        spin_unlock(&chip->reg_lock);
        return IRQ_HANDLED;
 }
@@ -1137,36 +1145,119 @@ static snd_pcm_uframes_t snd_wss_capture_pointer(struct snd_pcm_substream *subst
 
  */
 
+static int snd_ad1848_probe(struct snd_wss *chip)
+{
+       unsigned long timeout = jiffies + msecs_to_jiffies(1000);
+       unsigned long flags;
+       unsigned char r;
+       unsigned short hardware = 0;
+       int err = 0;
+       int i;
+
+       while (wss_inb(chip, CS4231P(REGSEL)) & CS4231_INIT) {
+               if (time_after(jiffies, timeout))
+                       return -ENODEV;
+               cond_resched();
+       }
+       spin_lock_irqsave(&chip->reg_lock, flags);
+
+       /* set CS423x MODE 1 */
+       snd_wss_dout(chip, CS4231_MISC_INFO, 0);
+
+       snd_wss_dout(chip, CS4231_RIGHT_INPUT, 0x45); /* 0x55 & ~0x10 */
+       r = snd_wss_in(chip, CS4231_RIGHT_INPUT);
+       if (r != 0x45) {
+               /* RMGE always high on AD1847 */
+               if ((r & ~CS4231_ENABLE_MIC_GAIN) != 0x45) {
+                       err = -ENODEV;
+                       goto out;
+               }
+               hardware = WSS_HW_AD1847;
+       } else {
+               snd_wss_dout(chip, CS4231_LEFT_INPUT,  0xaa);
+               r = snd_wss_in(chip, CS4231_LEFT_INPUT);
+               /* L/RMGE always low on AT2320 */
+               if ((r | CS4231_ENABLE_MIC_GAIN) != 0xaa) {
+                       err = -ENODEV;
+                       goto out;
+               }
+       }
+
+       /* clear pending IRQ */
+       wss_inb(chip, CS4231P(STATUS));
+       wss_outb(chip, CS4231P(STATUS), 0);
+       mb();
+
+       if ((chip->hardware & WSS_HW_TYPE_MASK) != WSS_HW_DETECT)
+               goto out;
+
+       if (hardware) {
+               chip->hardware = hardware;
+               goto out;
+       }
+
+       r = snd_wss_in(chip, CS4231_MISC_INFO);
+
+       /* set CS423x MODE 2 */
+       snd_wss_dout(chip, CS4231_MISC_INFO, CS4231_MODE2);
+       for (i = 0; i < 16; i++) {
+               if (snd_wss_in(chip, i) != snd_wss_in(chip, 16 + i)) {
+                       /* we have more than 16 registers: check ID */
+                       if ((r & 0xf) != 0xa)
+                               goto out_mode;
+                       /*
+                        * on CMI8330, CS4231_VERSION is volume control and
+                        * can be set to 0
+                        */
+                       snd_wss_dout(chip, CS4231_VERSION, 0);
+                       r = snd_wss_in(chip, CS4231_VERSION) & 0xe7;
+                       if (!r)
+                               chip->hardware = WSS_HW_CMI8330;
+                       goto out_mode;
+               }
+       }
+       if (r & 0x80)
+               chip->hardware = WSS_HW_CS4248;
+       else
+               chip->hardware = WSS_HW_AD1848;
+out_mode:
+       snd_wss_dout(chip, CS4231_MISC_INFO, 0);
+out:
+       spin_unlock_irqrestore(&chip->reg_lock, flags);
+       return err;
+}
+
 static int snd_wss_probe(struct snd_wss *chip)
 {
        unsigned long flags;
-       int i, id, rev;
+       int i, id, rev, regnum;
        unsigned char *ptr;
        unsigned int hw;
 
-#if 0
-       snd_wss_debug(chip);
-#endif
-       id = 0;
-       for (i = 0; i < 50; i++) {
-               mb();
-               if (wss_inb(chip, CS4231P(REGSEL)) & CS4231_INIT)
-                       udelay(2000);
-               else {
-                       spin_lock_irqsave(&chip->reg_lock, flags);
-                       snd_wss_out(chip, CS4231_MISC_INFO, CS4231_MODE2);
-                       id = snd_wss_in(chip, CS4231_MISC_INFO) & 0x0f;
-                       spin_unlock_irqrestore(&chip->reg_lock, flags);
-                       if (id == 0x0a)
-                               break;  /* this is valid value */
-               }
-       }
-       snd_printdd("wss: port = 0x%lx, id = 0x%x\n", chip->port, id);
-       if (id != 0x0a)
-               return -ENODEV; /* no valid device found */
+       id = snd_ad1848_probe(chip);
+       if (id < 0)
+               return id;
 
        hw = chip->hardware;
        if ((hw & WSS_HW_TYPE_MASK) == WSS_HW_DETECT) {
+               for (i = 0; i < 50; i++) {
+                       mb();
+                       if (wss_inb(chip, CS4231P(REGSEL)) & CS4231_INIT)
+                               msleep(2);
+                       else {
+                               spin_lock_irqsave(&chip->reg_lock, flags);
+                               snd_wss_out(chip, CS4231_MISC_INFO,
+                                           CS4231_MODE2);
+                               id = snd_wss_in(chip, CS4231_MISC_INFO) & 0x0f;
+                               spin_unlock_irqrestore(&chip->reg_lock, flags);
+                               if (id == 0x0a)
+                                       break;  /* this is valid value */
+                       }
+               }
+               snd_printdd("wss: port = 0x%lx, id = 0x%x\n", chip->port, id);
+               if (id != 0x0a)
+                       return -ENODEV; /* no valid device found */
+
                rev = snd_wss_in(chip, CS4231_VERSION) & 0xe7;
                snd_printdd("CS4231: VERSION (I25) = 0x%x\n", rev);
                if (rev == 0x80) {
@@ -1197,7 +1288,8 @@ static int snd_wss_probe(struct snd_wss *chip)
        mb();
        spin_unlock_irqrestore(&chip->reg_lock, flags);
 
-       chip->image[CS4231_MISC_INFO] = CS4231_MODE2;
+       if (!(chip->hardware & WSS_HW_AD1848_MASK))
+               chip->image[CS4231_MISC_INFO] = CS4231_MODE2;
        switch (chip->hardware) {
        case WSS_HW_INTERWAVE:
                chip->image[CS4231_MISC_INFO] = CS4231_IW_MODE3;
@@ -1223,9 +1315,10 @@ static int snd_wss_probe(struct snd_wss *chip)
                        chip->hardware == WSS_HW_INTERWAVE ? 0xc2 : 0x01;
        }
        ptr = (unsigned char *) &chip->image;
+       regnum = (chip->hardware & WSS_HW_AD1848_MASK) ? 16 : 32;
        snd_wss_mce_down(chip);
        spin_lock_irqsave(&chip->reg_lock, flags);
-       for (i = 0; i < 32; i++)        /* ok.. fill all CS4231 registers */
+       for (i = 0; i < regnum; i++)    /* ok.. fill all registers */
                snd_wss_out(chip, i, *ptr++);
        spin_unlock_irqrestore(&chip->reg_lock, flags);
        snd_wss_mce_up(chip);
@@ -1409,8 +1502,10 @@ static int snd_wss_capture_open(struct snd_pcm_substream *substream)
 
        /* hardware limitation of cheap chips */
        if (chip->hardware == WSS_HW_CS4235 ||
-           chip->hardware == WSS_HW_CS4239)
-               runtime->hw.formats = SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S16_LE;
+           chip->hardware == WSS_HW_CS4239 ||
+           chip->hardware == WSS_HW_OPTI93X)
+               runtime->hw.formats = SNDRV_PCM_FMTBIT_U8 |
+                                     SNDRV_PCM_FMTBIT_S16_LE;
 
        snd_pcm_limit_isa_dma_size(chip->dma2, &runtime->hw.buffer_bytes_max);
        snd_pcm_limit_isa_dma_size(chip->dma2, &runtime->hw.period_bytes_max);
@@ -1635,6 +1730,10 @@ static int snd_wss_new(struct snd_card *card,
        else
                memcpy(&chip->image, &snd_wss_original_image,
                       sizeof(snd_wss_original_image));
+       if (chip->hardware & WSS_HW_AD1848_MASK) {
+               chip->image[CS4231_PIN_CTRL] = 0;
+               chip->image[CS4231_TEST_INIT] = 0;
+       }
 
        *rchip = chip;
        return 0;
@@ -1662,7 +1761,7 @@ int snd_wss_create(struct snd_card *card,
        chip->dma1 = -1;
        chip->dma2 = -1;
 
-       chip->res_port = request_region(port, 4, "CS4231");
+       chip->res_port = request_region(port, 4, "WSS");
        if (!chip->res_port) {
                snd_printk(KERN_ERR "wss: can't grab port 0x%lx\n", port);
                snd_wss_free(chip);
@@ -1681,20 +1780,20 @@ int snd_wss_create(struct snd_card *card,
        chip->cport = cport;
        if (!(hwshare & WSS_HWSHARE_IRQ))
                if (request_irq(irq, snd_wss_interrupt, IRQF_DISABLED,
-                               "CS4231", (void *) chip)) {
+                               "WSS", (void *) chip)) {
                        snd_printk(KERN_ERR "wss: can't grab IRQ %d\n", irq);
                        snd_wss_free(chip);
                        return -EBUSY;
                }
        chip->irq = irq;
-       if (!(hwshare & WSS_HWSHARE_DMA1) && request_dma(dma1, "CS4231 - 1")) {
+       if (!(hwshare & WSS_HWSHARE_DMA1) && request_dma(dma1, "WSS - 1")) {
                snd_printk(KERN_ERR "wss: can't grab DMA1 %d\n", dma1);
                snd_wss_free(chip);
                return -EBUSY;
        }
        chip->dma1 = dma1;
        if (!(hwshare & WSS_HWSHARE_DMA2) && dma1 != dma2 &&
-             dma2 >= 0 && request_dma(dma2, "CS4231 - 2")) {
+             dma2 >= 0 && request_dma(dma2, "WSS - 2")) {
                snd_printk(KERN_ERR "wss: can't grab DMA2 %d\n", dma2);
                snd_wss_free(chip);
                return -EBUSY;
@@ -1705,6 +1804,12 @@ int snd_wss_create(struct snd_card *card,
        } else
                chip->dma2 = dma2;
 
+       if (hardware == WSS_HW_THINKPAD) {
+               chip->thinkpad_flag = 1;
+               chip->hardware = WSS_HW_DETECT; /* reset */
+               snd_wss_thinkpad_twiddle(chip, 1);
+       }
+
        /* global setup */
        if (snd_wss_probe(chip) < 0) {
                snd_wss_free(chip);
@@ -1768,19 +1873,9 @@ int snd_wss_pcm(struct snd_wss *chip, int device, struct snd_pcm **rpcm)
        if (err < 0)
                return err;
 
-       spin_lock_init(&chip->reg_lock);
-       mutex_init(&chip->mce_mutex);
-       mutex_init(&chip->open_mutex);
-
        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_wss_playback_ops);
        snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_wss_capture_ops);
 
-       /* temporary */
-       if (chip->hardware & WSS_HW_AD1848_MASK) {
-               chip->rate_constraint = snd_wss_xrate;
-               chip->set_playback_format = snd_wss_playback_format;
-               chip->set_capture_format = snd_wss_capture_format;
-       }
        /* global setup */
        pcm->private_data = chip;
        pcm->info_flags = 0;
@@ -1851,7 +1946,8 @@ static int snd_wss_info_mux(struct snd_kcontrol *kcontrol,
        char **ptexts = texts;
        struct snd_wss *chip = snd_kcontrol_chip(kcontrol);
 
-       snd_assert(chip->card != NULL, return -EINVAL);
+       if (snd_BUG_ON(!chip->card))
+               return -EINVAL;
        uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
        uinfo->count = 2;
        uinfo->value.enumerated.items = 4;
@@ -2167,7 +2263,8 @@ int snd_wss_mixer(struct snd_wss *chip)
        unsigned int idx;
        int err;
 
-       snd_assert(chip != NULL && chip->pcm != NULL, return -EINVAL);
+       if (snd_BUG_ON(!chip || !chip->pcm))
+               return -EINVAL;
 
        card = chip->card;