123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957 |
- /*
- * Copyright (c) 2004 James Courtier-Dutton <James@superbug.demon.co.uk>
- * Driver CA0106 chips. e.g. Sound Blaster Audigy LS and Live 24bit
- * Version: 0.0.18
- *
- * FEATURES currently supported:
- * See ca0106_main.c for features.
- *
- * Changelog:
- * Support interrupts per period.
- * Removed noise from Center/LFE channel when in Analog mode.
- * Rename and remove mixer controls.
- * 0.0.6
- * Use separate card based DMA buffer for periods table list.
- * 0.0.7
- * Change remove and rename ctrls into lists.
- * 0.0.8
- * Try to fix capture sources.
- * 0.0.9
- * Fix AC3 output.
- * Enable S32_LE format support.
- * 0.0.10
- * Enable playback 48000 and 96000 rates. (Rates other that these do not work, even with "plug:front".)
- * 0.0.11
- * Add Model name recognition.
- * 0.0.12
- * Correct interrupt timing. interrupt at end of period, instead of in the middle of a playback period.
- * Remove redundent "voice" handling.
- * 0.0.13
- * Single trigger call for multi channels.
- * 0.0.14
- * Set limits based on what the sound card hardware can do.
- * playback periods_min=2, periods_max=8
- * capture hw constraints require period_size = n * 64 bytes.
- * playback hw constraints require period_size = n * 64 bytes.
- * 0.0.15
- * Separated ca0106.c into separate functional .c files.
- * 0.0.16
- * Modified Copyright message.
- * 0.0.17
- * Implement Mic and Line in Capture.
- * 0.0.18
- * Add support for mute control on SB Live 24bit (cards w/ SPI DAC)
- *
- * This code was initially based on code from ALSA's emu10k1x.c which is:
- * Copyright (c) by Francisco Moraes <fmoraes@nc.rr.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- *
- */
- #include <linux/delay.h>
- #include <linux/init.h>
- #include <linux/interrupt.h>
- #include <linux/moduleparam.h>
- #include <sound/core.h>
- #include <sound/initval.h>
- #include <sound/pcm.h>
- #include <sound/ac97_codec.h>
- #include <sound/info.h>
- #include <sound/tlv.h>
- #include <asm/io.h>
- #include "ca0106.h"
- static void ca0106_spdif_enable(struct snd_ca0106 *emu)
- {
- unsigned int val;
- if (emu->spdif_enable) {
- /* Digital */
- snd_ca0106_ptr_write(emu, SPDIF_SELECT1, 0, 0xf);
- snd_ca0106_ptr_write(emu, SPDIF_SELECT2, 0, 0x0b000000);
- val = snd_ca0106_ptr_read(emu, CAPTURE_CONTROL, 0) & ~0x1000;
- snd_ca0106_ptr_write(emu, CAPTURE_CONTROL, 0, val);
- val = inl(emu->port + GPIO) & ~0x101;
- outl(val, emu->port + GPIO);
- } else {
- /* Analog */
- snd_ca0106_ptr_write(emu, SPDIF_SELECT1, 0, 0xf);
- snd_ca0106_ptr_write(emu, SPDIF_SELECT2, 0, 0x000f0000);
- val = snd_ca0106_ptr_read(emu, CAPTURE_CONTROL, 0) | 0x1000;
- snd_ca0106_ptr_write(emu, CAPTURE_CONTROL, 0, val);
- val = inl(emu->port + GPIO) | 0x101;
- outl(val, emu->port + GPIO);
- }
- }
- static void ca0106_set_capture_source(struct snd_ca0106 *emu)
- {
- unsigned int val = emu->capture_source;
- unsigned int source, mask;
- source = (val << 28) | (val << 24) | (val << 20) | (val << 16);
- mask = snd_ca0106_ptr_read(emu, CAPTURE_SOURCE, 0) & 0xffff;
- snd_ca0106_ptr_write(emu, CAPTURE_SOURCE, 0, source | mask);
- }
- static void ca0106_set_i2c_capture_source(struct snd_ca0106 *emu,
- unsigned int val, int force)
- {
- unsigned int ngain, ogain;
- u32 source;
- snd_ca0106_i2c_write(emu, ADC_MUX, 0); /* Mute input */
- ngain = emu->i2c_capture_volume[val][0]; /* Left */
- ogain = emu->i2c_capture_volume[emu->i2c_capture_source][0]; /* Left */
- if (force || ngain != ogain)
- snd_ca0106_i2c_write(emu, ADC_ATTEN_ADCL, ngain & 0xff);
- ngain = emu->i2c_capture_volume[val][1]; /* Right */
- ogain = emu->i2c_capture_volume[emu->i2c_capture_source][1]; /* Right */
- if (force || ngain != ogain)
- snd_ca0106_i2c_write(emu, ADC_ATTEN_ADCR, ngain & 0xff);
- source = 1 << val;
- snd_ca0106_i2c_write(emu, ADC_MUX, source); /* Set source */
- emu->i2c_capture_source = val;
- }
- static void ca0106_set_capture_mic_line_in(struct snd_ca0106 *emu)
- {
- u32 tmp;
- if (emu->capture_mic_line_in) {
- /* snd_ca0106_i2c_write(emu, ADC_MUX, 0); */ /* Mute input */
- tmp = inl(emu->port+GPIO) & ~0x400;
- tmp = tmp | 0x400;
- outl(tmp, emu->port+GPIO);
- /* snd_ca0106_i2c_write(emu, ADC_MUX, ADC_MUX_MIC); */
- } else {
- /* snd_ca0106_i2c_write(emu, ADC_MUX, 0); */ /* Mute input */
- tmp = inl(emu->port+GPIO) & ~0x400;
- outl(tmp, emu->port+GPIO);
- /* snd_ca0106_i2c_write(emu, ADC_MUX, ADC_MUX_LINEIN); */
- }
- }
- static void ca0106_set_spdif_bits(struct snd_ca0106 *emu, int idx)
- {
- snd_ca0106_ptr_write(emu, SPCS0 + idx, 0, emu->spdif_str_bits[idx]);
- }
- /*
- */
- static const DECLARE_TLV_DB_SCALE(snd_ca0106_db_scale1, -5175, 25, 1);
- static const DECLARE_TLV_DB_SCALE(snd_ca0106_db_scale2, -10350, 50, 1);
- #define snd_ca0106_shared_spdif_info snd_ctl_boolean_mono_info
- static int snd_ca0106_shared_spdif_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- ucontrol->value.integer.value[0] = emu->spdif_enable;
- return 0;
- }
- static int snd_ca0106_shared_spdif_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int val;
- int change = 0;
- val = !!ucontrol->value.integer.value[0];
- change = (emu->spdif_enable != val);
- if (change) {
- emu->spdif_enable = val;
- ca0106_spdif_enable(emu);
- }
- return change;
- }
- static int snd_ca0106_capture_source_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- static char *texts[6] = {
- "IEC958 out", "i2s mixer out", "IEC958 in", "i2s in", "AC97 in", "SRC out"
- };
- uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
- uinfo->count = 1;
- uinfo->value.enumerated.items = 6;
- if (uinfo->value.enumerated.item > 5)
- uinfo->value.enumerated.item = 5;
- strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
- return 0;
- }
- static int snd_ca0106_capture_source_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- ucontrol->value.enumerated.item[0] = emu->capture_source;
- return 0;
- }
- static int snd_ca0106_capture_source_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int val;
- int change = 0;
- val = ucontrol->value.enumerated.item[0] ;
- if (val >= 6)
- return -EINVAL;
- change = (emu->capture_source != val);
- if (change) {
- emu->capture_source = val;
- ca0106_set_capture_source(emu);
- }
- return change;
- }
- static int snd_ca0106_i2c_capture_source_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- static char *texts[6] = {
- "Phone", "Mic", "Line in", "Aux"
- };
- uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
- uinfo->count = 1;
- uinfo->value.enumerated.items = 4;
- if (uinfo->value.enumerated.item > 3)
- uinfo->value.enumerated.item = 3;
- strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
- return 0;
- }
- static int snd_ca0106_i2c_capture_source_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- ucontrol->value.enumerated.item[0] = emu->i2c_capture_source;
- return 0;
- }
- static int snd_ca0106_i2c_capture_source_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int source_id;
- int change = 0;
- /* If the capture source has changed,
- * update the capture volume from the cached value
- * for the particular source.
- */
- source_id = ucontrol->value.enumerated.item[0] ;
- if (source_id >= 4)
- return -EINVAL;
- change = (emu->i2c_capture_source != source_id);
- if (change) {
- ca0106_set_i2c_capture_source(emu, source_id, 0);
- }
- return change;
- }
- static int snd_ca0106_capture_line_in_side_out_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- static char *texts[2] = { "Side out", "Line in" };
- uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
- uinfo->count = 1;
- uinfo->value.enumerated.items = 2;
- if (uinfo->value.enumerated.item > 1)
- uinfo->value.enumerated.item = 1;
- strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
- return 0;
- }
- static int snd_ca0106_capture_mic_line_in_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- static char *texts[2] = { "Line in", "Mic in" };
- uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
- uinfo->count = 1;
- uinfo->value.enumerated.items = 2;
- if (uinfo->value.enumerated.item > 1)
- uinfo->value.enumerated.item = 1;
- strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);
- return 0;
- }
- static int snd_ca0106_capture_mic_line_in_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- ucontrol->value.enumerated.item[0] = emu->capture_mic_line_in;
- return 0;
- }
- static int snd_ca0106_capture_mic_line_in_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int val;
- int change = 0;
- val = ucontrol->value.enumerated.item[0] ;
- if (val > 1)
- return -EINVAL;
- change = (emu->capture_mic_line_in != val);
- if (change) {
- emu->capture_mic_line_in = val;
- ca0106_set_capture_mic_line_in(emu);
- }
- return change;
- }
- static struct snd_kcontrol_new snd_ca0106_capture_mic_line_in __devinitdata =
- {
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
- .name = "Shared Mic/Line in Capture Switch",
- .info = snd_ca0106_capture_mic_line_in_info,
- .get = snd_ca0106_capture_mic_line_in_get,
- .put = snd_ca0106_capture_mic_line_in_put
- };
- static struct snd_kcontrol_new snd_ca0106_capture_line_in_side_out __devinitdata =
- {
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
- .name = "Shared Line in/Side out Capture Switch",
- .info = snd_ca0106_capture_line_in_side_out_info,
- .get = snd_ca0106_capture_mic_line_in_get,
- .put = snd_ca0106_capture_mic_line_in_put
- };
- static int snd_ca0106_spdif_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- uinfo->type = SNDRV_CTL_ELEM_TYPE_IEC958;
- uinfo->count = 1;
- return 0;
- }
- static void decode_spdif_bits(unsigned char *status, unsigned int bits)
- {
- status[0] = (bits >> 0) & 0xff;
- status[1] = (bits >> 8) & 0xff;
- status[2] = (bits >> 16) & 0xff;
- status[3] = (bits >> 24) & 0xff;
- }
- static int snd_ca0106_spdif_get_default(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
- decode_spdif_bits(ucontrol->value.iec958.status,
- emu->spdif_bits[idx]);
- return 0;
- }
- static int snd_ca0106_spdif_get_stream(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
- decode_spdif_bits(ucontrol->value.iec958.status,
- emu->spdif_str_bits[idx]);
- return 0;
- }
- static int snd_ca0106_spdif_get_mask(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- ucontrol->value.iec958.status[0] = 0xff;
- ucontrol->value.iec958.status[1] = 0xff;
- ucontrol->value.iec958.status[2] = 0xff;
- ucontrol->value.iec958.status[3] = 0xff;
- return 0;
- }
- static unsigned int encode_spdif_bits(unsigned char *status)
- {
- return ((unsigned int)status[0] << 0) |
- ((unsigned int)status[1] << 8) |
- ((unsigned int)status[2] << 16) |
- ((unsigned int)status[3] << 24);
- }
- static int snd_ca0106_spdif_put_default(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
- unsigned int val;
- val = encode_spdif_bits(ucontrol->value.iec958.status);
- if (val != emu->spdif_bits[idx]) {
- emu->spdif_bits[idx] = val;
- /* FIXME: this isn't safe, but needed to keep the compatibility
- * with older alsa-lib config
- */
- emu->spdif_str_bits[idx] = val;
- ca0106_set_spdif_bits(emu, idx);
- return 1;
- }
- return 0;
- }
- static int snd_ca0106_spdif_put_stream(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
- unsigned int val;
- val = encode_spdif_bits(ucontrol->value.iec958.status);
- if (val != emu->spdif_str_bits[idx]) {
- emu->spdif_str_bits[idx] = val;
- ca0106_set_spdif_bits(emu, idx);
- return 1;
- }
- return 0;
- }
- static int snd_ca0106_volume_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
- uinfo->count = 2;
- uinfo->value.integer.min = 0;
- uinfo->value.integer.max = 255;
- return 0;
- }
- static int snd_ca0106_volume_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int value;
- int channel_id, reg;
- channel_id = (kcontrol->private_value >> 8) & 0xff;
- reg = kcontrol->private_value & 0xff;
- value = snd_ca0106_ptr_read(emu, reg, channel_id);
- ucontrol->value.integer.value[0] = 0xff - ((value >> 24) & 0xff); /* Left */
- ucontrol->value.integer.value[1] = 0xff - ((value >> 16) & 0xff); /* Right */
- return 0;
- }
- static int snd_ca0106_volume_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int oval, nval;
- int channel_id, reg;
- channel_id = (kcontrol->private_value >> 8) & 0xff;
- reg = kcontrol->private_value & 0xff;
- oval = snd_ca0106_ptr_read(emu, reg, channel_id);
- nval = ((0xff - ucontrol->value.integer.value[0]) << 24) |
- ((0xff - ucontrol->value.integer.value[1]) << 16);
- nval |= ((0xff - ucontrol->value.integer.value[0]) << 8) |
- ((0xff - ucontrol->value.integer.value[1]) );
- if (oval == nval)
- return 0;
- snd_ca0106_ptr_write(emu, reg, channel_id, nval);
- return 1;
- }
- static int snd_ca0106_i2c_volume_info(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_info *uinfo)
- {
- uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
- uinfo->count = 2;
- uinfo->value.integer.min = 0;
- uinfo->value.integer.max = 255;
- return 0;
- }
- static int snd_ca0106_i2c_volume_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- int source_id;
- source_id = kcontrol->private_value;
- ucontrol->value.integer.value[0] = emu->i2c_capture_volume[source_id][0];
- ucontrol->value.integer.value[1] = emu->i2c_capture_volume[source_id][1];
- return 0;
- }
- static int snd_ca0106_i2c_volume_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int ogain;
- unsigned int ngain;
- int source_id;
- int change = 0;
- source_id = kcontrol->private_value;
- ogain = emu->i2c_capture_volume[source_id][0]; /* Left */
- ngain = ucontrol->value.integer.value[0];
- if (ngain > 0xff)
- return -EINVAL;
- if (ogain != ngain) {
- if (emu->i2c_capture_source == source_id)
- snd_ca0106_i2c_write(emu, ADC_ATTEN_ADCL, ((ngain) & 0xff) );
- emu->i2c_capture_volume[source_id][0] = ucontrol->value.integer.value[0];
- change = 1;
- }
- ogain = emu->i2c_capture_volume[source_id][1]; /* Right */
- ngain = ucontrol->value.integer.value[1];
- if (ngain > 0xff)
- return -EINVAL;
- if (ogain != ngain) {
- if (emu->i2c_capture_source == source_id)
- snd_ca0106_i2c_write(emu, ADC_ATTEN_ADCR, ((ngain) & 0xff));
- emu->i2c_capture_volume[source_id][1] = ucontrol->value.integer.value[1];
- change = 1;
- }
- return change;
- }
- #define spi_mute_info snd_ctl_boolean_mono_info
- static int spi_mute_get(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int reg = kcontrol->private_value >> SPI_REG_SHIFT;
- unsigned int bit = kcontrol->private_value & SPI_REG_MASK;
- ucontrol->value.integer.value[0] = !(emu->spi_dac_reg[reg] & bit);
- return 0;
- }
- static int spi_mute_put(struct snd_kcontrol *kcontrol,
- struct snd_ctl_elem_value *ucontrol)
- {
- struct snd_ca0106 *emu = snd_kcontrol_chip(kcontrol);
- unsigned int reg = kcontrol->private_value >> SPI_REG_SHIFT;
- unsigned int bit = kcontrol->private_value & SPI_REG_MASK;
- int ret;
- ret = emu->spi_dac_reg[reg] & bit;
- if (ucontrol->value.integer.value[0]) {
- if (!ret) /* bit already cleared, do nothing */
- return 0;
- emu->spi_dac_reg[reg] &= ~bit;
- } else {
- if (ret) /* bit already set, do nothing */
- return 0;
- emu->spi_dac_reg[reg] |= bit;
- }
- ret = snd_ca0106_spi_write(emu, emu->spi_dac_reg[reg]);
- return ret ? -EINVAL : 1;
- }
- #define CA_VOLUME(xname,chid,reg) \
- { \
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
- .access = SNDRV_CTL_ELEM_ACCESS_READWRITE | \
- SNDRV_CTL_ELEM_ACCESS_TLV_READ, \
- .info = snd_ca0106_volume_info, \
- .get = snd_ca0106_volume_get, \
- .put = snd_ca0106_volume_put, \
- .tlv = { .p = snd_ca0106_db_scale1 }, \
- .private_value = ((chid) << 8) | (reg) \
- }
- static struct snd_kcontrol_new snd_ca0106_volume_ctls[] __devinitdata = {
- CA_VOLUME("Analog Front Playback Volume",
- CONTROL_FRONT_CHANNEL, PLAYBACK_VOLUME2),
- CA_VOLUME("Analog Rear Playback Volume",
- CONTROL_REAR_CHANNEL, PLAYBACK_VOLUME2),
- CA_VOLUME("Analog Center/LFE Playback Volume",
- CONTROL_CENTER_LFE_CHANNEL, PLAYBACK_VOLUME2),
- CA_VOLUME("Analog Side Playback Volume",
- CONTROL_UNKNOWN_CHANNEL, PLAYBACK_VOLUME2),
- CA_VOLUME("IEC958 Front Playback Volume",
- CONTROL_FRONT_CHANNEL, PLAYBACK_VOLUME1),
- CA_VOLUME("IEC958 Rear Playback Volume",
- CONTROL_REAR_CHANNEL, PLAYBACK_VOLUME1),
- CA_VOLUME("IEC958 Center/LFE Playback Volume",
- CONTROL_CENTER_LFE_CHANNEL, PLAYBACK_VOLUME1),
- CA_VOLUME("IEC958 Unknown Playback Volume",
- CONTROL_UNKNOWN_CHANNEL, PLAYBACK_VOLUME1),
- CA_VOLUME("CAPTURE feedback Playback Volume",
- 1, CAPTURE_CONTROL),
- {
- .access = SNDRV_CTL_ELEM_ACCESS_READ,
- .iface = SNDRV_CTL_ELEM_IFACE_PCM,
- .name = SNDRV_CTL_NAME_IEC958("",PLAYBACK,MASK),
- .count = 4,
- .info = snd_ca0106_spdif_info,
- .get = snd_ca0106_spdif_get_mask
- },
- {
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
- .name = "IEC958 Playback Switch",
- .info = snd_ca0106_shared_spdif_info,
- .get = snd_ca0106_shared_spdif_get,
- .put = snd_ca0106_shared_spdif_put
- },
- {
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
- .name = "Digital Source Capture Enum",
- .info = snd_ca0106_capture_source_info,
- .get = snd_ca0106_capture_source_get,
- .put = snd_ca0106_capture_source_put
- },
- {
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
- .name = "Analog Source Capture Enum",
- .info = snd_ca0106_i2c_capture_source_info,
- .get = snd_ca0106_i2c_capture_source_get,
- .put = snd_ca0106_i2c_capture_source_put
- },
- {
- .iface = SNDRV_CTL_ELEM_IFACE_PCM,
- .name = SNDRV_CTL_NAME_IEC958("",PLAYBACK,DEFAULT),
- .count = 4,
- .info = snd_ca0106_spdif_info,
- .get = snd_ca0106_spdif_get_default,
- .put = snd_ca0106_spdif_put_default
- },
- {
- .iface = SNDRV_CTL_ELEM_IFACE_PCM,
- .name = SNDRV_CTL_NAME_IEC958("",PLAYBACK,PCM_STREAM),
- .count = 4,
- .info = snd_ca0106_spdif_info,
- .get = snd_ca0106_spdif_get_stream,
- .put = snd_ca0106_spdif_put_stream
- },
- };
- #define I2C_VOLUME(xname,chid) \
- { \
- .iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
- .access = SNDRV_CTL_ELEM_ACCESS_READWRITE | \
- SNDRV_CTL_ELEM_ACCESS_TLV_READ, \
- .info = snd_ca0106_i2c_volume_info, \
- .get = snd_ca0106_i2c_volume_get, \
- .put = snd_ca0106_i2c_volume_put, \
- .tlv = { .p = snd_ca0106_db_scale2 }, \
- .private_value = chid \
- }
- static struct snd_kcontrol_new snd_ca0106_volume_i2c_adc_ctls[] __devinitdata = {
- I2C_VOLUME("Phone Capture Volume", 0),
- I2C_VOLUME("Mic Capture Volume", 1),
- I2C_VOLUME("Line in Capture Volume", 2),
- I2C_VOLUME("Aux Capture Volume", 3),
- };
- static const int spi_dmute_reg[] = {
- SPI_DMUTE0_REG,
- SPI_DMUTE1_REG,
- SPI_DMUTE2_REG,
- 0,
- SPI_DMUTE4_REG,
- };
- static const int spi_dmute_bit[] = {
- SPI_DMUTE0_BIT,
- SPI_DMUTE1_BIT,
- SPI_DMUTE2_BIT,
- 0,
- SPI_DMUTE4_BIT,
- };
- static struct snd_kcontrol_new __devinit
- snd_ca0106_volume_spi_dac_ctl(struct snd_ca0106_details *details,
- int channel_id)
- {
- struct snd_kcontrol_new spi_switch = {0};
- int reg, bit;
- int dac_id;
- spi_switch.iface = SNDRV_CTL_ELEM_IFACE_MIXER;
- spi_switch.access = SNDRV_CTL_ELEM_ACCESS_READWRITE;
- spi_switch.info = spi_mute_info;
- spi_switch.get = spi_mute_get;
- spi_switch.put = spi_mute_put;
- switch (channel_id) {
- case PCM_FRONT_CHANNEL:
- spi_switch.name = "Analog Front Playback Switch";
- dac_id = (details->spi_dac & 0xf000) >> (4 * 3);
- break;
- case PCM_REAR_CHANNEL:
- spi_switch.name = "Analog Rear Playback Switch";
- dac_id = (details->spi_dac & 0x0f00) >> (4 * 2);
- break;
- case PCM_CENTER_LFE_CHANNEL:
- spi_switch.name = "Analog Center/LFE Playback Switch";
- dac_id = (details->spi_dac & 0x00f0) >> (4 * 1);
- break;
- case PCM_UNKNOWN_CHANNEL:
- spi_switch.name = "Analog Side Playback Switch";
- dac_id = (details->spi_dac & 0x000f) >> (4 * 0);
- break;
- default:
- /* Unused channel */
- spi_switch.name = NULL;
- dac_id = 0;
- }
- reg = spi_dmute_reg[dac_id];
- bit = spi_dmute_bit[dac_id];
- spi_switch.private_value = (reg << SPI_REG_SHIFT) | bit;
- return spi_switch;
- }
- static int __devinit remove_ctl(struct snd_card *card, const char *name)
- {
- struct snd_ctl_elem_id id;
- memset(&id, 0, sizeof(id));
- strcpy(id.name, name);
- id.iface = SNDRV_CTL_ELEM_IFACE_MIXER;
- return snd_ctl_remove_id(card, &id);
- }
- static struct snd_kcontrol __devinit *ctl_find(struct snd_card *card, const char *name)
- {
- struct snd_ctl_elem_id sid;
- memset(&sid, 0, sizeof(sid));
- /* FIXME: strcpy is bad. */
- strcpy(sid.name, name);
- sid.iface = SNDRV_CTL_ELEM_IFACE_MIXER;
- return snd_ctl_find_id(card, &sid);
- }
- static int __devinit rename_ctl(struct snd_card *card, const char *src, const char *dst)
- {
- struct snd_kcontrol *kctl = ctl_find(card, src);
- if (kctl) {
- strcpy(kctl->id.name, dst);
- return 0;
- }
- return -ENOENT;
- }
- #define ADD_CTLS(emu, ctls) \
- do { \
- int i, _err; \
- for (i = 0; i < ARRAY_SIZE(ctls); i++) { \
- _err = snd_ctl_add(card, snd_ctl_new1(&ctls[i], emu)); \
- if (_err < 0) \
- return _err; \
- } \
- } while (0)
- static __devinitdata
- DECLARE_TLV_DB_SCALE(snd_ca0106_master_db_scale, -6375, 25, 1);
- static char *slave_vols[] __devinitdata = {
- "Analog Front Playback Volume",
- "Analog Rear Playback Volume",
- "Analog Center/LFE Playback Volume",
- "Analog Side Playback Volume",
- "IEC958 Front Playback Volume",
- "IEC958 Rear Playback Volume",
- "IEC958 Center/LFE Playback Volume",
- "IEC958 Unknown Playback Volume",
- "CAPTURE feedback Playback Volume",
- NULL
- };
- static char *slave_sws[] __devinitdata = {
- "Analog Front Playback Switch",
- "Analog Rear Playback Switch",
- "Analog Center/LFE Playback Switch",
- "Analog Side Playback Switch",
- "IEC958 Playback Switch",
- NULL
- };
- static void __devinit add_slaves(struct snd_card *card,
- struct snd_kcontrol *master, char **list)
- {
- for (; *list; list++) {
- struct snd_kcontrol *slave = ctl_find(card, *list);
- if (slave)
- snd_ctl_add_slave(master, slave);
- }
- }
- int __devinit snd_ca0106_mixer(struct snd_ca0106 *emu)
- {
- int err;
- struct snd_card *card = emu->card;
- char **c;
- struct snd_kcontrol *vmaster;
- static char *ca0106_remove_ctls[] = {
- "Master Mono Playback Switch",
- "Master Mono Playback Volume",
- "3D Control - Switch",
- "3D Control Sigmatel - Depth",
- "PCM Playback Switch",
- "PCM Playback Volume",
- "CD Playback Switch",
- "CD Playback Volume",
- "Phone Playback Switch",
- "Phone Playback Volume",
- "Video Playback Switch",
- "Video Playback Volume",
- "Beep Playback Switch",
- "Beep Playback Volume",
- "Mono Output Select",
- "Capture Source",
- "Capture Switch",
- "Capture Volume",
- "External Amplifier",
- "Sigmatel 4-Speaker Stereo Playback Switch",
- "Surround Phase Inversion Playback Switch",
- NULL
- };
- static char *ca0106_rename_ctls[] = {
- "Master Playback Switch", "Capture Switch",
- "Master Playback Volume", "Capture Volume",
- "Line Playback Switch", "AC97 Line Capture Switch",
- "Line Playback Volume", "AC97 Line Capture Volume",
- "Aux Playback Switch", "AC97 Aux Capture Switch",
- "Aux Playback Volume", "AC97 Aux Capture Volume",
- "Mic Playback Switch", "AC97 Mic Capture Switch",
- "Mic Playback Volume", "AC97 Mic Capture Volume",
- "Mic Select", "AC97 Mic Select",
- "Mic Boost (+20dB)", "AC97 Mic Boost (+20dB)",
- NULL
- };
- #if 1
- for (c = ca0106_remove_ctls; *c; c++)
- remove_ctl(card, *c);
- for (c = ca0106_rename_ctls; *c; c += 2)
- rename_ctl(card, c[0], c[1]);
- #endif
- ADD_CTLS(emu, snd_ca0106_volume_ctls);
- if (emu->details->i2c_adc == 1) {
- ADD_CTLS(emu, snd_ca0106_volume_i2c_adc_ctls);
- if (emu->details->gpio_type == 1)
- err = snd_ctl_add(card, snd_ctl_new1(&snd_ca0106_capture_mic_line_in, emu));
- else /* gpio_type == 2 */
- err = snd_ctl_add(card, snd_ctl_new1(&snd_ca0106_capture_line_in_side_out, emu));
- if (err < 0)
- return err;
- }
- if (emu->details->spi_dac) {
- int i;
- for (i = 0;; i++) {
- struct snd_kcontrol_new ctl;
- ctl = snd_ca0106_volume_spi_dac_ctl(emu->details, i);
- if (!ctl.name)
- break;
- err = snd_ctl_add(card, snd_ctl_new1(&ctl, emu));
- if (err < 0)
- return err;
- }
- }
- /* Create virtual master controls */
- vmaster = snd_ctl_make_virtual_master("Master Playback Volume",
- snd_ca0106_master_db_scale);
- if (!vmaster)
- return -ENOMEM;
- err = snd_ctl_add(card, vmaster);
- if (err < 0)
- return err;
- add_slaves(card, vmaster, slave_vols);
- if (emu->details->spi_dac) {
- vmaster = snd_ctl_make_virtual_master("Master Playback Switch",
- NULL);
- if (!vmaster)
- return -ENOMEM;
- err = snd_ctl_add(card, vmaster);
- if (err < 0)
- return err;
- add_slaves(card, vmaster, slave_sws);
- }
- strcpy(card->mixername, "CA0106");
- return 0;
- }
- #ifdef CONFIG_PM
- struct ca0106_vol_tbl {
- unsigned int channel_id;
- unsigned int reg;
- };
- static struct ca0106_vol_tbl saved_volumes[NUM_SAVED_VOLUMES] = {
- { CONTROL_FRONT_CHANNEL, PLAYBACK_VOLUME2 },
- { CONTROL_REAR_CHANNEL, PLAYBACK_VOLUME2 },
- { CONTROL_CENTER_LFE_CHANNEL, PLAYBACK_VOLUME2 },
- { CONTROL_UNKNOWN_CHANNEL, PLAYBACK_VOLUME2 },
- { CONTROL_FRONT_CHANNEL, PLAYBACK_VOLUME1 },
- { CONTROL_REAR_CHANNEL, PLAYBACK_VOLUME1 },
- { CONTROL_CENTER_LFE_CHANNEL, PLAYBACK_VOLUME1 },
- { CONTROL_UNKNOWN_CHANNEL, PLAYBACK_VOLUME1 },
- { 1, CAPTURE_CONTROL },
- };
- void snd_ca0106_mixer_suspend(struct snd_ca0106 *chip)
- {
- int i;
- /* save volumes */
- for (i = 0; i < NUM_SAVED_VOLUMES; i++)
- chip->saved_vol[i] =
- snd_ca0106_ptr_read(chip, saved_volumes[i].reg,
- saved_volumes[i].channel_id);
- }
- void snd_ca0106_mixer_resume(struct snd_ca0106 *chip)
- {
- int i;
- for (i = 0; i < NUM_SAVED_VOLUMES; i++)
- snd_ca0106_ptr_write(chip, saved_volumes[i].reg,
- saved_volumes[i].channel_id,
- chip->saved_vol[i]);
- ca0106_spdif_enable(chip);
- ca0106_set_capture_source(chip);
- ca0106_set_i2c_capture_source(chip, chip->i2c_capture_source, 1);
- for (i = 0; i < 4; i++)
- ca0106_set_spdif_bits(chip, i);
- if (chip->details->i2c_adc)
- ca0106_set_capture_mic_line_in(chip);
- }
- #endif /* CONFIG_PM */
|