/* Software stereo volume control.
*/ staticvoid change_volume(struct urb *urb_out, int volume[], int bytes_per_frame)
{ int chn = 0;
if (volume[0] == 256 && volume[1] == 256) return; /* maximum volume - no change */
if (bytes_per_frame == 4) {
__le16 *p, *buf_end;
p = (__le16 *)urb_out->transfer_buffer;
buf_end = p + urb_out->transfer_buffer_length / sizeof(*p);
for (; p < buf_end; ++p) { short pv = le16_to_cpu(*p); int val = (pv * volume[chn & 1]) >> 8;
pv = clamp(val, -0x8000, 0x7fff);
*p = cpu_to_le16(pv);
++chn;
}
} elseif (bytes_per_frame == 6) { unsignedchar *p, *buf_end;
p = (unsignedchar *)urb_out->transfer_buffer;
buf_end = p + urb_out->transfer_buffer_length;
for (; p < buf_end; p += 3) { int val;
val = p[0] + (p[1] << 8) + ((signedchar)p[2] << 16);
val = (val * volume[chn & 1]) >> 8;
val = clamp(val, -0x800000, 0x7fffff);
p[0] = val;
p[1] = val >> 8;
p[2] = val >> 16;
++chn;
}
}
}
/* Create signal for impulse response test.
*/ staticvoid create_impulse_test_signal(struct snd_line6_pcm *line6pcm, struct urb *urb_out, int bytes_per_frame)
{ int frames = urb_out->transfer_buffer_length / bytes_per_frame;
if (bytes_per_frame == 4) { int i; short *pi = (short *)line6pcm->prev_fbuf; short *po = (short *)urb_out->transfer_buffer;
for (i = 0; i < frames; ++i) {
po[0] = pi[0];
po[1] = 0;
pi += 2;
po += 2;
}
} elseif (bytes_per_frame == 6) { int i, j; unsignedchar *pi = line6pcm->prev_fbuf; unsignedchar *po = urb_out->transfer_buffer;
for (i = 0; i < frames; ++i) { for (j = 0; j < bytes_per_frame / 2; ++j)
po[j] = pi[j];
for (; j < bytes_per_frame; ++j)
po[j] = 0;
pi += bytes_per_frame;
po += bytes_per_frame;
}
} if (--line6pcm->impulse_count <= 0) {
((unsignedchar *)(urb_out->transfer_buffer))[bytes_per_frame -
1] =
line6pcm->impulse_volume;
line6pcm->impulse_count = line6pcm->impulse_period;
}
}
/* Add signal to buffer for software monitoring.
*/ staticvoid add_monitor_signal(struct urb *urb_out, unsignedchar *signal, int volume, int bytes_per_frame)
{ if (volume == 0) return; /* zero volume - no change */
if (bytes_per_frame == 4) {
__le16 *pi, *po, *buf_end;
pi = (__le16 *)signal;
po = (__le16 *)urb_out->transfer_buffer;
buf_end = po + urb_out->transfer_buffer_length / sizeof(*po);
for (; po < buf_end; ++pi, ++po) { short pov = le16_to_cpu(*po); short piv = le16_to_cpu(*pi); int val = pov + ((piv * volume) >> 8);
pov = clamp(val, -0x8000, 0x7fff);
*po = cpu_to_le16(pov);
}
}
/* We don't need to handle devices with 6 bytes per frame here since they all support hardware monitoring.
*/
}
/* Find a free URB, prepare audio data, and submit URB. must be called in line6pcm->out.lock context
*/ staticint submit_audio_out_urb(struct snd_line6_pcm *line6pcm)
{ int index; int i, urb_size, urb_frames; int ret; constint bytes_per_frame =
line6pcm->properties->bytes_per_channel *
line6pcm->properties->playback_hw.channels_max; constint frame_increment =
line6pcm->properties->rates.rats[0].num_min; constint frame_factor =
line6pcm->properties->rates.rats[0].den *
(line6pcm->line6->intervals_per_second / LINE6_ISO_INTERVAL); struct urb *urb_out;
index = find_first_zero_bit(&line6pcm->out.active_urbs,
line6pcm->line6->iso_buffers);
if (index < 0 || index >= line6pcm->line6->iso_buffers) {
dev_err(line6pcm->line6->ifcdev, "no free URB found\n"); return -EINVAL;
}
/* TODO: this may not work for LINE6_ISO_PACKETS != 1 */ for (i = 0; i < LINE6_ISO_PACKETS; ++i) { /* compute frame size for given sampling rate */ int fsize = 0; struct usb_iso_packet_descriptor *fout =
&urb_out->iso_frame_desc[i];
fsize = line6pcm->prev_fsize; if (fsize == 0) { int n;
line6pcm->out.count += frame_increment;
n = line6pcm->out.count / frame_factor;
line6pcm->out.count -= n * frame_factor;
fsize = n;
}
if (line6pcm->out.pos + urb_frames > runtime->buffer_size) { /* The transferred area goes over buffer boundary, copy the data to the temp buffer.
*/ int len;
if (ret == 0)
set_bit(index, &line6pcm->out.active_urbs); else
dev_err(line6pcm->line6->ifcdev, "URB out #%d submission failed (%d)\n", index, ret);
return 0;
}
/* Submit all currently available playback URBs. must be called in line6pcm->out.lock context
*/ int line6_submit_audio_out_all_urbs(struct snd_line6_pcm *line6pcm)
{ int ret = 0, i;
for (i = 0; i < line6pcm->line6->iso_buffers; ++i) {
ret = submit_audio_out_urb(line6pcm); if (ret < 0) break;
}
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.