/** * FreeRDP: A Remote Desktop Protocol Implementation * Audio Output Virtual Channel * * Copyright 2009-2011 Jay Sorg * Copyright 2010-2011 Vic Lee * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include #include #include #include "rdpsnd_main.h" typedef struct rdpsnd_alsa_plugin rdpsndAlsaPlugin; struct rdpsnd_alsa_plugin { rdpsndDevicePlugin device; char* device_name; snd_pcm_t* pcm_handle; snd_mixer_t* mixer_handle; UINT32 source_rate; UINT32 actual_rate; snd_pcm_format_t format; UINT32 source_channels; UINT32 actual_channels; int bytes_per_channel; int wformat; int block_size; int latency; BYTE* audio_data; UINT32 audio_data_size; UINT32 audio_data_left; snd_pcm_uframes_t period_size; snd_async_handler_t* pcm_callback; FREERDP_DSP_CONTEXT* dsp_context; }; void rdpsnd_alsa_async_handler(snd_async_handler_t* pcm_callback); static void rdpsnd_alsa_set_params(rdpsndAlsaPlugin* alsa) { int status; snd_pcm_hw_params_t* hw_params; snd_pcm_sw_params_t* sw_params; snd_pcm_uframes_t start_threshold; snd_pcm_uframes_t buffer_size; snd_pcm_drop(alsa->pcm_handle); //snd_async_add_pcm_handler(&alsa->pcm_callback, alsa->pcm_handle, rdpsnd_alsa_async_handler, (void*) alsa); status = snd_pcm_hw_params_malloc(&hw_params); if (status < 0) { DEBUG_WARN("snd_pcm_hw_params_malloc failed"); return; } snd_pcm_hw_params_any(alsa->pcm_handle, hw_params); snd_pcm_hw_params_set_access(alsa->pcm_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED); snd_pcm_hw_params_set_format(alsa->pcm_handle, hw_params, alsa->format); snd_pcm_hw_params_set_rate_near(alsa->pcm_handle, hw_params, &alsa->actual_rate, NULL); snd_pcm_hw_params_set_channels_near(alsa->pcm_handle, hw_params, &alsa->actual_channels); snd_pcm_hw_params_get_period_size(hw_params, &alsa->period_size, 0); alsa->audio_data_left = 0; if (alsa->latency < 0) buffer_size = alsa->actual_rate * 4 / 10; /* Default to 400ms buffer */ else buffer_size = alsa->latency * alsa->actual_rate * 2 / 1000; /* Double of the latency */ if (buffer_size < alsa->actual_rate / 2) buffer_size = alsa->actual_rate / 2; /* Minimum 0.5-second buffer */ snd_pcm_hw_params_set_buffer_size_near(alsa->pcm_handle, hw_params, &buffer_size); //snd_pcm_hw_params_set_period_size_near(alsa->out_handle, hw_params, &alsa->period_size, NULL); snd_pcm_hw_params(alsa->pcm_handle, hw_params); snd_pcm_hw_params_free(hw_params); status = snd_pcm_sw_params_malloc(&sw_params); if (status < 0) { DEBUG_WARN("snd_pcm_sw_params_malloc failed"); return; } snd_pcm_sw_params_current(alsa->pcm_handle, sw_params); if (alsa->latency == 0) start_threshold = 0; else start_threshold = buffer_size / 2; snd_pcm_sw_params_set_start_threshold(alsa->pcm_handle, sw_params, start_threshold); snd_pcm_sw_params(alsa->pcm_handle, sw_params); snd_pcm_sw_params_free(sw_params); snd_pcm_prepare(alsa->pcm_handle); DEBUG_SVC("hardware buffer %d frames, playback buffer %.2g seconds", (int) buffer_size, (double) buffer_size / 2.0 / (double) alsa->actual_rate); if ((alsa->actual_rate != alsa->source_rate) || (alsa->actual_channels != alsa->source_channels)) { DEBUG_SVC("actual rate %d / channel %d is different from source rate %d / channel %d, resampling required.", alsa->actual_rate, alsa->actual_channels, alsa->source_rate, alsa->source_channels); } } static void rdpsnd_alsa_set_format(rdpsndDevicePlugin* device, rdpsndFormat* format, int latency) { rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; if (format != NULL) { alsa->source_rate = format->nSamplesPerSec; alsa->actual_rate = format->nSamplesPerSec; alsa->source_channels = format->nChannels; alsa->actual_channels = format->nChannels; switch (format->wFormatTag) { case WAVE_FORMAT_PCM: switch (format->wBitsPerSample) { case 4: break; case 8: alsa->format = SND_PCM_FORMAT_S8; alsa->bytes_per_channel = 1; break; case 16: alsa->format = SND_PCM_FORMAT_S16_LE; alsa->bytes_per_channel = 2; break; } break; case WAVE_FORMAT_ADPCM: case WAVE_FORMAT_DVI_ADPCM: alsa->format = SND_PCM_FORMAT_S16_LE; alsa->bytes_per_channel = 2; break; } alsa->wformat = format->wFormatTag; alsa->block_size = format->nBlockAlign; } alsa->latency = latency; rdpsnd_alsa_set_params(alsa); } static void rdpsnd_alsa_open_mixer(rdpsndAlsaPlugin* alsa) { int status; snd_mixer_t* handle; status = snd_mixer_open(&handle, 0); if (status < 0) { DEBUG_WARN("snd_mixer_open failed"); return; } status = snd_mixer_attach(handle, alsa->device_name); if (status < 0) { DEBUG_WARN("snd_mixer_attach failed"); snd_mixer_close(handle); return; } status = snd_mixer_selem_register(handle, NULL, NULL); if (status < 0) { DEBUG_WARN("snd_mixer_selem_register failed"); snd_mixer_close(handle); return; } status = snd_mixer_load(handle); if (status < 0) { DEBUG_WARN("snd_mixer_load failed"); snd_mixer_close(handle); return; } alsa->mixer_handle = handle; } static void rdpsnd_alsa_open(rdpsndDevicePlugin* device, rdpsndFormat* format, int latency) { int mode; int status; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; if (alsa->pcm_handle != 0) return; DEBUG_SVC("opening"); mode = 0; //mode |= SND_PCM_NONBLOCK; status = snd_pcm_open(&alsa->pcm_handle, alsa->device_name, SND_PCM_STREAM_PLAYBACK, mode); if (status < 0) { DEBUG_WARN("snd_pcm_open failed"); } else { freerdp_dsp_context_reset_adpcm(alsa->dsp_context); rdpsnd_alsa_set_format(device, format, latency); rdpsnd_alsa_open_mixer(alsa); } } static void rdpsnd_alsa_close(rdpsndDevicePlugin* device) { rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*)device; if (alsa->pcm_handle != 0) { DEBUG_SVC("close"); snd_pcm_drain(alsa->pcm_handle); snd_pcm_close(alsa->pcm_handle); alsa->pcm_handle = 0; } if (alsa->mixer_handle) { snd_mixer_close(alsa->mixer_handle); alsa->mixer_handle = NULL; } } static void rdpsnd_alsa_free(rdpsndDevicePlugin* device) { rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; rdpsnd_alsa_close(device); free(alsa->device_name); if (alsa->audio_data) free(alsa->audio_data); freerdp_dsp_context_free(alsa->dsp_context); free(alsa); } static BOOL rdpsnd_alsa_format_supported(rdpsndDevicePlugin* device, rdpsndFormat* format) { switch (format->wFormatTag) { case WAVE_FORMAT_PCM: if (format->cbSize == 0 && format->nSamplesPerSec <= 48000 && (format->wBitsPerSample == 8 || format->wBitsPerSample == 16) && (format->nChannels == 1 || format->nChannels == 2)) { return TRUE; } break; case WAVE_FORMAT_ADPCM: case WAVE_FORMAT_DVI_ADPCM: if (format->nSamplesPerSec <= 48000 && format->wBitsPerSample == 4 && (format->nChannels == 1 || format->nChannels == 2)) { return TRUE; } break; case WAVE_FORMAT_ALAW: break; case WAVE_FORMAT_MULAW: break; case WAVE_FORMAT_GSM610: break; } return FALSE; } static void rdpsnd_alsa_set_volume(rdpsndDevicePlugin* device, UINT32 value) { long left; long right; long volume_min; long volume_max; long volume_left; long volume_right; snd_mixer_elem_t* elem; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; if (!alsa->mixer_handle) return; left = (value & 0xFFFF); right = ((value >> 16) & 0xFFFF); for (elem = snd_mixer_first_elem(alsa->mixer_handle); elem; elem = snd_mixer_elem_next(elem)) { if (snd_mixer_selem_has_playback_volume(elem)) { snd_mixer_selem_get_playback_volume_range(elem, &volume_min, &volume_max); volume_left = volume_min + (left * (volume_max - volume_min)) / 0xFFFF; volume_right = volume_min + (right * (volume_max - volume_min)) / 0xFFFF; snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, volume_left); snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, volume_right); } } } void rdpsnd_alsa_async_handler(snd_async_handler_t* pcm_callback) { snd_pcm_t* pcm_handle; snd_pcm_sframes_t avail; rdpsndAlsaPlugin* alsa; pcm_handle = snd_async_handler_get_pcm(pcm_callback); alsa = (rdpsndAlsaPlugin*) snd_async_handler_get_callback_private(pcm_callback); avail = snd_pcm_avail_update(pcm_handle); } static void rdpsnd_alsa_play(rdpsndDevicePlugin* device, BYTE* data, int size) { BYTE* src; int len; int status; int frames; int rbytes_per_frame; int sbytes_per_frame; BYTE* pindex; BYTE* end; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; if (alsa->pcm_handle == 0) return; if (alsa->wformat == WAVE_FORMAT_ADPCM) { alsa->dsp_context->decode_ms_adpcm(alsa->dsp_context, data, size, alsa->source_channels, alsa->block_size); size = alsa->dsp_context->adpcm_size; src = alsa->dsp_context->adpcm_buffer; } else if (alsa->wformat == WAVE_FORMAT_DVI_ADPCM) { alsa->dsp_context->decode_ima_adpcm(alsa->dsp_context, data, size, alsa->source_channels, alsa->block_size); size = alsa->dsp_context->adpcm_size; src = alsa->dsp_context->adpcm_buffer; } else { src = data; } sbytes_per_frame = alsa->source_channels * alsa->bytes_per_channel; rbytes_per_frame = alsa->actual_channels * alsa->bytes_per_channel; if ((size % sbytes_per_frame) != 0) { DEBUG_WARN("error len mod"); return; } if ((alsa->source_rate == alsa->actual_rate) && (alsa->source_channels == alsa->actual_channels)) { } else { alsa->dsp_context->resample(alsa->dsp_context, src, alsa->bytes_per_channel, alsa->source_channels, alsa->source_rate, size / sbytes_per_frame, alsa->actual_channels, alsa->actual_rate); frames = alsa->dsp_context->resampled_frames; DEBUG_SVC("resampled %d frames at %d to %d frames at %d", size / sbytes_per_frame, alsa->source_rate, frames, alsa->actual_rate); size = frames * rbytes_per_frame; src = alsa->dsp_context->resampled_buffer; } if (alsa->audio_data_left + size > alsa->audio_data_size) { alsa->audio_data = realloc(alsa->audio_data, alsa->audio_data_left + size); alsa->audio_data_size = alsa->audio_data_left + size; } CopyMemory(alsa->audio_data + alsa->audio_data_left, src, size); alsa->audio_data_left += size; pindex = alsa->audio_data; end = pindex + alsa->audio_data_left; while (pindex + alsa->period_size * rbytes_per_frame <= end) { len = end - pindex; status = snd_pcm_writei(alsa->pcm_handle, pindex, alsa->period_size); if (status == -EPIPE) { snd_pcm_recover(alsa->pcm_handle, status, 0); status = 0; } else if (status < 0) { DEBUG_WARN("snd_pcm_writei status %d", status); snd_pcm_close(alsa->pcm_handle); alsa->pcm_handle = 0; rdpsnd_alsa_open(device, NULL, alsa->latency); break; } pindex += status * rbytes_per_frame; } if ((pindex <= end) && (pindex != alsa->audio_data)) { CopyMemory(alsa->audio_data, pindex, end - pindex); alsa->audio_data_left = end - pindex; } } static void rdpsnd_alsa_start(rdpsndDevicePlugin* device) { rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; if (!alsa->pcm_handle) return; snd_pcm_start(alsa->pcm_handle); } COMMAND_LINE_ARGUMENT_A rdpsnd_alsa_args[] = { { "dev", COMMAND_LINE_VALUE_REQUIRED, "", NULL, NULL, -1, NULL, "device" }, { NULL, 0, NULL, NULL, NULL, -1, NULL, NULL } }; static void rdpsnd_alsa_parse_addin_args(rdpsndDevicePlugin* device, ADDIN_ARGV* args) { int status; DWORD flags; COMMAND_LINE_ARGUMENT_A* arg; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON; status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, rdpsnd_alsa_args, flags, alsa, NULL, NULL); arg = rdpsnd_alsa_args; do { if (!(arg->Flags & COMMAND_LINE_VALUE_PRESENT)) continue; CommandLineSwitchStart(arg) CommandLineSwitchCase(arg, "dev") { alsa->device_name = _strdup(arg->Value); } CommandLineSwitchEnd(arg) } while ((arg = CommandLineFindNextArgumentA(arg)) != NULL); } #ifdef STATIC_CHANNELS #define freerdp_rdpsnd_client_subsystem_entry alsa_freerdp_rdpsnd_client_subsystem_entry #endif int freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS pEntryPoints) { ADDIN_ARGV* args; rdpsndAlsaPlugin* alsa; alsa = (rdpsndAlsaPlugin*) malloc(sizeof(rdpsndAlsaPlugin)); ZeroMemory(alsa, sizeof(rdpsndAlsaPlugin)); alsa->device.Open = rdpsnd_alsa_open; alsa->device.FormatSupported = rdpsnd_alsa_format_supported; alsa->device.SetFormat = rdpsnd_alsa_set_format; alsa->device.SetVolume = rdpsnd_alsa_set_volume; alsa->device.Play = rdpsnd_alsa_play; alsa->device.Start = rdpsnd_alsa_start; alsa->device.Close = rdpsnd_alsa_close; alsa->device.Free = rdpsnd_alsa_free; args = pEntryPoints->args; rdpsnd_alsa_parse_addin_args((rdpsndDevicePlugin*) alsa, args); if (!alsa->device_name) alsa->device_name = _strdup("default"); alsa->pcm_handle = 0; alsa->source_rate = 22050; alsa->actual_rate = 22050; alsa->format = SND_PCM_FORMAT_S16_LE; alsa->source_channels = 2; alsa->actual_channels = 2; alsa->bytes_per_channel = 2; alsa->dsp_context = freerdp_dsp_context_new(); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) alsa); return 0; }