Lines Matching +full:8 +full:- +full:pin
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
24 pin->nid = nid; in lola_init_pin()
27 dev_err(chip->card->dev, "Can't read wcaps for 0x%x\n", nid); in lola_init_pin()
32 pin->is_analog = false; in lola_init_pin()
34 pin->is_analog = true; 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()
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()
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()
108 dev_err(chip->card->dev, "Can't read wcaps for 0x%x\n", nid); 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()
124 if (!chip->mixer.array_saved) in lola_init_mixer_widget()
125 return -ENOMEM; in lola_init_mixer_widget()
128 chip->mixer.src_stream_outs = chip->pcm[PLAY].num_streams; in lola_init_mixer_widget()
129 chip->mixer.src_phys_ins = chip->pin[CAPT].num_pins; in lola_init_mixer_widget()
132 chip->mixer.dest_stream_ins = chip->pcm[CAPT].num_streams; in lola_init_mixer_widget()
133 chip->mixer.dest_phys_outs = chip->pin[PLAY].num_pins; in lola_init_mixer_widget()
138 chip->mixer.src_stream_out_ofs = chip->mixer.src_phys_ins + in lola_init_mixer_widget()
140 chip->mixer.dest_phys_out_ofs = chip->mixer.dest_stream_ins + in lola_init_mixer_widget()
144 * +-+ 0-------8------16-------8------16 in lola_init_mixer_widget()
147 * | |->| -> |unused | -> |unused | in lola_init_mixer_widget()
150 * |c| 8-------------------------------- in lola_init_mixer_widget()
156 * | | 16------------------------------- in lola_init_mixer_widget()
159 * |n|->| -> |unused | -> |unused | in lola_init_mixer_widget()
162 * |a| 8-------------------------------- in lola_init_mixer_widget()
168 * +++ 16--|---------------|------------ in lola_init_mixer_widget()
169 * +---V---------------V-----------+ in lola_init_mixer_widget()
171 * +-------------------------------+ in lola_init_mixer_widget()
174 * +-+ 0-------8-2 in lola_init_mixer_widget()
177 * |r|->| -> | | -> in lola_init_mixer_widget()
178 * |c| |CAPTURE| | <- OUTPUT in lola_init_mixer_widget()
180 * |g| 8---------- in lola_init_mixer_widget()
183 * |n|->| -> | | -> in lola_init_mixer_widget()
184 * | | |CAPTURE| | <- OUTPUT in lola_init_mixer_widget()
186 * |r| 8---|----|- in lola_init_mixer_widget()
187 * |r| +---V----V-------------------+ in lola_init_mixer_widget()
189 * |y| +----------------------------+ in lola_init_mixer_widget()
191 if (chip->mixer.src_stream_out_ofs > MAX_AUDIO_INOUT_COUNT || in lola_init_mixer_widget()
192 chip->mixer.dest_phys_out_ofs > MAX_STREAM_IN_COUNT) { in lola_init_mixer_widget()
193 dev_err(chip->card->dev, "Invalid mixer widget size\n"); in lola_init_mixer_widget()
194 return -EINVAL; in lola_init_mixer_widget()
197 chip->mixer.src_mask = ((1U << chip->mixer.src_phys_ins) - 1) | in lola_init_mixer_widget()
198 (((1U << chip->mixer.src_stream_outs) - 1) in lola_init_mixer_widget()
199 << chip->mixer.src_stream_out_ofs); in lola_init_mixer_widget()
200 chip->mixer.dest_mask = ((1U << chip->mixer.dest_stream_ins) - 1) | in lola_init_mixer_widget()
201 (((1U << chip->mixer.dest_phys_outs) - 1) in lola_init_mixer_widget()
202 << chip->mixer.dest_phys_out_ofs); in lola_init_mixer_widget()
204 dev_dbg(chip->card->dev, "Mixer src_mask=%x, dest_mask=%x\n", in lola_init_mixer_widget()
205 chip->mixer.src_mask, chip->mixer.dest_mask); in lola_init_mixer_widget()
215 if (!(chip->mixer.src_mask & (1 << id))) in lola_mixer_set_src_gain()
216 return -EINVAL; in lola_mixer_set_src_gain()
217 oldval = val = readl(&chip->mixer.array->src_gain_enable); in lola_mixer_set_src_gain()
224 (gain == readw(&chip->mixer.array->src_gain[id]))) in lola_mixer_set_src_gain()
227 dev_dbg(chip->card->dev, in lola_mixer_set_src_gain()
230 writew(gain, &chip->mixer.array->src_gain[id]); in lola_mixer_set_src_gain()
231 writel(val, &chip->mixer.array->src_gain_enable); in lola_mixer_set_src_gain()
233 /* inform micro-controller about the new source gain */ in lola_mixer_set_src_gain()
234 return lola_codec_write(chip, chip->mixer.nid, in lola_mixer_set_src_gain()
244 if ((chip->mixer.src_mask & mask) != mask)
245 return -EINVAL;
248 writew(*gains, &chip->mixer.array->src_gain[i]);
252 writel(mask, &chip->mixer.array->src_gain_enable);
254 if (chip->mixer.caps & LOLA_PEAK_METER_CAN_AGC_MASK) {
256 return lola_codec_write(chip, chip->mixer.nid,
262 lola_codec_write(chip, chip->mixer.nid,
276 if (!(chip->mixer.src_mask & (1 << src)) || in lola_mixer_set_mapping_gain()
277 !(chip->mixer.dest_mask & (1 << dest))) in lola_mixer_set_mapping_gain()
278 return -EINVAL; in lola_mixer_set_mapping_gain()
280 writew(gain, &chip->mixer.array->dest_mix_gain[dest][src]); in lola_mixer_set_mapping_gain()
281 val = readl(&chip->mixer.array->dest_mix_gain_enable[dest]); in lola_mixer_set_mapping_gain()
286 writel(val, &chip->mixer.array->dest_mix_gain_enable[dest]); in lola_mixer_set_mapping_gain()
288 return lola_codec_write(chip, chip->mixer.nid, LOLA_VERB_SET_MIX_GAIN, in lola_mixer_set_mapping_gain()
298 if (!(chip->mixer.dest_mask & (1 << id)) ||
299 (chip->mixer.src_mask & mask) != mask)
300 return -EINVAL;
303 writew(*gains, &chip->mixer.array->dest_mix_gain[id][i]);
307 writel(mask, &chip->mixer.array->dest_mix_gain_enable[id]);
310 return lola_codec_write(chip, chip->mixer.nid,
324 struct lola_pin *pin; in lola_setup_all_analog_gains() local
327 pin = chip->pin[dir].pins; in lola_setup_all_analog_gains()
328 max_idx = chip->pin[dir].num_pins; in lola_setup_all_analog_gains()
330 if (pin[idx].is_analog) { in lola_setup_all_analog_gains()
331 unsigned int val = mute ? 0 : pin[idx].cur_gain_step; in lola_setup_all_analog_gains()
346 struct lola_pin *pin; in set_analog_volume() local
349 if (idx >= chip->pin[dir].num_pins) in set_analog_volume()
350 return -EINVAL; in set_analog_volume()
351 pin = &chip->pin[dir].pins[idx]; in set_analog_volume()
352 if (!pin->is_analog || pin->amp_num_steps <= val) in set_analog_volume()
353 return -EINVAL; in set_analog_volume()
354 if (external_call && pin->cur_gain_step == val) in set_analog_volume()
358 dev_dbg(chip->card->dev, in set_analog_volume()
361 err = lola_codec_write(chip, pin->nid, in set_analog_volume()
366 pin->cur_gain_step = val; in set_analog_volume()
377 if ((chip->input_src_caps_mask & src_mask) != src_mask) in lola_set_src_config()
378 return -EINVAL; in lola_set_src_config()
379 /* handle all even Inputs - SRC is a stereo setting !!! */ in lola_set_src_config()
380 for (n = 0; n < chip->pin[CAPT].num_pins; n += 2) { in lola_set_src_config()
383 if (!(chip->input_src_caps_mask & mask)) in lola_set_src_config()
388 src_state = (chip->input_src_mask & mask) != 0; in lola_set_src_config()
392 err = lola_codec_write(chip, chip->pcm[CAPT].streams[n].nid, in lola_set_src_config()
402 chip->input_src_mask = src_mask; in lola_set_src_config()
413 lola_set_src_config(chip, (1 << chip->pin[CAPT].num_pins) - 1, false); in init_mixer_values()
416 memset_io(chip->mixer.array, 0, sizeof(*chip->mixer.array)); in init_mixer_values()
417 /* inform firmware about all updated matrix columns - capture part */ in init_mixer_values()
418 for (i = 0; i < chip->mixer.dest_stream_ins; i++) in init_mixer_values()
419 lola_codec_write(chip, chip->mixer.nid, in init_mixer_values()
422 /* inform firmware about all updated matrix columns - output part */ in init_mixer_values()
423 for (i = 0; i < chip->mixer.dest_phys_outs; i++) in init_mixer_values()
424 lola_codec_write(chip, chip->mixer.nid, in init_mixer_values()
426 chip->mixer.dest_phys_out_ofs + i, 0); in init_mixer_values()
429 for (i = 0; i < chip->mixer.src_phys_ins; i++) in init_mixer_values()
433 for (i = 0; i < chip->mixer.src_stream_outs; i++) in init_mixer_values()
435 i + chip->mixer.src_stream_out_ofs, in init_mixer_values()
437 /* set gain value 0dB diagonally in matrix - part INPUT -> CAPTURE */ in init_mixer_values()
438 for (i = 0; i < chip->mixer.dest_stream_ins; i++) { in init_mixer_values()
439 int src = i % chip->mixer.src_phys_ins; in init_mixer_values()
442 /* set gain value 0dB diagonally in matrix , part PLAYBACK -> OUTPUT in init_mixer_values()
446 for (i = 0; i < chip->mixer.src_stream_outs; i++) { in init_mixer_values()
447 int src = chip->mixer.src_stream_out_ofs + i; in init_mixer_values()
448 int dst = chip->mixer.dest_phys_out_ofs + in init_mixer_values()
449 i % chip->mixer.dest_phys_outs; in init_mixer_values()
462 int dir = kcontrol->private_value; in lola_analog_vol_info()
464 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; in lola_analog_vol_info()
465 uinfo->count = chip->pin[dir].num_pins; in lola_analog_vol_info()
466 uinfo->value.integer.min = 0; in lola_analog_vol_info()
467 uinfo->value.integer.max = chip->pin[dir].pins[0].amp_num_steps; in lola_analog_vol_info()
475 int dir = kcontrol->private_value; in lola_analog_vol_get()
478 for (i = 0; i < chip->pin[dir].num_pins; i++) in lola_analog_vol_get()
479 ucontrol->value.integer.value[i] = in lola_analog_vol_get()
480 chip->pin[dir].pins[i].cur_gain_step; in lola_analog_vol_get()
488 int dir = kcontrol->private_value; in lola_analog_vol_put()
491 for (i = 0; i < chip->pin[dir].num_pins; i++) { in lola_analog_vol_put()
493 ucontrol->value.integer.value[i], in lola_analog_vol_put()
505 int dir = kcontrol->private_value; in lola_analog_vol_tlv()
507 struct lola_pin *pin; in lola_analog_vol_tlv() local
510 return -ENOMEM; in lola_analog_vol_tlv()
511 pin = &chip->pin[dir].pins[0]; in lola_analog_vol_tlv()
513 val2 = pin->amp_step_size * 25; in lola_analog_vol_tlv()
514 val1 = -1 * (int)pin->amp_offset * (int)val2; in lola_analog_vol_tlv()
519 return -EFAULT; in lola_analog_vol_tlv()
521 return -EFAULT; in lola_analog_vol_tlv()
523 return -EFAULT; in lola_analog_vol_tlv()
525 return -EFAULT; in lola_analog_vol_tlv()
542 if (!chip->pin[dir].num_pins) in create_analog_mixer()
545 if (chip->pin[dir].num_pins != chip->pin[dir].num_analog_pins) in create_analog_mixer()
549 return snd_ctl_add(chip->card, in create_analog_mixer()
561 uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN; in lola_input_src_info()
562 uinfo->count = chip->pin[CAPT].num_pins; in lola_input_src_info()
563 uinfo->value.integer.min = 0; in lola_input_src_info()
564 uinfo->value.integer.max = 1; in lola_input_src_info()
574 for (i = 0; i < chip->pin[CAPT].num_pins; i++) in lola_input_src_get()
575 ucontrol->value.integer.value[i] = in lola_input_src_get()
576 !!(chip->input_src_mask & (1 << i)); in lola_input_src_get()
588 for (i = 0; i < chip->pin[CAPT].num_pins; i++) in lola_input_src_put()
589 if (ucontrol->value.integer.value[i]) in lola_input_src_put()
608 if (!chip->input_src_caps_mask) in create_input_src_mixer()
611 return snd_ctl_add(chip->card, in create_input_src_mixer()
621 unsigned int count = (kcontrol->private_value >> 8) & 0xff; in lola_src_gain_info()
623 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; in lola_src_gain_info()
624 uinfo->count = count; in lola_src_gain_info()
625 uinfo->value.integer.min = 0; in lola_src_gain_info()
626 uinfo->value.integer.max = 409; in lola_src_gain_info()
634 unsigned int ofs = kcontrol->private_value & 0xff; in lola_src_gain_get()
635 unsigned int count = (kcontrol->private_value >> 8) & 0xff; in lola_src_gain_get()
638 mask = readl(&chip->mixer.array->src_gain_enable); in lola_src_gain_get()
642 if (!(chip->mixer.src_mask & (1 << idx))) in lola_src_gain_get()
643 return -EINVAL; in lola_src_gain_get()
645 val = readw(&chip->mixer.array->src_gain[idx]) + 1; in lola_src_gain_get()
648 ucontrol->value.integer.value[i] = val; in lola_src_gain_get()
657 unsigned int ofs = kcontrol->private_value & 0xff; in lola_src_gain_put()
658 unsigned int count = (kcontrol->private_value >> 8) & 0xff; in lola_src_gain_put()
663 unsigned short val = ucontrol->value.integer.value[i]; in lola_src_gain_put()
665 val--; in lola_src_gain_put()
673 /* raw value: 0 = -84dB, 336 = 0dB, 408=18dB, incremented 1 for mute */
674 static const DECLARE_TLV_DB_SCALE(lola_src_gain_tlv, -8425, 25, 1);
690 lola_src_gain_mixer.private_value = ofs + (num << 8); in create_src_gain_mixer()
691 return snd_ctl_add(chip->card, in create_src_gain_mixer()
697 * destination gain (matrix-like) mixer
702 unsigned int src_num = (kcontrol->private_value >> 8) & 0xff;
704 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
705 uinfo->count = src_num;
706 uinfo->value.integer.min = 0;
707 uinfo->value.integer.max = 433;
715 unsigned int src_ofs = kcontrol->private_value & 0xff;
716 unsigned int src_num = (kcontrol->private_value >> 8) & 0xff;
717 unsigned int dst_ofs = (kcontrol->private_value >> 16) & 0xff;
720 dst = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) + dst_ofs;
721 mask = readl(&chip->mixer.array->dest_mix_gain_enable[dst]);
725 if (!(chip->mixer.src_mask & (1 << src)))
726 return -EINVAL;
728 val = readw(&chip->mixer.array->dest_mix_gain[dst][src]) + 1;
731 ucontrol->value.integer.value[i] = val;
740 unsigned int src_ofs = kcontrol->private_value & 0xff;
741 unsigned int src_num = (kcontrol->private_value >> 8) & 0xff;
742 unsigned int dst_ofs = (kcontrol->private_value >> 16) & 0xff;
750 unsigned short val = ucontrol->value.integer.value[i];
752 gains[num++] = val - 1;
757 dst = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) + dst_ofs;
761 static const DECLARE_TLV_DB_SCALE(lola_dest_gain_tlv, -8425, 25, 1);
780 src_ofs + (src_num << 8) + (ofs << 16) + (num << 24);
781 return snd_ctl_add(chip->card,
801 err = create_src_gain_mixer(chip, chip->mixer.src_phys_ins, 0, in lola_create_mixer()
805 err = create_src_gain_mixer(chip, chip->mixer.src_stream_outs, in lola_create_mixer()
806 chip->mixer.src_stream_out_ofs, in lola_create_mixer()
813 chip->mixer.src_phys_ins, 0, in lola_create_mixer()
814 chip->mixer.dest_stream_ins, 0, in lola_create_mixer()
819 chip->mixer.src_stream_outs, in lola_create_mixer()
820 chip->mixer.src_stream_out_ofs, in lola_create_mixer()
821 chip->mixer.dest_stream_ins, 0, in lola_create_mixer()
822 "Stream-Loopback Capture Volume"); in lola_create_mixer()
826 chip->mixer.src_phys_ins, 0, in lola_create_mixer()
827 chip->mixer.dest_phys_outs, in lola_create_mixer()
828 chip->mixer.dest_phys_out_ofs, in lola_create_mixer()
829 "Line-Loopback Playback Volume"); in lola_create_mixer()
833 chip->mixer.src_stream_outs, in lola_create_mixer()
834 chip->mixer.src_stream_out_ofs, in lola_create_mixer()
835 chip->mixer.dest_phys_outs, in lola_create_mixer()
836 chip->mixer.dest_phys_out_ofs, in lola_create_mixer()