Lines Matching +full:pin +full:- +full:val
1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Support for Digigram Lola PCI-e boards
18 static int lola_init_pin(struct lola *chip, struct lola_pin *pin, in lola_init_pin() argument
21 unsigned int val; in lola_init_pin() local
24 pin->nid = nid; in lola_init_pin()
25 err = lola_read_param(chip, nid, LOLA_PAR_AUDIO_WIDGET_CAP, &val); in lola_init_pin()
27 dev_err(chip->card->dev, "Can't read wcaps for 0x%x\n", nid); in lola_init_pin()
30 val &= 0x00f00fff; /* test TYPE and bits 0..11 */ in lola_init_pin()
31 if (val == 0x00400200) /* Type = 4, Digital = 1 */ in lola_init_pin()
32 pin->is_analog = false; in lola_init_pin()
33 else if (val == 0x0040000a && dir == CAPT) /* Dig=0, InAmp/ovrd */ in lola_init_pin()
34 pin->is_analog = true; in lola_init_pin()
35 else if (val == 0x0040000c && dir == PLAY) /* Dig=0, OutAmp/ovrd */ in lola_init_pin()
36 pin->is_analog = true; in lola_init_pin()
38 dev_err(chip->card->dev, "Invalid wcaps 0x%x for 0x%x\n", val, nid); in lola_init_pin()
39 return -EINVAL; in lola_init_pin()
42 /* analog parameters only following, so continue in case of Digital pin in lola_init_pin()
44 if (!pin->is_analog) in lola_init_pin()
48 err = lola_read_param(chip, nid, LOLA_PAR_AMP_OUT_CAP, &val); in lola_init_pin()
50 err = lola_read_param(chip, nid, LOLA_PAR_AMP_IN_CAP, &val); in lola_init_pin()
52 dev_err(chip->card->dev, "Can't read AMP-caps for 0x%x\n", nid); in lola_init_pin()
56 pin->amp_mute = LOLA_AMP_MUTE_CAPABLE(val); in lola_init_pin()
57 pin->amp_step_size = LOLA_AMP_STEP_SIZE(val); in lola_init_pin()
58 pin->amp_num_steps = LOLA_AMP_NUM_STEPS(val); in lola_init_pin()
59 if (pin->amp_num_steps) { in lola_init_pin()
61 pin->amp_num_steps++; in lola_init_pin()
62 pin->amp_step_size++; in lola_init_pin()
64 pin->amp_offset = LOLA_AMP_OFFSET(val); in lola_init_pin()
66 err = lola_codec_read(chip, nid, LOLA_VERB_GET_MAX_LEVEL, 0, 0, &val, in lola_init_pin()
69 dev_err(chip->card->dev, "Can't get MAX_LEVEL 0x%x\n", nid); in lola_init_pin()
72 pin->max_level = val & 0x3ff; /* 10 bits */ in lola_init_pin()
74 pin->config_default_reg = 0; in lola_init_pin()
75 pin->fixed_gain_list_len = 0; in lola_init_pin()
76 pin->cur_gain_step = 0; in lola_init_pin()
85 for (i = 0; i < chip->pin[dir].num_pins; i++, nid++) { in lola_init_pins()
86 err = lola_init_pin(chip, &chip->pin[dir].pins[i], dir, nid); in lola_init_pins()
89 if (chip->pin[dir].pins[i].is_analog) in lola_init_pins()
90 chip->pin[dir].num_analog_pins++; in lola_init_pins()
98 vfree(chip->mixer.array_saved); in lola_free_mixer()
103 unsigned int val; in lola_init_mixer_widget() local
106 err = lola_read_param(chip, nid, LOLA_PAR_AUDIO_WIDGET_CAP, &val); in lola_init_mixer_widget()
108 dev_err(chip->card->dev, "Can't read wcaps for 0x%x\n", nid); in lola_init_mixer_widget()
112 if ((val & 0xfff00000) != 0x02f00000) { /* test SubType and Type */ in lola_init_mixer_widget()
113 dev_dbg(chip->card->dev, "No valid mixer widget\n"); in lola_init_mixer_widget()
117 chip->mixer.nid = nid; in lola_init_mixer_widget()
118 chip->mixer.caps = val; in lola_init_mixer_widget()
119 chip->mixer.array = (struct lola_mixer_array __iomem *) in lola_init_mixer_widget()
120 (chip->bar[BAR1].remap_addr + LOLA_BAR1_SOURCE_GAIN_ENABLE); in lola_init_mixer_widget()
123 chip->mixer.array_saved = vmalloc(sizeof(struct lola_mixer_array)); in lola_init_mixer_widget()
126 chip->mixer.src_stream_outs = chip->pcm[PLAY].num_streams; in lola_init_mixer_widget()
127 chip->mixer.src_phys_ins = chip->pin[CAPT].num_pins; in lola_init_mixer_widget()
130 chip->mixer.dest_stream_ins = chip->pcm[CAPT].num_streams; in lola_init_mixer_widget()
131 chip->mixer.dest_phys_outs = chip->pin[PLAY].num_pins; in lola_init_mixer_widget()
136 chip->mixer.src_stream_out_ofs = chip->mixer.src_phys_ins + in lola_init_mixer_widget()
137 LOLA_MIXER_SRC_INPUT_PLAY_SEPARATION(val); in lola_init_mixer_widget()
138 chip->mixer.dest_phys_out_ofs = chip->mixer.dest_stream_ins + in lola_init_mixer_widget()
139 LOLA_MIXER_DEST_REC_OUTPUT_SEPARATION(val); in lola_init_mixer_widget()
142 * +-+ 0-------8------16-------8------16 in lola_init_mixer_widget()
145 * | |->| -> |unused | -> |unused | in lola_init_mixer_widget()
148 * |c| 8-------------------------------- in lola_init_mixer_widget()
154 * | | 16------------------------------- in lola_init_mixer_widget()
157 * |n|->| -> |unused | -> |unused | in lola_init_mixer_widget()
160 * |a| 8-------------------------------- in lola_init_mixer_widget()
166 * +++ 16--|---------------|------------ in lola_init_mixer_widget()
167 * +---V---------------V-----------+ in lola_init_mixer_widget()
169 * +-------------------------------+ in lola_init_mixer_widget()
172 * +-+ 0-------8-2 in lola_init_mixer_widget()
175 * |r|->| -> | | -> in lola_init_mixer_widget()
176 * |c| |CAPTURE| | <- OUTPUT in lola_init_mixer_widget()
178 * |g| 8---------- in lola_init_mixer_widget()
181 * |n|->| -> | | -> in lola_init_mixer_widget()
182 * | | |CAPTURE| | <- OUTPUT in lola_init_mixer_widget()
184 * |r| 8---|----|- in lola_init_mixer_widget()
185 * |r| +---V----V-------------------+ in lola_init_mixer_widget()
187 * |y| +----------------------------+ in lola_init_mixer_widget()
189 if (chip->mixer.src_stream_out_ofs > MAX_AUDIO_INOUT_COUNT || in lola_init_mixer_widget()
190 chip->mixer.dest_phys_out_ofs > MAX_STREAM_IN_COUNT) { in lola_init_mixer_widget()
191 dev_err(chip->card->dev, "Invalid mixer widget size\n"); in lola_init_mixer_widget()
192 return -EINVAL; in lola_init_mixer_widget()
195 chip->mixer.src_mask = ((1U << chip->mixer.src_phys_ins) - 1) | in lola_init_mixer_widget()
196 (((1U << chip->mixer.src_stream_outs) - 1) in lola_init_mixer_widget()
197 << chip->mixer.src_stream_out_ofs); in lola_init_mixer_widget()
198 chip->mixer.dest_mask = ((1U << chip->mixer.dest_stream_ins) - 1) | in lola_init_mixer_widget()
199 (((1U << chip->mixer.dest_phys_outs) - 1) in lola_init_mixer_widget()
200 << chip->mixer.dest_phys_out_ofs); in lola_init_mixer_widget()
202 dev_dbg(chip->card->dev, "Mixer src_mask=%x, dest_mask=%x\n", in lola_init_mixer_widget()
203 chip->mixer.src_mask, chip->mixer.dest_mask); in lola_init_mixer_widget()
211 unsigned int oldval, val; in lola_mixer_set_src_gain() local
213 if (!(chip->mixer.src_mask & (1 << id))) in lola_mixer_set_src_gain()
214 return -EINVAL; in lola_mixer_set_src_gain()
215 oldval = val = readl(&chip->mixer.array->src_gain_enable); in lola_mixer_set_src_gain()
217 val |= (1 << id); in lola_mixer_set_src_gain()
219 val &= ~(1 << id); in lola_mixer_set_src_gain()
221 if ((val == oldval) && in lola_mixer_set_src_gain()
222 (gain == readw(&chip->mixer.array->src_gain[id]))) in lola_mixer_set_src_gain()
225 dev_dbg(chip->card->dev, in lola_mixer_set_src_gain()
227 id, gain, val); in lola_mixer_set_src_gain()
228 writew(gain, &chip->mixer.array->src_gain[id]); in lola_mixer_set_src_gain()
229 writel(val, &chip->mixer.array->src_gain_enable); in lola_mixer_set_src_gain()
231 /* inform micro-controller about the new source gain */ in lola_mixer_set_src_gain()
232 return lola_codec_write(chip, chip->mixer.nid, in lola_mixer_set_src_gain()
242 if ((chip->mixer.src_mask & mask) != mask)
243 return -EINVAL;
246 writew(*gains, &chip->mixer.array->src_gain[i]);
250 writel(mask, &chip->mixer.array->src_gain_enable);
252 if (chip->mixer.caps & LOLA_PEAK_METER_CAN_AGC_MASK) {
254 return lola_codec_write(chip, chip->mixer.nid,
260 lola_codec_write(chip, chip->mixer.nid,
272 unsigned int val; in lola_mixer_set_mapping_gain() local
274 if (!(chip->mixer.src_mask & (1 << src)) || in lola_mixer_set_mapping_gain()
275 !(chip->mixer.dest_mask & (1 << dest))) in lola_mixer_set_mapping_gain()
276 return -EINVAL; in lola_mixer_set_mapping_gain()
278 writew(gain, &chip->mixer.array->dest_mix_gain[dest][src]); in lola_mixer_set_mapping_gain()
279 val = readl(&chip->mixer.array->dest_mix_gain_enable[dest]); in lola_mixer_set_mapping_gain()
281 val |= (1 << src); in lola_mixer_set_mapping_gain()
283 val &= ~(1 << src); in lola_mixer_set_mapping_gain()
284 writel(val, &chip->mixer.array->dest_mix_gain_enable[dest]); in lola_mixer_set_mapping_gain()
286 return lola_codec_write(chip, chip->mixer.nid, LOLA_VERB_SET_MIX_GAIN, in lola_mixer_set_mapping_gain()
296 if (!(chip->mixer.dest_mask & (1 << id)) ||
297 (chip->mixer.src_mask & mask) != mask)
298 return -EINVAL;
301 writew(*gains, &chip->mixer.array->dest_mix_gain[id][i]);
305 writel(mask, &chip->mixer.array->dest_mix_gain_enable[id]);
308 return lola_codec_write(chip, chip->mixer.nid,
317 unsigned int idx, unsigned int val,
322 struct lola_pin *pin; in lola_setup_all_analog_gains() local
325 pin = chip->pin[dir].pins; in lola_setup_all_analog_gains()
326 max_idx = chip->pin[dir].num_pins; in lola_setup_all_analog_gains()
328 if (pin[idx].is_analog) { in lola_setup_all_analog_gains()
329 unsigned int val = mute ? 0 : pin[idx].cur_gain_step; in lola_setup_all_analog_gains() local
331 set_analog_volume(chip, dir, idx, val, false); in lola_setup_all_analog_gains()
340 if (chip->mixer.array_saved) { in lola_save_mixer()
342 memcpy_fromio(chip->mixer.array_saved, chip->mixer.array, in lola_save_mixer()
343 sizeof(*chip->mixer.array)); in lola_save_mixer()
353 if (chip->mixer.array_saved) { in lola_restore_mixer()
355 memcpy_toio(chip->mixer.array, chip->mixer.array_saved, in lola_restore_mixer()
356 sizeof(*chip->mixer.array)); in lola_restore_mixer()
357 /* inform micro-controller about all restored values in lola_restore_mixer()
360 for (i = 0; i < chip->mixer.src_phys_ins; i++) in lola_restore_mixer()
361 lola_codec_write(chip, chip->mixer.nid, in lola_restore_mixer()
364 for (i = 0; i < chip->mixer.src_stream_outs; i++) in lola_restore_mixer()
365 lola_codec_write(chip, chip->mixer.nid, in lola_restore_mixer()
367 chip->mixer.src_stream_out_ofs + i, 0); in lola_restore_mixer()
368 for (i = 0; i < chip->mixer.dest_stream_ins; i++) in lola_restore_mixer()
369 lola_codec_write(chip, chip->mixer.nid, in lola_restore_mixer()
372 for (i = 0; i < chip->mixer.dest_phys_outs; i++) in lola_restore_mixer()
373 lola_codec_write(chip, chip->mixer.nid, in lola_restore_mixer()
375 chip->mixer.dest_phys_out_ofs + i, 0); in lola_restore_mixer()
384 unsigned int idx, unsigned int val, in set_analog_volume() argument
387 struct lola_pin *pin; in set_analog_volume() local
390 if (idx >= chip->pin[dir].num_pins) in set_analog_volume()
391 return -EINVAL; in set_analog_volume()
392 pin = &chip->pin[dir].pins[idx]; in set_analog_volume()
393 if (!pin->is_analog || pin->amp_num_steps <= val) in set_analog_volume()
394 return -EINVAL; in set_analog_volume()
395 if (external_call && pin->cur_gain_step == val) in set_analog_volume()
399 dev_dbg(chip->card->dev, in set_analog_volume()
401 dir, idx, val); in set_analog_volume()
402 err = lola_codec_write(chip, pin->nid, in set_analog_volume()
403 LOLA_VERB_SET_AMP_GAIN_MUTE, val, 0); in set_analog_volume()
407 pin->cur_gain_step = val; in set_analog_volume()
418 if ((chip->input_src_caps_mask & src_mask) != src_mask) in lola_set_src_config()
419 return -EINVAL; in lola_set_src_config()
420 /* handle all even Inputs - SRC is a stereo setting !!! */ in lola_set_src_config()
421 for (n = 0; n < chip->pin[CAPT].num_pins; n += 2) { in lola_set_src_config()
424 if (!(chip->input_src_caps_mask & mask)) in lola_set_src_config()
429 src_state = (chip->input_src_mask & mask) != 0; in lola_set_src_config()
433 err = lola_codec_write(chip, chip->pcm[CAPT].streams[n].nid, in lola_set_src_config()
443 chip->input_src_mask = src_mask; in lola_set_src_config()
454 lola_set_src_config(chip, (1 << chip->pin[CAPT].num_pins) - 1, false); in init_mixer_values()
457 memset_io(chip->mixer.array, 0, sizeof(*chip->mixer.array)); in init_mixer_values()
458 /* inform firmware about all updated matrix columns - capture part */ in init_mixer_values()
459 for (i = 0; i < chip->mixer.dest_stream_ins; i++) in init_mixer_values()
460 lola_codec_write(chip, chip->mixer.nid, in init_mixer_values()
463 /* inform firmware about all updated matrix columns - output part */ in init_mixer_values()
464 for (i = 0; i < chip->mixer.dest_phys_outs; i++) in init_mixer_values()
465 lola_codec_write(chip, chip->mixer.nid, in init_mixer_values()
467 chip->mixer.dest_phys_out_ofs + i, 0); in init_mixer_values()
470 for (i = 0; i < chip->mixer.src_phys_ins; i++) in init_mixer_values()
474 for (i = 0; i < chip->mixer.src_stream_outs; i++) in init_mixer_values()
476 i + chip->mixer.src_stream_out_ofs, in init_mixer_values()
478 /* set gain value 0dB diagonally in matrix - part INPUT -> CAPTURE */ in init_mixer_values()
479 for (i = 0; i < chip->mixer.dest_stream_ins; i++) { in init_mixer_values()
480 int src = i % chip->mixer.src_phys_ins; in init_mixer_values()
483 /* set gain value 0dB diagonally in matrix , part PLAYBACK -> OUTPUT in init_mixer_values()
487 for (i = 0; i < chip->mixer.src_stream_outs; i++) { in init_mixer_values()
488 int src = chip->mixer.src_stream_out_ofs + i; in init_mixer_values()
489 int dst = chip->mixer.dest_phys_out_ofs + in init_mixer_values()
490 i % chip->mixer.dest_phys_outs; in init_mixer_values()
503 int dir = kcontrol->private_value; in lola_analog_vol_info()
505 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; in lola_analog_vol_info()
506 uinfo->count = chip->pin[dir].num_pins; in lola_analog_vol_info()
507 uinfo->value.integer.min = 0; in lola_analog_vol_info()
508 uinfo->value.integer.max = chip->pin[dir].pins[0].amp_num_steps; in lola_analog_vol_info()
516 int dir = kcontrol->private_value; in lola_analog_vol_get()
519 for (i = 0; i < chip->pin[dir].num_pins; i++) in lola_analog_vol_get()
520 ucontrol->value.integer.value[i] = in lola_analog_vol_get()
521 chip->pin[dir].pins[i].cur_gain_step; in lola_analog_vol_get()
529 int dir = kcontrol->private_value; in lola_analog_vol_put()
532 for (i = 0; i < chip->pin[dir].num_pins; i++) { in lola_analog_vol_put()
534 ucontrol->value.integer.value[i], in lola_analog_vol_put()
546 int dir = kcontrol->private_value; in lola_analog_vol_tlv()
548 struct lola_pin *pin; in lola_analog_vol_tlv() local
551 return -ENOMEM; in lola_analog_vol_tlv()
552 pin = &chip->pin[dir].pins[0]; in lola_analog_vol_tlv()
554 val2 = pin->amp_step_size * 25; in lola_analog_vol_tlv()
555 val1 = -1 * (int)pin->amp_offset * (int)val2; in lola_analog_vol_tlv()
560 return -EFAULT; in lola_analog_vol_tlv()
562 return -EFAULT; in lola_analog_vol_tlv()
564 return -EFAULT; in lola_analog_vol_tlv()
566 return -EFAULT; in lola_analog_vol_tlv()
583 if (!chip->pin[dir].num_pins) in create_analog_mixer()
586 if (chip->pin[dir].num_pins != chip->pin[dir].num_analog_pins) in create_analog_mixer()
590 return snd_ctl_add(chip->card, in create_analog_mixer()
602 uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN; in lola_input_src_info()
603 uinfo->count = chip->pin[CAPT].num_pins; in lola_input_src_info()
604 uinfo->value.integer.min = 0; in lola_input_src_info()
605 uinfo->value.integer.max = 1; in lola_input_src_info()
615 for (i = 0; i < chip->pin[CAPT].num_pins; i++) in lola_input_src_get()
616 ucontrol->value.integer.value[i] = in lola_input_src_get()
617 !!(chip->input_src_mask & (1 << i)); in lola_input_src_get()
629 for (i = 0; i < chip->pin[CAPT].num_pins; i++) in lola_input_src_put()
630 if (ucontrol->value.integer.value[i]) in lola_input_src_put()
649 if (!chip->input_src_caps_mask) in create_input_src_mixer()
652 return snd_ctl_add(chip->card, in create_input_src_mixer()
662 unsigned int count = (kcontrol->private_value >> 8) & 0xff; in lola_src_gain_info()
664 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; in lola_src_gain_info()
665 uinfo->count = count; in lola_src_gain_info()
666 uinfo->value.integer.min = 0; in lola_src_gain_info()
667 uinfo->value.integer.max = 409; in lola_src_gain_info()
675 unsigned int ofs = kcontrol->private_value & 0xff; in lola_src_gain_get()
676 unsigned int count = (kcontrol->private_value >> 8) & 0xff; in lola_src_gain_get()
679 mask = readl(&chip->mixer.array->src_gain_enable); in lola_src_gain_get()
682 unsigned short val; in lola_src_gain_get() local
683 if (!(chip->mixer.src_mask & (1 << idx))) in lola_src_gain_get()
684 return -EINVAL; in lola_src_gain_get()
686 val = readw(&chip->mixer.array->src_gain[idx]) + 1; in lola_src_gain_get()
688 val = 0; in lola_src_gain_get()
689 ucontrol->value.integer.value[i] = val; in lola_src_gain_get()
698 unsigned int ofs = kcontrol->private_value & 0xff; in lola_src_gain_put()
699 unsigned int count = (kcontrol->private_value >> 8) & 0xff; in lola_src_gain_put()
704 unsigned short val = ucontrol->value.integer.value[i]; in lola_src_gain_put() local
705 if (val) in lola_src_gain_put()
706 val--; in lola_src_gain_put()
707 err = lola_mixer_set_src_gain(chip, idx, val, !!val); in lola_src_gain_put()
714 /* raw value: 0 = -84dB, 336 = 0dB, 408=18dB, incremented 1 for mute */
715 static const DECLARE_TLV_DB_SCALE(lola_src_gain_tlv, -8425, 25, 1);
732 return snd_ctl_add(chip->card, in create_src_gain_mixer()
738 * destination gain (matrix-like) mixer
743 unsigned int src_num = (kcontrol->private_value >> 8) & 0xff;
745 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
746 uinfo->count = src_num;
747 uinfo->value.integer.min = 0;
748 uinfo->value.integer.max = 433;
756 unsigned int src_ofs = kcontrol->private_value & 0xff;
757 unsigned int src_num = (kcontrol->private_value >> 8) & 0xff;
758 unsigned int dst_ofs = (kcontrol->private_value >> 16) & 0xff;
761 dst = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) + dst_ofs;
762 mask = readl(&chip->mixer.array->dest_mix_gain_enable[dst]);
765 unsigned short val;
766 if (!(chip->mixer.src_mask & (1 << src)))
767 return -EINVAL;
769 val = readw(&chip->mixer.array->dest_mix_gain[dst][src]) + 1;
771 val = 0;
772 ucontrol->value.integer.value[i] = val;
781 unsigned int src_ofs = kcontrol->private_value & 0xff;
782 unsigned int src_num = (kcontrol->private_value >> 8) & 0xff;
783 unsigned int dst_ofs = (kcontrol->private_value >> 16) & 0xff;
791 unsigned short val = ucontrol->value.integer.value[i];
792 if (val) {
793 gains[num++] = val - 1;
798 dst = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) + dst_ofs;
802 static const DECLARE_TLV_DB_SCALE(lola_dest_gain_tlv, -8425, 25, 1);
822 return snd_ctl_add(chip->card,
842 err = create_src_gain_mixer(chip, chip->mixer.src_phys_ins, 0, in lola_create_mixer()
846 err = create_src_gain_mixer(chip, chip->mixer.src_stream_outs, in lola_create_mixer()
847 chip->mixer.src_stream_out_ofs, in lola_create_mixer()
854 chip->mixer.src_phys_ins, 0, in lola_create_mixer()
855 chip->mixer.dest_stream_ins, 0, in lola_create_mixer()
860 chip->mixer.src_stream_outs, in lola_create_mixer()
861 chip->mixer.src_stream_out_ofs, in lola_create_mixer()
862 chip->mixer.dest_stream_ins, 0, in lola_create_mixer()
863 "Stream-Loopback Capture Volume"); in lola_create_mixer()
867 chip->mixer.src_phys_ins, 0, in lola_create_mixer()
868 chip->mixer.dest_phys_outs, in lola_create_mixer()
869 chip->mixer.dest_phys_out_ofs, in lola_create_mixer()
870 "Line-Loopback Playback Volume"); in lola_create_mixer()
874 chip->mixer.src_stream_outs, in lola_create_mixer()
875 chip->mixer.src_stream_out_ofs, in lola_create_mixer()
876 chip->mixer.dest_phys_outs, in lola_create_mixer()
877 chip->mixer.dest_phys_out_ofs, in lola_create_mixer()