Merge pull request #4453 from akallabeth/sound_channel_refactor

Sound channel refactoring
This commit is contained in:
David Fort 2018-05-03 11:56:58 +02:00 committed by GitHub
commit 456b0e8934
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
45 changed files with 3416 additions and 3203 deletions

View File

@ -763,6 +763,18 @@ set(GSM_FEATURE_TYPE "OPTIONAL")
set(GSM_FEATURE_PURPOSE "codec") set(GSM_FEATURE_PURPOSE "codec")
set(GSM_FEATURE_DESCRIPTION "GSM audio codec library") set(GSM_FEATURE_DESCRIPTION "GSM audio codec library")
set(LAME_FEATURE_TYPE "OPTIONAL")
set(LAME_FEATURE_PURPOSE "codec")
set(LAME_FEATURE_DESCRIPTION "lame MP3 audio codec library")
set(FAAD2_FEATURE_TYPE "OPTIONAL")
set(FAAD2_FEATURE_PURPOSE "codec")
set(FAAD2_FEATURE_DESCRIPTION "FAAD2 AAC audio codec library")
set(FAAC_FEATURE_TYPE "OPTIONAL")
set(FAAC_FEATURE_PURPOSE "codec")
set(FAAC_FEATURE_DESCRIPTION "FAAC AAC audio codec library")
set(GSSAPI_FEATURE_TYPE "OPTIONAL") set(GSSAPI_FEATURE_TYPE "OPTIONAL")
set(GSSAPI_FEATURE_PURPOSE "auth") set(GSSAPI_FEATURE_PURPOSE "auth")
set(GSSAPI_FEATURE_DESCRIPTION "add kerberos support") set(GSSAPI_FEATURE_DESCRIPTION "add kerberos support")
@ -863,6 +875,9 @@ find_feature(JPEG ${JPEG_FEATURE_TYPE} ${JPEG_FEATURE_PURPOSE} ${JPEG_FEATURE_DE
find_feature(x264 ${X264_FEATURE_TYPE} ${X264_FEATURE_PURPOSE} ${X264_FEATURE_DESCRIPTION}) find_feature(x264 ${X264_FEATURE_TYPE} ${X264_FEATURE_PURPOSE} ${X264_FEATURE_DESCRIPTION})
find_feature(OpenH264 ${OPENH264_FEATURE_TYPE} ${OPENH264_FEATURE_PURPOSE} ${OPENH264_FEATURE_DESCRIPTION}) find_feature(OpenH264 ${OPENH264_FEATURE_TYPE} ${OPENH264_FEATURE_PURPOSE} ${OPENH264_FEATURE_DESCRIPTION})
find_feature(GSM ${GSM_FEATURE_TYPE} ${GSM_FEATURE_PURPOSE} ${GSM_FEATURE_DESCRIPTION}) find_feature(GSM ${GSM_FEATURE_TYPE} ${GSM_FEATURE_PURPOSE} ${GSM_FEATURE_DESCRIPTION})
find_feature(LAME ${LAME_FEATURE_TYPE} ${LAME_FEATURE_PURPOSE} ${LAME_FEATURE_DESCRIPTION})
find_feature(FAAD2 ${FAAD2_FEATURE_TYPE} ${FAAD2_FEATURE_PURPOSE} ${FAAD2_FEATURE_DESCRIPTION})
find_feature(FAAC ${FAAC_FEATURE_TYPE} ${FAAC_FEATURE_PURPOSE} ${FAAC_FEATURE_DESCRIPTION})
find_feature(GSSAPI ${GSSAPI_FEATURE_TYPE} ${GSSAPI_FEATURE_PURPOSE} ${GSSAPI_FEATURE_DESCRIPTION}) find_feature(GSSAPI ${GSSAPI_FEATURE_TYPE} ${GSSAPI_FEATURE_PURPOSE} ${GSSAPI_FEATURE_DESCRIPTION})

View File

@ -35,7 +35,6 @@
#include <alsa/asoundlib.h> #include <alsa/asoundlib.h>
#include <freerdp/addin.h> #include <freerdp/addin.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/rdpsnd.h> #include <freerdp/channels/rdpsnd.h>
#include "audin_main.h" #include "audin_main.h"
@ -46,34 +45,52 @@ typedef struct _AudinALSADevice
char* device_name; char* device_name;
UINT32 frames_per_packet; UINT32 frames_per_packet;
UINT32 target_rate; AUDIO_FORMAT aformat;
UINT32 actual_rate;
snd_pcm_format_t format;
UINT32 target_channels;
UINT32 actual_channels;
int bytes_per_channel;
int wformat;
int block_size;
FREERDP_DSP_CONTEXT* dsp_context;
HANDLE thread; HANDLE thread;
HANDLE stopEvent; HANDLE stopEvent;
BYTE* buffer;
int buffer_frames;
AudinReceive receive; AudinReceive receive;
void* user_data; void* user_data;
rdpContext* rdpcontext; rdpContext* rdpcontext;
} AudinALSADevice; } AudinALSADevice;
static snd_pcm_format_t audin_alsa_format(UINT32 wFormatTag, UINT32 bitPerChannel)
{
switch (wFormatTag)
{
case WAVE_FORMAT_PCM:
switch (bitPerChannel)
{
case 16:
return SND_PCM_FORMAT_S16_LE;
case 8:
return SND_PCM_FORMAT_S8;
default:
return SND_PCM_FORMAT_UNKNOWN;
}
case WAVE_FORMAT_ALAW:
return SND_PCM_FORMAT_A_LAW;
case WAVE_FORMAT_MULAW:
return SND_PCM_FORMAT_MU_LAW;
default:
return SND_PCM_FORMAT_UNKNOWN;
}
}
static BOOL audin_alsa_set_params(AudinALSADevice* alsa, static BOOL audin_alsa_set_params(AudinALSADevice* alsa,
snd_pcm_t* capture_handle) snd_pcm_t* capture_handle)
{ {
int error; int error;
UINT32 channels = alsa->aformat.nChannels;
snd_pcm_hw_params_t* hw_params; snd_pcm_hw_params_t* hw_params;
snd_pcm_format_t format = audin_alsa_format(alsa->aformat.wFormatTag, alsa->aformat.wBitsPerSample);
if ((error = snd_pcm_hw_params_malloc(&hw_params)) < 0) if ((error = snd_pcm_hw_params_malloc(&hw_params)) < 0)
{ {
@ -85,150 +102,27 @@ static BOOL audin_alsa_set_params(AudinALSADevice* alsa,
snd_pcm_hw_params_any(capture_handle, hw_params); snd_pcm_hw_params_any(capture_handle, hw_params);
snd_pcm_hw_params_set_access(capture_handle, hw_params, snd_pcm_hw_params_set_access(capture_handle, hw_params,
SND_PCM_ACCESS_RW_INTERLEAVED); SND_PCM_ACCESS_RW_INTERLEAVED);
snd_pcm_hw_params_set_format(capture_handle, hw_params, alsa->format); snd_pcm_hw_params_set_format(capture_handle, hw_params, format);
snd_pcm_hw_params_set_rate_near(capture_handle, hw_params, &alsa->actual_rate, snd_pcm_hw_params_set_rate_near(capture_handle, hw_params,
NULL); &alsa->aformat.nSamplesPerSec, NULL);
snd_pcm_hw_params_set_channels_near(capture_handle, hw_params, snd_pcm_hw_params_set_channels_near(capture_handle, hw_params,
&alsa->actual_channels); &channels);
snd_pcm_hw_params(capture_handle, hw_params); snd_pcm_hw_params(capture_handle, hw_params);
snd_pcm_hw_params_free(hw_params); snd_pcm_hw_params_free(hw_params);
snd_pcm_prepare(capture_handle); snd_pcm_prepare(capture_handle);
alsa->aformat.nChannels = channels;
if ((alsa->actual_rate != alsa->target_rate) ||
(alsa->actual_channels != alsa->target_channels))
{
DEBUG_DVC("actual rate %"PRIu32" / channel %"PRIu32" is "
"different from target rate %"PRIu32" / channel %"PRIu32", resampling required.",
alsa->actual_rate, alsa->actual_channels,
alsa->target_rate, alsa->target_channels);
}
return TRUE; return TRUE;
} }
/**
* Function description
*
* @return 0 on success, otherwise a Win32 error code
*/
static UINT audin_alsa_thread_receive(AudinALSADevice* alsa, BYTE* src,
int size)
{
int frames;
int cframes;
UINT ret = CHANNEL_RC_OK;
int encoded_size;
BYTE* encoded_data;
int rbytes_per_frame;
int tbytes_per_frame;
int status;
rbytes_per_frame = alsa->actual_channels * alsa->bytes_per_channel;
tbytes_per_frame = alsa->target_channels * alsa->bytes_per_channel;
if ((alsa->target_rate == alsa->actual_rate) &&
(alsa->target_channels == alsa->actual_channels))
{
frames = size / rbytes_per_frame;
}
else
{
alsa->dsp_context->resample(alsa->dsp_context, src, alsa->bytes_per_channel,
alsa->actual_channels, alsa->actual_rate, size / rbytes_per_frame,
alsa->target_channels, alsa->target_rate);
frames = alsa->dsp_context->resampled_frames;
DEBUG_DVC("resampled %d frames at %"PRIu32" to %d frames at %"PRIu32"",
size / rbytes_per_frame, alsa->actual_rate, frames, alsa->target_rate);
src = alsa->dsp_context->resampled_buffer;
}
while (frames > 0)
{
status = WaitForSingleObject(alsa->stopEvent, 0);
if (status == WAIT_FAILED)
{
ret = GetLastError();
WLog_ERR(TAG, "WaitForSingleObject failed with error %"PRIu32"!", ret);
break;
}
if (status == WAIT_OBJECT_0)
break;
cframes = alsa->frames_per_packet - alsa->buffer_frames;
if (cframes > frames)
cframes = frames;
CopyMemory(alsa->buffer + alsa->buffer_frames * tbytes_per_frame, src,
cframes * tbytes_per_frame);
alsa->buffer_frames += cframes;
if (alsa->buffer_frames >= alsa->frames_per_packet)
{
if (alsa->wformat == WAVE_FORMAT_DVI_ADPCM)
{
if (!alsa->dsp_context->encode_ima_adpcm(alsa->dsp_context,
alsa->buffer, alsa->buffer_frames * tbytes_per_frame,
alsa->target_channels, alsa->block_size))
{
ret = ERROR_INTERNAL_ERROR;
break;
}
encoded_data = alsa->dsp_context->adpcm_buffer;
encoded_size = alsa->dsp_context->adpcm_size;
DEBUG_DVC("encoded %d to %d",
alsa->buffer_frames * tbytes_per_frame, encoded_size);
}
else
{
encoded_data = alsa->buffer;
encoded_size = alsa->buffer_frames * tbytes_per_frame;
}
status = WaitForSingleObject(alsa->stopEvent, 0);
if (status == WAIT_FAILED)
{
ret = GetLastError();
WLog_ERR(TAG, "WaitForSingleObject failed with error %"PRIu32"!", ret);
break;
}
if (status == WAIT_OBJECT_0)
break;
else
{
DEBUG_DVC("encoded %d [%d] to %d [%X]", alsa->buffer_frames,
tbytes_per_frame, encoded_size,
alsa->wformat);
ret = alsa->receive(encoded_data, encoded_size, alsa->user_data);
}
alsa->buffer_frames = 0;
if (ret != CHANNEL_RC_OK)
break;
}
src += cframes * tbytes_per_frame;
frames -= cframes;
}
return ret;
}
static DWORD WINAPI audin_alsa_thread_func(LPVOID arg) static DWORD WINAPI audin_alsa_thread_func(LPVOID arg)
{ {
long error; long error;
BYTE* buffer; BYTE* buffer;
int rbytes_per_frame;
snd_pcm_t* capture_handle = NULL; snd_pcm_t* capture_handle = NULL;
AudinALSADevice* alsa = (AudinALSADevice*) arg; AudinALSADevice* alsa = (AudinALSADevice*) arg;
const size_t rbytes_per_frame = alsa->aformat.nChannels * 4;
DWORD status; DWORD status;
DEBUG_DVC("in"); DEBUG_DVC("in");
rbytes_per_frame = alsa->actual_channels * alsa->bytes_per_channel;
buffer = (BYTE*) calloc(alsa->frames_per_packet, rbytes_per_frame); buffer = (BYTE*) calloc(alsa->frames_per_packet, rbytes_per_frame);
if (!buffer) if (!buffer)
@ -238,8 +132,6 @@ static DWORD WINAPI audin_alsa_thread_func(LPVOID arg)
goto out; goto out;
} }
freerdp_dsp_context_reset_adpcm(alsa->dsp_context);
if ((error = snd_pcm_open(&capture_handle, alsa->device_name, if ((error = snd_pcm_open(&capture_handle, alsa->device_name,
SND_PCM_STREAM_CAPTURE, 0)) < 0) SND_PCM_STREAM_CAPTURE, 0)) < 0)
{ {
@ -281,7 +173,10 @@ static DWORD WINAPI audin_alsa_thread_func(LPVOID arg)
break; break;
} }
if ((error = audin_alsa_thread_receive(alsa, buffer, error * rbytes_per_frame))) error = alsa->receive(&alsa->aformat,
buffer, error, alsa->user_data);
if (error)
{ {
WLog_ERR(TAG, "audin_alsa_thread_receive failed with error %ld", error); WLog_ERR(TAG, "audin_alsa_thread_receive failed with error %ld", error);
break; break;
@ -312,14 +207,13 @@ out:
static UINT audin_alsa_free(IAudinDevice* device) static UINT audin_alsa_free(IAudinDevice* device)
{ {
AudinALSADevice* alsa = (AudinALSADevice*) device; AudinALSADevice* alsa = (AudinALSADevice*) device;
freerdp_dsp_context_free(alsa->dsp_context);
free(alsa->device_name); free(alsa->device_name);
free(alsa); free(alsa);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
static BOOL audin_alsa_format_supported(IAudinDevice* device, static BOOL audin_alsa_format_supported(IAudinDevice* device,
audinFormat* format) const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
@ -334,15 +228,12 @@ static BOOL audin_alsa_format_supported(IAudinDevice* device,
break; break;
case WAVE_FORMAT_DVI_ADPCM: case WAVE_FORMAT_ALAW:
if ((format->nSamplesPerSec <= 48000) && case WAVE_FORMAT_MULAW:
(format->wBitsPerSample == 4) &&
(format->nChannels == 1 || format->nChannels == 2))
{
return TRUE; return TRUE;
}
break; default:
return FALSE;
} }
return FALSE; return FALSE;
@ -353,47 +244,16 @@ static BOOL audin_alsa_format_supported(IAudinDevice* device,
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_alsa_set_format(IAudinDevice* device, audinFormat* format, static UINT audin_alsa_set_format(IAudinDevice* device, const AUDIO_FORMAT* format,
UINT32 FramesPerPacket) UINT32 FramesPerPacket)
{ {
int bs;
AudinALSADevice* alsa = (AudinALSADevice*) device; AudinALSADevice* alsa = (AudinALSADevice*) device;
alsa->target_rate = format->nSamplesPerSec; alsa->aformat = *format;
alsa->actual_rate = format->nSamplesPerSec; alsa->frames_per_packet = FramesPerPacket;
alsa->target_channels = format->nChannels;
alsa->actual_channels = format->nChannels;
switch (format->wFormatTag) if (audin_alsa_format(format->wFormatTag, format->wBitsPerSample) == SND_PCM_FORMAT_UNKNOWN)
{ return ERROR_INTERNAL_ERROR;
case WAVE_FORMAT_PCM:
switch (format->wBitsPerSample)
{
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_DVI_ADPCM:
alsa->format = SND_PCM_FORMAT_S16_LE;
alsa->bytes_per_channel = 2;
bs = (format->nBlockAlign - 4 * format->nChannels) * 4;
alsa->frames_per_packet = (alsa->frames_per_packet * format->nChannels * 2 /
bs + 1) * bs / (format->nChannels * 2);
DEBUG_DVC("aligned FramesPerPacket=%"PRIu32"",
alsa->frames_per_packet);
break;
}
alsa->wformat = format->wFormatTag;
alsa->block_size = format->nBlockAlign;
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -405,20 +265,9 @@ static UINT audin_alsa_set_format(IAudinDevice* device, audinFormat* format,
static UINT audin_alsa_open(IAudinDevice* device, AudinReceive receive, static UINT audin_alsa_open(IAudinDevice* device, AudinReceive receive,
void* user_data) void* user_data)
{ {
int tbytes_per_frame;
AudinALSADevice* alsa = (AudinALSADevice*) device; AudinALSADevice* alsa = (AudinALSADevice*) device;
alsa->receive = receive; alsa->receive = receive;
alsa->user_data = user_data; alsa->user_data = user_data;
tbytes_per_frame = alsa->target_channels * alsa->bytes_per_channel;
alsa->buffer = (BYTE*) calloc(alsa->frames_per_packet, tbytes_per_frame);
if (!alsa->buffer)
{
WLog_ERR(TAG, "calloc failed!");
return ERROR_NOT_ENOUGH_MEMORY;
}
alsa->buffer_frames = 0;
if (!(alsa->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL))) if (!(alsa->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL)))
{ {
@ -435,8 +284,6 @@ static UINT audin_alsa_open(IAudinDevice* device, AudinReceive receive,
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_out: error_out:
free(alsa->buffer);
alsa->buffer = NULL;
CloseHandle(alsa->stopEvent); CloseHandle(alsa->stopEvent);
alsa->stopEvent = NULL; alsa->stopEvent = NULL;
return ERROR_INTERNAL_ERROR; return ERROR_INTERNAL_ERROR;
@ -469,14 +316,12 @@ static UINT audin_alsa_close(IAudinDevice* device)
alsa->thread = NULL; alsa->thread = NULL;
} }
free(alsa->buffer);
alsa->buffer = NULL;
alsa->receive = NULL; alsa->receive = NULL;
alsa->user_data = NULL; alsa->user_data = NULL;
return error; return error;
} }
COMMAND_LINE_ARGUMENT_A audin_alsa_args[] = static COMMAND_LINE_ARGUMENT_A audin_alsa_args[] =
{ {
{ "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "audio device name" }, { "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "audio device name" },
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL } { NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
@ -579,20 +424,10 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS
} }
alsa->frames_per_packet = 128; alsa->frames_per_packet = 128;
alsa->target_rate = 22050; alsa->aformat.nChannels = 2;
alsa->actual_rate = 22050; alsa->aformat.wBitsPerSample = 16;
alsa->format = SND_PCM_FORMAT_S16_LE; alsa->aformat.wFormatTag = WAVE_FORMAT_PCM;
alsa->target_channels = 2; alsa->aformat.nSamplesPerSec = 44100;
alsa->actual_channels = 2;
alsa->bytes_per_channel = 2;
alsa->dsp_context = freerdp_dsp_context_new();
if (!alsa->dsp_context)
{
WLog_ERR(TAG, "freerdp_dsp_context_new failed!");
error = CHANNEL_RC_NO_MEMORY;
goto error_out;
}
if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin, if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin,
(IAudinDevice*) alsa))) (IAudinDevice*) alsa)))
@ -603,7 +438,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_out: error_out:
freerdp_dsp_context_free(alsa->dsp_context);
free(alsa->device_name); free(alsa->device_name);
free(alsa); free(alsa);
return error; return error;

View File

@ -32,11 +32,13 @@
#include <winpr/crt.h> #include <winpr/crt.h>
#include <winpr/cmdline.h> #include <winpr/cmdline.h>
#include <winpr/stream.h>
#include <winpr/wlog.h>
#include <freerdp/addin.h> #include <freerdp/addin.h>
#include <winpr/stream.h>
#include <freerdp/freerdp.h> #include <freerdp/freerdp.h>
#include <freerdp/codec/dsp.h>
#include "audin_main.h" #include "audin_main.h"
#define MSG_SNDIN_VERSION 0x01 #define MSG_SNDIN_VERSION 0x01
@ -70,7 +72,7 @@ struct _AUDIN_CHANNEL_CALLBACK
* be stored as reference when the server sends the format index in * be stored as reference when the server sends the format index in
* Open PDU and Format Change PDU * Open PDU and Format Change PDU
*/ */
audinFormat* formats; AUDIO_FORMAT* formats;
UINT32 formats_count; UINT32 formats_count;
}; };
@ -93,45 +95,66 @@ struct _AUDIN_PLUGIN
rdpContext* rdpcontext; rdpContext* rdpcontext;
BOOL attached; BOOL attached;
wLog* log; wStream* data;
AUDIO_FORMAT* format;
UINT32 FramesPerPacket;
FREERDP_DSP_CONTEXT* dsp_context;
}; };
static BOOL audin_process_addin_args(AUDIN_PLUGIN* audin, ADDIN_ARGV* args); static BOOL audin_process_addin_args(AUDIN_PLUGIN* audin, ADDIN_ARGV* args);
static UINT audin_write_and_free_stream(AUDIN_CHANNEL_CALLBACK* callback, wStream* s) static UINT audin_channel_write_and_free(AUDIN_CHANNEL_CALLBACK* callback, wStream* out,
BOOL freeStream)
{ {
UINT error = ERROR_INTERNAL_ERROR; UINT error;
const size_t length = Stream_GetPosition(s);
const BYTE* data = Stream_Buffer(s);
if (callback && callback->channel && callback->channel->Write) if (!callback || !out)
error = callback->channel->Write(callback->channel, length, data, NULL); return ERROR_INVALID_PARAMETER;
if (!callback->channel || !callback->channel->Write)
return ERROR_INTERNAL_ERROR;
Stream_SealLength(out);
error = callback->channel->Write(callback->channel,
Stream_Length(out),
Stream_Buffer(out), NULL);
if (freeStream)
Stream_Free(out, TRUE);
Stream_Free(s, TRUE);
return error; return error;
} }
/** /**
* Function description * Function description
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_process_version(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback, wStream* s) static UINT audin_process_version(IWTSVirtualChannelCallback* pChannelCallback, wStream* s)
{ {
wStream* out; wStream* out;
UINT32 Version; const UINT32 ClientVersion = 0x01;
Stream_Read_UINT32(s, Version); UINT32 ServerVersion;
DEBUG_DVC("Version=%"PRIu32"", Version); AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
if (Stream_GetRemainingLength(s) < 4)
return ERROR_INVALID_DATA;
Stream_Read_UINT32(s, ServerVersion);
DEBUG_DVC("ServerVersion=%"PRIu32", ClientVersion=%"PRIu32, ServerVersion, ClientVersion);
out = Stream_New(NULL, 5); out = Stream_New(NULL, 5);
if (!out) if (!out)
{ {
WLog_Print(audin->log, WLOG_ERROR, "Stream_New failed!"); WLog_ERR(TAG, "Stream_New failed!");
return ERROR_OUTOFMEMORY; return ERROR_OUTOFMEMORY;
} }
Stream_Write_UINT8(out, MSG_SNDIN_VERSION); Stream_Write_UINT8(out, MSG_SNDIN_VERSION);
Stream_Write_UINT32(out, Version); Stream_Write_UINT32(out, ClientVersion);
return audin_write_and_free_stream(callback, out); return audin_channel_write_and_free(callback, out, TRUE);
} }
/** /**
@ -139,13 +162,11 @@ static UINT audin_process_version(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* c
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_send_incoming_data_pdu(AUDIN_CHANNEL_CALLBACK* callback) static UINT audin_send_incoming_data_pdu(IWTSVirtualChannelCallback* pChannelCallback)
{ {
BYTE out_data[1] = { MSG_SNDIN_DATA_INCOMING }; BYTE out_data[1];
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
if (!callback || !callback->channel || !callback->channel->Write) out_data[0] = MSG_SNDIN_DATA_INCOMING;
return ERROR_INTERNAL_ERROR;
return callback->channel->Write(callback->channel, 1, out_data, NULL); return callback->channel->Write(callback->channel, 1, out_data, NULL);
} }
@ -154,34 +175,35 @@ static UINT audin_send_incoming_data_pdu(AUDIN_CHANNEL_CALLBACK* callback)
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_process_formats(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback, wStream* s) static UINT audin_process_formats(IWTSVirtualChannelCallback* pChannelCallback, wStream* s)
{ {
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
AUDIN_PLUGIN* audin = (AUDIN_PLUGIN*) callback->plugin;
UINT32 i; UINT32 i;
BYTE* fm;
UINT error; UINT error;
wStream* out; wStream* out;
UINT32 NumFormats; UINT32 NumFormats;
audinFormat format; AUDIO_FORMAT format;
size_t cbSizeFormatsPacket; UINT32 cbSizeFormatsPacket;
if (Stream_GetRemainingLength(s) < 8) if (Stream_GetRemainingLength(s) < 8)
return ERROR_NO_DATA; return ERROR_INVALID_DATA;
Stream_Read_UINT32(s, NumFormats); Stream_Read_UINT32(s, NumFormats);
DEBUG_DVC("NumFormats %"PRIu32"", NumFormats); DEBUG_DVC("NumFormats %"PRIu32"", NumFormats);
if ((NumFormats < 1) || (NumFormats > 1000)) if ((NumFormats < 1) || (NumFormats > 1000))
{ {
WLog_Print(audin->log, WLOG_ERROR, "bad NumFormats %"PRIu32"", NumFormats); WLog_ERR(TAG, "bad NumFormats %"PRIu32"", NumFormats);
return ERROR_INVALID_DATA; return ERROR_INVALID_DATA;
} }
Stream_Seek_UINT32(s); /* cbSizeFormatsPacket */ Stream_Seek_UINT32(s); /* cbSizeFormatsPacket */
callback->formats = (audinFormat*) calloc(NumFormats, sizeof(audinFormat)); callback->formats = (AUDIO_FORMAT*) calloc(NumFormats, sizeof(AUDIO_FORMAT));
if (!callback->formats) if (!callback->formats)
{ {
WLog_Print(audin->log, WLOG_ERROR, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
return ERROR_INVALID_DATA; return ERROR_INVALID_DATA;
} }
@ -190,7 +212,7 @@ static UINT audin_process_formats(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* c
if (!out) if (!out)
{ {
error = CHANNEL_RC_NO_MEMORY; error = CHANNEL_RC_NO_MEMORY;
WLog_Print(audin->log, WLOG_ERROR, "Stream_New failed!"); WLog_ERR(TAG, "Stream_New failed!");
goto out; goto out;
} }
@ -199,21 +221,23 @@ static UINT audin_process_formats(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* c
/* SoundFormats (variable) */ /* SoundFormats (variable) */
for (i = 0; i < NumFormats; i++) for (i = 0; i < NumFormats; i++)
{ {
BYTE* fm;
if (Stream_GetRemainingLength(s) < 18) if (Stream_GetRemainingLength(s) < 18)
return ERROR_NO_DATA; return ERROR_INVALID_DATA;
Stream_GetPointer(s, fm); Stream_GetPointer(s, fm);
Stream_Read_UINT16(s, format.wFormatTag); Stream_Read_UINT16(s, format.wFormatTag);
Stream_Read_UINT16(s, format.nChannels); Stream_Read_UINT16(s, format.nChannels);
Stream_Read_UINT32(s, format.nSamplesPerSec); Stream_Read_UINT32(s, format.nSamplesPerSec);
Stream_Seek_UINT32(s); /* nAvgBytesPerSec */ Stream_Read_UINT32(s, format.nAvgBytesPerSec);
Stream_Read_UINT16(s, format.nBlockAlign); Stream_Read_UINT16(s, format.nBlockAlign);
Stream_Read_UINT16(s, format.wBitsPerSample); Stream_Read_UINT16(s, format.wBitsPerSample);
Stream_Read_UINT16(s, format.cbSize); Stream_Read_UINT16(s, format.cbSize);
format.data = Stream_Pointer(s); format.data = Stream_Pointer(s);
if (Stream_GetRemainingLength(s) < format.cbSize) if (Stream_GetRemainingLength(s) < format.cbSize)
return ERROR_NO_DATA; return ERROR_INVALID_DATA;
Stream_Seek(s, format.cbSize); Stream_Seek(s, format.cbSize);
DEBUG_DVC("wFormatTag=%"PRIu16" nChannels=%"PRIu16" nSamplesPerSec=%"PRIu32" " DEBUG_DVC("wFormatTag=%"PRIu16" nChannels=%"PRIu16" nSamplesPerSec=%"PRIu32" "
@ -230,9 +254,9 @@ static UINT audin_process_formats(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* c
if (audin->fixed_rate > 0 && audin->fixed_rate != format.nSamplesPerSec) if (audin->fixed_rate > 0 && audin->fixed_rate != format.nSamplesPerSec)
continue; continue;
if (audin->device && audin->device->FormatSupported(audin->device, &format)) if (freerdp_dsp_supports_format(&format, TRUE) ||
audin->device->FormatSupported(audin->device, &format))
{ {
DEBUG_DVC("format ok");
/* Store the agreed format in the corresponding index */ /* Store the agreed format in the corresponding index */
callback->formats[callback->formats_count++] = format; callback->formats[callback->formats_count++] = format;
@ -240,7 +264,7 @@ static UINT audin_process_formats(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* c
if (!Stream_EnsureRemainingCapacity(out, 18 + format.cbSize)) if (!Stream_EnsureRemainingCapacity(out, 18 + format.cbSize))
{ {
error = CHANNEL_RC_NO_MEMORY; error = CHANNEL_RC_NO_MEMORY;
WLog_Print(audin->log, WLOG_ERROR, "Stream_EnsureRemainingCapacity failed!"); WLog_ERR(TAG, "Stream_EnsureRemainingCapacity failed!");
goto out; goto out;
} }
@ -248,19 +272,19 @@ static UINT audin_process_formats(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* c
} }
} }
if ((error = audin_send_incoming_data_pdu(callback))) if ((error = audin_send_incoming_data_pdu(pChannelCallback)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_send_incoming_data_pdu failed!"); WLog_ERR(TAG, "audin_send_incoming_data_pdu failed!");
goto out; goto out;
} }
cbSizeFormatsPacket = Stream_GetPosition(out); cbSizeFormatsPacket = (UINT32) Stream_GetPosition(out);
Stream_SetPosition(out, 0); Stream_SetPosition(out, 0);
Stream_Write_UINT8(out, MSG_SNDIN_FORMATS); /* Header (1 byte) */ Stream_Write_UINT8(out, MSG_SNDIN_FORMATS); /* Header (1 byte) */
Stream_Write_UINT32(out, callback->formats_count); /* NumFormats (4 bytes) */ Stream_Write_UINT32(out, callback->formats_count); /* NumFormats (4 bytes) */
Stream_Write_UINT32(out, cbSizeFormatsPacket); /* cbSizeFormatsPacket (4 bytes) */ Stream_Write_UINT32(out, cbSizeFormatsPacket); /* cbSizeFormatsPacket (4 bytes) */
Stream_SetPosition(out, cbSizeFormatsPacket); Stream_SetPosition(out, cbSizeFormatsPacket);
error = audin_write_and_free_stream(callback, out); error = audin_channel_write_and_free(callback, out, TRUE);
out: out:
if (error != CHANNEL_RC_OK) if (error != CHANNEL_RC_OK)
@ -277,20 +301,22 @@ out:
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_send_format_change_pdu(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback, static UINT audin_send_format_change_pdu(IWTSVirtualChannelCallback* pChannelCallback,
UINT32 NewFormat) UINT32 NewFormat)
{ {
wStream* out = Stream_New(NULL, 5); wStream* out;
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
out = Stream_New(NULL, 5);
if (!out) if (!out)
{ {
WLog_Print(audin->log, WLOG_ERROR, "Stream_New failed!"); WLog_ERR(TAG, "Stream_New failed!");
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
Stream_Write_UINT8(out, MSG_SNDIN_FORMATCHANGE); Stream_Write_UINT8(out, MSG_SNDIN_FORMATCHANGE);
Stream_Write_UINT32(out, NewFormat); Stream_Write_UINT32(out, NewFormat);
return audin_write_and_free_stream(callback, out); return audin_channel_write_and_free(callback, out, TRUE);
} }
/** /**
@ -298,20 +324,21 @@ static UINT audin_send_format_change_pdu(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALL
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_send_open_reply_pdu(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback, static UINT audin_send_open_reply_pdu(IWTSVirtualChannelCallback* pChannelCallback, UINT32 Result)
UINT32 Result)
{ {
wStream* out = Stream_New(NULL, 5); wStream* out;
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
out = Stream_New(NULL, 5);
if (!out) if (!out)
{ {
WLog_Print(audin->log, WLOG_ERROR, "Stream_New failed!"); WLog_ERR(TAG, "Stream_New failed!");
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
Stream_Write_UINT8(out, MSG_SNDIN_OPEN_REPLY); Stream_Write_UINT8(out, MSG_SNDIN_OPEN_REPLY);
Stream_Write_UINT32(out, Result); Stream_Write_UINT32(out, Result);
return audin_write_and_free_stream(callback, out); return audin_channel_write_and_free(callback, out, TRUE);
} }
/** /**
@ -319,10 +346,10 @@ static UINT audin_send_open_reply_pdu(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBAC
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_receive_wave_data(const BYTE* data, int size, void* user_data) static UINT audin_receive_wave_data(const AUDIO_FORMAT* format,
const BYTE* data, size_t size, void* user_data)
{ {
UINT error; UINT error;
wStream* out;
AUDIN_PLUGIN* audin; AUDIN_PLUGIN* audin;
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) user_data; AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) user_data;
@ -337,84 +364,124 @@ static UINT audin_receive_wave_data(const BYTE* data, int size, void* user_data)
if (!audin->attached) if (!audin->attached)
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
if ((error = audin_send_incoming_data_pdu(callback))) Stream_SetPosition(audin->data, 0);
if (!Stream_EnsureRemainingCapacity(audin->data, 1))
return CHANNEL_RC_NO_MEMORY;
Stream_Write_UINT8(audin->data, MSG_SNDIN_DATA);
if (audin->device->FormatSupported(audin->device, audin->format))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_send_incoming_data_pdu failed!"); if (!Stream_EnsureRemainingCapacity(audin->data, size))
return CHANNEL_RC_NO_MEMORY;
Stream_Write(audin->data, data, size);
}
else if (!freerdp_dsp_encode(audin->dsp_context, format, data, size, audin->data))
return ERROR_INTERNAL_ERROR;
if (Stream_GetPosition(audin->data) <= 1)
return CHANNEL_RC_OK;
if ((error = audin_send_incoming_data_pdu((IWTSVirtualChannelCallback*) callback)))
{
WLog_ERR(TAG, "audin_send_incoming_data_pdu failed!");
return error; return error;
} }
out = Stream_New(NULL, size + 1); return audin_channel_write_and_free(callback, audin->data, FALSE);
}
if (!out) static BOOL audin_open_device(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback)
{ {
WLog_Print(audin->log, WLOG_ERROR, "Stream_New failed!"); UINT error = ERROR_INTERNAL_ERROR;
return ERROR_NOT_ENOUGH_MEMORY; BOOL supported;
AUDIO_FORMAT format;
if (!audin || !audin->device)
return FALSE;
format = *audin->format;
supported = IFCALLRESULT(FALSE, audin->device->FormatSupported, audin->device, &format);
WLog_DBG(TAG, "microphone uses %s codec",
rdpsnd_get_audio_tag_string(format.wFormatTag));
if (!supported)
{
format.wFormatTag = WAVE_FORMAT_PCM;
format.wBitsPerSample = 16;
} }
Stream_Write_UINT8(out, MSG_SNDIN_DATA); IFCALLRET(audin->device->SetFormat, error,
Stream_Write(out, data, size); audin->device, &format,
return audin_write_and_free_stream(callback, out); audin->FramesPerPacket);
if (error != CHANNEL_RC_OK)
{
WLog_ERR(TAG, "SetFormat failed with errorcode %"PRIu32"", error);
return FALSE;
} }
if (!audin->device->FormatSupported(audin->device, audin->format))
{
if (!freerdp_dsp_context_reset(audin->dsp_context, audin->format))
return FALSE;
}
IFCALLRET(audin->device->Open, error, audin->device,
audin_receive_wave_data, callback);
if (error != CHANNEL_RC_OK)
{
WLog_ERR(TAG, "Open failed with errorcode %"PRIu32"", error);
return FALSE;
}
return TRUE;
}
/** /**
* Function description * Function description
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_process_open(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback, wStream* s) static UINT audin_process_open(IWTSVirtualChannelCallback* pChannelCallback, wStream* s)
{ {
audinFormat* format; AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
AUDIN_PLUGIN* audin = (AUDIN_PLUGIN*) callback->plugin;
UINT32 initialFormat; UINT32 initialFormat;
UINT32 FramesPerPacket; UINT32 FramesPerPacket;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
if (!audin || !callback || !s)
return ERROR_INTERNAL_ERROR;
if (Stream_GetRemainingLength(s) < 8) if (Stream_GetRemainingLength(s) < 8)
return ERROR_NO_DATA; return ERROR_INVALID_DATA;
Stream_Read_UINT32(s, FramesPerPacket); Stream_Read_UINT32(s, FramesPerPacket);
Stream_Read_UINT32(s, initialFormat); Stream_Read_UINT32(s, initialFormat);
DEBUG_DVC("FramesPerPacket=%"PRIu32" initialFormat=%"PRIu32"", DEBUG_DVC("FramesPerPacket=%"PRIu32" initialFormat=%"PRIu32"",
FramesPerPacket, initialFormat); FramesPerPacket, initialFormat);
audin->FramesPerPacket = FramesPerPacket;
if (initialFormat >= (UINT32) callback->formats_count) if (initialFormat >= callback->formats_count)
{ {
WLog_Print(audin->log, WLOG_ERROR, "invalid format index %"PRIu32" (total %d)", WLog_ERR(TAG, "invalid format index %"PRIu32" (total %d)",
initialFormat, callback->formats_count); initialFormat, callback->formats_count);
return ERROR_INVALID_DATA; return ERROR_INVALID_DATA;
} }
format = &callback->formats[initialFormat]; audin->format = &callback->formats[initialFormat];
if (audin->device) if (!audin_open_device(audin, callback))
{ return ERROR_INTERNAL_ERROR;
IFCALLRET(audin->device->SetFormat, error, audin->device, format, FramesPerPacket);
if (error != CHANNEL_RC_OK) if ((error = audin_send_format_change_pdu(pChannelCallback, initialFormat)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "SetFormat failed with errorcode %"PRIu32"", error); WLog_ERR(TAG, "audin_send_format_change_pdu failed!");
return error; return error;
} }
IFCALLRET(audin->device->Open, error, audin->device, audin_receive_wave_data, callback); if ((error = audin_send_open_reply_pdu(pChannelCallback, 0)))
WLog_ERR(TAG, "audin_send_open_reply_pdu failed!");
if (error != CHANNEL_RC_OK)
{
WLog_Print(audin->log, WLOG_ERROR, "Open failed with errorcode %"PRIu32"", error);
return error;
}
}
if ((error = audin_send_format_change_pdu(audin, callback, initialFormat)))
{
WLog_Print(audin->log, WLOG_ERROR, "audin_send_format_change_pdu failed!");
return error;
}
if ((error = audin_send_open_reply_pdu(audin, callback, 0)))
WLog_Print(audin->log, WLOG_ERROR, "audin_send_open_reply_pdu failed!");
return error; return error;
} }
@ -424,30 +491,27 @@ static UINT audin_process_open(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* call
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_process_format_change(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLBACK* callback, static UINT audin_process_format_change(IWTSVirtualChannelCallback* pChannelCallback, wStream* s)
wStream* s)
{ {
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
AUDIN_PLUGIN* audin = (AUDIN_PLUGIN*) callback->plugin;
UINT32 NewFormat; UINT32 NewFormat;
audinFormat* format;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
if (!audin || !callback || !s)
return ERROR_INTERNAL_ERROR;
if (Stream_GetRemainingLength(s) < 4) if (Stream_GetRemainingLength(s) < 4)
return ERROR_NO_DATA; return ERROR_INVALID_DATA;
Stream_Read_UINT32(s, NewFormat); Stream_Read_UINT32(s, NewFormat);
DEBUG_DVC("NewFormat=%"PRIu32"", NewFormat); DEBUG_DVC("NewFormat=%"PRIu32"", NewFormat);
if (NewFormat >= callback->formats_count) if (NewFormat >= (UINT32) callback->formats_count)
{ {
WLog_Print(audin->log, WLOG_ERROR, "invalid format index %"PRIu32" (total %d)", WLog_ERR(TAG, "invalid format index %"PRIu32" (total %d)",
NewFormat, callback->formats_count); NewFormat, callback->formats_count);
return ERROR_INVALID_DATA; return ERROR_INVALID_DATA;
} }
format = &callback->formats[NewFormat]; audin->format = &callback->formats[NewFormat];
if (audin->device) if (audin->device)
{ {
@ -455,29 +519,16 @@ static UINT audin_process_format_change(AUDIN_PLUGIN* audin, AUDIN_CHANNEL_CALLB
if (error != CHANNEL_RC_OK) if (error != CHANNEL_RC_OK)
{ {
WLog_Print(audin->log, WLOG_ERROR, "Close failed with errorcode %"PRIu32"", error); WLog_ERR(TAG, "Close failed with errorcode %"PRIu32"", error);
return error;
}
IFCALLRET(audin->device->SetFormat, error, audin->device, format, 0);
if (error != CHANNEL_RC_OK)
{
WLog_Print(audin->log, WLOG_ERROR, "SetFormat failed with errorcode %"PRIu32"", error);
return error;
}
IFCALLRET(audin->device->Open, error, audin->device, audin_receive_wave_data, callback);
if (error != CHANNEL_RC_OK)
{
WLog_Print(audin->log, WLOG_ERROR, "Open failed with errorcode %"PRIu32"", error);
return error; return error;
} }
} }
if ((error = audin_send_format_change_pdu(audin, callback, NewFormat))) if (!audin_open_device(audin, callback))
WLog_Print(audin->log, WLOG_ERROR, "audin_send_format_change_pdu failed!"); return ERROR_INTERNAL_ERROR;
if ((error = audin_send_format_change_pdu(pChannelCallback, NewFormat)))
WLog_ERR(TAG, "audin_send_format_change_pdu failed!");
return error; return error;
} }
@ -491,19 +542,9 @@ static UINT audin_on_data_received(IWTSVirtualChannelCallback* pChannelCallback,
{ {
UINT error; UINT error;
BYTE MessageId; BYTE MessageId;
AUDIN_PLUGIN* audin;
AUDIN_CHANNEL_CALLBACK* callback = (AUDIN_CHANNEL_CALLBACK*) pChannelCallback;
if (!callback || !data) if (Stream_GetRemainingLength(data) < 1)
return ERROR_INVALID_PARAMETER; return ERROR_INVALID_DATA;
audin = (AUDIN_PLUGIN*) callback->plugin;
if (!audin)
return ERROR_INTERNAL_ERROR;
if (Stream_GetRemainingCapacity(data) < 1)
return ERROR_NO_DATA;
Stream_Read_UINT8(data, MessageId); Stream_Read_UINT8(data, MessageId);
DEBUG_DVC("MessageId=0x%02"PRIx8"", MessageId); DEBUG_DVC("MessageId=0x%02"PRIx8"", MessageId);
@ -511,23 +552,23 @@ static UINT audin_on_data_received(IWTSVirtualChannelCallback* pChannelCallback,
switch (MessageId) switch (MessageId)
{ {
case MSG_SNDIN_VERSION: case MSG_SNDIN_VERSION:
error = audin_process_version(audin, callback, data); error = audin_process_version(pChannelCallback, data);
break; break;
case MSG_SNDIN_FORMATS: case MSG_SNDIN_FORMATS:
error = audin_process_formats(audin, callback, data); error = audin_process_formats(pChannelCallback, data);
break; break;
case MSG_SNDIN_OPEN: case MSG_SNDIN_OPEN:
error = audin_process_open(audin, callback, data); error = audin_process_open(pChannelCallback, data);
break; break;
case MSG_SNDIN_FORMATCHANGE: case MSG_SNDIN_FORMATCHANGE:
error = audin_process_format_change(audin, callback, data); error = audin_process_format_change(pChannelCallback, data);
break; break;
default: default:
WLog_Print(audin->log, WLOG_ERROR, "unknown MessageId=0x%02"PRIx8"", MessageId); WLog_ERR(TAG, "unknown MessageId=0x%02"PRIx8"", MessageId);
error = ERROR_INVALID_DATA; error = ERROR_INVALID_DATA;
break; break;
} }
@ -552,9 +593,10 @@ static UINT audin_on_close(IWTSVirtualChannelCallback* pChannelCallback)
IFCALLRET(audin->device->Close, error, audin->device); IFCALLRET(audin->device->Close, error, audin->device);
if (error != CHANNEL_RC_OK) if (error != CHANNEL_RC_OK)
WLog_Print(audin->log, WLOG_ERROR, "Close failed with errorcode %"PRIu32"", error); WLog_ERR(TAG, "Close failed with errorcode %"PRIu32"", error);
} }
audin->format = NULL;
free(callback->formats); free(callback->formats);
free(callback); free(callback);
return error; return error;
@ -570,19 +612,13 @@ static UINT audin_on_new_channel_connection(IWTSListenerCallback* pListenerCallb
IWTSVirtualChannelCallback** ppCallback) IWTSVirtualChannelCallback** ppCallback)
{ {
AUDIN_CHANNEL_CALLBACK* callback; AUDIN_CHANNEL_CALLBACK* callback;
AUDIN_PLUGIN* audin;
AUDIN_LISTENER_CALLBACK* listener_callback = (AUDIN_LISTENER_CALLBACK*) pListenerCallback; AUDIN_LISTENER_CALLBACK* listener_callback = (AUDIN_LISTENER_CALLBACK*) pListenerCallback;
if (!listener_callback || !listener_callback->plugin)
return ERROR_INTERNAL_ERROR;
audin = (AUDIN_PLUGIN*) listener_callback->plugin;
DEBUG_DVC("..."); DEBUG_DVC("...");
callback = (AUDIN_CHANNEL_CALLBACK*) calloc(1, sizeof(AUDIN_CHANNEL_CALLBACK)); callback = (AUDIN_CHANNEL_CALLBACK*) calloc(1, sizeof(AUDIN_CHANNEL_CALLBACK));
if (!callback) if (!callback)
{ {
WLog_Print(audin->log, WLOG_ERROR, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
@ -608,7 +644,7 @@ static UINT audin_plugin_initialize(IWTSPlugin* pPlugin, IWTSVirtualChannelManag
if (!audin->listener_callback) if (!audin->listener_callback)
{ {
WLog_Print(audin->log, WLOG_ERROR, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
@ -636,17 +672,17 @@ static UINT audin_plugin_terminated(IWTSPlugin* pPlugin)
if (error != CHANNEL_RC_OK) if (error != CHANNEL_RC_OK)
{ {
WLog_Print(audin->log, WLOG_ERROR, "Free failed with errorcode %"PRIu32"", error); WLog_ERR(TAG, "Free failed with errorcode %"PRIu32"", error);
// dont stop on error // dont stop on error
} }
audin->device = NULL; audin->device = NULL;
} }
freerdp_dsp_context_free(audin->dsp_context);
Stream_Free(audin->data, TRUE);
free(audin->subsystem); free(audin->subsystem);
audin->subsystem = NULL;
free(audin->device_name); free(audin->device_name);
audin->device_name = NULL;
free(audin->listener_callback); free(audin->listener_callback);
free(audin); free(audin);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
@ -687,7 +723,7 @@ static UINT audin_register_device_plugin(IWTSPlugin* pPlugin, IAudinDevice* devi
if (audin->device) if (audin->device)
{ {
WLog_Print(audin->log, WLOG_ERROR, "existing device, abort."); WLog_ERR(TAG, "existing device, abort.");
return ERROR_ALREADY_EXISTS; return ERROR_ALREADY_EXISTS;
} }
@ -701,34 +737,33 @@ static UINT audin_register_device_plugin(IWTSPlugin* pPlugin, IAudinDevice* devi
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_load_device_plugin(AUDIN_PLUGIN* audin, char* name, ADDIN_ARGV* args) static UINT audin_load_device_plugin(IWTSPlugin* pPlugin, const char* name, ADDIN_ARGV* args)
{ {
PFREERDP_AUDIN_DEVICE_ENTRY entry; PFREERDP_AUDIN_DEVICE_ENTRY entry;
FREERDP_AUDIN_DEVICE_ENTRY_POINTS entryPoints; FREERDP_AUDIN_DEVICE_ENTRY_POINTS entryPoints;
UINT error; UINT error;
entry = (PFREERDP_AUDIN_DEVICE_ENTRY) freerdp_load_channel_addin_entry("audin", name, NULL, entry = (PFREERDP_AUDIN_DEVICE_ENTRY) freerdp_load_channel_addin_entry("audin", (LPSTR) name, NULL,
0); 0);
if (entry == NULL) if (entry == NULL)
{ {
WLog_Print(audin->log, WLOG_ERROR, WLog_ERR(TAG, "freerdp_load_channel_addin_entry did not return any function pointers for %s ",
"freerdp_load_channel_addin_entry did not return any function pointers for %s ",
name); name);
return ERROR_INVALID_FUNCTION; return ERROR_INVALID_FUNCTION;
} }
entryPoints.plugin = (IWTSPlugin*) audin; entryPoints.plugin = pPlugin;
entryPoints.pRegisterAudinDevice = audin_register_device_plugin; entryPoints.pRegisterAudinDevice = audin_register_device_plugin;
entryPoints.args = args; entryPoints.args = args;
entryPoints.rdpcontext = audin->rdpcontext; entryPoints.rdpcontext = ((AUDIN_PLUGIN*)pPlugin)->rdpcontext;
if ((error = entry(&entryPoints))) if ((error = entry(&entryPoints)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "%s entry returned error %"PRIu32".", name, error); WLog_ERR(TAG, "%s entry returned error %"PRIu32".", name, error);
return error; return error;
} }
WLog_Print(audin->log, WLOG_INFO, "Loaded %s backend for audin", name); WLog_INFO(TAG, "Loaded %s backend for audin", name);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -744,7 +779,7 @@ static UINT audin_set_subsystem(AUDIN_PLUGIN* audin, const char* subsystem)
if (!audin->subsystem) if (!audin->subsystem)
{ {
WLog_Print(audin->log, WLOG_ERROR, "_strdup failed!"); WLog_ERR(TAG, "_strdup failed!");
return ERROR_NOT_ENOUGH_MEMORY; return ERROR_NOT_ENOUGH_MEMORY;
} }
@ -763,7 +798,7 @@ static UINT audin_set_device_name(AUDIN_PLUGIN* audin, char* device_name)
if (!audin->device_name) if (!audin->device_name)
{ {
WLog_Print(audin->log, WLOG_ERROR, "_strdup failed!"); WLog_ERR(TAG, "_strdup failed!");
return ERROR_NOT_ENOUGH_MEMORY; return ERROR_NOT_ENOUGH_MEMORY;
} }
@ -810,7 +845,7 @@ BOOL audin_process_addin_args(AUDIN_PLUGIN* audin, ADDIN_ARGV* args)
{ {
if ((error = audin_set_subsystem(audin, arg->Value))) if ((error = audin_set_subsystem(audin, arg->Value)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_set_subsystem failed with error %"PRIu32"!", error); WLog_ERR(TAG, "audin_set_subsystem failed with error %"PRIu32"!", error);
return FALSE; return FALSE;
} }
} }
@ -818,7 +853,7 @@ BOOL audin_process_addin_args(AUDIN_PLUGIN* audin, ADDIN_ARGV* args)
{ {
if ((error = audin_set_device_name(audin, arg->Value))) if ((error = audin_set_device_name(audin, arg->Value)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_set_device_name failed with error %"PRIu32"!", error); WLog_ERR(TAG, "audin_set_device_name failed with error %"PRIu32"!", error);
return FALSE; return FALSE;
} }
} }
@ -916,7 +951,16 @@ UINT DVCPluginEntry(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
audin->log = WLog_Get(TAG); audin->data = Stream_New(NULL, 4096);
if (!audin->data)
goto out;
audin->dsp_context = freerdp_dsp_context_new(TRUE);
if (!audin->dsp_context)
goto out;
audin->attached = TRUE; audin->attached = TRUE;
audin->iface.Initialize = audin_plugin_initialize; audin->iface.Initialize = audin_plugin_initialize;
audin->iface.Connected = NULL; audin->iface.Connected = NULL;
@ -936,9 +980,9 @@ UINT DVCPluginEntry(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
if (audin->subsystem) if (audin->subsystem)
{ {
if ((error = audin_load_device_plugin(audin, audin->subsystem, args))) if ((error = audin_load_device_plugin((IWTSPlugin*) audin, audin->subsystem, args)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_load_device_plugin %s failed with error %"PRIu32"!", WLog_ERR(TAG, "audin_load_device_plugin %s failed with error %"PRIu32"!",
audin->subsystem, error); audin->subsystem, error);
goto out; goto out;
} }
@ -949,17 +993,17 @@ UINT DVCPluginEntry(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
{ {
if ((error = audin_set_subsystem(audin, entry->subsystem))) if ((error = audin_set_subsystem(audin, entry->subsystem)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_set_subsystem for %s failed with error %"PRIu32"!", WLog_ERR(TAG, "audin_set_subsystem for %s failed with error %"PRIu32"!",
entry->subsystem, error); entry->subsystem, error);
} }
else if ((error = audin_set_device_name(audin, entry->device))) else if ((error = audin_set_device_name(audin, entry->device)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_set_device_name for %s failed with error %"PRIu32"!", WLog_ERR(TAG, "audin_set_device_name for %s failed with error %"PRIu32"!",
entry->subsystem, error); entry->subsystem, error);
} }
else if ((error = audin_load_device_plugin(audin, audin->subsystem, args))) else if ((error = audin_load_device_plugin((IWTSPlugin*) audin, audin->subsystem, args)))
{ {
WLog_Print(audin->log, WLOG_ERROR, "audin_load_device_plugin %s failed with error %"PRIu32"!", WLog_ERR(TAG, "audin_load_device_plugin %s failed with error %"PRIu32"!",
entry->subsystem, error); entry->subsystem, error);
} }
@ -968,7 +1012,7 @@ UINT DVCPluginEntry(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
} }
if (audin->device == NULL) if (audin->device == NULL)
WLog_Print(audin->log, WLOG_ERROR, "no sound device."); WLog_ERR(TAG, "no sound device.");
error = pEntryPoints->RegisterPlugin(pEntryPoints, "audin", (IWTSPlugin*) audin); error = pEntryPoints->RegisterPlugin(pEntryPoints, "audin", (IWTSPlugin*) audin);
out: out:

View File

@ -42,7 +42,6 @@
#include <AudioToolbox/AudioQueue.h> #include <AudioToolbox/AudioQueue.h>
#include <freerdp/addin.h> #include <freerdp/addin.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/rdpsnd.h> #include <freerdp/channels/rdpsnd.h>
#include "audin_main.h" #include "audin_main.h"
@ -50,13 +49,22 @@
#define MAC_AUDIO_QUEUE_NUM_BUFFERS 100 #define MAC_AUDIO_QUEUE_NUM_BUFFERS 100
#define MAC_AUDIO_QUEUE_BUFFER_SIZE 32768 #define MAC_AUDIO_QUEUE_BUFFER_SIZE 32768
/* Fix for #4462: Provide type alias if not declared (Mac OS < 10.10)
* https://developer.apple.com/documentation/coreaudio/audioformatid
*/
#ifndef AudioFormatID
typedef UInt32 AudioFormatID;
#endif
#ifndef AudioFormatFlags
typedef UInt32 AudioFormatFlags;
#endif
typedef struct _AudinMacDevice typedef struct _AudinMacDevice
{ {
IAudinDevice iface; IAudinDevice iface;
FREERDP_DSP_CONTEXT* dsp_context; AUDIO_FORMAT format;
audinFormat format;
UINT32 FramesPerPacket; UINT32 FramesPerPacket;
int dev_unit; int dev_unit;
@ -71,7 +79,7 @@ typedef struct _AudinMacDevice
AudioQueueBufferRef audioBuffers[MAC_AUDIO_QUEUE_NUM_BUFFERS]; AudioQueueBufferRef audioBuffers[MAC_AUDIO_QUEUE_NUM_BUFFERS];
} AudinMacDevice; } AudinMacDevice;
static AudioFormatID audin_mac_get_format(const audinFormat* format) static AudioFormatID audin_mac_get_format(const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
@ -95,7 +103,7 @@ static AudioFormatID audin_mac_get_format(const audinFormat* format)
return 0; return 0;
} }
static AudioFormatFlags audin_mac_get_flags_for_format(const audinFormat* format) static AudioFormatFlags audin_mac_get_flags_for_format(const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
@ -103,12 +111,13 @@ static AudioFormatFlags audin_mac_get_flags_for_format(const audinFormat* format
case WAVE_FORMAT_ADPCM: case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_PCM: case WAVE_FORMAT_PCM:
return kAudioFormatFlagIsSignedInteger; return kAudioFormatFlagIsSignedInteger;
default: default:
return 0; return 0;
} }
} }
static BOOL audin_mac_format_supported(IAudinDevice* device, audinFormat* format) static BOOL audin_mac_format_supported(IAudinDevice* device, const AUDIO_FORMAT* format)
{ {
AudioFormatID req_fmt = 0; AudioFormatID req_fmt = 0;
@ -128,7 +137,8 @@ static BOOL audin_mac_format_supported(IAudinDevice* device, audinFormat* format
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_mac_set_format(IAudinDevice* device, audinFormat* format, UINT32 FramesPerPacket) static UINT audin_mac_set_format(IAudinDevice* device, const AUDIO_FORMAT* format,
UINT32 FramesPerPacket)
{ {
AudinMacDevice* mac = (AudinMacDevice*)device; AudinMacDevice* mac = (AudinMacDevice*)device;
@ -136,8 +146,7 @@ static UINT audin_mac_set_format(IAudinDevice* device, audinFormat* format, UINT
return ERROR_INVALID_PARAMETER; return ERROR_INVALID_PARAMETER;
mac->FramesPerPacket = FramesPerPacket; mac->FramesPerPacket = FramesPerPacket;
CopyMemory(&(mac->format), format, sizeof(audinFormat)); mac->format = *format;
WLog_INFO(TAG, "Audio Format %s [channels=%d, samples=%d, bits=%d]", WLog_INFO(TAG, "Audio Format %s [channels=%d, samples=%d, bits=%d]",
rdpsnd_get_audio_tag_string(format->wFormatTag), rdpsnd_get_audio_tag_string(format->wFormatTag),
format->nChannels, format->nSamplesPerSec, format->wBitsPerSample); format->nChannels, format->nSamplesPerSec, format->wBitsPerSample);
@ -161,7 +170,6 @@ static UINT audin_mac_set_format(IAudinDevice* device, audinFormat* format, UINT
mac->audioFormat.mChannelsPerFrame * mac->audioFormat.mBitsPerChannel / 8; mac->audioFormat.mChannelsPerFrame * mac->audioFormat.mBitsPerChannel / 8;
mac->audioFormat.mBytesPerPacket = mac->audioFormat.mBytesPerPacket =
mac->audioFormat.mBytesPerFrame * mac->audioFormat.mFramesPerPacket; mac->audioFormat.mBytesPerFrame * mac->audioFormat.mFramesPerPacket;
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -174,52 +182,19 @@ static void mac_audio_queue_input_cb(void *aqData,
{ {
AudinMacDevice* mac = (AudinMacDevice*)aqData; AudinMacDevice* mac = (AudinMacDevice*)aqData;
UINT error; UINT error;
int encoded_size = 0; const BYTE* buffer = inBuffer->mAudioData;
const BYTE *encoded_data;
BYTE *buffer = inBuffer->mAudioData;
int buffer_size = inBuffer->mAudioDataByteSize; int buffer_size = inBuffer->mAudioDataByteSize;
(void)inAQ; (void)inAQ;
(void)inStartTime; (void)inStartTime;
(void)inNumPackets; (void)inNumPackets;
(void)inPacketDesc; (void)inPacketDesc;
if ((error = mac->receive(&mac->format, buffer, buffer_size, mac->user_data)))
/* Process. */
switch (mac->format.wFormatTag) {
case WAVE_FORMAT_ADPCM:
if (!mac->dsp_context->encode_ms_adpcm(mac->dsp_context,
buffer, buffer_size, mac->format.nChannels, mac->format.nBlockAlign))
{
SetLastError(ERROR_INTERNAL_ERROR);
return;
}
encoded_data = mac->dsp_context->adpcm_buffer;
encoded_size = mac->dsp_context->adpcm_size;
break;
case WAVE_FORMAT_DVI_ADPCM:
if (!mac->dsp_context->encode_ima_adpcm(mac->dsp_context,
buffer, buffer_size, mac->format.nChannels, mac->format.nBlockAlign))
{
SetLastError(ERROR_INTERNAL_ERROR);
break;
}
encoded_data = mac->dsp_context->adpcm_buffer;
encoded_size = mac->dsp_context->adpcm_size;
break;
default:
encoded_data = buffer;
encoded_size = buffer_size;
break;
}
if ((error = mac->receive(encoded_data, encoded_size, mac->user_data)))
{ {
WLog_ERR(TAG, "mac->receive failed with error %"PRIu32"", error); WLog_ERR(TAG, "mac->receive failed with error %"PRIu32"", error);
SetLastError(ERROR_INTERNAL_ERROR); SetLastError(ERROR_INTERNAL_ERROR);
return; return;
} }
} }
static UINT audin_mac_close(IAudinDevice* device) static UINT audin_mac_close(IAudinDevice* device)
@ -235,18 +210,21 @@ static UINT audin_mac_close(IAudinDevice *device)
if (mac->isOpen) if (mac->isOpen)
{ {
devStat = AudioQueueStop(mac->audioQueue, true); devStat = AudioQueueStop(mac->audioQueue, true);
if (devStat != 0) if (devStat != 0)
{ {
errCode = GetLastError(); errCode = GetLastError();
WLog_ERR(TAG, "AudioQueueStop failed with %s [%"PRIu32"]", WLog_ERR(TAG, "AudioQueueStop failed with %s [%"PRIu32"]",
winpr_strerror(errCode, errString, sizeof(errString)), errCode); winpr_strerror(errCode, errString, sizeof(errString)), errCode);
} }
mac->isOpen = false; mac->isOpen = false;
} }
if (mac->audioQueue) if (mac->audioQueue)
{ {
devStat = AudioQueueDispose(mac->audioQueue, true); devStat = AudioQueueDispose(mac->audioQueue, true);
if (devStat != 0) if (devStat != 0)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -259,22 +237,21 @@ static UINT audin_mac_close(IAudinDevice *device)
mac->receive = NULL; mac->receive = NULL;
mac->user_data = NULL; mac->user_data = NULL;
return errCode; return errCode;
} }
static UINT audin_mac_open(IAudinDevice *device, AudinReceive receive, void *user_data) { static UINT audin_mac_open(IAudinDevice* device, AudinReceive receive, void* user_data)
{
AudinMacDevice* mac = (AudinMacDevice*)device; AudinMacDevice* mac = (AudinMacDevice*)device;
DWORD errCode; DWORD errCode;
char errString[1024]; char errString[1024];
OSStatus devStat; OSStatus devStat;
size_t index; size_t index;
mac->receive = receive; mac->receive = receive;
mac->user_data = user_data; mac->user_data = user_data;
devStat = AudioQueueNewInput(&(mac->audioFormat), mac_audio_queue_input_cb, devStat = AudioQueueNewInput(&(mac->audioFormat), mac_audio_queue_input_cb,
mac, NULL, kCFRunLoopCommonModes, 0, &(mac->audioQueue)); mac, NULL, kCFRunLoopCommonModes, 0, &(mac->audioQueue));
if (devStat != 0) if (devStat != 0)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -287,6 +264,7 @@ static UINT audin_mac_open(IAudinDevice *device, AudinReceive receive, void *use
{ {
devStat = AudioQueueAllocateBuffer(mac->audioQueue, MAC_AUDIO_QUEUE_BUFFER_SIZE, devStat = AudioQueueAllocateBuffer(mac->audioQueue, MAC_AUDIO_QUEUE_BUFFER_SIZE,
&mac->audioBuffers[index]); &mac->audioBuffers[index]);
if (devStat != 0) if (devStat != 0)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -299,6 +277,7 @@ static UINT audin_mac_open(IAudinDevice *device, AudinReceive receive, void *use
mac->audioBuffers[index], mac->audioBuffers[index],
0, 0,
NULL); NULL);
if (devStat != 0) if (devStat != 0)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -308,9 +287,8 @@ static UINT audin_mac_open(IAudinDevice *device, AudinReceive receive, void *use
} }
} }
freerdp_dsp_context_reset_adpcm(mac->dsp_context);
devStat = AudioQueueStart(mac->audioQueue, NULL); devStat = AudioQueueStart(mac->audioQueue, NULL);
if (devStat != 0) if (devStat != 0)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -318,10 +296,9 @@ static UINT audin_mac_open(IAudinDevice *device, AudinReceive receive, void *use
winpr_strerror(errCode, errString, sizeof(errString)), errCode); winpr_strerror(errCode, errString, sizeof(errString)), errCode);
goto err_out; goto err_out;
} }
mac->isOpen = true; mac->isOpen = true;
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
err_out: err_out:
audin_mac_close(device); audin_mac_close(device);
return CHANNEL_RC_INITIALIZATION_ERROR; return CHANNEL_RC_INITIALIZATION_ERROR;
@ -330,7 +307,6 @@ err_out:
static UINT audin_mac_free(IAudinDevice* device) static UINT audin_mac_free(IAudinDevice* device)
{ {
AudinMacDevice* mac = (AudinMacDevice*)device; AudinMacDevice* mac = (AudinMacDevice*)device;
int error; int error;
if (device == NULL) if (device == NULL)
@ -340,9 +316,8 @@ static UINT audin_mac_free(IAudinDevice* device)
{ {
WLog_ERR(TAG, "audin_oss_close failed with error code %d!", error); WLog_ERR(TAG, "audin_oss_close failed with error code %d!", error);
} }
freerdp_dsp_context_free(mac->dsp_context);
free(mac);
free(mac);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -366,7 +341,8 @@ static UINT audin_mac_parse_addin_args(AudinMacDevice *device, ADDIN_ARGV *args)
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD; flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
status = CommandLineParseArgumentsA(args->argc, (const char**)args->argv, audin_mac_args, flags, mac, NULL, NULL); status = CommandLineParseArgumentsA(args->argc, (const char**)args->argv, audin_mac_args, flags,
mac, NULL, NULL);
if (status < 0) if (status < 0)
return ERROR_INVALID_PARAMETER; return ERROR_INVALID_PARAMETER;
@ -382,6 +358,7 @@ static UINT audin_mac_parse_addin_args(AudinMacDevice *device, ADDIN_ARGV *args)
CommandLineSwitchCase(arg, "dev") CommandLineSwitchCase(arg, "dev")
{ {
str_num = _strdup(arg->Value); str_num = _strdup(arg->Value);
if (!str_num) if (!str_num)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -389,6 +366,7 @@ static UINT audin_mac_parse_addin_args(AudinMacDevice *device, ADDIN_ARGV *args)
winpr_strerror(errCode, errString, sizeof(errString)), errCode); winpr_strerror(errCode, errString, sizeof(errString)), errCode);
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
mac->dev_unit = strtol(str_num, &eptr, 10); mac->dev_unit = strtol(str_num, &eptr, 10);
if (mac->dev_unit < 0 || *eptr != '\0') if (mac->dev_unit < 0 || *eptr != '\0')
@ -416,8 +394,8 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
ADDIN_ARGV* args; ADDIN_ARGV* args;
AudinMacDevice* mac; AudinMacDevice* mac;
UINT error; UINT error;
mac = (AudinMacDevice*)calloc(1, sizeof(AudinMacDevice)); mac = (AudinMacDevice*)calloc(1, sizeof(AudinMacDevice));
if (!mac) if (!mac)
{ {
errCode = GetLastError(); errCode = GetLastError();
@ -432,7 +410,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
mac->iface.Close = audin_mac_close; mac->iface.Close = audin_mac_close;
mac->iface.Free = audin_mac_free; mac->iface.Free = audin_mac_free;
mac->rdpcontext = pEntryPoints->rdpcontext; mac->rdpcontext = pEntryPoints->rdpcontext;
mac->dev_unit = -1; mac->dev_unit = -1;
args = pEntryPoints->args; args = pEntryPoints->args;
@ -442,14 +419,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
goto error_out; goto error_out;
} }
mac->dsp_context = freerdp_dsp_context_new();
if (!mac->dsp_context)
{
WLog_ERR(TAG, "freerdp_dsp_context_new failed!");
error = CHANNEL_RC_NO_MEMORY;
goto error_out;
}
if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin, (IAudinDevice*) mac))) if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin, (IAudinDevice*) mac)))
{ {
WLog_ERR(TAG, "RegisterAudinDevice failed with error %"PRIu32"!", error); WLog_ERR(TAG, "RegisterAudinDevice failed with error %"PRIu32"!", error);
@ -457,10 +426,7 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
} }
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_out: error_out:
freerdp_dsp_context_free(mac->dsp_context);
free(mac); free(mac);
return error; return error;
} }

View File

@ -34,7 +34,6 @@
#include <winpr/cmdline.h> #include <winpr/cmdline.h>
#include <freerdp/addin.h> #include <freerdp/addin.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/rdpsnd.h> #include <freerdp/channels/rdpsnd.h>
#include <SLES/OpenSLES.h> #include <SLES/OpenSLES.h>
@ -50,16 +49,11 @@ typedef struct _AudinOpenSLESDevice
char* device_name; char* device_name;
OPENSL_STREAM* stream; OPENSL_STREAM* stream;
AUDIO_FORMAT format;
UINT32 frames_per_packet; UINT32 frames_per_packet;
UINT32 rate;
UINT32 channels;
UINT32 bytes_per_channel; UINT32 bytes_per_channel;
UINT32 format;
UINT32 block_size;
FREERDP_DSP_CONTEXT* dsp_context;
AudinReceive receive; AudinReceive receive;
HANDLE thread; HANDLE thread;
@ -83,30 +77,27 @@ static DWORD WINAPI audin_opensles_thread_func(LPVOID arg)
int rc = CHANNEL_RC_OK; int rc = CHANNEL_RC_OK;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
DWORD status; DWORD status;
DEBUG_DVC("opensles=%p", (void*) opensles); DEBUG_DVC("opensles=%p", (void*) opensles);
assert(opensles); assert(opensles);
assert(opensles->frames_per_packet > 0); assert(opensles->frames_per_packet > 0);
assert(opensles->dsp_context);
assert(opensles->stopEvent); assert(opensles->stopEvent);
assert(opensles->stream); assert(opensles->stream);
buffer.v = calloc(1, raw_size); buffer.v = calloc(1, raw_size);
if (!buffer.v) if (!buffer.v)
{ {
error = CHANNEL_RC_NO_MEMORY; error = CHANNEL_RC_NO_MEMORY;
WLog_ERR(TAG, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
if (opensles->rdpcontext) if (opensles->rdpcontext)
setChannelError(opensles->rdpcontext, CHANNEL_RC_NO_MEMORY, "audin_opensles_thread_func reported an error"); setChannelError(opensles->rdpcontext, CHANNEL_RC_NO_MEMORY,
"audin_opensles_thread_func reported an error");
goto out; goto out;
} }
freerdp_dsp_context_reset_adpcm(opensles->dsp_context);
while (1) while (1)
{ {
status = WaitForSingleObject(opensles->stopEvent, 0); status = WaitForSingleObject(opensles->stopEvent, 0);
if (status == WAIT_FAILED) if (status == WAIT_FAILED)
@ -119,49 +110,17 @@ static DWORD WINAPI audin_opensles_thread_func(LPVOID arg)
if (status == WAIT_OBJECT_0) if (status == WAIT_OBJECT_0)
break; break;
size_t encoded_size;
void *encoded_data;
rc = android_RecIn(opensles->stream, buffer.s, raw_size); rc = android_RecIn(opensles->stream, buffer.s, raw_size);
if (rc < 0) if (rc < 0)
{ {
WLog_ERR(TAG, "android_RecIn %d", rc); WLog_ERR(TAG, "android_RecIn %d", rc);
continue; continue;
} }
assert(rc == raw_size); error = opensles->receive(&opensles->format,
if (opensles->format == WAVE_FORMAT_ADPCM) buffer.v, raw_size, opensles->user_data);
{
if (!opensles->dsp_context->encode_ms_adpcm(opensles->dsp_context,
buffer.b, rc, opensles->channels, opensles->block_size))
{
error = ERROR_INTERNAL_ERROR;
break;
}
encoded_data = opensles->dsp_context->adpcm_buffer;
encoded_size = opensles->dsp_context->adpcm_size;
}
else if (opensles->format == WAVE_FORMAT_DVI_ADPCM)
{
if (!opensles->dsp_context->encode_ima_adpcm(opensles->dsp_context,
buffer.b, rc,
opensles->channels, opensles->block_size))
{
error = ERROR_INTERNAL_ERROR;
break;
}
encoded_data = opensles->dsp_context->adpcm_buffer;
encoded_size = opensles->dsp_context->adpcm_size;
}
else
{
encoded_data = buffer.v;
encoded_size = rc;
}
error = opensles->receive(encoded_data, encoded_size, opensles->user_data);
if (error) if (error)
break; break;
} }
@ -185,7 +144,6 @@ out:
static UINT audin_opensles_free(IAudinDevice* device) static UINT audin_opensles_free(IAudinDevice* device)
{ {
AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device; AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device;
DEBUG_DVC("device=%p", (void*) device); DEBUG_DVC("device=%p", (void*) device);
/* The function may have been called out of order, /* The function may have been called out of order,
@ -194,26 +152,19 @@ static UINT audin_opensles_free(IAudinDevice* device)
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
assert(opensles); assert(opensles);
assert(opensles->dsp_context);
assert(!opensles->stream); assert(!opensles->stream);
freerdp_dsp_context_free(opensles->dsp_context);
free(opensles->device_name); free(opensles->device_name);
free(opensles); free(opensles);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
static BOOL audin_opensles_format_supported(IAudinDevice* device, audinFormat* format) static BOOL audin_opensles_format_supported(IAudinDevice* device,
const AUDIO_FORMAT* format)
{ {
#ifdef WITH_DEBUG_DVC #ifdef WITH_DEBUG_DVC
AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device; AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device;
#endif #endif
DEBUG_DVC("device=%p, format=%p", (void*) opensles, (void*) format); DEBUG_DVC("device=%p, format=%p", (void*) opensles, (void*) format);
assert(format); assert(format);
switch (format->wFormatTag) switch (format->wFormatTag)
@ -226,17 +177,9 @@ static BOOL audin_opensles_format_supported(IAudinDevice* device, audinFormat* f
{ {
return TRUE; return TRUE;
} }
break; break;
/* TODO: Deactivated format, does not work, find out why */
// case WAVE_FORMAT_ADPCM: /* IMA ADPCM */
case WAVE_FORMAT_DVI_ADPCM:
if ((format->nSamplesPerSec <= 48000) &&
(format->wBitsPerSample == 4) &&
(format->nChannels == 1 || format->nChannels == 2))
{
return TRUE;
}
break;
default: default:
DEBUG_DVC("Encoding '%s' [0x%04X"PRIX16"] not supported", DEBUG_DVC("Encoding '%s' [0x%04X"PRIX16"] not supported",
rdpsnd_get_audio_tag_string(format->wFormatTag), rdpsnd_get_audio_tag_string(format->wFormatTag),
@ -253,14 +196,11 @@ static BOOL audin_opensles_format_supported(IAudinDevice* device, audinFormat* f
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_opensles_set_format(IAudinDevice* device, static UINT audin_opensles_set_format(IAudinDevice* device,
audinFormat* format, UINT32 FramesPerPacket) const AUDIO_FORMAT* format, UINT32 FramesPerPacket)
{ {
int bs;
AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device; AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device;
DEBUG_DVC("device=%p, format=%p, FramesPerPacket=%"PRIu32"", DEBUG_DVC("device=%p, format=%p, FramesPerPacket=%"PRIu32"",
(void*) device, (void*) format, FramesPerPacket); (void*) device, (void*) format, FramesPerPacket);
assert(format); assert(format);
/* The function may have been called out of order, ignore /* The function may have been called out of order, ignore
@ -268,35 +208,31 @@ static UINT audin_opensles_set_format(IAudinDevice* device,
if (!opensles) if (!opensles)
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
opensles->format = *format;
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
case WAVE_FORMAT_PCM: case WAVE_FORMAT_PCM:
opensles->frames_per_packet = FramesPerPacket; opensles->frames_per_packet = FramesPerPacket;
switch (format->wBitsPerSample) switch (format->wBitsPerSample)
{ {
case 4: case 4:
opensles->bytes_per_channel = 1; opensles->bytes_per_channel = 1;
break; break;
case 8: case 8:
opensles->bytes_per_channel = 1; opensles->bytes_per_channel = 1;
break; break;
case 16: case 16:
opensles->bytes_per_channel = 2; opensles->bytes_per_channel = 2;
break; break;
}
break;
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
opensles->bytes_per_channel = 2;
bs = (format->nBlockAlign - 4 * format->nChannels) * 4;
opensles->frames_per_packet = default:
(FramesPerPacket * format->nChannels * 2 / return ERROR_UNSUPPORTED_TYPE;
bs + 1) * bs / (format->nChannels * 2); }
break;
case WAVE_FORMAT_ALAW:
case WAVE_FORMAT_MULAW:
opensles->frames_per_packet = FramesPerPacket;
break; break;
default: default:
@ -306,12 +242,6 @@ static UINT audin_opensles_set_format(IAudinDevice* device,
return ERROR_UNSUPPORTED_TYPE; return ERROR_UNSUPPORTED_TYPE;
} }
opensles->rate = format->nSamplesPerSec;
opensles->channels = format->nChannels;
opensles->format = format->wFormatTag;
opensles->block_size = format->nBlockAlign;
DEBUG_DVC("aligned frames_per_packet=%"PRIu32", block_size=%"PRIu32"", DEBUG_DVC("aligned frames_per_packet=%"PRIu32", block_size=%"PRIu32"",
opensles->frames_per_packet, opensles->block_size); opensles->frames_per_packet, opensles->block_size);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
@ -326,9 +256,8 @@ static UINT audin_opensles_open(IAudinDevice* device, AudinReceive receive,
void* user_data) void* user_data)
{ {
AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device; AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device;
DEBUG_DVC("device=%p, receive=%p, user_data=%p", (void*) device, (void*) receive,
DEBUG_DVC("device=%p, receive=%p, user_data=%p", (void*) device, (void*) receive, (void*) user_data); (void*) user_data);
assert(opensles); assert(opensles);
/* The function may have been called out of order, /* The function may have been called out of order,
@ -338,10 +267,10 @@ static UINT audin_opensles_open(IAudinDevice* device, AudinReceive receive,
if (!(opensles->stream = android_OpenRecDevice( if (!(opensles->stream = android_OpenRecDevice(
opensles->device_name, opensles->device_name,
opensles->rate, opensles->format.nSamplesPerSec,
opensles->channels, opensles->format.nChannels,
opensles->frames_per_packet, opensles->frames_per_packet,
opensles->bytes_per_channel * 8))) opensles->format.wBitsPerSample)))
{ {
WLog_ERR(TAG, "android_OpenRecDevice failed!"); WLog_ERR(TAG, "android_OpenRecDevice failed!");
return ERROR_INTERNAL_ERROR; return ERROR_INTERNAL_ERROR;
@ -355,13 +284,14 @@ static UINT audin_opensles_open(IAudinDevice* device, AudinReceive receive,
WLog_ERR(TAG, "CreateEvent failed!"); WLog_ERR(TAG, "CreateEvent failed!");
goto error_out; goto error_out;
} }
if (!(opensles->thread = CreateThread(NULL, 0, if (!(opensles->thread = CreateThread(NULL, 0,
audin_opensles_thread_func, audin_opensles_thread_func, opensles, 0, NULL)))
opensles, 0, NULL)))
{ {
WLog_ERR(TAG, "CreateThread failed!"); WLog_ERR(TAG, "CreateThread failed!");
goto error_out; goto error_out;
} }
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_out: error_out:
android_CloseRecDevice(opensles->stream); android_CloseRecDevice(opensles->stream);
@ -380,9 +310,7 @@ static UINT audin_opensles_close(IAudinDevice* device)
{ {
UINT error; UINT error;
AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device; AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device;
DEBUG_DVC("device=%p", (void*) device); DEBUG_DVC("device=%p", (void*) device);
assert(opensles); assert(opensles);
/* The function may have been called out of order, /* The function may have been called out of order,
@ -396,32 +324,32 @@ static UINT audin_opensles_close(IAudinDevice* device)
assert(opensles->stopEvent); assert(opensles->stopEvent);
assert(opensles->thread); assert(opensles->thread);
assert(opensles->stream); assert(opensles->stream);
SetEvent(opensles->stopEvent); SetEvent(opensles->stopEvent);
if (WaitForSingleObject(opensles->thread, INFINITE) == WAIT_FAILED) if (WaitForSingleObject(opensles->thread, INFINITE) == WAIT_FAILED)
{ {
error = GetLastError(); error = GetLastError();
WLog_ERR(TAG, "WaitForSingleObject failed with error %"PRIu32"", error); WLog_ERR(TAG, "WaitForSingleObject failed with error %"PRIu32"", error);
return error; return error;
} }
CloseHandle(opensles->stopEvent); CloseHandle(opensles->stopEvent);
CloseHandle(opensles->thread); CloseHandle(opensles->thread);
android_CloseRecDevice(opensles->stream); android_CloseRecDevice(opensles->stream);
opensles->stopEvent = NULL; opensles->stopEvent = NULL;
opensles->thread = NULL; opensles->thread = NULL;
opensles->receive = NULL; opensles->receive = NULL;
opensles->user_data = NULL; opensles->user_data = NULL;
opensles->stream = NULL; opensles->stream = NULL;
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
static COMMAND_LINE_ARGUMENT_A audin_opensles_args[] = static COMMAND_LINE_ARGUMENT_A audin_opensles_args[] =
{ {
{ "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", {
NULL, NULL, -1, NULL, "audio device name" }, "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>",
NULL, NULL, -1, NULL, "audio device name"
},
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL } { NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
}; };
@ -437,13 +365,11 @@ static UINT audin_opensles_parse_addin_args(AudinOpenSLESDevice* device,
DWORD flags; DWORD flags;
COMMAND_LINE_ARGUMENT_A* arg; COMMAND_LINE_ARGUMENT_A* arg;
AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device; AudinOpenSLESDevice* opensles = (AudinOpenSLESDevice*) device;
DEBUG_DVC("device=%p, args=%p", (void*) device, (void*) args); DEBUG_DVC("device=%p, args=%p", (void*) device, (void*) args);
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD; flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv,
audin_opensles_args, flags, opensles, NULL, NULL); audin_opensles_args, flags, opensles, NULL, NULL);
if (status < 0) if (status < 0)
return status; return status;
@ -455,17 +381,16 @@ static UINT audin_opensles_parse_addin_args(AudinOpenSLESDevice* device,
continue; continue;
CommandLineSwitchStart(arg) CommandLineSwitchStart(arg)
CommandLineSwitchCase(arg, "dev") CommandLineSwitchCase(arg, "dev")
{ {
opensles->device_name = _strdup(arg->Value); opensles->device_name = _strdup(arg->Value);
if (!opensles->device_name) if (!opensles->device_name)
{ {
WLog_ERR(TAG, "_strdup failed!"); WLog_ERR(TAG, "_strdup failed!");
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
} }
CommandLineSwitchEnd(arg) CommandLineSwitchEnd(arg)
} }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL); while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
@ -492,10 +417,9 @@ UINT freerdp_audin_client_subsystem_entry(
ADDIN_ARGV* args; ADDIN_ARGV* args;
AudinOpenSLESDevice* opensles; AudinOpenSLESDevice* opensles;
UINT error; UINT error;
DEBUG_DVC("pEntryPoints=%p", (void*) pEntryPoints); DEBUG_DVC("pEntryPoints=%p", (void*) pEntryPoints);
opensles = (AudinOpenSLESDevice*) calloc(1, sizeof(AudinOpenSLESDevice)); opensles = (AudinOpenSLESDevice*) calloc(1, sizeof(AudinOpenSLESDevice));
if (!opensles) if (!opensles)
{ {
WLog_ERR(TAG, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
@ -508,7 +432,6 @@ UINT freerdp_audin_client_subsystem_entry(
opensles->iface.Close = audin_opensles_close; opensles->iface.Close = audin_opensles_close;
opensles->iface.Free = audin_opensles_free; opensles->iface.Free = audin_opensles_free;
opensles->rdpcontext = pEntryPoints->rdpcontext; opensles->rdpcontext = pEntryPoints->rdpcontext;
args = pEntryPoints->args; args = pEntryPoints->args;
if ((error = audin_opensles_parse_addin_args(opensles, args))) if ((error = audin_opensles_parse_addin_args(opensles, args)))
@ -517,14 +440,6 @@ UINT freerdp_audin_client_subsystem_entry(
goto error_out; goto error_out;
} }
opensles->dsp_context = freerdp_dsp_context_new();
if (!opensles->dsp_context)
{
WLog_ERR(TAG, "freerdp_dsp_context_new failed!");
error = CHANNEL_RC_NO_MEMORY;
goto error_out;
}
if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin, (IAudinDevice*) opensles))) if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin, (IAudinDevice*) opensles)))
{ {
WLog_ERR(TAG, "RegisterAudinDevice failed with error %"PRIu32"!", error); WLog_ERR(TAG, "RegisterAudinDevice failed with error %"PRIu32"!", error);
@ -533,7 +448,6 @@ UINT freerdp_audin_client_subsystem_entry(
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_out: error_out:
freerdp_dsp_context_free(opensles->dsp_context);
free(opensles); free(opensles);
return error; return error;
} }

View File

@ -47,7 +47,6 @@
#include <sys/ioctl.h> #include <sys/ioctl.h>
#include <freerdp/addin.h> #include <freerdp/addin.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/rdpsnd.h> #include <freerdp/channels/rdpsnd.h>
#include "audin_main.h" #include "audin_main.h"
@ -56,12 +55,10 @@ typedef struct _AudinOSSDevice
{ {
IAudinDevice iface; IAudinDevice iface;
FREERDP_DSP_CONTEXT* dsp_context;
HANDLE thread; HANDLE thread;
HANDLE stopEvent; HANDLE stopEvent;
audinFormat format; AUDIO_FORMAT format;
UINT32 FramesPerPacket; UINT32 FramesPerPacket;
int dev_unit; int dev_unit;
@ -76,7 +73,7 @@ typedef struct _AudinOSSDevice
WLog_ERR(TAG, "%s: %i - %s\n", _text, _error, strerror(_error)); WLog_ERR(TAG, "%s: %i - %s\n", _text, _error, strerror(_error));
static int audin_oss_get_format(audinFormat* format) static int audin_oss_get_format(const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
@ -94,25 +91,17 @@ static int audin_oss_get_format(audinFormat* format)
case WAVE_FORMAT_ALAW: case WAVE_FORMAT_ALAW:
return AFMT_A_LAW; return AFMT_A_LAW;
#if 0 /* This does not work on my desktop. */
case WAVE_FORMAT_MULAW: case WAVE_FORMAT_MULAW:
return AFMT_MU_LAW; return AFMT_MU_LAW;
#endif
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
return AFMT_S16_LE;
} }
return 0; return 0;
} }
static BOOL audin_oss_format_supported(IAudinDevice* device, static BOOL audin_oss_format_supported(IAudinDevice* device,
audinFormat* format) const AUDIO_FORMAT* format)
{ {
int req_fmt = 0;
if (device == NULL || format == NULL) if (device == NULL || format == NULL)
return FALSE; return FALSE;
@ -127,21 +116,14 @@ static BOOL audin_oss_format_supported(IAudinDevice* device,
break; break;
case WAVE_FORMAT_ADPCM: case WAVE_FORMAT_ALAW:
case WAVE_FORMAT_DVI_ADPCM: case WAVE_FORMAT_MULAW:
if (format->nSamplesPerSec > 48000 || return TRUE;
format->wBitsPerSample != 4 ||
(format->nChannels != 1 && format->nChannels != 2))
return FALSE;
break; default:
return FALSE;
} }
req_fmt = audin_oss_get_format(format);
if (req_fmt == 0)
return FALSE;
return TRUE; return TRUE;
} }
@ -150,7 +132,7 @@ static BOOL audin_oss_format_supported(IAudinDevice* device,
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_oss_set_format(IAudinDevice* device, audinFormat* format, static UINT audin_oss_set_format(IAudinDevice* device, const AUDIO_FORMAT* format,
UINT32 FramesPerPacket) UINT32 FramesPerPacket)
{ {
AudinOSSDevice* oss = (AudinOSSDevice*)device; AudinOSSDevice* oss = (AudinOSSDevice*)device;
@ -159,17 +141,7 @@ static UINT audin_oss_set_format(IAudinDevice* device, audinFormat* format,
return ERROR_INVALID_PARAMETER; return ERROR_INVALID_PARAMETER;
oss->FramesPerPacket = FramesPerPacket; oss->FramesPerPacket = FramesPerPacket;
CopyMemory(&(oss->format), format, sizeof(audinFormat)); oss->format = *format;
switch (format->wFormatTag)
{
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
oss->FramesPerPacket *= 4; /* Compression ratio. */
oss->format.wBitsPerSample *= 4;
break;
}
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -178,8 +150,9 @@ static DWORD WINAPI audin_oss_thread_func(LPVOID arg)
char dev_name[PATH_MAX] = "/dev/dsp"; char dev_name[PATH_MAX] = "/dev/dsp";
char mixer_name[PATH_MAX] = "/dev/mixer"; char mixer_name[PATH_MAX] = "/dev/mixer";
int pcm_handle = -1, mixer_handle; int pcm_handle = -1, mixer_handle;
BYTE* buffer = NULL, *encoded_data = NULL; BYTE* buffer = NULL;
int tmp, buffer_size, encoded_size; int tmp;
size_t buffer_size;
AudinOSSDevice* oss = (AudinOSSDevice*)arg; AudinOSSDevice* oss = (AudinOSSDevice*)arg;
UINT error = 0; UINT error = 0;
DWORD status; DWORD status;
@ -271,8 +244,6 @@ static DWORD WINAPI audin_oss_thread_func(LPVOID arg)
goto err_out; goto err_out;
} }
freerdp_dsp_context_reset_adpcm(oss->dsp_context);
while (1) while (1)
{ {
status = WaitForSingleObject(oss->stopEvent, 0); status = WaitForSingleObject(oss->stopEvent, 0);
@ -299,40 +270,7 @@ static DWORD WINAPI audin_oss_thread_func(LPVOID arg)
if (tmp < buffer_size) /* Not enouth data. */ if (tmp < buffer_size) /* Not enouth data. */
continue; continue;
/* Process. */ if ((error = oss->receive(&oss->format, buffer, buffer_size, oss->user_data)))
switch (oss->format.wFormatTag)
{
case WAVE_FORMAT_ADPCM:
if (!oss->dsp_context->encode_ms_adpcm(oss->dsp_context,
buffer, buffer_size, oss->format.nChannels, oss->format.nBlockAlign))
{
error = ERROR_INTERNAL_ERROR;
goto err_out;
}
encoded_data = oss->dsp_context->adpcm_buffer;
encoded_size = oss->dsp_context->adpcm_size;
break;
case WAVE_FORMAT_DVI_ADPCM:
if (!oss->dsp_context->encode_ima_adpcm(oss->dsp_context,
buffer, buffer_size, oss->format.nChannels, oss->format.nBlockAlign))
{
error = ERROR_INTERNAL_ERROR;
goto err_out;
}
encoded_data = oss->dsp_context->adpcm_buffer;
encoded_size = oss->dsp_context->adpcm_size;
break;
default:
encoded_data = buffer;
encoded_size = buffer_size;
break;
}
if ((error = oss->receive(encoded_data, encoded_size, oss->user_data)))
{ {
WLog_ERR(TAG, "oss->receive failed with error %"PRIu32"", error); WLog_ERR(TAG, "oss->receive failed with error %"PRIu32"", error);
break; break;
@ -438,12 +376,11 @@ static UINT audin_oss_free(IAudinDevice* device)
WLog_ERR(TAG, "audin_oss_close failed with error code %d!", error); WLog_ERR(TAG, "audin_oss_close failed with error code %d!", error);
} }
freerdp_dsp_context_free(oss->dsp_context);
free(oss); free(oss);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
COMMAND_LINE_ARGUMENT_A audin_oss_args[] = static COMMAND_LINE_ARGUMENT_A audin_oss_args[] =
{ {
{ "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "audio device name" }, { "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "audio device name" },
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL } { NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
@ -552,15 +489,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS
goto error_out; goto error_out;
} }
oss->dsp_context = freerdp_dsp_context_new();
if (!oss->dsp_context)
{
WLog_ERR(TAG, "freerdp_dsp_context_new failed!");
error = CHANNEL_RC_NO_MEMORY;
goto error_out;
}
if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin, if ((error = pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin,
(IAudinDevice*) oss))) (IAudinDevice*) oss)))
{ {
@ -570,7 +498,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_out: error_out:
freerdp_dsp_context_free(oss->dsp_context);
free(oss); free(oss);
return error; return error;
} }

View File

@ -34,7 +34,7 @@
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/addin.h> #include <freerdp/addin.h>
#include <freerdp/codec/dsp.h> #include <freerdp/codec/audio.h>
#include <freerdp/client/audin.h> #include <freerdp/client/audin.h>
#include "audin_main.h" #include "audin_main.h"
@ -49,14 +49,10 @@ typedef struct _AudinPulseDevice
pa_context* context; pa_context* context;
pa_sample_spec sample_spec; pa_sample_spec sample_spec;
pa_stream* stream; pa_stream* stream;
int format; AUDIO_FORMAT format;
int block_size;
FREERDP_DSP_CONTEXT* dsp_context; size_t bytes_per_frame;
size_t buffer_frames;
int bytes_per_frame;
BYTE* buffer;
int buffer_frames;
AudinReceive receive; AudinReceive receive;
void* user_data; void* user_data;
@ -172,12 +168,11 @@ static UINT audin_pulse_free(IAudinDevice* device)
pulse->mainloop = NULL; pulse->mainloop = NULL;
} }
freerdp_dsp_context_free(pulse->dsp_context);
free(pulse); free(pulse);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
static BOOL audin_pulse_format_supported(IAudinDevice* device, audinFormat* format) static BOOL audin_pulse_format_supported(IAudinDevice* device, const AUDIO_FORMAT* format)
{ {
AudinPulseDevice* pulse = (AudinPulseDevice*) device; AudinPulseDevice* pulse = (AudinPulseDevice*) device;
@ -186,7 +181,7 @@ static BOOL audin_pulse_format_supported(IAudinDevice* device, audinFormat* form
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
case 1: /* PCM */ case WAVE_FORMAT_PCM:
if (format->cbSize == 0 && if (format->cbSize == 0 &&
(format->nSamplesPerSec <= PA_RATE_MAX) && (format->nSamplesPerSec <= PA_RATE_MAX) &&
(format->wBitsPerSample == 8 || format->wBitsPerSample == 16) && (format->wBitsPerSample == 8 || format->wBitsPerSample == 16) &&
@ -197,8 +192,8 @@ static BOOL audin_pulse_format_supported(IAudinDevice* device, audinFormat* form
break; break;
case 6: /* A-LAW */ case WAVE_FORMAT_ALAW: /* A-LAW */
case 7: /* U-LAW */ case WAVE_FORMAT_MULAW: /* U-LAW */
if (format->cbSize == 0 && if (format->cbSize == 0 &&
(format->nSamplesPerSec <= PA_RATE_MAX) && (format->nSamplesPerSec <= PA_RATE_MAX) &&
(format->wBitsPerSample == 8) && (format->wBitsPerSample == 8) &&
@ -209,15 +204,8 @@ static BOOL audin_pulse_format_supported(IAudinDevice* device, audinFormat* form
break; break;
case 0x11: /* IMA ADPCM */ default:
if ((format->nSamplesPerSec <= PA_RATE_MAX) && return FALSE;
(format->wBitsPerSample == 4) &&
(format->nChannels == 1 || format->nChannels == 2))
{
return TRUE;
}
break;
} }
return FALSE; return FALSE;
@ -228,10 +216,9 @@ static BOOL audin_pulse_format_supported(IAudinDevice* device, audinFormat* form
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_pulse_set_format(IAudinDevice* device, audinFormat* format, static UINT audin_pulse_set_format(IAudinDevice* device, const AUDIO_FORMAT* format,
UINT32 FramesPerPacket) UINT32 FramesPerPacket)
{ {
int bs;
pa_sample_spec sample_spec = { 0 }; pa_sample_spec sample_spec = { 0 };
AudinPulseDevice* pulse = (AudinPulseDevice*) device; AudinPulseDevice* pulse = (AudinPulseDevice*) device;
@ -239,16 +226,14 @@ static UINT audin_pulse_set_format(IAudinDevice* device, audinFormat* format,
return ERROR_INVALID_PARAMETER; return ERROR_INVALID_PARAMETER;
if (FramesPerPacket > 0) if (FramesPerPacket > 0)
{
pulse->frames_per_packet = FramesPerPacket; pulse->frames_per_packet = FramesPerPacket;
}
sample_spec.rate = format->nSamplesPerSec; sample_spec.rate = format->nSamplesPerSec;
sample_spec.channels = format->nChannels; sample_spec.channels = format->nChannels;
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
case 1: /* PCM */ case WAVE_FORMAT_PCM: /* PCM */
switch (format->wBitsPerSample) switch (format->wBitsPerSample)
{ {
case 8: case 8:
@ -258,31 +243,27 @@ static UINT audin_pulse_set_format(IAudinDevice* device, audinFormat* format,
case 16: case 16:
sample_spec.format = PA_SAMPLE_S16LE; sample_spec.format = PA_SAMPLE_S16LE;
break; break;
default:
return ERROR_INTERNAL_ERROR;
} }
break; break;
case 6: /* A-LAW */ case WAVE_FORMAT_ALAW: /* A-LAW */
sample_spec.format = PA_SAMPLE_ALAW; sample_spec.format = PA_SAMPLE_ALAW;
break; break;
case 7: /* U-LAW */ case WAVE_FORMAT_MULAW: /* U-LAW */
sample_spec.format = PA_SAMPLE_ULAW; sample_spec.format = PA_SAMPLE_ULAW;
break; break;
case 0x11: /* IMA ADPCM */ default:
sample_spec.format = PA_SAMPLE_S16LE; return ERROR_INTERNAL_ERROR;
bs = (format->nBlockAlign - 4 * format->nChannels) * 4;
pulse->frames_per_packet = (pulse->frames_per_packet * format->nChannels * 2 /
bs + 1) * bs / (format->nChannels * 2);
DEBUG_DVC("aligned FramesPerPacket=%"PRIu32"",
pulse->frames_per_packet);
break;
} }
pulse->sample_spec = sample_spec; pulse->sample_spec = sample_spec;
pulse->format = format->wFormatTag; pulse->format = *format;
pulse->block_size = format->nBlockAlign;
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -313,81 +294,15 @@ static void audin_pulse_stream_state_callback(pa_stream* stream, void* userdata)
static void audin_pulse_stream_request_callback(pa_stream* stream, size_t length, void* userdata) static void audin_pulse_stream_request_callback(pa_stream* stream, size_t length, void* userdata)
{ {
int frames;
int cframes;
const void* data; const void* data;
const BYTE* src;
int encoded_size;
BYTE* encoded_data;
AudinPulseDevice* pulse = (AudinPulseDevice*) userdata; AudinPulseDevice* pulse = (AudinPulseDevice*) userdata;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
/* There is a race condition here where we may receive this callback
* before the buffer has been set up in the main code. It's probably
* possible to fix with additional locking, but it's easier just to
* ignore input until the buffer is ready.
*/
if (pulse->buffer == NULL)
{
/* WLog_ERR(TAG, "%s: ignoring input, pulse buffer not ready.\n", __func__); */
return;
}
pa_stream_peek(stream, &data, &length); pa_stream_peek(stream, &data, &length);
frames = length / pulse->bytes_per_frame; error = pulse->receive(&pulse->format, data, length, pulse->user_data);
DEBUG_DVC("length %"PRIdz" frames %d", length, frames);
src = (const BYTE*) data;
while (frames > 0)
{
cframes = pulse->frames_per_packet - pulse->buffer_frames;
if (cframes > frames)
cframes = frames;
memcpy(pulse->buffer + pulse->buffer_frames * pulse->bytes_per_frame,
src, cframes * pulse->bytes_per_frame);
pulse->buffer_frames += cframes;
if (pulse->buffer_frames >= pulse->frames_per_packet)
{
if (pulse->format == 0x11)
{
if (!pulse->dsp_context->encode_ima_adpcm(pulse->dsp_context,
pulse->buffer, pulse->buffer_frames * pulse->bytes_per_frame,
pulse->sample_spec.channels, pulse->block_size))
{
error = ERROR_INTERNAL_ERROR;
break;
}
encoded_data = pulse->dsp_context->adpcm_buffer;
encoded_size = pulse->dsp_context->adpcm_size;
}
else
{
encoded_data = pulse->buffer;
encoded_size = pulse->buffer_frames * pulse->bytes_per_frame;
}
DEBUG_DVC("encoded %d [%d] to %d [%X]",
pulse->buffer_frames, pulse->bytes_per_frame, encoded_size,
pulse->format);
error = pulse->receive(encoded_data, encoded_size, pulse->user_data);
pulse->buffer_frames = 0;
if (!error)
break;
}
src += cframes * pulse->bytes_per_frame;
frames -= cframes;
}
pa_stream_drop(stream); pa_stream_drop(stream);
if (error && pulse->rdpcontext) if (error && pulse->rdpcontext)
setChannelError(pulse->rdpcontext, error, "audin_oss_thread_func reported an error"); setChannelError(pulse->rdpcontext, error, "audin_pulse_thread_func reported an error");
} }
@ -411,14 +326,6 @@ static UINT audin_pulse_close(IAudinDevice* device)
pulse->receive = NULL; pulse->receive = NULL;
pulse->user_data = NULL; pulse->user_data = NULL;
if (pulse->buffer)
{
free(pulse->buffer);
pulse->buffer = NULL;
pulse->buffer_frames = 0;
}
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -439,7 +346,6 @@ static UINT audin_pulse_open(IAudinDevice* device, AudinReceive receive, void* u
if (!pulse->sample_spec.rate || pulse->stream) if (!pulse->sample_spec.rate || pulse->stream)
return ERROR_INVALID_PARAMETER; return ERROR_INVALID_PARAMETER;
pulse->buffer = NULL;
pulse->receive = receive; pulse->receive = receive;
pulse->user_data = user_data; pulse->user_data = user_data;
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
@ -496,15 +402,6 @@ static UINT audin_pulse_open(IAudinDevice* device, AudinReceive receive, void* u
} }
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
freerdp_dsp_context_reset_adpcm(pulse->dsp_context);
pulse->buffer = calloc(pulse->frames_per_packet, pulse->bytes_per_frame);
if (!pulse->buffer)
{
WLog_ERR(TAG, "calloc failed!");
return CHANNEL_RC_NO_MEMORY;
}
pulse->buffer_frames = 0; pulse->buffer_frames = 0;
DEBUG_DVC("connected"); DEBUG_DVC("connected");
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
@ -597,15 +494,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
goto error_out; goto error_out;
} }
pulse->dsp_context = freerdp_dsp_context_new();
if (!pulse->dsp_context)
{
WLog_ERR(TAG, "freerdp_dsp_context_new failed!");
error = CHANNEL_RC_NO_MEMORY;
goto error_out;
}
pulse->mainloop = pa_threaded_mainloop_new(); pulse->mainloop = pa_threaded_mainloop_new();
if (!pulse->mainloop) if (!pulse->mainloop)

View File

@ -70,18 +70,32 @@ static void CALLBACK waveInProc(HWAVEIN hWaveIn, UINT uMsg, DWORD_PTR dwInstance
case WIM_DATA: case WIM_DATA:
pWaveHdr = (WAVEHDR*)dwParam1; pWaveHdr = (WAVEHDR*)dwParam1;
if (WHDR_DONE == (WHDR_DONE & pWaveHdr->dwFlags)) if (WHDR_DONE == (WHDR_DONE & pWaveHdr->dwFlags))
{ {
if (pWaveHdr->dwBytesRecorded if (pWaveHdr->dwBytesRecorded
&& !(WaitForSingleObject(winmm->stopEvent, 0) == WAIT_OBJECT_0)) && !(WaitForSingleObject(winmm->stopEvent, 0) == WAIT_OBJECT_0))
{ {
if ((error = winmm->receive(pWaveHdr->lpData, pWaveHdr->dwBytesRecorded, winmm->user_data))) AUDIO_FORMAT format;
format.cbSize = winmm->pwfx_cur->cbSize;
format.nBlockAlign = winmm->pwfx_cur->nBlockAlign;
format.nAvgBytesPerSec = winmm->pwfx_cur->nAvgBytesPerSec;
format.nChannels = winmm->pwfx_cur->nChannels;
format.nSamplesPerSec = winmm->pwfx_cur->nSamplesPerSec;
format.wBitsPerSample = winmm->pwfx_cur->wBitsPerSample;
format.wFormatTag = winmm->pwfx_cur->wFormatTag;
if ((error = winmm->receive(&format, pWaveHdr->lpData, pWaveHdr->dwBytesRecorded,
winmm->user_data)))
break; break;
mmResult = waveInAddBuffer(hWaveIn, pWaveHdr, sizeof(WAVEHDR)); mmResult = waveInAddBuffer(hWaveIn, pWaveHdr, sizeof(WAVEHDR));
if (mmResult != MMSYSERR_NOERROR) if (mmResult != MMSYSERR_NOERROR)
error = ERROR_INTERNAL_ERROR; error = ERROR_INTERNAL_ERROR;
} }
} }
break; break;
case WIM_OPEN: case WIM_OPEN:
@ -90,6 +104,7 @@ static void CALLBACK waveInProc(HWAVEIN hWaveIn, UINT uMsg, DWORD_PTR dwInstance
default: default:
break; break;
} }
if (error && winmm->rdpcontext) if (error && winmm->rdpcontext)
setChannelError(winmm->rdpcontext, error, "waveInProc reported an error"); setChannelError(winmm->rdpcontext, error, "waveInProc reported an error");
} }
@ -109,44 +124,58 @@ static DWORD WINAPI audin_winmm_thread_func(LPVOID arg)
(DWORD_PTR)waveInProc, (DWORD_PTR)winmm, CALLBACK_FUNCTION)) (DWORD_PTR)waveInProc, (DWORD_PTR)winmm, CALLBACK_FUNCTION))
{ {
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
return 0; "audin_winmm_thread_func reported an error");
return ERROR_INTERNAL_ERROR;
} }
} }
size = (winmm->pwfx_cur->wBitsPerSample * winmm->pwfx_cur->nChannels * winmm->frames_per_packet + 7) / 8; size = (winmm->pwfx_cur->wBitsPerSample * winmm->pwfx_cur->nChannels * winmm->frames_per_packet +
7) / 8;
for (i = 0; i < 4; i++) for (i = 0; i < 4; i++)
{ {
buffer = (char*) malloc(size); buffer = (char*) malloc(size);
if (!buffer) if (!buffer)
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
waveHdr[i].dwBufferLength = size; waveHdr[i].dwBufferLength = size;
waveHdr[i].dwFlags = 0; waveHdr[i].dwFlags = 0;
waveHdr[i].lpData = buffer; waveHdr[i].lpData = buffer;
rc = waveInPrepareHeader(winmm->hWaveIn, &waveHdr[i], sizeof(waveHdr[i])); rc = waveInPrepareHeader(winmm->hWaveIn, &waveHdr[i], sizeof(waveHdr[i]));
if (MMSYSERR_NOERROR != rc) if (MMSYSERR_NOERROR != rc)
{ {
DEBUG_DVC("waveInPrepareHeader failed. %"PRIu32"", rc); DEBUG_DVC("waveInPrepareHeader failed. %"PRIu32"", rc);
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
} }
rc = waveInAddBuffer(winmm->hWaveIn, &waveHdr[i], sizeof(waveHdr[i])); rc = waveInAddBuffer(winmm->hWaveIn, &waveHdr[i], sizeof(waveHdr[i]));
if (MMSYSERR_NOERROR != rc) if (MMSYSERR_NOERROR != rc)
{ {
DEBUG_DVC("waveInAddBuffer failed. %"PRIu32"", rc); DEBUG_DVC("waveInAddBuffer failed. %"PRIu32"", rc);
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
} }
} }
rc = waveInStart(winmm->hWaveIn); rc = waveInStart(winmm->hWaveIn);
if (MMSYSERR_NOERROR != rc) if (MMSYSERR_NOERROR != rc)
{ {
DEBUG_DVC("waveInStart failed. %"PRIu32"", rc); DEBUG_DVC("waveInStart failed. %"PRIu32"", rc);
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
} }
status = WaitForSingleObject(winmm->stopEvent, INFINITE); status = WaitForSingleObject(winmm->stopEvent, INFINITE);
@ -154,39 +183,51 @@ static DWORD WINAPI audin_winmm_thread_func(LPVOID arg)
if (status == WAIT_FAILED) if (status == WAIT_FAILED)
{ {
DEBUG_DVC("WaitForSingleObject failed."); DEBUG_DVC("WaitForSingleObject failed.");
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
} }
rc = waveInReset(winmm->hWaveIn); rc = waveInReset(winmm->hWaveIn);
if (MMSYSERR_NOERROR != rc) if (MMSYSERR_NOERROR != rc)
{ {
DEBUG_DVC("waveInReset failed. %"PRIu32"", rc); DEBUG_DVC("waveInReset failed. %"PRIu32"", rc);
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
} }
for (i = 0; i < 4; i++) for (i = 0; i < 4; i++)
{ {
rc = waveInUnprepareHeader(winmm->hWaveIn, &waveHdr[i], sizeof(waveHdr[i])); rc = waveInUnprepareHeader(winmm->hWaveIn, &waveHdr[i], sizeof(waveHdr[i]));
if (MMSYSERR_NOERROR != rc) if (MMSYSERR_NOERROR != rc)
{ {
DEBUG_DVC("waveInUnprepareHeader failed. %"PRIu32"", rc); DEBUG_DVC("waveInUnprepareHeader failed. %"PRIu32"", rc);
if (winmm->rdpcontext) if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error"); setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
} }
free(waveHdr[i].lpData); free(waveHdr[i].lpData);
} }
rc = waveInClose(winmm->hWaveIn); rc = waveInClose(winmm->hWaveIn);
if (MMSYSERR_NOERROR != rc) if (MMSYSERR_NOERROR != rc)
{ {
DEBUG_DVC("waveInClose failed. %"PRIu32"", rc); DEBUG_DVC("waveInClose failed. %"PRIu32"", rc);
if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR, "audin_winmm_thread_func reported an error");
}
winmm->hWaveIn = NULL;
if (winmm->rdpcontext)
setChannelError(winmm->rdpcontext, ERROR_INTERNAL_ERROR,
"audin_winmm_thread_func reported an error");
}
winmm->hWaveIn = NULL;
return 0; return 0;
} }
@ -208,7 +249,6 @@ static UINT audin_winmm_free(IAudinDevice* device)
free(winmm->ppwfx); free(winmm->ppwfx);
free(winmm->device_name); free(winmm->device_name);
free(winmm); free(winmm);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -222,9 +262,7 @@ static UINT audin_winmm_close(IAudinDevice* device)
DWORD status; DWORD status;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
AudinWinmmDevice* winmm = (AudinWinmmDevice*) device; AudinWinmmDevice* winmm = (AudinWinmmDevice*) device;
SetEvent(winmm->stopEvent); SetEvent(winmm->stopEvent);
status = WaitForSingleObject(winmm->thread, INFINITE); status = WaitForSingleObject(winmm->thread, INFINITE);
if (status == WAIT_FAILED) if (status == WAIT_FAILED)
@ -236,12 +274,10 @@ static UINT audin_winmm_close(IAudinDevice* device)
CloseHandle(winmm->thread); CloseHandle(winmm->thread);
CloseHandle(winmm->stopEvent); CloseHandle(winmm->stopEvent);
winmm->thread = NULL; winmm->thread = NULL;
winmm->stopEvent = NULL; winmm->stopEvent = NULL;
winmm->receive = NULL; winmm->receive = NULL;
winmm->user_data = NULL; winmm->user_data = NULL;
return error; return error;
} }
@ -250,11 +286,11 @@ static UINT audin_winmm_close(IAudinDevice* device)
* *
* @return 0 on success, otherwise a Win32 error code * @return 0 on success, otherwise a Win32 error code
*/ */
static UINT audin_winmm_set_format(IAudinDevice* device, audinFormat* format, UINT32 FramesPerPacket) static UINT audin_winmm_set_format(IAudinDevice* device, const AUDIO_FORMAT* format,
UINT32 FramesPerPacket)
{ {
UINT32 i; UINT32 i;
AudinWinmmDevice* winmm = (AudinWinmmDevice*) device; AudinWinmmDevice* winmm = (AudinWinmmDevice*) device;
winmm->frames_per_packet = FramesPerPacket; winmm->frames_per_packet = FramesPerPacket;
for (i = 0; i < winmm->cFormats; i++) for (i = 0; i < winmm->cFormats; i++)
@ -267,18 +303,20 @@ static UINT audin_winmm_set_format(IAudinDevice* device, audinFormat* format, UI
break; break;
} }
} }
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
static BOOL audin_winmm_format_supported(IAudinDevice* device, audinFormat* format) static BOOL audin_winmm_format_supported(IAudinDevice* device, const AUDIO_FORMAT* format)
{ {
AudinWinmmDevice* winmm = (AudinWinmmDevice*) device; AudinWinmmDevice* winmm = (AudinWinmmDevice*) device;
PWAVEFORMATEX pwfx; PWAVEFORMATEX pwfx;
BYTE* data; BYTE* data;
pwfx = (PWAVEFORMATEX)malloc(sizeof(WAVEFORMATEX) + format->cbSize); pwfx = (PWAVEFORMATEX)malloc(sizeof(WAVEFORMATEX) + format->cbSize);
if (!pwfx) if (!pwfx)
return FALSE; return FALSE;
pwfx->cbSize = format->cbSize; pwfx->cbSize = format->cbSize;
pwfx->wFormatTag = format->wFormatTag; pwfx->wFormatTag = format->wFormatTag;
pwfx->nChannels = format->nChannels; pwfx->nChannels = format->nChannels;
@ -286,26 +324,27 @@ static BOOL audin_winmm_format_supported(IAudinDevice* device, audinFormat* form
pwfx->nBlockAlign = format->nBlockAlign; pwfx->nBlockAlign = format->nBlockAlign;
pwfx->wBitsPerSample = format->wBitsPerSample; pwfx->wBitsPerSample = format->wBitsPerSample;
data = (BYTE*)pwfx + sizeof(WAVEFORMATEX); data = (BYTE*)pwfx + sizeof(WAVEFORMATEX);
memcpy(data, format->data, format->cbSize); memcpy(data, format->data, format->cbSize);
if (pwfx->wFormatTag == WAVE_FORMAT_PCM) if (pwfx->wFormatTag == WAVE_FORMAT_PCM)
{ {
pwfx->nAvgBytesPerSec = pwfx->nSamplesPerSec * pwfx->nBlockAlign; pwfx->nAvgBytesPerSec = pwfx->nSamplesPerSec * pwfx->nBlockAlign;
if (MMSYSERR_NOERROR == waveInOpen(NULL, WAVE_MAPPER, pwfx, 0, 0, WAVE_FORMAT_QUERY)) if (MMSYSERR_NOERROR == waveInOpen(NULL, WAVE_MAPPER, pwfx, 0, 0, WAVE_FORMAT_QUERY))
{ {
if (winmm->cFormats >= winmm->ppwfx_size) if (winmm->cFormats >= winmm->ppwfx_size)
{ {
PWAVEFORMATEX* tmp_ppwfx; PWAVEFORMATEX* tmp_ppwfx;
tmp_ppwfx = realloc(winmm->ppwfx, sizeof(PWAVEFORMATEX) * winmm->ppwfx_size * 2); tmp_ppwfx = realloc(winmm->ppwfx, sizeof(PWAVEFORMATEX) * winmm->ppwfx_size * 2);
if (!tmp_ppwfx) if (!tmp_ppwfx)
return FALSE; return FALSE;
winmm->ppwfx_size *= 2; winmm->ppwfx_size *= 2;
winmm->ppwfx = tmp_ppwfx; winmm->ppwfx = tmp_ppwfx;
} }
winmm->ppwfx[winmm->cFormats++] = pwfx;
winmm->ppwfx[winmm->cFormats++] = pwfx;
return TRUE; return TRUE;
} }
} }
@ -322,7 +361,6 @@ static BOOL audin_winmm_format_supported(IAudinDevice* device, audinFormat* form
static UINT audin_winmm_open(IAudinDevice* device, AudinReceive receive, void* user_data) static UINT audin_winmm_open(IAudinDevice* device, AudinReceive receive, void* user_data)
{ {
AudinWinmmDevice* winmm = (AudinWinmmDevice*) device; AudinWinmmDevice* winmm = (AudinWinmmDevice*) device;
winmm->receive = receive; winmm->receive = receive;
winmm->user_data = user_data; winmm->user_data = user_data;
@ -332,13 +370,15 @@ static UINT audin_winmm_open(IAudinDevice* device, AudinReceive receive, void* u
return ERROR_INTERNAL_ERROR; return ERROR_INTERNAL_ERROR;
} }
if (!(winmm->thread = CreateThread(NULL, 0, audin_winmm_thread_func, winmm, 0, NULL))) if (!(winmm->thread = CreateThread(NULL, 0,
audin_winmm_thread_func, winmm, 0, NULL)))
{ {
WLog_ERR(TAG, "CreateThread failed!"); WLog_ERR(TAG, "CreateThread failed!");
CloseHandle(winmm->stopEvent); CloseHandle(winmm->stopEvent);
winmm->stopEvent = NULL; winmm->stopEvent = NULL;
return ERROR_INTERNAL_ERROR; return ERROR_INTERNAL_ERROR;
} }
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -359,11 +399,9 @@ static UINT audin_winmm_parse_addin_args(AudinWinmmDevice* device, ADDIN_ARGV* a
DWORD flags; DWORD flags;
COMMAND_LINE_ARGUMENT_A* arg; COMMAND_LINE_ARGUMENT_A* arg;
AudinWinmmDevice* winmm = (AudinWinmmDevice*) device; AudinWinmmDevice* winmm = (AudinWinmmDevice*) device;
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD; flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, audin_winmm_args, flags,
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, audin_winmm_args, flags, winmm, NULL, NULL); winmm, NULL, NULL);
arg = audin_winmm_args; arg = audin_winmm_args;
do do
@ -372,17 +410,16 @@ static UINT audin_winmm_parse_addin_args(AudinWinmmDevice* device, ADDIN_ARGV* a
continue; continue;
CommandLineSwitchStart(arg) CommandLineSwitchStart(arg)
CommandLineSwitchCase(arg, "dev") CommandLineSwitchCase(arg, "dev")
{ {
winmm->device_name = _strdup(arg->Value); winmm->device_name = _strdup(arg->Value);
if (!winmm->device_name) if (!winmm->device_name)
{ {
WLog_ERR(TAG, "_strdup failed!"); WLog_ERR(TAG, "_strdup failed!");
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
} }
CommandLineSwitchEnd(arg) CommandLineSwitchEnd(arg)
} }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL); while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
@ -406,8 +443,8 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
ADDIN_ARGV* args; ADDIN_ARGV* args;
AudinWinmmDevice* winmm; AudinWinmmDevice* winmm;
UINT error; UINT error;
winmm = (AudinWinmmDevice*) calloc(1, sizeof(AudinWinmmDevice)); winmm = (AudinWinmmDevice*) calloc(1, sizeof(AudinWinmmDevice));
if (!winmm) if (!winmm)
{ {
WLog_ERR(TAG, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
@ -420,7 +457,6 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
winmm->iface.Close = audin_winmm_close; winmm->iface.Close = audin_winmm_close;
winmm->iface.Free = audin_winmm_free; winmm->iface.Free = audin_winmm_free;
winmm->rdpcontext = pEntryPoints->rdpcontext; winmm->rdpcontext = pEntryPoints->rdpcontext;
args = pEntryPoints->args; args = pEntryPoints->args;
if ((error = audin_winmm_parse_addin_args(winmm, args))) if ((error = audin_winmm_parse_addin_args(winmm, args)))
@ -432,6 +468,7 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
if (!winmm->device_name) if (!winmm->device_name)
{ {
winmm->device_name = _strdup("default"); winmm->device_name = _strdup("default");
if (!winmm->device_name) if (!winmm->device_name)
{ {
WLog_ERR(TAG, "_strdup failed!"); WLog_ERR(TAG, "_strdup failed!");
@ -442,6 +479,7 @@ UINT freerdp_audin_client_subsystem_entry(PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEn
winmm->ppwfx_size = 10; winmm->ppwfx_size = 10;
winmm->ppwfx = malloc(sizeof(PWAVEFORMATEX) * winmm->ppwfx_size); winmm->ppwfx = malloc(sizeof(PWAVEFORMATEX) * winmm->ppwfx_size);
if (!winmm->ppwfx) if (!winmm->ppwfx)
{ {
WLog_ERR(TAG, "malloc failed!"); WLog_ERR(TAG, "malloc failed!");

View File

@ -339,10 +339,9 @@ static UINT audin_server_recv_data(audin_server* audin, wStream* s,
AUDIO_FORMAT* format; AUDIO_FORMAT* format;
int sbytes_per_sample; int sbytes_per_sample;
int sbytes_per_frame; int sbytes_per_frame;
BYTE* src;
int size;
int frames; int frames;
UINT success = CHANNEL_RC_OK; wStream* out;
UINT success = ERROR_INTERNAL_ERROR;
if (audin->context.selected_client_format < 0) if (audin->context.selected_client_format < 0)
{ {
@ -351,53 +350,26 @@ static UINT audin_server_recv_data(audin_server* audin, wStream* s,
return ERROR_INVALID_DATA; return ERROR_INVALID_DATA;
} }
out = Stream_New(NULL, 4096);
if (!out)
return ERROR_OUTOFMEMORY;
format = &audin->context.client_formats[audin->context.selected_client_format]; format = &audin->context.client_formats[audin->context.selected_client_format];
if (format->wFormatTag == WAVE_FORMAT_ADPCM) if (freerdp_dsp_decode(audin->dsp_context, format, Stream_Pointer(s), length, out))
{ {
audin->dsp_context->decode_ms_adpcm(audin->dsp_context, Stream_SealLength(out);
Stream_Pointer(s), length, format->nChannels, format->nBlockAlign);
size = audin->dsp_context->adpcm_size;
src = audin->dsp_context->adpcm_buffer;
sbytes_per_sample = 2;
sbytes_per_frame = format->nChannels * 2;
}
else if (format->wFormatTag == WAVE_FORMAT_DVI_ADPCM)
{
audin->dsp_context->decode_ima_adpcm(audin->dsp_context,
Stream_Pointer(s), length, format->nChannels, format->nBlockAlign);
size = audin->dsp_context->adpcm_size;
src = audin->dsp_context->adpcm_buffer;
sbytes_per_sample = 2;
sbytes_per_frame = format->nChannels * 2;
}
else
{
size = length;
src = Stream_Pointer(s);
sbytes_per_sample = format->wBitsPerSample / 8; sbytes_per_sample = format->wBitsPerSample / 8;
sbytes_per_frame = format->nChannels * sbytes_per_sample; sbytes_per_frame = format->nChannels * sbytes_per_sample;
} frames = Stream_Length(out) / sbytes_per_frame;
IFCALLRET(audin->context.ReceiveSamples, success, &audin->context, Stream_Buffer(out), frames);
if (format->nSamplesPerSec == audin->context.dst_format.nSamplesPerSec
&& format->nChannels == audin->context.dst_format.nChannels)
{
frames = size / sbytes_per_frame;
}
else
{
audin->dsp_context->resample(audin->dsp_context, src, sbytes_per_sample,
format->nChannels, format->nSamplesPerSec, size / sbytes_per_frame,
audin->context.dst_format.nChannels, audin->context.dst_format.nSamplesPerSec);
frames = audin->dsp_context->resampled_frames;
src = audin->dsp_context->resampled_buffer;
}
IFCALLRET(audin->context.ReceiveSamples, success, &audin->context, src, frames);
if (success) if (success)
WLog_ERR(TAG, "context.ReceiveSamples failed with error %"PRIu32"", success); WLog_ERR(TAG, "context.ReceiveSamples failed with error %"PRIu32"", success);
}
Stream_Free(out, TRUE);
return success; return success;
} }
@ -694,7 +666,7 @@ audin_server_context* audin_server_context_new(HANDLE vcm)
audin->context.SelectFormat = audin_server_select_format; audin->context.SelectFormat = audin_server_select_format;
audin->context.Open = audin_server_open; audin->context.Open = audin_server_open;
audin->context.Close = audin_server_close; audin->context.Close = audin_server_close;
audin->dsp_context = freerdp_dsp_context_new(); audin->dsp_context = freerdp_dsp_context_new(FALSE);
if (!audin->dsp_context) if (!audin->dsp_context)
{ {

View File

@ -23,11 +23,8 @@ set(${MODULE_PREFIX}_SRCS
add_channel_client_library(${MODULE_PREFIX} ${MODULE_NAME} ${CHANNEL_NAME} FALSE "VirtualChannelEntryEx") add_channel_client_library(${MODULE_PREFIX} ${MODULE_NAME} ${CHANNEL_NAME} FALSE "VirtualChannelEntryEx")
target_link_libraries(${MODULE_NAME} winpr freerdp ${CMAKE_THREAD_LIBS_INIT}) target_link_libraries(${MODULE_NAME} winpr freerdp ${CMAKE_THREAD_LIBS_INIT})
set_property(TARGET ${MODULE_NAME} PROPERTY FOLDER "Channels/${CHANNEL_NAME}/Client") set_property(TARGET ${MODULE_NAME} PROPERTY FOLDER "Channels/${CHANNEL_NAME}/Client")
if(WITH_OSS) if(WITH_OSS)
@ -57,3 +54,5 @@ endif()
if(WITH_OPENSLES) if(WITH_OPENSLES)
add_channel_client_subsystem(${MODULE_PREFIX} ${CHANNEL_NAME} "opensles" "") add_channel_client_subsystem(${MODULE_PREFIX} ${CHANNEL_NAME} "opensles" "")
endif() endif()
add_channel_client_subsystem(${MODULE_PREFIX} ${CHANNEL_NAME} "fake" "")

View File

@ -47,22 +47,18 @@ struct rdpsnd_alsa_plugin
{ {
rdpsndDevicePlugin device; rdpsndDevicePlugin device;
int latency; UINT32 latency;
int wformat; AUDIO_FORMAT aformat;
int block_size;
char* device_name; char* device_name;
snd_pcm_t* pcm_handle; snd_pcm_t* pcm_handle;
snd_mixer_t* mixer_handle; snd_mixer_t* mixer_handle;
UINT32 source_rate;
UINT32 actual_rate; UINT32 actual_rate;
UINT32 wLocalTimeClose;
snd_pcm_format_t format; snd_pcm_format_t format;
UINT32 source_channels;
UINT32 actual_channels; UINT32 actual_channels;
int bytes_per_channel;
snd_pcm_uframes_t buffer_size; snd_pcm_uframes_t buffer_size;
snd_pcm_uframes_t period_size; snd_pcm_uframes_t period_size;
FREERDP_DSP_CONTEXT* dsp_context;
}; };
#define SND_PCM_CHECK(_func, _status) \ #define SND_PCM_CHECK(_func, _status) \
@ -77,33 +73,25 @@ static int rdpsnd_alsa_set_hw_params(rdpsndAlsaPlugin* alsa)
int status; int status;
snd_pcm_hw_params_t* hw_params; snd_pcm_hw_params_t* hw_params;
snd_pcm_uframes_t buffer_size_max; snd_pcm_uframes_t buffer_size_max;
status = snd_pcm_hw_params_malloc(&hw_params); status = snd_pcm_hw_params_malloc(&hw_params);
SND_PCM_CHECK("snd_pcm_hw_params_malloc", status); SND_PCM_CHECK("snd_pcm_hw_params_malloc", status);
status = snd_pcm_hw_params_any(alsa->pcm_handle, hw_params); status = snd_pcm_hw_params_any(alsa->pcm_handle, hw_params);
SND_PCM_CHECK("snd_pcm_hw_params_any", status); SND_PCM_CHECK("snd_pcm_hw_params_any", status);
/* Set interleaved read/write access */ /* Set interleaved read/write access */
status = snd_pcm_hw_params_set_access(alsa->pcm_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED); status = snd_pcm_hw_params_set_access(alsa->pcm_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);
SND_PCM_CHECK("snd_pcm_hw_params_set_access", status); SND_PCM_CHECK("snd_pcm_hw_params_set_access", status);
/* Set sample format */ /* Set sample format */
status = snd_pcm_hw_params_set_format(alsa->pcm_handle, hw_params, alsa->format); status = snd_pcm_hw_params_set_format(alsa->pcm_handle, hw_params, alsa->format);
SND_PCM_CHECK("snd_pcm_hw_params_set_format", status); SND_PCM_CHECK("snd_pcm_hw_params_set_format", status);
/* Set sample rate */ /* Set sample rate */
status = snd_pcm_hw_params_set_rate_near(alsa->pcm_handle, hw_params, &alsa->actual_rate, NULL); status = snd_pcm_hw_params_set_rate_near(alsa->pcm_handle, hw_params, &alsa->actual_rate, NULL);
SND_PCM_CHECK("snd_pcm_hw_params_set_rate_near", status); SND_PCM_CHECK("snd_pcm_hw_params_set_rate_near", status);
/* Set number of channels */ /* Set number of channels */
status = snd_pcm_hw_params_set_channels(alsa->pcm_handle, hw_params, alsa->actual_channels); status = snd_pcm_hw_params_set_channels(alsa->pcm_handle, hw_params, alsa->actual_channels);
SND_PCM_CHECK("snd_pcm_hw_params_set_channels", status); SND_PCM_CHECK("snd_pcm_hw_params_set_channels", status);
/* Get maximum buffer size */ /* Get maximum buffer size */
status = snd_pcm_hw_params_get_buffer_size_max(hw_params, &buffer_size_max); status = snd_pcm_hw_params_get_buffer_size_max(hw_params, &buffer_size_max);
SND_PCM_CHECK("snd_pcm_hw_params_get_buffer_size_max", status); SND_PCM_CHECK("snd_pcm_hw_params_get_buffer_size_max", status);
/** /**
* ALSA Parameters * ALSA Parameters
* *
@ -122,10 +110,8 @@ static int rdpsnd_alsa_set_hw_params(rdpsndAlsaPlugin* alsa)
* Commonly this is (2 * period_size), but some hardware can do 8 periods per buffer. * Commonly this is (2 * period_size), but some hardware can do 8 periods per buffer.
* It is also possible for the buffer size to not be an integer multiple of the period size. * It is also possible for the buffer size to not be an integer multiple of the period size.
*/ */
int interrupts_per_sec_near = 50; int interrupts_per_sec_near = 50;
int bytes_per_sec = (alsa->actual_rate * alsa->bytes_per_channel * alsa->actual_channels); int bytes_per_sec = (alsa->actual_rate * alsa->aformat.wBitsPerSample / 8 * alsa->actual_channels);
alsa->buffer_size = buffer_size_max; alsa->buffer_size = buffer_size_max;
alsa->period_size = (bytes_per_sec / interrupts_per_sec_near); alsa->period_size = (bytes_per_sec / interrupts_per_sec_near);
@ -139,16 +125,13 @@ static int rdpsnd_alsa_set_hw_params(rdpsndAlsaPlugin* alsa)
/* Set buffer size */ /* Set buffer size */
status = snd_pcm_hw_params_set_buffer_size_near(alsa->pcm_handle, hw_params, &alsa->buffer_size); status = snd_pcm_hw_params_set_buffer_size_near(alsa->pcm_handle, hw_params, &alsa->buffer_size);
SND_PCM_CHECK("snd_pcm_hw_params_set_buffer_size_near", status); SND_PCM_CHECK("snd_pcm_hw_params_set_buffer_size_near", status);
/* Set period size */ /* Set period size */
status = snd_pcm_hw_params_set_period_size_near(alsa->pcm_handle, hw_params, &alsa->period_size, NULL); status = snd_pcm_hw_params_set_period_size_near(alsa->pcm_handle, hw_params, &alsa->period_size,
NULL);
SND_PCM_CHECK("snd_pcm_hw_params_set_period_size_near", status); SND_PCM_CHECK("snd_pcm_hw_params_set_period_size_near", status);
status = snd_pcm_hw_params(alsa->pcm_handle, hw_params); status = snd_pcm_hw_params(alsa->pcm_handle, hw_params);
SND_PCM_CHECK("snd_pcm_hw_params", status); SND_PCM_CHECK("snd_pcm_hw_params", status);
snd_pcm_hw_params_free(hw_params); snd_pcm_hw_params_free(hw_params);
return 0; return 0;
} }
@ -156,27 +139,21 @@ static int rdpsnd_alsa_set_sw_params(rdpsndAlsaPlugin* alsa)
{ {
int status; int status;
snd_pcm_sw_params_t* sw_params; snd_pcm_sw_params_t* sw_params;
status = snd_pcm_sw_params_malloc(&sw_params); status = snd_pcm_sw_params_malloc(&sw_params);
SND_PCM_CHECK("snd_pcm_sw_params_malloc", status); SND_PCM_CHECK("snd_pcm_sw_params_malloc", status);
status = snd_pcm_sw_params_current(alsa->pcm_handle, sw_params); status = snd_pcm_sw_params_current(alsa->pcm_handle, sw_params);
SND_PCM_CHECK("snd_pcm_sw_params_current", status); SND_PCM_CHECK("snd_pcm_sw_params_current", status);
status = snd_pcm_sw_params_set_avail_min(alsa->pcm_handle, sw_params,
status = snd_pcm_sw_params_set_avail_min(alsa->pcm_handle, sw_params, (alsa->bytes_per_channel * alsa->actual_channels)); (alsa->aformat.nChannels * alsa->actual_channels));
SND_PCM_CHECK("snd_pcm_sw_params_set_avail_min", status); SND_PCM_CHECK("snd_pcm_sw_params_set_avail_min", status);
status = snd_pcm_sw_params_set_start_threshold(alsa->pcm_handle, sw_params,
status = snd_pcm_sw_params_set_start_threshold(alsa->pcm_handle, sw_params, alsa->block_size); alsa->aformat.nBlockAlign);
SND_PCM_CHECK("snd_pcm_sw_params_set_start_threshold", status); SND_PCM_CHECK("snd_pcm_sw_params_set_start_threshold", status);
status = snd_pcm_sw_params(alsa->pcm_handle, sw_params); status = snd_pcm_sw_params(alsa->pcm_handle, sw_params);
SND_PCM_CHECK("snd_pcm_sw_params", status); SND_PCM_CHECK("snd_pcm_sw_params", status);
snd_pcm_sw_params_free(sw_params); snd_pcm_sw_params_free(sw_params);
status = snd_pcm_prepare(alsa->pcm_handle); status = snd_pcm_prepare(alsa->pcm_handle);
SND_PCM_CHECK("snd_pcm_prepare", status); SND_PCM_CHECK("snd_pcm_prepare", status);
return 0; return 0;
} }
@ -185,10 +162,8 @@ static int rdpsnd_alsa_validate_params(rdpsndAlsaPlugin* alsa)
int status; int status;
snd_pcm_uframes_t buffer_size; snd_pcm_uframes_t buffer_size;
snd_pcm_uframes_t period_size; snd_pcm_uframes_t period_size;
status = snd_pcm_get_params(alsa->pcm_handle, &buffer_size, &period_size); status = snd_pcm_get_params(alsa->pcm_handle, &buffer_size, &period_size);
SND_PCM_CHECK("snd_pcm_get_params", status); SND_PCM_CHECK("snd_pcm_get_params", status);
return 0; return 0;
} }
@ -205,15 +180,15 @@ static int rdpsnd_alsa_set_params(rdpsndAlsaPlugin* alsa)
return rdpsnd_alsa_validate_params(alsa); return rdpsnd_alsa_validate_params(alsa);
} }
static BOOL rdpsnd_alsa_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_alsa_set_format(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format,
UINT32 latency)
{ {
rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device;
if (format) if (format)
{ {
alsa->source_rate = format->nSamplesPerSec; alsa->aformat = *format;
alsa->actual_rate = format->nSamplesPerSec; alsa->actual_rate = format->nSamplesPerSec;
alsa->source_channels = format->nChannels;
alsa->actual_channels = format->nChannels; alsa->actual_channels = format->nChannels;
switch (format->wFormatTag) switch (format->wFormatTag)
@ -223,29 +198,28 @@ static BOOL rdpsnd_alsa_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* for
{ {
case 8: case 8:
alsa->format = SND_PCM_FORMAT_S8; alsa->format = SND_PCM_FORMAT_S8;
alsa->bytes_per_channel = 1;
break; break;
case 16: case 16:
alsa->format = SND_PCM_FORMAT_S16_LE; alsa->format = SND_PCM_FORMAT_S16_LE;
alsa->bytes_per_channel = 2;
break;
}
break; break;
case WAVE_FORMAT_ADPCM: default:
case WAVE_FORMAT_DVI_ADPCM: return FALSE;
alsa->format = SND_PCM_FORMAT_S16_LE;
alsa->bytes_per_channel = 2;
break;
} }
alsa->wformat = format->wFormatTag; break;
alsa->block_size = format->nBlockAlign;
case WAVE_FORMAT_ALAW:
case WAVE_FORMAT_MULAW:
break;
default:
return FALSE;
}
} }
alsa->latency = latency; alsa->latency = latency;
return (rdpsnd_alsa_set_params(alsa) == 0); return (rdpsnd_alsa_set_params(alsa) == 0);
} }
@ -257,6 +231,7 @@ static BOOL rdpsnd_alsa_open_mixer(rdpsndAlsaPlugin* alsa)
return TRUE; return TRUE;
status = snd_mixer_open(&alsa->mixer_handle, 0); status = snd_mixer_open(&alsa->mixer_handle, 0);
if (status < 0) if (status < 0)
{ {
WLog_ERR(TAG, "snd_mixer_open failed"); WLog_ERR(TAG, "snd_mixer_open failed");
@ -264,6 +239,7 @@ static BOOL rdpsnd_alsa_open_mixer(rdpsndAlsaPlugin* alsa)
} }
status = snd_mixer_attach(alsa->mixer_handle, alsa->device_name); status = snd_mixer_attach(alsa->mixer_handle, alsa->device_name);
if (status < 0) if (status < 0)
{ {
WLog_ERR(TAG, "snd_mixer_attach failed"); WLog_ERR(TAG, "snd_mixer_attach failed");
@ -272,6 +248,7 @@ static BOOL rdpsnd_alsa_open_mixer(rdpsndAlsaPlugin* alsa)
} }
status = snd_mixer_selem_register(alsa->mixer_handle, NULL, NULL); status = snd_mixer_selem_register(alsa->mixer_handle, NULL, NULL);
if (status < 0) if (status < 0)
{ {
WLog_ERR(TAG, "snd_mixer_selem_register failed"); WLog_ERR(TAG, "snd_mixer_selem_register failed");
@ -280,6 +257,7 @@ static BOOL rdpsnd_alsa_open_mixer(rdpsndAlsaPlugin* alsa)
} }
status = snd_mixer_load(alsa->mixer_handle); status = snd_mixer_load(alsa->mixer_handle);
if (status < 0) if (status < 0)
{ {
WLog_ERR(TAG, "snd_mixer_load failed"); WLog_ERR(TAG, "snd_mixer_load failed");
@ -290,7 +268,7 @@ static BOOL rdpsnd_alsa_open_mixer(rdpsndAlsaPlugin* alsa)
return TRUE; return TRUE;
} }
static BOOL rdpsnd_alsa_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_alsa_open(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format, UINT32 latency)
{ {
int mode; int mode;
int status; int status;
@ -301,16 +279,14 @@ static BOOL rdpsnd_alsa_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, i
mode = 0; mode = 0;
/*mode |= SND_PCM_NONBLOCK;*/ /*mode |= SND_PCM_NONBLOCK;*/
status = snd_pcm_open(&alsa->pcm_handle, alsa->device_name, SND_PCM_STREAM_PLAYBACK, mode); status = snd_pcm_open(&alsa->pcm_handle, alsa->device_name, SND_PCM_STREAM_PLAYBACK, mode);
if (status < 0) if (status < 0)
{ {
WLog_ERR(TAG, "snd_pcm_open failed"); WLog_ERR(TAG, "snd_pcm_open failed");
return FALSE; return FALSE;
} }
freerdp_dsp_context_reset_adpcm(alsa->dsp_context);
return rdpsnd_alsa_set_format(device, format, latency) && return rdpsnd_alsa_set_format(device, format, latency) &&
rdpsnd_alsa_open_mixer(alsa); rdpsnd_alsa_open_mixer(alsa);
} }
@ -329,9 +305,6 @@ static void rdpsnd_alsa_close(rdpsndDevicePlugin* device)
if (status != 0) if (status != 0)
frames = 0; frames = 0;
alsa->wLocalTimeClose = GetTickCount();
alsa->wLocalTimeClose += (((frames * 1000) / alsa->actual_rate) / alsa->actual_channels);
} }
static void rdpsnd_alsa_free(rdpsndDevicePlugin* device) static void rdpsnd_alsa_free(rdpsndDevicePlugin* device)
@ -352,13 +325,10 @@ static void rdpsnd_alsa_free(rdpsndDevicePlugin* device)
} }
free(alsa->device_name); free(alsa->device_name);
freerdp_dsp_context_free(alsa->dsp_context);
free(alsa); free(alsa);
} }
static BOOL rdpsnd_alsa_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT* format) static BOOL rdpsnd_alsa_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
@ -370,22 +340,7 @@ static BOOL rdpsnd_alsa_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMA
{ {
return TRUE; 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:
case WAVE_FORMAT_MULAW:
case WAVE_FORMAT_GSM610:
default:
break; break;
} }
@ -403,7 +358,6 @@ static UINT32 rdpsnd_alsa_get_volume(rdpsndDevicePlugin* device)
UINT16 dwVolumeRight; UINT16 dwVolumeRight;
snd_mixer_elem_t* elem; snd_mixer_elem_t* elem;
rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device;
dwVolumeLeft = ((50 * 0xFFFF) / 100); /* 50% */ dwVolumeLeft = ((50 * 0xFFFF) / 100); /* 50% */
dwVolumeRight = ((50 * 0xFFFF) / 100); /* 50% */ dwVolumeRight = ((50 * 0xFFFF) / 100); /* 50% */
@ -417,16 +371,13 @@ static UINT32 rdpsnd_alsa_get_volume(rdpsndDevicePlugin* device)
snd_mixer_selem_get_playback_volume_range(elem, &volume_min, &volume_max); snd_mixer_selem_get_playback_volume_range(elem, &volume_min, &volume_max);
snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, &volume_left); snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, &volume_left);
snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, &volume_right); snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, &volume_right);
dwVolumeLeft = (UINT16)(((volume_left * 0xFFFF) - volume_min) / (volume_max - volume_min)); dwVolumeLeft = (UINT16)(((volume_left * 0xFFFF) - volume_min) / (volume_max - volume_min));
dwVolumeRight = (UINT16)(((volume_right * 0xFFFF) - volume_min) / (volume_max - volume_min)); dwVolumeRight = (UINT16)(((volume_right * 0xFFFF) - volume_min) / (volume_max - volume_min));
break; break;
} }
} }
dwVolume = (dwVolumeLeft << 16) | dwVolumeRight; dwVolume = (dwVolumeLeft << 16) | dwVolumeRight;
return dwVolume; return dwVolume;
} }
@ -454,6 +405,7 @@ static BOOL rdpsnd_alsa_set_volume(rdpsndDevicePlugin* device, UINT32 value)
snd_mixer_selem_get_playback_volume_range(elem, &volume_min, &volume_max); snd_mixer_selem_get_playback_volume_range(elem, &volume_min, &volume_max);
volume_left = volume_min + (left * (volume_max - volume_min)) / 0xFFFF; volume_left = volume_min + (left * (volume_max - volume_min)) / 0xFFFF;
volume_right = volume_min + (right * (volume_max - volume_min)) / 0xFFFF; volume_right = volume_min + (right * (volume_max - volume_min)) / 0xFFFF;
if ((snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, volume_left) < 0) || if ((snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, volume_left) < 0) ||
(snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, volume_right) < 0)) (snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, volume_right) < 0))
{ {
@ -466,108 +418,18 @@ static BOOL rdpsnd_alsa_set_volume(rdpsndDevicePlugin* device, UINT32 value)
return TRUE; return TRUE;
} }
static BYTE* rdpsnd_alsa_process_audio_sample(rdpsndDevicePlugin* device, BYTE* data, int* size) static UINT rdpsnd_alsa_play(rdpsndDevicePlugin* device, const BYTE* data, size_t size)
{ {
int frames; size_t offset;
BYTE* srcData;
int srcFrameSize;
int dstFrameSize;
rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device;
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;
srcData = 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;
srcData = alsa->dsp_context->adpcm_buffer;
}
else
{
srcData = data;
}
srcFrameSize = alsa->source_channels * alsa->bytes_per_channel;
dstFrameSize = alsa->actual_channels * alsa->bytes_per_channel;
if ((*size % srcFrameSize) != 0)
return NULL;
if (!((alsa->source_rate == alsa->actual_rate) && (alsa->source_channels == alsa->actual_channels)))
{
alsa->dsp_context->resample(alsa->dsp_context, srcData, alsa->bytes_per_channel,
alsa->source_channels, alsa->source_rate, *size / srcFrameSize,
alsa->actual_channels, alsa->actual_rate);
frames = alsa->dsp_context->resampled_frames;
*size = frames * dstFrameSize;
srcData = alsa->dsp_context->resampled_buffer;
}
data = srcData;
return data;
}
static BOOL rdpsnd_alsa_wave_decode(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
{
int size;
BYTE* data;
size = wave->length;
data = rdpsnd_alsa_process_audio_sample(device, wave->data, &size);
wave->data = (BYTE*) malloc(size);
if (!wave->data)
return FALSE;
CopyMemory(wave->data, data, size);
wave->length = size;
return TRUE;
}
static void rdpsnd_alsa_wave_play(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
{
BYTE* data;
int offset;
int length;
int status;
int frame_size; int frame_size;
UINT32 wCurrentTime;
snd_htimestamp_t tstamp;
snd_pcm_uframes_t frames;
rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device;
offset = 0; offset = 0;
data = wave->data; frame_size = alsa->actual_channels * alsa->aformat.wBitsPerSample / 8;
length = wave->length;
frame_size = alsa->actual_channels * alsa->bytes_per_channel;
if (alsa->wLocalTimeClose) while (offset < size)
{ {
wCurrentTime = GetTickCount(); snd_pcm_sframes_t status = snd_pcm_writei(alsa->pcm_handle, &data[offset],
(size - offset) / frame_size);
if (snd_pcm_htimestamp(alsa->pcm_handle, &frames, &tstamp) == -EPIPE)
{
if (wCurrentTime > alsa->wLocalTimeClose)
snd_pcm_recover(alsa->pcm_handle, -EPIPE, 1);
}
alsa->wLocalTimeClose = 0;
}
while (offset < length)
{
status = snd_pcm_writei(alsa->pcm_handle, &data[offset], (length - offset) / frame_size);
if (status == -EPIPE) if (status == -EPIPE)
{ {
@ -590,11 +452,7 @@ static void rdpsnd_alsa_wave_play(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
offset += status * frame_size; offset += status * frame_size;
} }
free(data); return 10; /* TODO: Get real latency in [ms] */
/* From rdpsnd_main.c */
wave->wTimeStampB = wave->wTimeStampA + wave->wAudioLength + 65;
wave->wLocalTimeB = wave->wLocalTimeA + wave->wAudioLength + 65;
} }
static COMMAND_LINE_ARGUMENT_A rdpsnd_alsa_args[] = static COMMAND_LINE_ARGUMENT_A rdpsnd_alsa_args[] =
@ -614,10 +472,10 @@ static UINT rdpsnd_alsa_parse_addin_args(rdpsndDevicePlugin* device, ADDIN_ARGV*
DWORD flags; DWORD flags;
COMMAND_LINE_ARGUMENT_A* arg; COMMAND_LINE_ARGUMENT_A* arg;
rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device; rdpsndAlsaPlugin* alsa = (rdpsndAlsaPlugin*) device;
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD; flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, rdpsnd_alsa_args, flags,
alsa, NULL, NULL);
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, rdpsnd_alsa_args, flags, alsa, NULL, NULL);
if (status < 0) if (status < 0)
{ {
WLog_ERR(TAG, "CommandLineParseArgumentsA failed!"); WLog_ERR(TAG, "CommandLineParseArgumentsA failed!");
@ -632,14 +490,13 @@ static UINT rdpsnd_alsa_parse_addin_args(rdpsndDevicePlugin* device, ADDIN_ARGV*
continue; continue;
CommandLineSwitchStart(arg) CommandLineSwitchStart(arg)
CommandLineSwitchCase(arg, "dev") CommandLineSwitchCase(arg, "dev")
{ {
alsa->device_name = _strdup(arg->Value); alsa->device_name = _strdup(arg->Value);
if (!alsa->device_name) if (!alsa->device_name)
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
CommandLineSwitchEnd(arg) CommandLineSwitchEnd(arg)
} }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL); while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
@ -663,8 +520,8 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
ADDIN_ARGV* args; ADDIN_ARGV* args;
rdpsndAlsaPlugin* alsa; rdpsndAlsaPlugin* alsa;
UINT error; UINT error;
alsa = (rdpsndAlsaPlugin*) calloc(1, sizeof(rdpsndAlsaPlugin)); alsa = (rdpsndAlsaPlugin*) calloc(1, sizeof(rdpsndAlsaPlugin));
if (!alsa) if (!alsa)
{ {
WLog_ERR(TAG, "calloc failed!"); WLog_ERR(TAG, "calloc failed!");
@ -673,15 +530,13 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
alsa->device.Open = rdpsnd_alsa_open; alsa->device.Open = rdpsnd_alsa_open;
alsa->device.FormatSupported = rdpsnd_alsa_format_supported; alsa->device.FormatSupported = rdpsnd_alsa_format_supported;
alsa->device.SetFormat = rdpsnd_alsa_set_format;
alsa->device.GetVolume = rdpsnd_alsa_get_volume; alsa->device.GetVolume = rdpsnd_alsa_get_volume;
alsa->device.SetVolume = rdpsnd_alsa_set_volume; alsa->device.SetVolume = rdpsnd_alsa_set_volume;
alsa->device.WaveDecode = rdpsnd_alsa_wave_decode; alsa->device.Play = rdpsnd_alsa_play;
alsa->device.WavePlay = rdpsnd_alsa_wave_play;
alsa->device.Close = rdpsnd_alsa_close; alsa->device.Close = rdpsnd_alsa_close;
alsa->device.Free = rdpsnd_alsa_free; alsa->device.Free = rdpsnd_alsa_free;
args = pEntryPoints->args; args = pEntryPoints->args;
if (args->argc > 1) if (args->argc > 1)
{ {
if ((error = rdpsnd_alsa_parse_addin_args((rdpsndDevicePlugin*) alsa, args))) if ((error = rdpsnd_alsa_parse_addin_args((rdpsndDevicePlugin*) alsa, args)))
@ -691,10 +546,10 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
} }
} }
if (!alsa->device_name) if (!alsa->device_name)
{ {
alsa->device_name = _strdup("default"); alsa->device_name = _strdup("default");
if (!alsa->device_name) if (!alsa->device_name)
{ {
WLog_ERR(TAG, "_strdup failed!"); WLog_ERR(TAG, "_strdup failed!");
@ -704,27 +559,11 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
} }
alsa->pcm_handle = 0; alsa->pcm_handle = 0;
alsa->source_rate = 22050;
alsa->actual_rate = 22050; alsa->actual_rate = 22050;
alsa->format = SND_PCM_FORMAT_S16_LE; alsa->format = SND_PCM_FORMAT_S16_LE;
alsa->source_channels = 2;
alsa->actual_channels = 2; alsa->actual_channels = 2;
alsa->bytes_per_channel = 2;
alsa->dsp_context = freerdp_dsp_context_new();
if (!alsa->dsp_context)
{
WLog_ERR(TAG, "freerdp_dsp_context_new failed!");
error = CHANNEL_RC_NO_MEMORY;
goto error_dsp_context;
}
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) alsa); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) alsa);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error_dsp_context:
freerdp_dsp_context_free(alsa->dsp_context);
error_strdup: error_strdup:
free(alsa->device_name); free(alsa->device_name);
error_parse_args: error_parse_args:

View File

@ -0,0 +1,33 @@
# FreeRDP: A Remote Desktop Protocol Implementation
# FreeRDP cmake build script
#
# Copyright 2019 Armin Novak <armin.novak@thincast.com>
# Copyright 2019 Thincast Technologies GmbH
#
# 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.
define_channel_client_subsystem("rdpsnd" "fake" "")
set(${MODULE_PREFIX}_SRCS
rdpsnd_fake.c)
include_directories(..)
add_channel_client_subsystem_library(${MODULE_PREFIX} ${MODULE_NAME} ${CHANNEL_NAME} "" TRUE "")
list(APPEND ${MODULE_PREFIX}_LIBS freerdp)
list(APPEND ${MODULE_PREFIX}_LIBS winpr)
target_link_libraries(${MODULE_NAME} ${${MODULE_PREFIX}_LIBS})
set_property(TARGET ${MODULE_NAME} PROPERTY FOLDER "Channels/${CHANNEL_NAME}/Client/Fake")

View File

@ -0,0 +1,175 @@
/**
* FreeRDP: A Remote Desktop Protocol Implementation
* Audio Output Virtual Channel
*
* Copyright 2019 Armin Novak <armin.novak@thincast.com>
* Copyright 2019 Thincast Technologies GmbH
*
* 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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <winpr/crt.h>
#include <winpr/stream.h>
#include <winpr/cmdline.h>
#include <freerdp/types.h>
#include "rdpsnd_main.h"
typedef struct rdpsnd_fake_plugin rdpsndFakePlugin;
struct rdpsnd_fake_plugin
{
rdpsndDevicePlugin device;
};
static BOOL rdpsnd_fake_open(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format, UINT32 latency)
{
return TRUE;
}
static void rdpsnd_fake_close(rdpsndDevicePlugin* device)
{
}
static BOOL rdpsnd_fake_set_volume(rdpsndDevicePlugin* device, UINT32 value)
{
return TRUE;
}
static void rdpsnd_fake_free(rdpsndDevicePlugin* device)
{
rdpsndFakePlugin* fake = (rdpsndFakePlugin*) device;
if (!fake)
return;
free(fake);
}
static BOOL rdpsnd_fake_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format)
{
return TRUE;
}
static BOOL rdpsnd_fake_set_format(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format, int latency)
{
return TRUE;
}
static UINT rdpsnd_fake_play(rdpsndDevicePlugin* device, const BYTE* data, size_t size)
{
return CHANNEL_RC_OK;
}
static void rdpsnd_fake_start(rdpsndDevicePlugin* device)
{
}
static COMMAND_LINE_ARGUMENT_A rdpsnd_fake_args[] =
{
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
};
/**
* Function description
*
* @return 0 on success, otherwise a Win32 error code
*/
static UINT rdpsnd_fake_parse_addin_args(rdpsndFakePlugin* fake, ADDIN_ARGV* args)
{
int status;
DWORD flags;
COMMAND_LINE_ARGUMENT_A* arg;
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv,
rdpsnd_fake_args, flags, fake, NULL, NULL);
if (status < 0)
return ERROR_INVALID_DATA;
arg = rdpsnd_fake_args;
do
{
if (!(arg->Flags & COMMAND_LINE_VALUE_PRESENT))
continue;
CommandLineSwitchStart(arg)
CommandLineSwitchEnd(arg)
}
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
return CHANNEL_RC_OK;
}
#ifdef BUILTIN_CHANNELS
#define freerdp_rdpsnd_client_subsystem_entry fake_freerdp_rdpsnd_client_subsystem_entry
#else
#define freerdp_rdpsnd_client_subsystem_entry FREERDP_API freerdp_rdpsnd_client_subsystem_entry
#endif
/**
* Function description
*
* @return 0 on success, otherwise a Win32 error code
*/
UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS pEntryPoints)
{
ADDIN_ARGV* args;
rdpsndFakePlugin* fake;
UINT ret;
fake = (rdpsndFakePlugin*) calloc(1, sizeof(rdpsndFakePlugin));
if (!fake)
return CHANNEL_RC_NO_MEMORY;
fake->device.Open = rdpsnd_fake_open;
fake->device.FormatSupported = rdpsnd_fake_format_supported;
fake->device.SetVolume = rdpsnd_fake_set_volume;
fake->device.Play = rdpsnd_fake_play;
fake->device.Start = rdpsnd_fake_start;
fake->device.Close = rdpsnd_fake_close;
fake->device.Free = rdpsnd_fake_free;
args = pEntryPoints->args;
if (args->argc > 1)
{
ret = rdpsnd_fake_parse_addin_args(fake, args);
if (ret != CHANNEL_RC_OK)
{
WLog_ERR(TAG, "error parsing arguments");
goto error;
}
}
ret = CHANNEL_RC_NO_MEMORY;
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, &fake->device);
return CHANNEL_RC_OK;
error:
rdpsnd_fake_free(&fake->device);
return ret;
}

View File

@ -68,16 +68,14 @@ static OSStatus rdpsnd_ios_render_cb(
for (i = 0; i < ioData->mNumberBuffers; i++) for (i = 0; i < ioData->mNumberBuffers; i++)
{ {
AudioBuffer* target_buffer = &ioData->mBuffers[i]; AudioBuffer* target_buffer = &ioData->mBuffers[i];
int32_t available_bytes = 0; int32_t available_bytes = 0;
const void* buffer = TPCircularBufferTail(&p->buffer, &available_bytes); const void* buffer = TPCircularBufferTail(&p->buffer, &available_bytes);
if (buffer != NULL && available_bytes > 0) if (buffer != NULL && available_bytes > 0)
{ {
const int bytes_to_copy = MIN((int32_t)target_buffer->mDataByteSize, available_bytes); const int bytes_to_copy = MIN((int32_t)target_buffer->mDataByteSize, available_bytes);
memcpy(target_buffer->mData, buffer, bytes_to_copy); memcpy(target_buffer->mData, buffer, bytes_to_copy);
target_buffer->mDataByteSize = bytes_to_copy; target_buffer->mDataByteSize = bytes_to_copy;
TPCircularBufferConsume(&p->buffer, bytes_to_copy); TPCircularBufferConsume(&p->buffer, bytes_to_copy);
} }
else else
@ -97,10 +95,12 @@ static BOOL rdpsnd_ios_format_supported(rdpsndDevicePlugin* __unused device, AUD
{ {
return 1; return 1;
} }
return 0; return 0;
} }
static BOOL rdpsnd_ios_set_format(rdpsndDevicePlugin* __unused device, AUDIO_FORMAT* __unused format, int __unused latency) static BOOL rdpsnd_ios_set_format(rdpsndDevicePlugin* __unused device,
AUDIO_FORMAT* __unused format, int __unused latency)
{ {
return TRUE; return TRUE;
} }
@ -120,6 +120,7 @@ static void rdpsnd_ios_start(rdpsndDevicePlugin* device)
/* Start the device. */ /* Start the device. */
int32_t available_bytes = 0; int32_t available_bytes = 0;
TPCircularBufferTail(&p->buffer, &available_bytes); TPCircularBufferTail(&p->buffer, &available_bytes);
if (available_bytes > 0) if (available_bytes > 0)
{ {
p->is_playing = 1; p->is_playing = 1;
@ -138,23 +139,21 @@ static void rdpsnd_ios_stop(rdpsndDevicePlugin* __unused device)
/* Stop the device. */ /* Stop the device. */
AudioOutputUnitStop(p->audio_unit); AudioOutputUnitStop(p->audio_unit);
p->is_playing = 0; p->is_playing = 0;
/* Free all buffers. */ /* Free all buffers. */
TPCircularBufferClear(&p->buffer); TPCircularBufferClear(&p->buffer);
} }
} }
static void rdpsnd_ios_play(rdpsndDevicePlugin* device, BYTE* data, int size) static UINT rdpsnd_ios_play(rdpsndDevicePlugin* device, BYTE* data, int size)
{ {
rdpsndIOSPlugin* p = THIS(device); rdpsndIOSPlugin* p = THIS(device);
const BOOL ok = TPCircularBufferProduceBytes(&p->buffer, data, size); const BOOL ok = TPCircularBufferProduceBytes(&p->buffer, data, size);
if (!ok) if (!ok)
{ return 0;
return;
}
rdpsnd_ios_start(device); rdpsnd_ios_start(device);
return 10; /* TODO: Get real latencry in [ms] */
} }
static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int __unused latency) static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int __unused latency)
@ -171,13 +170,14 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
desc.componentSubType = kAudioUnitSubType_RemoteIO; desc.componentSubType = kAudioUnitSubType_RemoteIO;
desc.componentFlags = 0; desc.componentFlags = 0;
desc.componentFlagsMask = 0; desc.componentFlagsMask = 0;
AudioComponent audioComponent = AudioComponentFindNext(NULL, &desc); AudioComponent audioComponent = AudioComponentFindNext(NULL, &desc);
if (audioComponent == NULL) if (audioComponent == NULL)
return FALSE; return FALSE;
/* Open the audio unit. */ /* Open the audio unit. */
OSStatus status = AudioComponentInstanceNew(audioComponent, &p->audio_unit); OSStatus status = AudioComponentInstanceNew(audioComponent, &p->audio_unit);
if (status != 0) if (status != 0)
return FALSE; return FALSE;
@ -191,7 +191,6 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
audioFormat.mBitsPerChannel = format->wBitsPerSample; audioFormat.mBitsPerChannel = format->wBitsPerSample;
audioFormat.mBytesPerFrame = (format->wBitsPerSample * format->nChannels) / 8; audioFormat.mBytesPerFrame = (format->wBitsPerSample * format->nChannels) / 8;
audioFormat.mBytesPerPacket = audioFormat.mBytesPerFrame * audioFormat.mFramesPerPacket; audioFormat.mBytesPerPacket = audioFormat.mBytesPerFrame * audioFormat.mFramesPerPacket;
status = AudioUnitSetProperty( status = AudioUnitSetProperty(
p->audio_unit, p->audio_unit,
kAudioUnitProperty_StreamFormat, kAudioUnitProperty_StreamFormat,
@ -199,6 +198,7 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
0, 0,
&audioFormat, &audioFormat,
sizeof(audioFormat)); sizeof(audioFormat));
if (status != 0) if (status != 0)
{ {
AudioComponentInstanceDispose(p->audio_unit); AudioComponentInstanceDispose(p->audio_unit);
@ -217,6 +217,7 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
0, 0,
&callbackStruct, &callbackStruct,
sizeof(callbackStruct)); sizeof(callbackStruct));
if (status != 0) if (status != 0)
{ {
AudioComponentInstanceDispose(p->audio_unit); AudioComponentInstanceDispose(p->audio_unit);
@ -226,6 +227,7 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
/* Initialize the AudioUnit. */ /* Initialize the AudioUnit. */
status = AudioUnitInitialize(p->audio_unit); status = AudioUnitInitialize(p->audio_unit);
if (status != 0) if (status != 0)
{ {
AudioComponentInstanceDispose(p->audio_unit); AudioComponentInstanceDispose(p->audio_unit);
@ -235,6 +237,7 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
/* Allocate the circular buffer. */ /* Allocate the circular buffer. */
const BOOL ok = TPCircularBufferInit(&p->buffer, CIRCULAR_BUFFER_SIZE); const BOOL ok = TPCircularBufferInit(&p->buffer, CIRCULAR_BUFFER_SIZE);
if (!ok) if (!ok)
{ {
AudioUnitUninitialize(p->audio_unit); AudioUnitUninitialize(p->audio_unit);
@ -250,7 +253,6 @@ static BOOL rdpsnd_ios_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
static void rdpsnd_ios_close(rdpsndDevicePlugin* device) static void rdpsnd_ios_close(rdpsndDevicePlugin* device)
{ {
rdpsndIOSPlugin* p = THIS(device); rdpsndIOSPlugin* p = THIS(device);
/* Make sure the device is stopped. */ /* Make sure the device is stopped. */
rdpsnd_ios_stop(device); rdpsnd_ios_stop(device);
@ -262,7 +264,6 @@ static void rdpsnd_ios_close(rdpsndDevicePlugin* device)
AudioComponentInstanceDispose(p->audio_unit); AudioComponentInstanceDispose(p->audio_unit);
p->audio_unit = NULL; p->audio_unit = NULL;
p->is_opened = 0; p->is_opened = 0;
/* Destroy the circular buffer. */ /* Destroy the circular buffer. */
TPCircularBufferCleanup(&p->buffer); TPCircularBufferCleanup(&p->buffer);
} }
@ -271,10 +272,8 @@ static void rdpsnd_ios_close(rdpsndDevicePlugin* device)
static void rdpsnd_ios_free(rdpsndDevicePlugin* device) static void rdpsnd_ios_free(rdpsndDevicePlugin* device)
{ {
rdpsndIOSPlugin* p = THIS(device); rdpsndIOSPlugin* p = THIS(device);
/* Ensure the device is closed. */ /* Ensure the device is closed. */
rdpsnd_ios_close(device); rdpsnd_ios_close(device);
/* Free memory associated with the device. */ /* Free memory associated with the device. */
free(p); free(p);
} }
@ -305,8 +304,6 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
p->device.Start = rdpsnd_ios_start; p->device.Start = rdpsnd_ios_start;
p->device.Close = rdpsnd_ios_close; p->device.Close = rdpsnd_ios_close;
p->device.Free = rdpsnd_ios_free; p->device.Free = rdpsnd_ios_free;
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*)p); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*)p);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }

View File

@ -32,7 +32,6 @@
#include <winpr/crt.h> #include <winpr/crt.h>
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/codec/dsp.h>
#define __COREFOUNDATION_CFPLUGINCOM__ 1 #define __COREFOUNDATION_CFPLUGINCOM__ 1
#define IUNKNOWN_C_GUTS void *_reserved; void* QueryInterface; void* AddRef; void* Release #define IUNKNOWN_C_GUTS void *_reserved; void* QueryInterface; void* AddRef; void* Release
@ -60,32 +59,27 @@ struct rdpsnd_mac_plugin
AudioQueueRef audioQueue; AudioQueueRef audioQueue;
AudioStreamBasicDescription audioFormat; AudioStreamBasicDescription audioFormat;
AudioQueueBufferRef audioBuffers[MAC_AUDIO_QUEUE_NUM_BUFFERS]; AudioQueueBufferRef audioBuffers[MAC_AUDIO_QUEUE_NUM_BUFFERS];
Float64 lastStartTime;
int wformat;
int block_size;
FREERDP_DSP_CONTEXT* dsp_context;
}; };
typedef struct rdpsnd_mac_plugin rdpsndMacPlugin; typedef struct rdpsnd_mac_plugin rdpsndMacPlugin;
static void mac_audio_queue_output_cb(void* inUserData, AudioQueueRef inAQ, AudioQueueBufferRef inBuffer) static void mac_audio_queue_output_cb(void* inUserData, AudioQueueRef inAQ,
AudioQueueBufferRef inBuffer)
{ {
rdpsndMacPlugin* mac = (rdpsndMacPlugin*)inUserData; rdpsndMacPlugin* mac = (rdpsndMacPlugin*)inUserData;
if (inBuffer == mac->audioBuffers[mac->lastAudioBufferIndex]) { if (inBuffer == mac->audioBuffers[mac->lastAudioBufferIndex])
{
AudioQueuePause(mac->audioQueue); AudioQueuePause(mac->audioQueue);
mac->isPlaying = FALSE; mac->isPlaying = FALSE;
} }
} }
static BOOL rdpsnd_mac_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_mac_set_format(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format,
UINT32 latency)
{ {
rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device; rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device;
mac->latency = (UINT32) latency; mac->latency = (UINT32) latency;
CopyMemory(&(mac->format), format, sizeof(AUDIO_FORMAT)); CopyMemory(&(mac->format), format, sizeof(AUDIO_FORMAT));
mac->audioFormat.mSampleRate = format->nSamplesPerSec; mac->audioFormat.mSampleRate = format->nSamplesPerSec;
mac->audioFormat.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagIsPacked; mac->audioFormat.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagIsPacked;
mac->audioFormat.mFramesPerPacket = 1; mac->audioFormat.mFramesPerPacket = 1;
@ -109,34 +103,18 @@ static BOOL rdpsnd_mac_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* form
mac->audioFormat.mFormatID = kAudioFormatLinearPCM; mac->audioFormat.mFormatID = kAudioFormatLinearPCM;
break; break;
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
mac->audioFormat.mFormatID = kAudioFormatLinearPCM;
mac->audioFormat.mBitsPerChannel = 16;
mac->audioFormat.mBytesPerFrame = (16 * format->nChannels) / 8;
mac->audioFormat.mBytesPerPacket = mac->audioFormat.mFramesPerPacket * mac->audioFormat.mBytesPerFrame;
break;
case WAVE_FORMAT_GSM610:
mac->audioFormat.mFormatID = kAudioFormatMicrosoftGSM;
break;
default: default:
break; break;
} }
mac->wformat = format->wFormatTag;
mac->block_size = format->nBlockAlign;
rdpsnd_print_audio_format(format); rdpsnd_print_audio_format(format);
return TRUE; return TRUE;
} }
static BOOL rdpsnd_mac_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_mac_open(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format, UINT32 latency)
{ {
int index; int index;
OSStatus status; OSStatus status;
rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device; rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device;
if (mac->isOpen) if (mac->isOpen)
@ -144,13 +122,8 @@ static BOOL rdpsnd_mac_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
mac->audioBufferIndex = 0; mac->audioBufferIndex = 0;
if (!device->SetFormat(device, format, 0)) if (!rdpsnd_mac_set_format(device, format, latency))
{
WLog_ERR(TAG, "SetFormat failure\n");
return FALSE; return FALSE;
}
freerdp_dsp_context_reset_adpcm(mac->dsp_context);
status = AudioQueueNewOutput(&(mac->audioFormat), status = AudioQueueNewOutput(&(mac->audioFormat),
mac_audio_queue_output_cb, mac, mac_audio_queue_output_cb, mac,
@ -164,7 +137,6 @@ static BOOL rdpsnd_mac_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
UInt32 DecodeBufferSizeFrames; UInt32 DecodeBufferSizeFrames;
UInt32 propertySize = sizeof(DecodeBufferSizeFrames); UInt32 propertySize = sizeof(DecodeBufferSizeFrames);
status = AudioQueueGetProperty(mac->audioQueue, status = AudioQueueGetProperty(mac->audioQueue,
kAudioQueueProperty_DecodeBufferSizeFrames, kAudioQueueProperty_DecodeBufferSizeFrames,
&DecodeBufferSizeFrames, &DecodeBufferSizeFrames,
@ -178,7 +150,8 @@ static BOOL rdpsnd_mac_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
for (index = 0; index < MAC_AUDIO_QUEUE_NUM_BUFFERS; index++) for (index = 0; index < MAC_AUDIO_QUEUE_NUM_BUFFERS; index++)
{ {
status = AudioQueueAllocateBuffer(mac->audioQueue, MAC_AUDIO_QUEUE_BUFFER_SIZE, &mac->audioBuffers[index]); status = AudioQueueAllocateBuffer(mac->audioQueue, MAC_AUDIO_QUEUE_BUFFER_SIZE,
&mac->audioBuffers[index]);
if (status != 0) if (status != 0)
{ {
@ -187,8 +160,6 @@ static BOOL rdpsnd_mac_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
} }
} }
mac->lastStartTime = 0;
mac->isOpen = TRUE; mac->isOpen = TRUE;
return TRUE; return TRUE;
} }
@ -201,7 +172,6 @@ static void rdpsnd_mac_close(rdpsndDevicePlugin* device)
{ {
size_t index; size_t index;
mac->isOpen = FALSE; mac->isOpen = FALSE;
AudioQueueStop(mac->audioQueue, true); AudioQueueStop(mac->audioQueue, true);
for (index = 0; index < MAC_AUDIO_QUEUE_NUM_BUFFERS; index++) for (index = 0; index < MAC_AUDIO_QUEUE_NUM_BUFFERS; index++)
@ -211,7 +181,6 @@ static void rdpsnd_mac_close(rdpsndDevicePlugin* device)
AudioQueueDispose(mac->audioQueue, true); AudioQueueDispose(mac->audioQueue, true);
mac->audioQueue = NULL; mac->audioQueue = NULL;
mac->isPlaying = FALSE; mac->isPlaying = FALSE;
} }
} }
@ -219,25 +188,21 @@ static void rdpsnd_mac_close(rdpsndDevicePlugin* device)
static void rdpsnd_mac_free(rdpsndDevicePlugin* device) static void rdpsnd_mac_free(rdpsndDevicePlugin* device)
{ {
rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device; rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device;
device->Close(device); device->Close(device);
freerdp_dsp_context_free(mac->dsp_context);
free(mac); free(mac);
} }
static BOOL rdpsnd_mac_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT* format) static BOOL rdpsnd_mac_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
case WAVE_FORMAT_PCM: case WAVE_FORMAT_PCM:
case WAVE_FORMAT_ALAW: case WAVE_FORMAT_ALAW:
case WAVE_FORMAT_MULAW: case WAVE_FORMAT_MULAW:
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
return TRUE;
case WAVE_FORMAT_GSM610: case WAVE_FORMAT_GSM610:
return TRUE;
default:
return FALSE; return FALSE;
} }
@ -257,9 +222,7 @@ static BOOL rdpsnd_mac_set_volume(rdpsndDevicePlugin* device, UINT32 value)
volumeLeft = (value & 0xFFFF); volumeLeft = (value & 0xFFFF);
volumeRight = ((value >> 16) & 0xFFFF); volumeRight = ((value >> 16) & 0xFFFF);
fVolume = ((float) volumeLeft) / 65535.0; fVolume = ((float) volumeLeft) / 65535.0;
status = AudioQueueSetParameter(mac->audioQueue, kAudioQueueParam_Volume, fVolume); status = AudioQueueSetParameter(mac->audioQueue, kAudioQueueParam_Volume, fVolume);
if (status != 0) if (status != 0)
@ -293,68 +256,28 @@ static void rdpsnd_mac_start(rdpsndDevicePlugin* device)
} }
} }
static BOOL rdpsnd_mac_wave_decode(rdpsndDevicePlugin* device, RDPSND_WAVE* wave) static UINT rdpsnd_mac_play(rdpsndDevicePlugin* device, const BYTE* data, size_t size)
{ {
int length; size_t length;
BYTE* data;
rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device;
if (mac->wformat == WAVE_FORMAT_ADPCM)
{
mac->dsp_context->decode_ms_adpcm(mac->dsp_context, wave->data, wave->length, mac->format.nChannels, mac->block_size);
length = mac->dsp_context->adpcm_size;
data = mac->dsp_context->adpcm_buffer;
}
else if (mac->wformat == WAVE_FORMAT_DVI_ADPCM)
{
mac->dsp_context->decode_ima_adpcm(mac->dsp_context, wave->data, wave->length, mac->format.nChannels, mac->block_size);
length = mac->dsp_context->adpcm_size;
data = mac->dsp_context->adpcm_buffer;
}
else
{
length = wave->length;
data = wave->data;
}
wave->data = (BYTE*) malloc(length);
CopyMemory(wave->data, data, length);
wave->length = length;
return TRUE;
}
static void rdpsnd_mac_waveplay(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
{
int length;
AudioQueueBufferRef audioBuffer; AudioQueueBufferRef audioBuffer;
AudioTimeStamp outActualStartTime; AudioTimeStamp outActualStartTime;
rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device; rdpsndMacPlugin* mac = (rdpsndMacPlugin*) device;
if (!mac->isOpen) if (!mac->isOpen)
return; return 0;
audioBuffer = mac->audioBuffers[mac->audioBufferIndex]; audioBuffer = mac->audioBuffers[mac->audioBufferIndex];
length = size > audioBuffer->mAudioDataBytesCapacity ? audioBuffer->mAudioDataBytesCapacity : size;
length = wave->length > audioBuffer->mAudioDataBytesCapacity ? audioBuffer->mAudioDataBytesCapacity : wave->length; CopyMemory(audioBuffer->mAudioData, data, length);
CopyMemory(audioBuffer->mAudioData, wave->data, length);
free(wave->data);
wave->data = NULL;
audioBuffer->mAudioDataByteSize = length; audioBuffer->mAudioDataByteSize = length;
audioBuffer->mUserData = wave; audioBuffer->mUserData = mac;
AudioQueueEnqueueBufferWithParameters(mac->audioQueue, audioBuffer, 0, 0, 0, 0, 0, NULL, NULL,
AudioQueueEnqueueBufferWithParameters(mac->audioQueue, audioBuffer, 0, 0, 0, 0, 0, NULL, NULL, &outActualStartTime); &outActualStartTime);
UInt64 startTimeDelta = (outActualStartTime.mSampleTime - mac->lastStartTime) / 100.0;
wave->wLocalTimeB = wave->wLocalTimeA + startTimeDelta + wave->wAudioLength;
wave->wTimeStampB = wave->wTimeStampA + wave->wLocalTimeB - wave->wLocalTimeA;
mac->lastStartTime = outActualStartTime.mSampleTime;
mac->lastAudioBufferIndex = mac->audioBufferIndex; mac->lastAudioBufferIndex = mac->audioBufferIndex;
mac->audioBufferIndex++; mac->audioBufferIndex++;
mac->audioBufferIndex %= MAC_AUDIO_QUEUE_NUM_BUFFERS; mac->audioBufferIndex %= MAC_AUDIO_QUEUE_NUM_BUFFERS;
rdpsnd_mac_start(device);
device->Start(device); return 10; /* TODO: Get real latencry in [ms] */
} }
#ifdef BUILTIN_CHANNELS #ifdef BUILTIN_CHANNELS
@ -371,7 +294,6 @@ static void rdpsnd_mac_waveplay(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS pEntryPoints) UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS pEntryPoints)
{ {
rdpsndMacPlugin* mac; rdpsndMacPlugin* mac;
mac = (rdpsndMacPlugin*) calloc(1, sizeof(rdpsndMacPlugin)); mac = (rdpsndMacPlugin*) calloc(1, sizeof(rdpsndMacPlugin));
if (!mac) if (!mac)
@ -379,17 +301,10 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
mac->device.Open = rdpsnd_mac_open; mac->device.Open = rdpsnd_mac_open;
mac->device.FormatSupported = rdpsnd_mac_format_supported; mac->device.FormatSupported = rdpsnd_mac_format_supported;
mac->device.SetFormat = rdpsnd_mac_set_format;
mac->device.SetVolume = rdpsnd_mac_set_volume; mac->device.SetVolume = rdpsnd_mac_set_volume;
mac->device.WaveDecode = rdpsnd_mac_wave_decode; mac->device.Play = rdpsnd_mac_play;
mac->device.WavePlay = rdpsnd_mac_waveplay;
mac->device.Start = rdpsnd_mac_start;
mac->device.Close = rdpsnd_mac_close; mac->device.Close = rdpsnd_mac_close;
mac->device.Free = rdpsnd_mac_free; mac->device.Free = rdpsnd_mac_free;
mac->dsp_context = freerdp_dsp_context_new();
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) mac); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) mac);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }

View File

@ -36,7 +36,6 @@
#include <winpr/collections.h> #include <winpr/collections.h>
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/log.h> #include <freerdp/channels/log.h>
#include "opensl_io.h" #include "opensl_io.h"
@ -48,7 +47,7 @@ struct rdpsnd_opensles_plugin
{ {
rdpsndDevicePlugin device; rdpsndDevicePlugin device;
int latency; UINT32 latency;
int wformat; int wformat;
int block_size; int block_size;
char* device_name; char* device_name;
@ -60,7 +59,6 @@ struct rdpsnd_opensles_plugin
UINT32 rate; UINT32 rate;
UINT32 channels; UINT32 channels;
int format; int format;
FREERDP_DSP_CONTEXT* dsp_context;
}; };
static int rdpsnd_opensles_volume_to_millibel(unsigned short level, int max) static int rdpsnd_opensles_volume_to_millibel(unsigned short level, int max)
@ -91,9 +89,6 @@ static bool rdpsnd_opensles_check_handle(const rdpsndopenslesPlugin* hdl)
rc = false; rc = false;
else else
{ {
if (!hdl->dsp_context)
rc = false;
if (!hdl->stream) if (!hdl->stream)
rc = false; rc = false;
} }
@ -120,11 +115,11 @@ static int rdpsnd_opensles_set_params(rdpsndopenslesPlugin* opensles)
} }
static BOOL rdpsnd_opensles_set_format(rdpsndDevicePlugin* device, static BOOL rdpsnd_opensles_set_format(rdpsndDevicePlugin* device,
AUDIO_FORMAT* format, int latency) const AUDIO_FORMAT* format, UINT32 latency)
{ {
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device; rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
rdpsnd_opensles_check_handle(opensles); rdpsnd_opensles_check_handle(opensles);
DEBUG_SND("opensles=%p format=%p, latency=%d", (void*) opensles, (void*) format, latency); DEBUG_SND("opensles=%p format=%p, latency=%"PRIu32, (void*) opensles, (void*) format, latency);
if (format) if (format)
{ {
@ -143,10 +138,10 @@ static BOOL rdpsnd_opensles_set_format(rdpsndDevicePlugin* device,
} }
static BOOL rdpsnd_opensles_open(rdpsndDevicePlugin* device, static BOOL rdpsnd_opensles_open(rdpsndDevicePlugin* device,
AUDIO_FORMAT* format, int latency) const AUDIO_FORMAT* format, UINT32 latency)
{ {
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device; rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
DEBUG_SND("opensles=%p format=%p, latency=%d, rate=%"PRIu32"", DEBUG_SND("opensles=%p format=%p, latency=%"PRIu32", rate=%"PRIu32"",
(void*) opensles, (void*) format, latency, opensles->rate); (void*) opensles, (void*) format, latency, opensles->rate);
if (rdpsnd_opensles_check_handle(opensles)) if (rdpsnd_opensles_check_handle(opensles))
@ -161,8 +156,7 @@ static BOOL rdpsnd_opensles_open(rdpsndDevicePlugin* device,
else else
rdpsnd_opensles_set_volume(device, opensles->volume); rdpsnd_opensles_set_volume(device, opensles->volume);
rdpsnd_opensles_set_format(device, format, latency); return rdpsnd_opensles_set_format(device, format, latency);
return TRUE;
} }
static void rdpsnd_opensles_close(rdpsndDevicePlugin* device) static void rdpsnd_opensles_close(rdpsndDevicePlugin* device)
@ -184,13 +178,11 @@ static void rdpsnd_opensles_free(rdpsndDevicePlugin* device)
assert(opensles); assert(opensles);
assert(opensles->device_name); assert(opensles->device_name);
free(opensles->device_name); free(opensles->device_name);
assert(opensles->dsp_context);
freerdp_dsp_context_free(opensles->dsp_context);
free(opensles); free(opensles);
} }
static BOOL rdpsnd_opensles_format_supported(rdpsndDevicePlugin* device, static BOOL rdpsnd_opensles_format_supported(rdpsndDevicePlugin* device,
AUDIO_FORMAT* format) const AUDIO_FORMAT* format)
{ {
DEBUG_SND("format=%"PRIu16", cbsize=%"PRIu16", samples=%"PRIu32", bits=%"PRIu16", channels=%"PRIu16", align=%"PRIu16"", DEBUG_SND("format=%"PRIu16", cbsize=%"PRIu16", samples=%"PRIu32", bits=%"PRIu16", channels=%"PRIu16", align=%"PRIu16"",
format->wFormatTag, format->cbSize, format->nSamplesPerSec, format->wFormatTag, format->cbSize, format->nSamplesPerSec,
@ -211,20 +203,6 @@ static BOOL rdpsnd_opensles_format_supported(rdpsndDevicePlugin* device,
break; 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:
case WAVE_FORMAT_MULAW:
case WAVE_FORMAT_GSM610:
default: default:
break; break;
} }
@ -283,46 +261,22 @@ static BOOL rdpsnd_opensles_set_volume(rdpsndDevicePlugin* device,
return TRUE; return TRUE;
} }
static void rdpsnd_opensles_play(rdpsndDevicePlugin* device, static UINT rdpsnd_opensles_play(rdpsndDevicePlugin* device,
BYTE* data, int size) const BYTE* data, size_t size)
{ {
union union
{ {
BYTE* b; const BYTE* b;
short* s; const short* s;
} src; } src;
int ret; int ret;
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device; rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
DEBUG_SND("opensles=%p, data=%p, size=%d", (void*) opensles, (void*) data, size); DEBUG_SND("opensles=%p, data=%p, size=%d", (void*) opensles, (void*) data, size);
if (!rdpsnd_opensles_check_handle(opensles)) if (!rdpsnd_opensles_check_handle(opensles))
return; return 0;
if (opensles->format == WAVE_FORMAT_ADPCM)
{
DEBUG_SND("dsp_context=%p, channels=%"PRIu32", block_size=%d",
(void*) opensles->dsp_context, opensles->channels, opensles->block_size);
opensles->dsp_context->decode_ms_adpcm(opensles->dsp_context,
data, size, opensles->channels, opensles->block_size);
size = opensles->dsp_context->adpcm_size;
src.b = opensles->dsp_context->adpcm_buffer;
}
else if (opensles->format == WAVE_FORMAT_DVI_ADPCM)
{
DEBUG_SND("dsp_context=%p, channels=%"PRIu32", block_size=%d",
(void*) opensles->dsp_context, opensles->channels, opensles->block_size);
opensles->dsp_context->decode_ima_adpcm(opensles->dsp_context,
data, size, opensles->channels, opensles->block_size);
size = opensles->dsp_context->adpcm_size;
src.b = opensles->dsp_context->adpcm_buffer;
}
else
{
src.b = data; src.b = data;
}
DEBUG_SND("size=%d, src=%p", size, (void*) src.b); DEBUG_SND("size=%d, src=%p", size, (void*) src.b);
assert(0 == size % 2); assert(0 == size % 2);
assert(size > 0); assert(size > 0);
@ -331,6 +285,8 @@ static void rdpsnd_opensles_play(rdpsndDevicePlugin* device,
if (ret < 0) if (ret < 0)
WLog_ERR(TAG, "android_AudioOut failed (%d)", ret); WLog_ERR(TAG, "android_AudioOut failed (%d)", ret);
return 10; /* TODO: Get real latencry in [ms] */
} }
static void rdpsnd_opensles_start(rdpsndDevicePlugin* device) static void rdpsnd_opensles_start(rdpsndDevicePlugin* device)
@ -416,7 +372,6 @@ UINT freerdp_rdpsnd_client_subsystem_entry(
opensles->device.Open = rdpsnd_opensles_open; opensles->device.Open = rdpsnd_opensles_open;
opensles->device.FormatSupported = rdpsnd_opensles_format_supported; opensles->device.FormatSupported = rdpsnd_opensles_format_supported;
opensles->device.SetFormat = rdpsnd_opensles_set_format;
opensles->device.GetVolume = rdpsnd_opensles_get_volume; opensles->device.GetVolume = rdpsnd_opensles_get_volume;
opensles->device.SetVolume = rdpsnd_opensles_set_volume; opensles->device.SetVolume = rdpsnd_opensles_set_volume;
opensles->device.Start = rdpsnd_opensles_start; opensles->device.Start = rdpsnd_opensles_start;
@ -440,20 +395,10 @@ UINT freerdp_rdpsnd_client_subsystem_entry(
opensles->rate = 44100; opensles->rate = 44100;
opensles->channels = 2; opensles->channels = 2;
opensles->format = WAVE_FORMAT_ADPCM; opensles->format = WAVE_FORMAT_ADPCM;
opensles->dsp_context = freerdp_dsp_context_new();
if (!opensles->dsp_context)
{
error = CHANNEL_RC_NO_MEMORY;
goto out_dsp_new;
}
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd,
(rdpsndDevicePlugin*) opensles); (rdpsndDevicePlugin*) opensles);
DEBUG_SND("success"); DEBUG_SND("success");
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
out_dsp_new:
free(opensles->device_name);
outstrdup: outstrdup:
free(opensles); free(opensles);
return error; return error;

View File

@ -47,7 +47,6 @@
#include <sys/ioctl.h> #include <sys/ioctl.h>
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/log.h> #include <freerdp/channels/log.h>
#include "rdpsnd_main.h" #include "rdpsnd_main.h"
@ -64,10 +63,8 @@ struct rdpsnd_oss_plugin
int supported_formats; int supported_formats;
int latency; UINT32 latency;
AUDIO_FORMAT format; AUDIO_FORMAT format;
FREERDP_DSP_CONTEXT* dsp_context;
}; };
#define OSS_LOG_ERR(_text, _error) \ #define OSS_LOG_ERR(_text, _error) \
@ -77,7 +74,7 @@ struct rdpsnd_oss_plugin
} }
static int rdpsnd_oss_get_format(AUDIO_FORMAT* format) static int rdpsnd_oss_get_format(const AUDIO_FORMAT* format)
{ {
switch (format->wFormatTag) switch (format->wFormatTag)
{ {
@ -95,21 +92,15 @@ static int rdpsnd_oss_get_format(AUDIO_FORMAT* format)
case WAVE_FORMAT_ALAW: case WAVE_FORMAT_ALAW:
return AFMT_A_LAW; return AFMT_A_LAW;
#if 0 /* This does not work on my desktop. */
case WAVE_FORMAT_MULAW: case WAVE_FORMAT_MULAW:
return AFMT_MU_LAW; return AFMT_MU_LAW;
#endif
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
return AFMT_S16_LE;
} }
return 0; return 0;
} }
static BOOL rdpsnd_oss_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT* format) static BOOL rdpsnd_oss_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format)
{ {
int req_fmt = 0; int req_fmt = 0;
rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device; rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device;
@ -128,14 +119,12 @@ static BOOL rdpsnd_oss_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT
break; break;
case WAVE_FORMAT_ADPCM: case WAVE_FORMAT_MULAW:
case WAVE_FORMAT_DVI_ADPCM: case WAVE_FORMAT_ALAW:
if (format->nSamplesPerSec > 48000 ||
format->wBitsPerSample != 4 ||
(format->nChannels != 1 && format->nChannels != 2))
return FALSE;
break; break;
default:
return FALSE;
} }
req_fmt = rdpsnd_oss_get_format(format); req_fmt = rdpsnd_oss_get_format(format);
@ -155,7 +144,8 @@ static BOOL rdpsnd_oss_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT
return TRUE; return TRUE;
} }
static BOOL rdpsnd_oss_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_oss_set_format(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format,
UINT32 latency)
{ {
int tmp; int tmp;
rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device; rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device;
@ -227,7 +217,7 @@ static void rdpsnd_oss_open_mixer(rdpsndOssPlugin* oss)
} }
} }
static BOOL rdpsnd_oss_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_oss_open(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format, UINT32 latency)
{ {
char dev_name[PATH_MAX] = "/dev/dsp"; char dev_name[PATH_MAX] = "/dev/dsp";
rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device; rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device;
@ -272,7 +262,6 @@ static BOOL rdpsnd_oss_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, in
return FALSE; return FALSE;
} }
freerdp_dsp_context_reset_adpcm(oss->dsp_context);
rdpsnd_oss_set_format(device, format, latency); rdpsnd_oss_set_format(device, format, latency);
rdpsnd_oss_open_mixer(oss); rdpsnd_oss_open_mixer(oss);
return TRUE; return TRUE;
@ -308,7 +297,6 @@ static void rdpsnd_oss_free(rdpsndDevicePlugin* device)
return; return;
rdpsnd_oss_close(device); rdpsnd_oss_close(device);
freerdp_dsp_context_free(oss->dsp_context);
free(oss); free(oss);
} }
@ -370,68 +358,36 @@ static BOOL rdpsnd_oss_set_volume(rdpsndDevicePlugin* device, UINT32 value)
return TRUE; return TRUE;
} }
static BOOL rdpsnd_oss_wave_decode(rdpsndDevicePlugin* device, RDPSND_WAVE* wave) static UINT rdpsnd_oss_play(rdpsndDevicePlugin* device, const BYTE* data, size_t size)
{ {
rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device; rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device;
if (device == NULL || wave == NULL) if (device == NULL || oss->mixer_handle == -1)
return FALSE; return 0;
switch (oss->format.wFormatTag) while (size > 0)
{ {
case WAVE_FORMAT_ADPCM: ssize_t status = write(oss->pcm_handle, data, size);
oss->dsp_context->decode_ms_adpcm(oss->dsp_context,
wave->data, wave->length, oss->format.nChannels, oss->format.nBlockAlign);
wave->length = oss->dsp_context->adpcm_size;
wave->data = oss->dsp_context->adpcm_buffer;
break;
case WAVE_FORMAT_DVI_ADPCM:
oss->dsp_context->decode_ima_adpcm(oss->dsp_context,
wave->data, wave->length, oss->format.nChannels, oss->format.nBlockAlign);
wave->length = oss->dsp_context->adpcm_size;
wave->data = oss->dsp_context->adpcm_buffer;
break;
}
return TRUE;
}
static void rdpsnd_oss_wave_play(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
{
BYTE* data;
int offset, size, status, latency;
rdpsndOssPlugin* oss = (rdpsndOssPlugin*)device;
if (device == NULL || wave == NULL)
return;
offset = 0;
data = wave->data;
size = wave->length;
latency = oss->latency;
while (offset < size)
{
status = write(oss->pcm_handle, &data[offset], (size - offset));
if (status < 0) if (status < 0)
{ {
OSS_LOG_ERR("write fail", errno); OSS_LOG_ERR("write fail", errno);
rdpsnd_oss_close(device); rdpsnd_oss_close(device);
rdpsnd_oss_open(device, NULL, latency); rdpsnd_oss_open(device, NULL, oss->latency);
break; break;
} }
offset += status; data += status;
if (status <= size)
size -= status;
else
size = 0;
} }
/* From rdpsnd_main.c */ return 10; /* TODO: Get real latency in [ms] */
wave->wTimeStampB = wave->wTimeStampA + wave->wAudioLength + 65 + latency;
wave->wLocalTimeB = wave->wLocalTimeA + wave->wAudioLength + 65 + latency;
} }
static COMMAND_LINE_ARGUMENT_A rdpsnd_oss_args[] = static COMMAND_LINE_ARGUMENT_A rdpsnd_oss_args[] =
{ {
{ "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "device" }, { "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "device" },
@ -514,11 +470,9 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
oss->device.Open = rdpsnd_oss_open; oss->device.Open = rdpsnd_oss_open;
oss->device.FormatSupported = rdpsnd_oss_format_supported; oss->device.FormatSupported = rdpsnd_oss_format_supported;
oss->device.SetFormat = rdpsnd_oss_set_format;
oss->device.GetVolume = rdpsnd_oss_get_volume; oss->device.GetVolume = rdpsnd_oss_get_volume;
oss->device.SetVolume = rdpsnd_oss_set_volume; oss->device.SetVolume = rdpsnd_oss_set_volume;
oss->device.WaveDecode = rdpsnd_oss_wave_decode; oss->device.Play = rdpsnd_oss_play;
oss->device.WavePlay = rdpsnd_oss_wave_play;
oss->device.Close = rdpsnd_oss_close; oss->device.Close = rdpsnd_oss_close;
oss->device.Free = rdpsnd_oss_free; oss->device.Free = rdpsnd_oss_free;
oss->pcm_handle = -1; oss->pcm_handle = -1;
@ -526,14 +480,6 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
oss->dev_unit = -1; oss->dev_unit = -1;
args = pEntryPoints->args; args = pEntryPoints->args;
rdpsnd_oss_parse_addin_args((rdpsndDevicePlugin*)oss, args); rdpsnd_oss_parse_addin_args((rdpsndDevicePlugin*)oss, args);
oss->dsp_context = freerdp_dsp_context_new();
if (!oss->dsp_context)
{
free(oss);
return CHANNEL_RC_NO_MEMORY;
}
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*)oss); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*)oss);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }

View File

@ -25,16 +25,10 @@ include_directories(${PULSE_INCLUDE_DIR})
add_channel_client_subsystem_library(${MODULE_PREFIX} ${MODULE_NAME} ${CHANNEL_NAME} "" TRUE "") add_channel_client_subsystem_library(${MODULE_PREFIX} ${MODULE_NAME} ${CHANNEL_NAME} "" TRUE "")
list(APPEND ${MODULE_PREFIX}_LIBS ${PULSE_LIBRARY}) list(APPEND ${MODULE_PREFIX}_LIBS ${PULSE_LIBRARY})
list(APPEND ${MODULE_PREFIX}_LIBS freerdp) list(APPEND ${MODULE_PREFIX}_LIBS freerdp)
list(APPEND ${MODULE_PREFIX}_LIBS winpr) list(APPEND ${MODULE_PREFIX}_LIBS winpr)
if(GSM_FOUND)
list(APPEND ${MODULE_PREFIX}_LIBS ${GSM_LIBRARIES})
endif()
target_link_libraries(${MODULE_NAME} ${${MODULE_PREFIX}_LIBS}) target_link_libraries(${MODULE_NAME} ${${MODULE_PREFIX}_LIBS})
set_property(TARGET ${MODULE_NAME} PROPERTY FOLDER "Channels/${CHANNEL_NAME}/Client/Pulse") set_property(TARGET ${MODULE_NAME} PROPERTY FOLDER "Channels/${CHANNEL_NAME}/Client/Pulse")

View File

@ -33,10 +33,6 @@
#include <pulse/pulseaudio.h> #include <pulse/pulseaudio.h>
#ifdef WITH_GSM
#include <gsm/gsm.h>
#endif
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/codec/dsp.h> #include <freerdp/codec/dsp.h>
@ -53,23 +49,15 @@ struct rdpsnd_pulse_plugin
pa_context* context; pa_context* context;
pa_sample_spec sample_spec; pa_sample_spec sample_spec;
pa_stream* stream; pa_stream* stream;
int format; UINT32 latency;
int block_size;
int latency;
FREERDP_DSP_CONTEXT* dsp_context;
#ifdef WITH_GSM
gsm gsm_context;
wStream* gsmBuffer;
#endif
}; };
static BOOL rdpsnd_pulse_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format);
static void rdpsnd_pulse_context_state_callback(pa_context* context, void* userdata) static void rdpsnd_pulse_context_state_callback(pa_context* context, void* userdata)
{ {
pa_context_state_t state; pa_context_state_t state;
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata;
state = pa_context_get_state(context); state = pa_context_get_state(context);
switch (state) switch (state)
@ -140,7 +128,6 @@ static BOOL rdpsnd_pulse_connect(rdpsndDevicePlugin* device)
static void rdpsnd_pulse_stream_success_callback(pa_stream* stream, int success, void* userdata) static void rdpsnd_pulse_stream_success_callback(pa_stream* stream, int success, void* userdata)
{ {
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata;
pa_threaded_mainloop_signal(pulse->mainloop, 0); pa_threaded_mainloop_signal(pulse->mainloop, 0);
} }
@ -161,7 +148,6 @@ static void rdpsnd_pulse_stream_state_callback(pa_stream* stream, void* userdata
{ {
pa_stream_state_t state; pa_stream_state_t state;
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata;
state = pa_stream_get_state(stream); state = pa_stream_get_state(stream);
switch (state) switch (state)
@ -183,7 +169,6 @@ static void rdpsnd_pulse_stream_state_callback(pa_stream* stream, void* userdata
static void rdpsnd_pulse_stream_request_callback(pa_stream* stream, size_t length, void* userdata) static void rdpsnd_pulse_stream_request_callback(pa_stream* stream, size_t length, void* userdata)
{ {
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) userdata;
pa_threaded_mainloop_signal(pulse->mainloop, 0); pa_threaded_mainloop_signal(pulse->mainloop, 0);
} }
@ -191,30 +176,27 @@ static void rdpsnd_pulse_close(rdpsndDevicePlugin* device)
{ {
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device;
#ifdef WITH_GSM
if (pulse->gsm_context)
gsm_destroy(pulse->gsm_context);
#endif
if (!pulse->context || !pulse->stream) if (!pulse->context || !pulse->stream)
return; return;
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
rdpsnd_pulse_wait_for_operation(pulse, pa_stream_drain(pulse->stream,
rdpsnd_pulse_wait_for_operation(pulse, pa_stream_drain(pulse->stream, rdpsnd_pulse_stream_success_callback, pulse)); rdpsnd_pulse_stream_success_callback, pulse));
pa_stream_disconnect(pulse->stream); pa_stream_disconnect(pulse->stream);
pa_stream_unref(pulse->stream); pa_stream_unref(pulse->stream);
pulse->stream = NULL; pulse->stream = NULL;
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
} }
static void rdpsnd_pulse_set_format_spec(rdpsndPulsePlugin* pulse, AUDIO_FORMAT* format) static BOOL rdpsnd_pulse_set_format_spec(rdpsndPulsePlugin* pulse, const AUDIO_FORMAT* format)
{ {
pa_sample_spec sample_spec = { 0 }; pa_sample_spec sample_spec = { 0 };
if (!pulse->context) if (!pulse->context)
return; return FALSE;
if (!rdpsnd_pulse_format_supported(&pulse->device, format))
return FALSE;
sample_spec.rate = format->nSamplesPerSec; sample_spec.rate = format->nSamplesPerSec;
sample_spec.channels = format->nChannels; sample_spec.channels = format->nChannels;
@ -227,15 +209,15 @@ static void rdpsnd_pulse_set_format_spec(rdpsndPulsePlugin* pulse, AUDIO_FORMAT*
case 8: case 8:
sample_spec.format = PA_SAMPLE_U8; sample_spec.format = PA_SAMPLE_U8;
break; break;
case 16: case 16:
sample_spec.format = PA_SAMPLE_S16LE; sample_spec.format = PA_SAMPLE_S16LE;
break; break;
}
break;
case WAVE_FORMAT_ADPCM: default:
case WAVE_FORMAT_DVI_ADPCM: return FALSE;
sample_spec.format = PA_SAMPLE_S16LE; }
break; break;
case WAVE_FORMAT_ALAW: case WAVE_FORMAT_ALAW:
@ -246,17 +228,16 @@ static void rdpsnd_pulse_set_format_spec(rdpsndPulsePlugin* pulse, AUDIO_FORMAT*
sample_spec.format = PA_SAMPLE_ULAW; sample_spec.format = PA_SAMPLE_ULAW;
break; break;
case WAVE_FORMAT_GSM610: default:
sample_spec.format = PA_SAMPLE_S16LE; return FALSE;
break;
} }
pulse->sample_spec = sample_spec; pulse->sample_spec = sample_spec;
pulse->format = format->wFormatTag; return TRUE;
pulse->block_size = format->nBlockAlign;
} }
static BOOL rdpsnd_pulse_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_pulse_open(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format,
UINT32 latency)
{ {
pa_stream_state_t state; pa_stream_state_t state;
pa_stream_flags_t flags; pa_stream_flags_t flags;
@ -267,7 +248,9 @@ static BOOL rdpsnd_pulse_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format,
if (!pulse->context || pulse->stream) if (!pulse->context || pulse->stream)
return TRUE; return TRUE;
rdpsnd_pulse_set_format_spec(pulse, format); if (!rdpsnd_pulse_set_format_spec(pulse, format))
return FALSE;
pulse->latency = latency; pulse->latency = latency;
if (pa_sample_spec_valid(&pulse->sample_spec) == 0) if (pa_sample_spec_valid(&pulse->sample_spec) == 0)
@ -277,8 +260,8 @@ static BOOL rdpsnd_pulse_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format,
} }
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
pulse->stream = pa_stream_new(pulse->context, "freerdp", &pulse->sample_spec, NULL); pulse->stream = pa_stream_new(pulse->context, "freerdp", &pulse->sample_spec, NULL);
if (!pulse->stream) if (!pulse->stream)
{ {
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
@ -288,7 +271,6 @@ static BOOL rdpsnd_pulse_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format,
/* register essential callbacks */ /* register essential callbacks */
pa_stream_set_state_callback(pulse->stream, rdpsnd_pulse_stream_state_callback, pulse); pa_stream_set_state_callback(pulse->stream, rdpsnd_pulse_stream_state_callback, pulse);
pa_stream_set_write_callback(pulse->stream, rdpsnd_pulse_stream_request_callback, pulse); pa_stream_set_write_callback(pulse->stream, rdpsnd_pulse_stream_request_callback, pulse);
flags = PA_STREAM_INTERPOLATE_TIMING | PA_STREAM_AUTO_TIMING_UPDATE; flags = PA_STREAM_INTERPOLATE_TIMING | PA_STREAM_AUTO_TIMING_UPDATE;
if (pulse->latency > 0) if (pulse->latency > 0)
@ -326,17 +308,7 @@ static BOOL rdpsnd_pulse_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format,
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
if (state == PA_STREAM_READY) if (state == PA_STREAM_READY)
{
freerdp_dsp_context_reset_adpcm(pulse->dsp_context);
#ifdef WITH_GSM
if (pulse->gsm_context)
gsm_destroy(pulse->gsm_context);
pulse->gsm_context = gsm_create();
#endif
return TRUE; return TRUE;
}
rdpsnd_pulse_close(device); rdpsnd_pulse_close(device);
return FALSE; return FALSE;
@ -369,16 +341,11 @@ static void rdpsnd_pulse_free(rdpsndDevicePlugin* device)
pulse->mainloop = NULL; pulse->mainloop = NULL;
} }
#ifdef WITH_GSM
Stream_Free(pulse->gsmBuffer, TRUE);
#endif
free(pulse->device_name); free(pulse->device_name);
freerdp_dsp_context_free(pulse->dsp_context);
free(pulse); free(pulse);
} }
static BOOL rdpsnd_pulse_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT* format) BOOL rdpsnd_pulse_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format)
{ {
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*)device; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*)device;
@ -395,6 +362,7 @@ static BOOL rdpsnd_pulse_format_supported(rdpsndDevicePlugin* device, AUDIO_FORM
{ {
return TRUE; return TRUE;
} }
break; break;
case WAVE_FORMAT_ALAW: case WAVE_FORMAT_ALAW:
@ -406,48 +374,13 @@ static BOOL rdpsnd_pulse_format_supported(rdpsndDevicePlugin* device, AUDIO_FORM
{ {
return TRUE; return TRUE;
} }
break;
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
if ((format->nSamplesPerSec <= PA_RATE_MAX) &&
(format->wBitsPerSample == 4) &&
(format->nChannels == 1 || format->nChannels == 2))
{
return TRUE;
}
break; break;
#ifdef WITH_GSM
case WAVE_FORMAT_GSM610:
if ((format->nSamplesPerSec <= PA_RATE_MAX) &&
(format->nBlockAlign == 65) && (format->nChannels == 1))
{
return TRUE;
}
break;
#endif
} }
return FALSE; return FALSE;
} }
static BOOL rdpsnd_pulse_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency)
{
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device;
if (pulse->stream)
{
pa_threaded_mainloop_lock(pulse->mainloop);
pa_stream_disconnect(pulse->stream);
pa_stream_unref(pulse->stream);
pulse->stream = NULL;
pa_threaded_mainloop_unlock(pulse->mainloop);
}
return rdpsnd_pulse_open(device, format, latency);
}
static BOOL rdpsnd_pulse_set_volume(rdpsndDevicePlugin* device, UINT32 value) static BOOL rdpsnd_pulse_set_volume(rdpsndDevicePlugin* device, UINT32 value)
{ {
pa_cvolume cv; pa_cvolume cv;
@ -461,15 +394,13 @@ static BOOL rdpsnd_pulse_set_volume(rdpsndDevicePlugin* device, UINT32 value)
left = (pa_volume_t)(value & 0xFFFF); left = (pa_volume_t)(value & 0xFFFF);
right = (pa_volume_t)((value >> 16) & 0xFFFF); right = (pa_volume_t)((value >> 16) & 0xFFFF);
pa_cvolume_init(&cv); pa_cvolume_init(&cv);
cv.channels = 2; cv.channels = 2;
cv.values[0] = PA_VOLUME_MUTED + (left * (PA_VOLUME_NORM - PA_VOLUME_MUTED)) / 0xFFFF; cv.values[0] = PA_VOLUME_MUTED + (left * (PA_VOLUME_NORM - PA_VOLUME_MUTED)) / 0xFFFF;
cv.values[1] = PA_VOLUME_MUTED + (right * (PA_VOLUME_NORM - PA_VOLUME_MUTED)) / 0xFFFF; cv.values[1] = PA_VOLUME_MUTED + (right * (PA_VOLUME_NORM - PA_VOLUME_MUTED)) / 0xFFFF;
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
operation = pa_context_set_sink_input_volume(pulse->context, pa_stream_get_index(pulse->stream),
operation = pa_context_set_sink_input_volume(pulse->context, pa_stream_get_index(pulse->stream), &cv, NULL, NULL); &cv, NULL, NULL);
if (operation) if (operation)
pa_operation_unref(operation); pa_operation_unref(operation);
@ -478,93 +409,23 @@ static BOOL rdpsnd_pulse_set_volume(rdpsndDevicePlugin* device, UINT32 value)
return TRUE; return TRUE;
} }
static BYTE* rdpsnd_pulse_convert_audio(rdpsndDevicePlugin* device, BYTE* data, int* size) static UINT rdpsnd_pulse_play(rdpsndDevicePlugin* device, const BYTE* data, size_t size)
{ {
BYTE* pcmData = NULL; size_t length;
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device;
if (pulse->format == WAVE_FORMAT_ADPCM)
{
pulse->dsp_context->decode_ms_adpcm(pulse->dsp_context,
data, *size, pulse->sample_spec.channels, pulse->block_size);
*size = pulse->dsp_context->adpcm_size;
pcmData = pulse->dsp_context->adpcm_buffer;
}
else if (pulse->format == WAVE_FORMAT_DVI_ADPCM)
{
pulse->dsp_context->decode_ima_adpcm(pulse->dsp_context,
data, *size, pulse->sample_spec.channels, pulse->block_size);
*size = pulse->dsp_context->adpcm_size;
pcmData = pulse->dsp_context->adpcm_buffer;
}
#ifdef WITH_GSM
else if (pulse->format == WAVE_FORMAT_GSM610)
{
int inPos = 0;
int inSize = *size;
UINT16 gsmBlockBuffer[160];
Stream_SetPosition(pulse->gsmBuffer, 0);
while (inSize)
{
ZeroMemory(gsmBlockBuffer, sizeof(gsmBlockBuffer));
gsm_decode(pulse->gsm_context, (gsm_byte*) &data[inPos], (gsm_signal*) gsmBlockBuffer);
if ((inPos % 65) == 0)
{
inPos += 33;
inSize -= 33;
}
else
{
inPos += 32;
inSize -= 32;
}
if (!Stream_EnsureRemainingCapacity(pulse->gsmBuffer, 160 * 2))
return NULL;
Stream_Write(pulse->gsmBuffer, (void*) gsmBlockBuffer, 160 * 2);
}
Stream_SealLength(pulse->gsmBuffer);
pcmData = Stream_Buffer(pulse->gsmBuffer);
*size = Stream_Length(pulse->gsmBuffer);
}
#endif
else
{
pcmData = data;
}
return pcmData;
}
static void rdpsnd_pulse_play(rdpsndDevicePlugin* device, BYTE* data, int size)
{
int length;
int status; int status;
BYTE* pcmData; pa_usec_t latency;
int negative;
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device;
if (!pulse->stream) if (!pulse->stream || !data)
return; return 0;
pcmData = rdpsnd_pulse_convert_audio(device, data, &size);
if (!pcmData)
return;
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
while (size > 0) while (size > 0)
{ {
while ((length = pa_stream_writable_size(pulse->stream)) == 0) while ((length = pa_stream_writable_size(pulse->stream)) == 0)
{
pa_threaded_mainloop_wait(pulse->mainloop); pa_threaded_mainloop_wait(pulse->mainloop);
}
if (length < 0) if (length < 0)
break; break;
@ -572,18 +433,22 @@ static void rdpsnd_pulse_play(rdpsndDevicePlugin* device, BYTE* data, int size)
if (length > size) if (length > size)
length = size; length = size;
status = pa_stream_write(pulse->stream, pcmData, length, NULL, 0LL, PA_SEEK_RELATIVE); status = pa_stream_write(pulse->stream, data, length, NULL, 0LL, PA_SEEK_RELATIVE);
if (status < 0) if (status < 0)
{ {
break; break;
} }
pcmData += length; data += length;
size -= length; size -= length;
} }
if (pa_stream_get_latency(pulse->stream, &latency, &negative) != 0)
latency = 0;
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
return latency / 1000;
} }
static void rdpsnd_pulse_start(rdpsndDevicePlugin* device) static void rdpsnd_pulse_start(rdpsndDevicePlugin* device)
@ -598,7 +463,7 @@ static void rdpsnd_pulse_start(rdpsndDevicePlugin* device)
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
} }
COMMAND_LINE_ARGUMENT_A rdpsnd_pulse_args[] = static COMMAND_LINE_ARGUMENT_A rdpsnd_pulse_args[] =
{ {
{ "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "device" }, { "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>", NULL, NULL, -1, NULL, "device" },
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL } { NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
@ -615,29 +480,28 @@ static UINT rdpsnd_pulse_parse_addin_args(rdpsndDevicePlugin* device, ADDIN_ARGV
DWORD flags; DWORD flags;
COMMAND_LINE_ARGUMENT_A* arg; COMMAND_LINE_ARGUMENT_A* arg;
rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device; rdpsndPulsePlugin* pulse = (rdpsndPulsePlugin*) device;
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD; flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON | COMMAND_LINE_IGN_UNKNOWN_KEYWORD;
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv, status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv,
rdpsnd_pulse_args, flags, pulse, NULL, NULL); rdpsnd_pulse_args, flags, pulse, NULL, NULL);
if (status < 0) if (status < 0)
return ERROR_INVALID_DATA; return ERROR_INVALID_DATA;
arg = rdpsnd_pulse_args; arg = rdpsnd_pulse_args;
do do
{ {
if (!(arg->Flags & COMMAND_LINE_VALUE_PRESENT)) if (!(arg->Flags & COMMAND_LINE_VALUE_PRESENT))
continue; continue;
CommandLineSwitchStart(arg) CommandLineSwitchStart(arg)
CommandLineSwitchCase(arg, "dev") CommandLineSwitchCase(arg, "dev")
{ {
pulse->device_name = _strdup(arg->Value); pulse->device_name = _strdup(arg->Value);
if (!pulse->device_name) if (!pulse->device_name)
return ERROR_OUTOFMEMORY; return ERROR_OUTOFMEMORY;
} }
CommandLineSwitchEnd(arg) CommandLineSwitchEnd(arg)
} }
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL); while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
@ -661,24 +525,24 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
ADDIN_ARGV* args; ADDIN_ARGV* args;
rdpsndPulsePlugin* pulse; rdpsndPulsePlugin* pulse;
UINT ret; UINT ret;
pulse = (rdpsndPulsePlugin*) calloc(1, sizeof(rdpsndPulsePlugin)); pulse = (rdpsndPulsePlugin*) calloc(1, sizeof(rdpsndPulsePlugin));
if (!pulse) if (!pulse)
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
pulse->device.Open = rdpsnd_pulse_open; pulse->device.Open = rdpsnd_pulse_open;
pulse->device.FormatSupported = rdpsnd_pulse_format_supported; pulse->device.FormatSupported = rdpsnd_pulse_format_supported;
pulse->device.SetFormat = rdpsnd_pulse_set_format;
pulse->device.SetVolume = rdpsnd_pulse_set_volume; pulse->device.SetVolume = rdpsnd_pulse_set_volume;
pulse->device.Play = rdpsnd_pulse_play; pulse->device.Play = rdpsnd_pulse_play;
pulse->device.Start = rdpsnd_pulse_start; pulse->device.Start = rdpsnd_pulse_start;
pulse->device.Close = rdpsnd_pulse_close; pulse->device.Close = rdpsnd_pulse_close;
pulse->device.Free = rdpsnd_pulse_free; pulse->device.Free = rdpsnd_pulse_free;
args = pEntryPoints->args; args = pEntryPoints->args;
if (args->argc > 1) if (args->argc > 1)
{ {
ret = rdpsnd_pulse_parse_addin_args((rdpsndDevicePlugin*) pulse, args); ret = rdpsnd_pulse_parse_addin_args((rdpsndDevicePlugin*) pulse, args);
if (ret != CHANNEL_RC_OK) if (ret != CHANNEL_RC_OK)
{ {
WLog_ERR(TAG, "error parsing arguments"); WLog_ERR(TAG, "error parsing arguments");
@ -687,17 +551,6 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
} }
ret = CHANNEL_RC_NO_MEMORY; ret = CHANNEL_RC_NO_MEMORY;
pulse->dsp_context = freerdp_dsp_context_new();
if (!pulse->dsp_context)
goto error;
#ifdef WITH_GSM
pulse->gsmBuffer = Stream_New(NULL, 4096);
if (!pulse->gsmBuffer)
goto error;
#endif
pulse->mainloop = pa_threaded_mainloop_new(); pulse->mainloop = pa_threaded_mainloop_new();
if (!pulse->mainloop) if (!pulse->mainloop)
@ -709,14 +562,13 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
goto error; goto error;
pa_context_set_state_callback(pulse->context, rdpsnd_pulse_context_state_callback, pulse); pa_context_set_state_callback(pulse->context, rdpsnd_pulse_context_state_callback, pulse);
ret = ERROR_INVALID_OPERATION; ret = ERROR_INVALID_OPERATION;
if (!rdpsnd_pulse_connect((rdpsndDevicePlugin*)pulse)) if (!rdpsnd_pulse_connect((rdpsndDevicePlugin*)pulse))
goto error; goto error;
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) pulse); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) pulse);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
error: error:
rdpsnd_pulse_free((rdpsndDevicePlugin*)pulse); rdpsnd_pulse_free((rdpsndDevicePlugin*)pulse);
return ret; return ret;

File diff suppressed because it is too large Load Diff

View File

@ -35,6 +35,4 @@
#define DEBUG_SND(...) do { } while (0) #define DEBUG_SND(...) do { } while (0)
#endif #endif
UINT rdpsnd_virtual_channel_write(rdpsndPlugin* rdpsnd, wStream* s);
#endif /* FREERDP_CHANNEL_RDPSND_CLIENT_MAIN_H */ #endif /* FREERDP_CHANNEL_RDPSND_CLIENT_MAIN_H */

View File

@ -36,7 +36,6 @@
#include <winpr/cmdline.h> #include <winpr/cmdline.h>
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/codec/dsp.h>
#include <freerdp/channels/log.h> #include <freerdp/channels/log.h>
#include "rdpsnd_main.h" #include "rdpsnd_main.h"
@ -49,15 +48,14 @@ struct rdpsnd_winmm_plugin
HWAVEOUT hWaveOut; HWAVEOUT hWaveOut;
WAVEFORMATEX format; WAVEFORMATEX format;
int wformat;
int block_size;
UINT32 volume; UINT32 volume;
FREERDP_DSP_CONTEXT* dsp_context; HANDLE next;
}; };
static BOOL rdpsnd_winmm_convert_format(const AUDIO_FORMAT* in, WAVEFORMATEX* out) static BOOL rdpsnd_winmm_convert_format(const AUDIO_FORMAT* in, WAVEFORMATEX* out)
{ {
BOOL result = FALSE; if (!in || !out)
return FALSE;
ZeroMemory(out, sizeof(WAVEFORMATEX)); ZeroMemory(out, sizeof(WAVEFORMATEX));
out->wFormatTag = WAVE_FORMAT_PCM; out->wFormatTag = WAVE_FORMAT_PCM;
@ -68,40 +66,31 @@ static BOOL rdpsnd_winmm_convert_format(const AUDIO_FORMAT* in, WAVEFORMATEX* ou
{ {
case WAVE_FORMAT_PCM: case WAVE_FORMAT_PCM:
out->wBitsPerSample = in->wBitsPerSample; out->wBitsPerSample = in->wBitsPerSample;
result = TRUE;
break; break;
case WAVE_FORMAT_ADPCM: default:
case WAVE_FORMAT_DVI_ADPCM: return FALSE;
out->wBitsPerSample = 16;
result = TRUE;
break;
} }
out->nBlockAlign = out->nChannels * out->wBitsPerSample / 8; out->nBlockAlign = out->nChannels * out->wBitsPerSample / 8;
out->nAvgBytesPerSec = out->nSamplesPerSec * out->nBlockAlign; out->nAvgBytesPerSec = out->nSamplesPerSec * out->nBlockAlign;
return result;
}
static BOOL rdpsnd_winmm_set_format(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency)
{
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
if (format)
{
if (!rdpsnd_winmm_convert_format(format, &winmm->format))
return FALSE;
winmm->wformat = format->wFormatTag;
winmm->block_size = format->nBlockAlign;
}
return TRUE; return TRUE;
} }
static void CALLBACK rdpsnd_winmm_callback_function(HWAVEOUT hwo, UINT uMsg, DWORD_PTR dwInstance, DWORD_PTR dwParam1, DWORD_PTR dwParam2) static BOOL rdpsnd_winmm_set_format(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format,
UINT32 latency)
{
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
if (!rdpsnd_winmm_convert_format(format, &winmm->format))
return FALSE;
return TRUE;
}
static void CALLBACK rdpsnd_winmm_callback_function(HWAVEOUT hwo, UINT uMsg, DWORD_PTR dwInstance,
DWORD_PTR dwParam1, DWORD_PTR dwParam2)
{ {
RDPSND_WAVE* wave;
LPWAVEHDR lpWaveHdr; LPWAVEHDR lpWaveHdr;
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) dwInstance; rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) dwInstance;
@ -113,39 +102,24 @@ static void CALLBACK rdpsnd_winmm_callback_function(HWAVEOUT hwo, UINT uMsg, DWO
case MM_WOM_CLOSE: case MM_WOM_CLOSE:
WLog_DBG(TAG, "MM_WOM_CLOSE"); WLog_DBG(TAG, "MM_WOM_CLOSE");
SetEvent(winmm->next);
break; break;
case MM_WOM_DONE: case MM_WOM_DONE:
{ WLog_DBG(TAG, "MM_WOM_DONE");
UINT32 wTimeDelta;
lpWaveHdr = (LPWAVEHDR) dwParam1; lpWaveHdr = (LPWAVEHDR) dwParam1;
if (!lpWaveHdr)
return;
wave = (RDPSND_WAVE*) lpWaveHdr->dwUser;
if (!wave)
return;
WLog_DBG(TAG, "MM_WOM_DONE: dwBufferLength: %"PRIu32" cBlockNo: %"PRIu8"",
lpWaveHdr->dwBufferLength, wave->cBlockNo);
wave->wLocalTimeB = GetTickCount();
wTimeDelta = wave->wLocalTimeB - wave->wLocalTimeA;
wave->wTimeStampB = wave->wTimeStampA + wTimeDelta;
winmm->device.WaveConfirm(&(winmm->device), wave);
free(lpWaveHdr->lpData);
free(lpWaveHdr); free(lpWaveHdr);
SetEvent(winmm->next);
break;
free(wave); default:
} WLog_DBG(TAG, "UNKNOWN [0x%08"PRIx32"]", uMsg);
break; break;
} }
} }
static BOOL rdpsnd_winmm_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency) static BOOL rdpsnd_winmm_open(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format,
UINT32 latency)
{ {
MMRESULT mmResult; MMRESULT mmResult;
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device; rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
@ -153,8 +127,8 @@ static BOOL rdpsnd_winmm_open(rdpsndDevicePlugin* device, AUDIO_FORMAT* format,
if (winmm->hWaveOut) if (winmm->hWaveOut)
return TRUE; return TRUE;
rdpsnd_winmm_set_format(device, format, latency); if (!rdpsnd_winmm_set_format(device, format, latency))
freerdp_dsp_context_reset_adpcm(winmm->dsp_context); return FALSE;
mmResult = waveOutOpen(&winmm->hWaveOut, WAVE_MAPPER, &winmm->format, mmResult = waveOutOpen(&winmm->hWaveOut, WAVE_MAPPER, &winmm->format,
(DWORD_PTR) rdpsnd_winmm_callback_function, (DWORD_PTR) winmm, CALLBACK_FUNCTION); (DWORD_PTR) rdpsnd_winmm_callback_function, (DWORD_PTR) winmm, CALLBACK_FUNCTION);
@ -184,7 +158,6 @@ static void rdpsnd_winmm_close(rdpsndDevicePlugin* device)
if (winmm->hWaveOut) if (winmm->hWaveOut)
{ {
mmResult = waveOutReset(winmm->hWaveOut); mmResult = waveOutReset(winmm->hWaveOut);
mmResult = waveOutClose(winmm->hWaveOut); mmResult = waveOutClose(winmm->hWaveOut);
if (mmResult != MMSYSERR_NOERROR) if (mmResult != MMSYSERR_NOERROR)
@ -203,14 +176,12 @@ static void rdpsnd_winmm_free(rdpsndDevicePlugin* device)
if (winmm) if (winmm)
{ {
rdpsnd_winmm_close(device); rdpsnd_winmm_close(device);
CloseHandle(winmm->next);
freerdp_dsp_context_free(winmm->dsp_context);
free(winmm); free(winmm);
} }
} }
static BOOL rdpsnd_winmm_format_supported(rdpsndDevicePlugin* device, AUDIO_FORMAT* format) static BOOL rdpsnd_winmm_format_supported(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format)
{ {
MMRESULT result; MMRESULT result;
WAVEFORMATEX out; WAVEFORMATEX out;
@ -231,9 +202,7 @@ static UINT32 rdpsnd_winmm_get_volume(rdpsndDevicePlugin* device)
DWORD dwVolume; DWORD dwVolume;
UINT16 dwVolumeLeft; UINT16 dwVolumeLeft;
UINT16 dwVolumeRight; UINT16 dwVolumeRight;
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device; rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
dwVolumeLeft = ((50 * 0xFFFF) / 100); /* 50% */ dwVolumeLeft = ((50 * 0xFFFF) / 100); /* 50% */
dwVolumeRight = ((50 * 0xFFFF) / 100); /* 50% */ dwVolumeRight = ((50 * 0xFFFF) / 100); /* 50% */
dwVolume = (dwVolumeLeft << 16) | dwVolumeRight; dwVolume = (dwVolumeLeft << 16) | dwVolumeRight;
@ -242,14 +211,12 @@ static UINT32 rdpsnd_winmm_get_volume(rdpsndDevicePlugin* device)
return dwVolume; return dwVolume;
waveOutGetVolume(winmm->hWaveOut, &dwVolume); waveOutGetVolume(winmm->hWaveOut, &dwVolume);
return dwVolume; return dwVolume;
} }
static BOOL rdpsnd_winmm_set_volume(rdpsndDevicePlugin* device, UINT32 value) static BOOL rdpsnd_winmm_set_volume(rdpsndDevicePlugin* device, UINT32 value)
{ {
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device; rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
winmm->volume = value; winmm->volume = value;
if (!winmm->hWaveOut) if (!winmm->hWaveOut)
@ -258,72 +225,38 @@ static BOOL rdpsnd_winmm_set_volume(rdpsndDevicePlugin* device, UINT32 value)
return (waveOutSetVolume(winmm->hWaveOut, value) == MMSYSERR_NOERROR); return (waveOutSetVolume(winmm->hWaveOut, value) == MMSYSERR_NOERROR);
} }
static BOOL rdpsnd_winmm_wave_decode(rdpsndDevicePlugin* device, RDPSND_WAVE* wave) static void rdpsnd_winmm_start(rdpsndDevicePlugin* device)
{ {
int length; //rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
BYTE* data;
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
if (winmm->wformat == WAVE_FORMAT_ADPCM)
{
winmm->dsp_context->decode_ms_adpcm(winmm->dsp_context,
wave->data, wave->length, winmm->format.nChannels, winmm->block_size);
length = winmm->dsp_context->adpcm_size;
data = winmm->dsp_context->adpcm_buffer;
}
else if (winmm->wformat == WAVE_FORMAT_DVI_ADPCM)
{
winmm->dsp_context->decode_ima_adpcm(winmm->dsp_context,
wave->data, wave->length, winmm->format.nChannels, winmm->block_size);
length = winmm->dsp_context->adpcm_size;
data = winmm->dsp_context->adpcm_buffer;
}
else
{
length = wave->length;
data = wave->data;
} }
wave->data = (BYTE*) malloc(length); static UINT rdpsnd_winmm_play(rdpsndDevicePlugin* device, const BYTE* data, size_t size)
if (!wave->data)
return FALSE;
CopyMemory(wave->data, data, length);
wave->length = length;
return TRUE;
}
void rdpsnd_winmm_wave_play(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
{ {
MMRESULT mmResult; MMRESULT mmResult;
LPWAVEHDR lpWaveHdr; LPWAVEHDR lpWaveHdr;
rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device; rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
if (!winmm->hWaveOut) if (!winmm->hWaveOut)
return; return 0;
wave->AutoConfirm = FALSE;
lpWaveHdr = (LPWAVEHDR) malloc(sizeof(WAVEHDR)); lpWaveHdr = (LPWAVEHDR) malloc(sizeof(WAVEHDR));
if (!lpWaveHdr) if (!lpWaveHdr)
return; return 0;
ZeroMemory(lpWaveHdr, sizeof(WAVEHDR)); ZeroMemory(lpWaveHdr, sizeof(WAVEHDR));
lpWaveHdr->dwFlags = 0; lpWaveHdr->dwFlags = 0;
lpWaveHdr->dwLoops = 0; lpWaveHdr->dwLoops = 0;
lpWaveHdr->lpData = (LPSTR) wave->data; lpWaveHdr->lpData = (LPSTR) data;
lpWaveHdr->dwBufferLength = wave->length; lpWaveHdr->dwBufferLength = size;
lpWaveHdr->dwUser = (DWORD_PTR) wave; lpWaveHdr->dwUser = NULL;
lpWaveHdr->lpNext = NULL; lpWaveHdr->lpNext = NULL;
mmResult = waveOutPrepareHeader(winmm->hWaveOut, lpWaveHdr, sizeof(WAVEHDR)); mmResult = waveOutPrepareHeader(winmm->hWaveOut, lpWaveHdr, sizeof(WAVEHDR));
if (mmResult != MMSYSERR_NOERROR) if (mmResult != MMSYSERR_NOERROR)
{ {
WLog_ERR(TAG, "waveOutPrepareHeader failure: %"PRIu32"", mmResult); WLog_ERR(TAG, "waveOutPrepareHeader failure: %"PRIu32"", mmResult);
return; return 0;
} }
mmResult = waveOutWrite(winmm->hWaveOut, lpWaveHdr, sizeof(WAVEHDR)); mmResult = waveOutWrite(winmm->hWaveOut, lpWaveHdr, sizeof(WAVEHDR));
@ -333,18 +266,15 @@ void rdpsnd_winmm_wave_play(rdpsndDevicePlugin* device, RDPSND_WAVE* wave)
WLog_ERR(TAG, "waveOutWrite failure: %"PRIu32"", mmResult); WLog_ERR(TAG, "waveOutWrite failure: %"PRIu32"", mmResult);
waveOutUnprepareHeader(winmm->hWaveOut, lpWaveHdr, sizeof(WAVEHDR)); waveOutUnprepareHeader(winmm->hWaveOut, lpWaveHdr, sizeof(WAVEHDR));
free(lpWaveHdr); free(lpWaveHdr);
return; return 0;
}
} }
static void rdpsnd_winmm_start(rdpsndDevicePlugin* device) WaitForSingleObject(winmm->next, INFINITE);
{ return 10; /* TODO: Get real latencry in [ms] */
//rdpsndWinmmPlugin* winmm = (rdpsndWinmmPlugin*) device;
} }
static void rdpsnd_winmm_parse_addin_args(rdpsndDevicePlugin* device, ADDIN_ARGV* args) static void rdpsnd_winmm_parse_addin_args(rdpsndDevicePlugin* device, ADDIN_ARGV* args)
{ {
} }
#ifdef BUILTIN_CHANNELS #ifdef BUILTIN_CHANNELS
@ -362,38 +292,30 @@ UINT freerdp_rdpsnd_client_subsystem_entry(PFREERDP_RDPSND_DEVICE_ENTRY_POINTS p
{ {
ADDIN_ARGV* args; ADDIN_ARGV* args;
rdpsndWinmmPlugin* winmm; rdpsndWinmmPlugin* winmm;
winmm = (rdpsndWinmmPlugin*) calloc(1, sizeof(rdpsndWinmmPlugin)); winmm = (rdpsndWinmmPlugin*) calloc(1, sizeof(rdpsndWinmmPlugin));
if (!winmm) if (!winmm)
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
winmm->device.DisableConfirmThread = TRUE;
winmm->device.Open = rdpsnd_winmm_open; winmm->device.Open = rdpsnd_winmm_open;
winmm->device.FormatSupported = rdpsnd_winmm_format_supported; winmm->device.FormatSupported = rdpsnd_winmm_format_supported;
winmm->device.SetFormat = rdpsnd_winmm_set_format;
winmm->device.GetVolume = rdpsnd_winmm_get_volume; winmm->device.GetVolume = rdpsnd_winmm_get_volume;
winmm->device.SetVolume = rdpsnd_winmm_set_volume; winmm->device.SetVolume = rdpsnd_winmm_set_volume;
winmm->device.WaveDecode = rdpsnd_winmm_wave_decode;
winmm->device.WavePlay = rdpsnd_winmm_wave_play;
winmm->device.Start = rdpsnd_winmm_start; winmm->device.Start = rdpsnd_winmm_start;
winmm->device.Play = rdpsnd_winmm_play;
winmm->device.Close = rdpsnd_winmm_close; winmm->device.Close = rdpsnd_winmm_close;
winmm->device.Free = rdpsnd_winmm_free; winmm->device.Free = rdpsnd_winmm_free;
winmm->next = CreateEventA(NULL, FALSE, FALSE, "winmm-play-event");
args = pEntryPoints->args; if (!winmm->next)
rdpsnd_winmm_parse_addin_args((rdpsndDevicePlugin*) winmm, args);
winmm->dsp_context = freerdp_dsp_context_new();
if (!winmm->dsp_context)
{ {
free(winmm); free(winmm);
return CHANNEL_RC_NO_MEMORY; return CHANNEL_RC_NO_MEMORY;
} }
args = pEntryPoints->args;
rdpsnd_winmm_parse_addin_args((rdpsndDevicePlugin*) winmm, args);
winmm->volume = 0xFFFFFFFF; winmm->volume = 0xFFFFFFFF;
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) winmm); pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd, (rdpsndDevicePlugin*) winmm);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }

View File

@ -246,7 +246,6 @@ static DWORD WINAPI rdpsnd_server_thread(LPVOID arg)
RdpsndServerContext* context; RdpsndServerContext* context;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
context = (RdpsndServerContext*)arg; context = (RdpsndServerContext*)arg;
nCount = 0; nCount = 0;
events[nCount++] = context->priv->channelEvent; events[nCount++] = context->priv->channelEvent;
events[nCount++] = context->priv->StopEvent; events[nCount++] = context->priv->StopEvent;
@ -393,7 +392,7 @@ static UINT rdpsnd_server_select_format(RdpsndServerContext* context,
context->priv->out_buffer_size = out_buffer_size; context->priv->out_buffer_size = out_buffer_size;
} }
freerdp_dsp_context_reset_adpcm(context->priv->dsp_context); freerdp_dsp_context_reset(context->priv->dsp_context, format);
out: out:
LeaveCriticalSection(&context->priv->lock); LeaveCriticalSection(&context->priv->lock);
return error; return error;
@ -408,103 +407,54 @@ out:
static UINT rdpsnd_server_send_audio_pdu(RdpsndServerContext* context, static UINT rdpsnd_server_send_audio_pdu(RdpsndServerContext* context,
UINT16 wTimestamp) UINT16 wTimestamp)
{ {
int size; size_t length;
BYTE* src; size_t start, end = 0;
int frames; const BYTE* src;
int fill_size;
BOOL status; BOOL status;
AUDIO_FORMAT* format; AUDIO_FORMAT* format;
int tbytes_per_frame;
ULONG written; ULONG written;
wStream* s = context->priv->rdpsnd_pdu; wStream* s = context->priv->rdpsnd_pdu;
UINT error = CHANNEL_RC_OK; UINT error = CHANNEL_RC_OK;
format = &context->client_formats[context->selected_client_format]; format = &context->client_formats[context->selected_client_format];
tbytes_per_frame = format->nChannels * context->priv->src_bytes_per_sample;
if ((format->nSamplesPerSec == context->src_format.nSamplesPerSec) &&
(format->nChannels == context->src_format.nChannels))
{
src = context->priv->out_buffer;
frames = context->priv->out_pending_frames;
}
else
{
context->priv->dsp_context->resample(context->priv->dsp_context,
context->priv->out_buffer,
context->priv->src_bytes_per_sample, context->src_format.nChannels,
context->src_format.nSamplesPerSec, context->priv->out_pending_frames,
format->nChannels, format->nSamplesPerSec);
frames = context->priv->dsp_context->resampled_frames;
src = context->priv->dsp_context->resampled_buffer;
}
size = frames * tbytes_per_frame;
if (format->wFormatTag == WAVE_FORMAT_DVI_ADPCM)
{
context->priv->dsp_context->encode_ima_adpcm(context->priv->dsp_context,
src, size, format->nChannels, format->nBlockAlign);
src = context->priv->dsp_context->adpcm_buffer;
size = context->priv->dsp_context->adpcm_size;
}
else if (format->wFormatTag == WAVE_FORMAT_ADPCM)
{
context->priv->dsp_context->encode_ms_adpcm(context->priv->dsp_context,
src, size, format->nChannels, format->nBlockAlign);
src = context->priv->dsp_context->adpcm_buffer;
size = context->priv->dsp_context->adpcm_size;
}
context->block_no = (context->block_no + 1) % 256;
/* Fill to nBlockAlign for the last audio packet */
fill_size = 0;
if ((format->wFormatTag == WAVE_FORMAT_DVI_ADPCM
|| format->wFormatTag == WAVE_FORMAT_ADPCM) &&
(context->priv->out_pending_frames < context->priv->out_frames)
&& ((size % format->nBlockAlign) != 0))
{
fill_size = format->nBlockAlign - (size % format->nBlockAlign);
}
/* WaveInfo PDU */ /* WaveInfo PDU */
Stream_SetPosition(s, 0); Stream_SetPosition(s, 0);
Stream_Write_UINT8(s, SNDC_WAVE); /* msgType */ Stream_Write_UINT8(s, SNDC_WAVE); /* msgType */
Stream_Write_UINT8(s, 0); /* bPad */ Stream_Write_UINT8(s, 0); /* bPad */
Stream_Write_UINT16(s, size + fill_size + 8); /* BodySize */ Stream_Write_UINT16(s, 0); /* BodySize */
Stream_Write_UINT16(s, wTimestamp); /* wTimeStamp */ Stream_Write_UINT16(s, wTimestamp); /* wTimeStamp */
Stream_Write_UINT16(s, context->selected_client_format); /* wFormatNo */ Stream_Write_UINT16(s, context->selected_client_format); /* wFormatNo */
Stream_Write_UINT8(s, context->block_no); /* cBlockNo */ Stream_Write_UINT8(s, context->block_no); /* cBlockNo */
Stream_Seek(s, 3); /* bPad */ Stream_Seek(s, 3); /* bPad */
Stream_Write(s, src, 4); start = Stream_GetPosition(s);
status = WTSVirtualChannelWrite(context->priv->ChannelHandle, src = context->priv->out_buffer;
(PCHAR) Stream_Buffer(s), Stream_GetPosition(s), &written); length = context->priv->out_pending_frames * context->priv->src_bytes_per_frame;
if (!status) if (!freerdp_dsp_encode(context->priv->dsp_context, format, src, length, s))
status = ERROR_INTERNAL_ERROR;
else
{
/* Set stream size */
end = Stream_GetPosition(s);
Stream_SetPosition(s, 2);
Stream_Write_UINT16(s, end - start + 8);
Stream_SetPosition(s, end);
context->block_no = (context->block_no + 1) % 256;
status = WTSVirtualChannelWrite(context->priv->ChannelHandle,
(PCHAR) Stream_Buffer(s), start + 4, &written);
}
if (status != CHANNEL_RC_OK)
{ {
WLog_ERR(TAG, "WTSVirtualChannelWrite failed!"); WLog_ERR(TAG, "WTSVirtualChannelWrite failed!");
error = ERROR_INTERNAL_ERROR; error = ERROR_INTERNAL_ERROR;
goto out; goto out;
} }
Stream_SetPosition(s, 0); Stream_SetPosition(s, start);
/* Wave PDU */
if (!Stream_EnsureRemainingCapacity(s, size + fill_size))
{
WLog_ERR(TAG, "Stream_EnsureRemainingCapacity failed!");
error = CHANNEL_RC_NO_MEMORY;
goto out;
}
Stream_Write_UINT32(s, 0); /* bPad */ Stream_Write_UINT32(s, 0); /* bPad */
Stream_Write(s, src + 4, size - 4); Stream_SetPosition(s, start);
if (fill_size > 0)
Stream_Zero(s, fill_size);
status = WTSVirtualChannelWrite(context->priv->ChannelHandle, status = WTSVirtualChannelWrite(context->priv->ChannelHandle,
(PCHAR) Stream_Buffer(s), Stream_GetPosition(s), &written); (PCHAR) Stream_Pointer(s), end - start, &written);
if (!status) if (!status)
{ {
@ -578,7 +528,6 @@ static UINT rdpsnd_server_set_volume(RdpsndServerContext* context, int left,
BOOL status; BOOL status;
ULONG written; ULONG written;
wStream* s = context->priv->rdpsnd_pdu; wStream* s = context->priv->rdpsnd_pdu;
Stream_Write_UINT8(s, SNDC_SETVOLUME); Stream_Write_UINT8(s, SNDC_SETVOLUME);
Stream_Write_UINT8(s, 0); Stream_Write_UINT8(s, 0);
Stream_Seek_UINT16(s); Stream_Seek_UINT16(s);
@ -790,7 +739,7 @@ RdpsndServerContext* rdpsnd_server_context_new(HANDLE vcm)
goto out_free; goto out_free;
} }
priv->dsp_context = freerdp_dsp_context_new(); priv->dsp_context = freerdp_dsp_context_new(TRUE);
if (!priv->dsp_context) if (!priv->dsp_context)
{ {

View File

@ -46,19 +46,19 @@ typedef struct _TSMFALSAAudioDevice
UINT32 source_channels; UINT32 source_channels;
UINT32 actual_channels; UINT32 actual_channels;
UINT32 bytes_per_sample; UINT32 bytes_per_sample;
FREERDP_DSP_CONTEXT *dsp_context;
} TSMFAlsaAudioDevice; } TSMFAlsaAudioDevice;
static BOOL tsmf_alsa_open_device(TSMFAlsaAudioDevice* alsa) static BOOL tsmf_alsa_open_device(TSMFAlsaAudioDevice* alsa)
{ {
int error; int error;
error = snd_pcm_open(&alsa->out_handle, alsa->device, SND_PCM_STREAM_PLAYBACK, 0); error = snd_pcm_open(&alsa->out_handle, alsa->device, SND_PCM_STREAM_PLAYBACK, 0);
if (error < 0) if (error < 0)
{ {
WLog_ERR(TAG, "failed to open device %s", alsa->device); WLog_ERR(TAG, "failed to open device %s", alsa->device);
return FALSE; return FALSE;
} }
DEBUG_TSMF("open device %s", alsa->device); DEBUG_TSMF("open device %s", alsa->device);
return TRUE; return TRUE;
} }
@ -66,6 +66,7 @@ static BOOL tsmf_alsa_open_device(TSMFAlsaAudioDevice *alsa)
static BOOL tsmf_alsa_open(ITSMFAudioDevice* audio, const char* device) static BOOL tsmf_alsa_open(ITSMFAudioDevice* audio, const char* device)
{ {
TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio; TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio;
if (!device) if (!device)
{ {
strncpy(alsa->device, "default", sizeof(alsa->device)); strncpy(alsa->device, "default", sizeof(alsa->device));
@ -74,6 +75,7 @@ static BOOL tsmf_alsa_open(ITSMFAudioDevice *audio, const char *device)
{ {
strncpy(alsa->device, device, sizeof(alsa->device) - 1); strncpy(alsa->device, device, sizeof(alsa->device) - 1);
} }
return tsmf_alsa_open_device(alsa); return tsmf_alsa_open_device(alsa);
} }
@ -85,18 +87,22 @@ static BOOL tsmf_alsa_set_format(ITSMFAudioDevice *audio,
snd_pcm_hw_params_t* hw_params; snd_pcm_hw_params_t* hw_params;
snd_pcm_sw_params_t* sw_params; snd_pcm_sw_params_t* sw_params;
TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio; TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio;
if (!alsa->out_handle) if (!alsa->out_handle)
return FALSE; return FALSE;
snd_pcm_drop(alsa->out_handle); snd_pcm_drop(alsa->out_handle);
alsa->actual_rate = alsa->source_rate = sample_rate; alsa->actual_rate = alsa->source_rate = sample_rate;
alsa->actual_channels = alsa->source_channels = channels; alsa->actual_channels = alsa->source_channels = channels;
alsa->bytes_per_sample = bits_per_sample / 8; alsa->bytes_per_sample = bits_per_sample / 8;
error = snd_pcm_hw_params_malloc(&hw_params); error = snd_pcm_hw_params_malloc(&hw_params);
if (error < 0) if (error < 0)
{ {
WLog_ERR(TAG, "snd_pcm_hw_params_malloc failed"); WLog_ERR(TAG, "snd_pcm_hw_params_malloc failed");
return FALSE; return FALSE;
} }
snd_pcm_hw_params_any(alsa->out_handle, hw_params); snd_pcm_hw_params_any(alsa->out_handle, hw_params);
snd_pcm_hw_params_set_access(alsa->out_handle, hw_params, snd_pcm_hw_params_set_access(alsa->out_handle, hw_params,
SND_PCM_ACCESS_RW_INTERLEAVED); SND_PCM_ACCESS_RW_INTERLEAVED);
@ -112,11 +118,13 @@ static BOOL tsmf_alsa_set_format(ITSMFAudioDevice *audio,
snd_pcm_hw_params(alsa->out_handle, hw_params); snd_pcm_hw_params(alsa->out_handle, hw_params);
snd_pcm_hw_params_free(hw_params); snd_pcm_hw_params_free(hw_params);
error = snd_pcm_sw_params_malloc(&sw_params); error = snd_pcm_sw_params_malloc(&sw_params);
if (error < 0) if (error < 0)
{ {
WLog_ERR(TAG, "snd_pcm_sw_params_malloc"); WLog_ERR(TAG, "snd_pcm_sw_params_malloc");
return FALSE; return FALSE;
} }
snd_pcm_sw_params_current(alsa->out_handle, sw_params); snd_pcm_sw_params_current(alsa->out_handle, sw_params);
snd_pcm_sw_params_set_start_threshold(alsa->out_handle, sw_params, snd_pcm_sw_params_set_start_threshold(alsa->out_handle, sw_params,
frames / 2); frames / 2);
@ -126,6 +134,7 @@ static BOOL tsmf_alsa_set_format(ITSMFAudioDevice *audio,
DEBUG_TSMF("sample_rate %"PRIu32" channels %"PRIu32" bits_per_sample %"PRIu32"", DEBUG_TSMF("sample_rate %"PRIu32" channels %"PRIu32" bits_per_sample %"PRIu32"",
sample_rate, channels, bits_per_sample); sample_rate, channels, bits_per_sample);
DEBUG_TSMF("hardware buffer %lu frames", frames); DEBUG_TSMF("hardware buffer %lu frames", frames);
if ((alsa->actual_rate != alsa->source_rate) || if ((alsa->actual_rate != alsa->source_rate) ||
(alsa->actual_channels != alsa->source_channels)) (alsa->actual_channels != alsa->source_channels))
{ {
@ -134,48 +143,35 @@ static BOOL tsmf_alsa_set_format(ITSMFAudioDevice *audio,
alsa->actual_rate, alsa->actual_channels, alsa->actual_rate, alsa->actual_channels,
alsa->source_rate, alsa->source_channels); alsa->source_rate, alsa->source_channels);
} }
return TRUE; return TRUE;
} }
static BOOL tsmf_alsa_play(ITSMFAudioDevice *audio, BYTE *data, UINT32 data_size) static BOOL tsmf_alsa_play(ITSMFAudioDevice* audio, const BYTE* src, UINT32 data_size)
{ {
int len; int len;
int error; int error;
int frames; int frames;
BYTE *end; const BYTE* end;
BYTE *src; const BYTE* pindex;
BYTE *pindex;
int rbytes_per_frame; int rbytes_per_frame;
int sbytes_per_frame; int sbytes_per_frame;
TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio; TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio;
DEBUG_TSMF("data_size %"PRIu32"", data_size); DEBUG_TSMF("data_size %"PRIu32"", data_size);
if (alsa->out_handle) if (alsa->out_handle)
{ {
sbytes_per_frame = alsa->source_channels * alsa->bytes_per_sample; sbytes_per_frame = alsa->source_channels * alsa->bytes_per_sample;
rbytes_per_frame = alsa->actual_channels * alsa->bytes_per_sample; rbytes_per_frame = alsa->actual_channels * alsa->bytes_per_sample;
if((alsa->source_rate == alsa->actual_rate) &&
(alsa->source_channels == alsa->actual_channels))
{
src = data;
}
else
{
alsa->dsp_context->resample(alsa->dsp_context, data, alsa->bytes_per_sample,
alsa->source_channels, alsa->source_rate, data_size / sbytes_per_frame,
alsa->actual_channels, alsa->actual_rate);
frames = alsa->dsp_context->resampled_frames;
DEBUG_TSMF("resampled %"PRIu32" frames at %"PRIu32" to %d frames at %"PRIu32"",
data_size / sbytes_per_frame, alsa->source_rate, frames, alsa->actual_rate);
data_size = frames * rbytes_per_frame;
src = alsa->dsp_context->resampled_buffer;
}
pindex = src; pindex = src;
end = pindex + data_size; end = pindex + data_size;
while (pindex < end) while (pindex < end)
{ {
len = end - pindex; len = end - pindex;
frames = len / rbytes_per_frame; frames = len / rbytes_per_frame;
error = snd_pcm_writei(alsa->out_handle, pindex, frames); error = snd_pcm_writei(alsa->out_handle, pindex, frames);
if (error == -EPIPE) if (error == -EPIPE)
{ {
snd_pcm_recover(alsa->out_handle, error, 0); snd_pcm_recover(alsa->out_handle, error, 0);
@ -189,13 +185,16 @@ static BOOL tsmf_alsa_play(ITSMFAudioDevice *audio, BYTE *data, UINT32 data_size
tsmf_alsa_open_device(alsa); tsmf_alsa_open_device(alsa);
break; break;
} }
DEBUG_TSMF("%d frames played.", error); DEBUG_TSMF("%d frames played.", error);
if (error == 0) if (error == 0)
break; break;
pindex += error * rbytes_per_frame; pindex += error * rbytes_per_frame;
} }
} }
free(data);
return TRUE; return TRUE;
} }
@ -204,12 +203,14 @@ static UINT64 tsmf_alsa_get_latency(ITSMFAudioDevice *audio)
UINT64 latency = 0; UINT64 latency = 0;
snd_pcm_sframes_t frames = 0; snd_pcm_sframes_t frames = 0;
TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio; TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio;
if (alsa->out_handle && alsa->actual_rate > 0 && if (alsa->out_handle && alsa->actual_rate > 0 &&
snd_pcm_delay(alsa->out_handle, &frames) == 0 && snd_pcm_delay(alsa->out_handle, &frames) == 0 &&
frames > 0) frames > 0)
{ {
latency = ((UINT64)frames) * 10000000LL / (UINT64) alsa->actual_rate; latency = ((UINT64)frames) * 10000000LL / (UINT64) alsa->actual_rate;
} }
return latency; return latency;
} }
@ -222,12 +223,13 @@ static void tsmf_alsa_free(ITSMFAudioDevice *audio)
{ {
TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio; TSMFAlsaAudioDevice* alsa = (TSMFAlsaAudioDevice*) audio;
DEBUG_TSMF(""); DEBUG_TSMF("");
if (alsa->out_handle) if (alsa->out_handle)
{ {
snd_pcm_drain(alsa->out_handle); snd_pcm_drain(alsa->out_handle);
snd_pcm_close(alsa->out_handle); snd_pcm_close(alsa->out_handle);
} }
freerdp_dsp_context_free(alsa->dsp_context);
free(alsa); free(alsa);
} }
@ -248,6 +250,5 @@ ITSMFAudioDevice *freerdp_tsmf_client_audio_subsystem_entry(void)
alsa->iface.GetLatency = tsmf_alsa_get_latency; alsa->iface.GetLatency = tsmf_alsa_get_latency;
alsa->iface.Flush = tsmf_alsa_flush; alsa->iface.Flush = tsmf_alsa_flush;
alsa->iface.Free = tsmf_alsa_free; alsa->iface.Free = tsmf_alsa_free;
alsa->dsp_context = freerdp_dsp_context_new();
return (ITSMFAudioDevice*) alsa; return (ITSMFAudioDevice*) alsa;
} }

View File

@ -129,7 +129,8 @@ static BOOL tsmf_oss_open(ITSMFAudioDevice* audio, const char* device)
return TRUE; return TRUE;
} }
static BOOL tsmf_oss_set_format(ITSMFAudioDevice* audio, UINT32 sample_rate, UINT32 channels, UINT32 bits_per_sample) static BOOL tsmf_oss_set_format(ITSMFAudioDevice* audio, UINT32 sample_rate, UINT32 channels,
UINT32 bits_per_sample)
{ {
int tmp; int tmp;
TSMFOssAudioDevice* oss = (TSMFOssAudioDevice*)audio; TSMFOssAudioDevice* oss = (TSMFOssAudioDevice*)audio;
@ -165,7 +166,7 @@ static BOOL tsmf_oss_set_format(ITSMFAudioDevice* audio, UINT32 sample_rate, UIN
return TRUE; return TRUE;
} }
static BOOL tsmf_oss_play(ITSMFAudioDevice* audio, BYTE* data, UINT32 data_size) static BOOL tsmf_oss_play(ITSMFAudioDevice* audio, const BYTE* data, UINT32 data_size)
{ {
int status; int status;
UINT32 offset; UINT32 offset;
@ -176,10 +177,7 @@ static BOOL tsmf_oss_play(ITSMFAudioDevice* audio, BYTE* data, UINT32 data_size)
return FALSE; return FALSE;
if (data == NULL || data_size == 0) if (data == NULL || data_size == 0)
{
free(data);
return TRUE; return TRUE;
}
offset = 0; offset = 0;
oss->data_size_last = data_size; oss->data_size_last = data_size;
@ -191,14 +189,12 @@ static BOOL tsmf_oss_play(ITSMFAudioDevice* audio, BYTE* data, UINT32 data_size)
if (status < 0) if (status < 0)
{ {
OSS_LOG_ERR("write fail", errno); OSS_LOG_ERR("write fail", errno);
free(data);
return FALSE; return FALSE;
} }
offset += status; offset += status;
} }
free(data);
return TRUE; return TRUE;
} }

View File

@ -48,17 +48,20 @@ static void tsmf_pulse_context_state_callback(pa_context *context, void *userdat
TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) userdata; TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) userdata;
pa_context_state_t state; pa_context_state_t state;
state = pa_context_get_state(context); state = pa_context_get_state(context);
switch (state) switch (state)
{ {
case PA_CONTEXT_READY: case PA_CONTEXT_READY:
DEBUG_TSMF("PA_CONTEXT_READY"); DEBUG_TSMF("PA_CONTEXT_READY");
pa_threaded_mainloop_signal(pulse->mainloop, 0); pa_threaded_mainloop_signal(pulse->mainloop, 0);
break; break;
case PA_CONTEXT_FAILED: case PA_CONTEXT_FAILED:
case PA_CONTEXT_TERMINATED: case PA_CONTEXT_TERMINATED:
DEBUG_TSMF("state %d", state); DEBUG_TSMF("state %d", state);
pa_threaded_mainloop_signal(pulse->mainloop, 0); pa_threaded_mainloop_signal(pulse->mainloop, 0);
break; break;
default: default:
DEBUG_TSMF("state %d", state); DEBUG_TSMF("state %d", state);
break; break;
@ -68,15 +71,19 @@ static void tsmf_pulse_context_state_callback(pa_context *context, void *userdat
static BOOL tsmf_pulse_connect(TSMFPulseAudioDevice* pulse) static BOOL tsmf_pulse_connect(TSMFPulseAudioDevice* pulse)
{ {
pa_context_state_t state; pa_context_state_t state;
if (!pulse->context) if (!pulse->context)
return FALSE; return FALSE;
if (pa_context_connect(pulse->context, NULL, 0, NULL)) if (pa_context_connect(pulse->context, NULL, 0, NULL))
{ {
WLog_ERR(TAG, "pa_context_connect failed (%d)", WLog_ERR(TAG, "pa_context_connect failed (%d)",
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
return FALSE; return FALSE;
} }
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
if (pa_threaded_mainloop_start(pulse->mainloop) < 0) if (pa_threaded_mainloop_start(pulse->mainloop) < 0)
{ {
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
@ -84,20 +91,26 @@ static BOOL tsmf_pulse_connect(TSMFPulseAudioDevice *pulse)
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
return FALSE; return FALSE;
} }
for (;;) for (;;)
{ {
state = pa_context_get_state(pulse->context); state = pa_context_get_state(pulse->context);
if (state == PA_CONTEXT_READY) if (state == PA_CONTEXT_READY)
break; break;
if (!PA_CONTEXT_IS_GOOD(state)) if (!PA_CONTEXT_IS_GOOD(state))
{ {
DEBUG_TSMF("bad context state (%d)", DEBUG_TSMF("bad context state (%d)",
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
break; break;
} }
pa_threaded_mainloop_wait(pulse->mainloop); pa_threaded_mainloop_wait(pulse->mainloop);
} }
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
if (state == PA_CONTEXT_READY) if (state == PA_CONTEXT_READY)
{ {
DEBUG_TSMF("connected"); DEBUG_TSMF("connected");
@ -113,28 +126,36 @@ static BOOL tsmf_pulse_connect(TSMFPulseAudioDevice *pulse)
static BOOL tsmf_pulse_open(ITSMFAudioDevice* audio, const char* device) static BOOL tsmf_pulse_open(ITSMFAudioDevice* audio, const char* device)
{ {
TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio; TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio;
if (device) if (device)
{ {
strncpy(pulse->device, device, sizeof(pulse->device) - 1); strncpy(pulse->device, device, sizeof(pulse->device) - 1);
} }
pulse->mainloop = pa_threaded_mainloop_new(); pulse->mainloop = pa_threaded_mainloop_new();
if (!pulse->mainloop) if (!pulse->mainloop)
{ {
WLog_ERR(TAG, "pa_threaded_mainloop_new failed"); WLog_ERR(TAG, "pa_threaded_mainloop_new failed");
return FALSE; return FALSE;
} }
pulse->context = pa_context_new(pa_threaded_mainloop_get_api(pulse->mainloop), "freerdp"); pulse->context = pa_context_new(pa_threaded_mainloop_get_api(pulse->mainloop), "freerdp");
if (!pulse->context) if (!pulse->context)
{ {
WLog_ERR(TAG, "pa_context_new failed"); WLog_ERR(TAG, "pa_context_new failed");
return FALSE; return FALSE;
} }
pa_context_set_state_callback(pulse->context, tsmf_pulse_context_state_callback, pulse); pa_context_set_state_callback(pulse->context, tsmf_pulse_context_state_callback, pulse);
if (!tsmf_pulse_connect(pulse)) if (!tsmf_pulse_connect(pulse))
{ {
WLog_ERR(TAG, "tsmf_pulse_connect failed"); WLog_ERR(TAG, "tsmf_pulse_connect failed");
return FALSE; return FALSE;
} }
DEBUG_TSMF("open device %s", pulse->device); DEBUG_TSMF("open device %s", pulse->device);
return TRUE; return TRUE;
} }
@ -149,10 +170,12 @@ static void tsmf_pulse_wait_for_operation(TSMFPulseAudioDevice *pulse, pa_operat
{ {
if (operation == NULL) if (operation == NULL)
return; return;
while (pa_operation_get_state(operation) == PA_OPERATION_RUNNING) while (pa_operation_get_state(operation) == PA_OPERATION_RUNNING)
{ {
pa_threaded_mainloop_wait(pulse->mainloop); pa_threaded_mainloop_wait(pulse->mainloop);
} }
pa_operation_unref(operation); pa_operation_unref(operation);
} }
@ -161,17 +184,20 @@ static void tsmf_pulse_stream_state_callback(pa_stream *stream, void *userdata)
TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) userdata; TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) userdata;
pa_stream_state_t state; pa_stream_state_t state;
state = pa_stream_get_state(stream); state = pa_stream_get_state(stream);
switch (state) switch (state)
{ {
case PA_STREAM_READY: case PA_STREAM_READY:
DEBUG_TSMF("PA_STREAM_READY"); DEBUG_TSMF("PA_STREAM_READY");
pa_threaded_mainloop_signal(pulse->mainloop, 0); pa_threaded_mainloop_signal(pulse->mainloop, 0);
break; break;
case PA_STREAM_FAILED: case PA_STREAM_FAILED:
case PA_STREAM_TERMINATED: case PA_STREAM_TERMINATED:
DEBUG_TSMF("state %d", state); DEBUG_TSMF("state %d", state);
pa_threaded_mainloop_signal(pulse->mainloop, 0); pa_threaded_mainloop_signal(pulse->mainloop, 0);
break; break;
default: default:
DEBUG_TSMF("state %d", state); DEBUG_TSMF("state %d", state);
break; break;
@ -189,6 +215,7 @@ static BOOL tsmf_pulse_close_stream(TSMFPulseAudioDevice *pulse)
{ {
if (!pulse->context || !pulse->stream) if (!pulse->context || !pulse->stream)
return FALSE; return FALSE;
DEBUG_TSMF(""); DEBUG_TSMF("");
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
pa_stream_set_write_callback(pulse->stream, NULL, NULL); pa_stream_set_write_callback(pulse->stream, NULL, NULL);
@ -205,12 +232,15 @@ static BOOL tsmf_pulse_open_stream(TSMFPulseAudioDevice *pulse)
{ {
pa_stream_state_t state; pa_stream_state_t state;
pa_buffer_attr buffer_attr = { 0 }; pa_buffer_attr buffer_attr = { 0 };
if (!pulse->context) if (!pulse->context)
return FALSE; return FALSE;
DEBUG_TSMF(""); DEBUG_TSMF("");
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
pulse->stream = pa_stream_new(pulse->context, "freerdp", pulse->stream = pa_stream_new(pulse->context, "freerdp",
&pulse->sample_spec, NULL); &pulse->sample_spec, NULL);
if (!pulse->stream) if (!pulse->stream)
{ {
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
@ -218,6 +248,7 @@ static BOOL tsmf_pulse_open_stream(TSMFPulseAudioDevice *pulse)
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
return FALSE; return FALSE;
} }
pa_stream_set_state_callback(pulse->stream, pa_stream_set_state_callback(pulse->stream,
tsmf_pulse_stream_state_callback, pulse); tsmf_pulse_stream_state_callback, pulse);
pa_stream_set_write_callback(pulse->stream, pa_stream_set_write_callback(pulse->stream,
@ -227,6 +258,7 @@ static BOOL tsmf_pulse_open_stream(TSMFPulseAudioDevice *pulse)
buffer_attr.prebuf = (UINT32) - 1; buffer_attr.prebuf = (UINT32) - 1;
buffer_attr.minreq = (UINT32) - 1; buffer_attr.minreq = (UINT32) - 1;
buffer_attr.fragsize = (UINT32) - 1; buffer_attr.fragsize = (UINT32) - 1;
if (pa_stream_connect_playback(pulse->stream, if (pa_stream_connect_playback(pulse->stream,
pulse->device[0] ? pulse->device : NULL, &buffer_attr, pulse->device[0] ? pulse->device : NULL, &buffer_attr,
PA_STREAM_ADJUST_LATENCY | PA_STREAM_INTERPOLATE_TIMING | PA_STREAM_AUTO_TIMING_UPDATE, PA_STREAM_ADJUST_LATENCY | PA_STREAM_INTERPOLATE_TIMING | PA_STREAM_AUTO_TIMING_UPDATE,
@ -237,20 +269,26 @@ static BOOL tsmf_pulse_open_stream(TSMFPulseAudioDevice *pulse)
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
return FALSE; return FALSE;
} }
for (;;) for (;;)
{ {
state = pa_stream_get_state(pulse->stream); state = pa_stream_get_state(pulse->stream);
if (state == PA_STREAM_READY) if (state == PA_STREAM_READY)
break; break;
if (!PA_STREAM_IS_GOOD(state)) if (!PA_STREAM_IS_GOOD(state))
{ {
WLog_ERR(TAG, "bad stream state (%d)", WLog_ERR(TAG, "bad stream state (%d)",
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
break; break;
} }
pa_threaded_mainloop_wait(pulse->mainloop); pa_threaded_mainloop_wait(pulse->mainloop);
} }
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
if (state == PA_STREAM_READY) if (state == PA_STREAM_READY)
{ {
DEBUG_TSMF("connected"); DEBUG_TSMF("connected");
@ -275,17 +313,19 @@ static BOOL tsmf_pulse_set_format(ITSMFAudioDevice *audio,
return tsmf_pulse_open_stream(pulse); return tsmf_pulse_open_stream(pulse);
} }
static BOOL tsmf_pulse_play(ITSMFAudioDevice *audio, BYTE *data, UINT32 data_size) static BOOL tsmf_pulse_play(ITSMFAudioDevice* audio, const BYTE* data, UINT32 data_size)
{ {
TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio; TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio;
BYTE *src; const BYTE* src;
int len; int len;
int ret; int ret;
DEBUG_TSMF("data_size %"PRIu32"", data_size); DEBUG_TSMF("data_size %"PRIu32"", data_size);
if (pulse->stream) if (pulse->stream)
{ {
pa_threaded_mainloop_lock(pulse->mainloop); pa_threaded_mainloop_lock(pulse->mainloop);
src = data; src = data;
while (data_size > 0) while (data_size > 0)
{ {
while ((len = pa_stream_writable_size(pulse->stream)) == 0) while ((len = pa_stream_writable_size(pulse->stream)) == 0)
@ -293,23 +333,29 @@ static BOOL tsmf_pulse_play(ITSMFAudioDevice *audio, BYTE *data, UINT32 data_siz
DEBUG_TSMF("waiting"); DEBUG_TSMF("waiting");
pa_threaded_mainloop_wait(pulse->mainloop); pa_threaded_mainloop_wait(pulse->mainloop);
} }
if (len < 0) if (len < 0)
break; break;
if (len > data_size) if (len > data_size)
len = data_size; len = data_size;
ret = pa_stream_write(pulse->stream, src, len, NULL, 0LL, PA_SEEK_RELATIVE); ret = pa_stream_write(pulse->stream, src, len, NULL, 0LL, PA_SEEK_RELATIVE);
if (ret < 0) if (ret < 0)
{ {
DEBUG_TSMF("pa_stream_write failed (%d)", DEBUG_TSMF("pa_stream_write failed (%d)",
pa_context_errno(pulse->context)); pa_context_errno(pulse->context));
break; break;
} }
src += len; src += len;
data_size -= len; data_size -= len;
} }
pa_threaded_mainloop_unlock(pulse->mainloop); pa_threaded_mainloop_unlock(pulse->mainloop);
} }
free(data);
return TRUE; return TRUE;
} }
@ -318,10 +364,12 @@ static UINT64 tsmf_pulse_get_latency(ITSMFAudioDevice *audio)
pa_usec_t usec; pa_usec_t usec;
UINT64 latency = 0; UINT64 latency = 0;
TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio; TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio;
if (pulse->stream && pa_stream_get_latency(pulse->stream, &usec, NULL) == 0) if (pulse->stream && pa_stream_get_latency(pulse->stream, &usec, NULL) == 0)
{ {
latency = ((UINT64)usec) * 10LL; latency = ((UINT64)usec) * 10LL;
} }
return latency; return latency;
} }
@ -340,21 +388,25 @@ static void tsmf_pulse_free(ITSMFAudioDevice *audio)
TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio; TSMFPulseAudioDevice* pulse = (TSMFPulseAudioDevice*) audio;
DEBUG_TSMF(""); DEBUG_TSMF("");
tsmf_pulse_close_stream(pulse); tsmf_pulse_close_stream(pulse);
if (pulse->mainloop) if (pulse->mainloop)
{ {
pa_threaded_mainloop_stop(pulse->mainloop); pa_threaded_mainloop_stop(pulse->mainloop);
} }
if (pulse->context) if (pulse->context)
{ {
pa_context_disconnect(pulse->context); pa_context_disconnect(pulse->context);
pa_context_unref(pulse->context); pa_context_unref(pulse->context);
pulse->context = NULL; pulse->context = NULL;
} }
if (pulse->mainloop) if (pulse->mainloop)
{ {
pa_threaded_mainloop_free(pulse->mainloop); pa_threaded_mainloop_free(pulse->mainloop);
pulse->mainloop = NULL; pulse->mainloop = NULL;
} }
free(pulse); free(pulse);
} }
@ -365,10 +417,11 @@ FREERDP_API ITSMFAudioDevice *freerdp_tsmf_client_audio_subsystem_entry(void)
#endif #endif
{ {
TSMFPulseAudioDevice* pulse; TSMFPulseAudioDevice* pulse;
pulse = (TSMFPulseAudioDevice*) calloc(1, sizeof(TSMFPulseAudioDevice)); pulse = (TSMFPulseAudioDevice*) calloc(1, sizeof(TSMFPulseAudioDevice));
if (!pulse) if (!pulse)
return NULL; return NULL;
pulse->iface.Open = tsmf_pulse_open; pulse->iface.Open = tsmf_pulse_open;
pulse->iface.SetFormat = tsmf_pulse_set_format; pulse->iface.SetFormat = tsmf_pulse_set_format;
pulse->iface.Play = tsmf_pulse_play; pulse->iface.Play = tsmf_pulse_play;

View File

@ -29,9 +29,10 @@ struct _ITSMFAudioDevice
/* Open the audio device. */ /* Open the audio device. */
BOOL (*Open)(ITSMFAudioDevice* audio, const char* device); BOOL (*Open)(ITSMFAudioDevice* audio, const char* device);
/* Set the audio data format. */ /* Set the audio data format. */
BOOL (*SetFormat) (ITSMFAudioDevice* audio, UINT32 sample_rate, UINT32 channels, UINT32 bits_per_sample); BOOL (*SetFormat)(ITSMFAudioDevice* audio, UINT32 sample_rate, UINT32 channels,
UINT32 bits_per_sample);
/* Play audio data. */ /* Play audio data. */
BOOL (*Play) (ITSMFAudioDevice* audio, BYTE* data, UINT32 data_size); BOOL (*Play)(ITSMFAudioDevice* audio, const BYTE* data, UINT32 data_size);
/* Get the latency of the last written sample, in 100ns */ /* Get the latency of the last written sample, in 100ns */
UINT64(*GetLatency)(ITSMFAudioDevice* audio); UINT64(*GetLatency)(ITSMFAudioDevice* audio);
/* Change the playback volume level */ /* Change the playback volume level */

View File

@ -159,8 +159,8 @@ struct _TSMF_SAMPLE
static wArrayList* presentation_list = NULL; static wArrayList* presentation_list = NULL;
static int TERMINATING = 0; static int TERMINATING = 0;
static void _tsmf_presentation_free(TSMF_PRESENTATION* presentation); static void _tsmf_presentation_free(void* obj);
static void _tsmf_stream_free(TSMF_STREAM* stream); static void _tsmf_stream_free(void* obj);
static UINT64 get_current_time(void) static UINT64 get_current_time(void)
{ {
@ -367,7 +367,7 @@ TSMF_PRESENTATION* tsmf_presentation_new(const BYTE* guid,
goto error_stream_list; goto error_stream_list;
ArrayList_Object(presentation->stream_list)->fnObjectFree = ArrayList_Object(presentation->stream_list)->fnObjectFree =
(OBJECT_FREE_FN) _tsmf_stream_free; _tsmf_stream_free;
if (ArrayList_Add(presentation_list, presentation) < 0) if (ArrayList_Add(presentation_list, presentation) < 0)
goto error_add; goto error_add;
@ -528,6 +528,7 @@ static BOOL tsmf_sample_playback_audio(TSMF_SAMPLE* sample)
{ {
ret = sample->stream->audio->Play(sample->stream->audio, sample->data, ret = sample->stream->audio->Play(sample->stream->audio, sample->data,
sample->decoded_size); sample->decoded_size);
free(sample->data);
sample->data = NULL; sample->data = NULL;
sample->decoded_size = 0; sample->decoded_size = 0;
@ -1193,7 +1194,11 @@ BOOL tsmf_stream_flush(TSMF_STREAM* stream)
return TRUE; return TRUE;
} }
void _tsmf_presentation_free(TSMF_PRESENTATION* presentation) void _tsmf_presentation_free(void* obj)
{
TSMF_PRESENTATION* presentation = (TSMF_PRESENTATION*)obj;
if (presentation)
{ {
tsmf_presentation_stop(presentation); tsmf_presentation_stop(presentation);
ArrayList_Clear(presentation->stream_list); ArrayList_Clear(presentation->stream_list);
@ -1202,6 +1207,7 @@ void _tsmf_presentation_free(TSMF_PRESENTATION* presentation)
ZeroMemory(presentation, sizeof(TSMF_PRESENTATION)); ZeroMemory(presentation, sizeof(TSMF_PRESENTATION));
free(presentation); free(presentation);
} }
}
void tsmf_presentation_free(TSMF_PRESENTATION* presentation) void tsmf_presentation_free(TSMF_PRESENTATION* presentation)
{ {
@ -1414,8 +1420,10 @@ void tsmf_stream_end(TSMF_STREAM* stream, UINT32 message_id,
stream->eos_channel_callback = pChannelCallback; stream->eos_channel_callback = pChannelCallback;
} }
void _tsmf_stream_free(TSMF_STREAM* stream) void _tsmf_stream_free(void* obj)
{ {
TSMF_STREAM* stream = (TSMF_STREAM*)obj;
if (!stream) if (!stream)
return; return;
@ -1552,8 +1560,7 @@ BOOL tsmf_media_init(void)
if (!presentation_list) if (!presentation_list)
return FALSE; return FALSE;
ArrayList_Object(presentation_list)->fnObjectFree = (OBJECT_FREE_FN) ArrayList_Object(presentation_list)->fnObjectFree = _tsmf_presentation_free;
_tsmf_presentation_free;
} }
return TRUE; return TRUE;

View File

@ -137,6 +137,11 @@ option(WITH_DEBUG_SYMBOLS "Pack debug symbols to installer" OFF)
option(WITH_CCACHE "Use ccache support if available" ON) option(WITH_CCACHE "Use ccache support if available" ON)
option(WITH_ICU "Use ICU for unicode conversion" OFF) option(WITH_ICU "Use ICU for unicode conversion" OFF)
option(WITH_DSP_EXPERIMENTAL "Enable experimental sound encoder/decoder formats" OFF)
if (WITH_FFMPEG)
option(WITH_DSP_FFMPEG "Use FFMPEG for audio encoding/decoding" OFF)
endif(WITH_FFMPEG)
option(USE_VERSION_FROM_GIT_TAG "Extract FreeRDP version from git tag." OFF) option(USE_VERSION_FROM_GIT_TAG "Extract FreeRDP version from git tag." OFF)
if(ANDROID) if(ANDROID)

13
cmake/FindFAAC.cmake Normal file
View File

@ -0,0 +1,13 @@
find_path(FAAC_INCLUDE_DIR faac.h)
find_library(FAAC_LIBRARY faac)
find_package_handle_standard_args(FAAC DEFAULT_MSG FAAC_INCLUDE_DIR FAAC_LIBRARY)
if(FAAC_FOUND)
set(FAAC_LIBRARIES ${FAAC_LIBRARY})
set(FAAC_INCLUDE_DIRS ${FAAC_INCLUDE_DIR})
endif()
mark_as_advanced(FAAC_INCLUDE_DIR FAAC_LIBRARY)

13
cmake/FindFAAD2.cmake Normal file
View File

@ -0,0 +1,13 @@
find_path(FAAD2_INCLUDE_DIR faad.h)
find_library(FAAD2_LIBRARY faad)
find_package_handle_standard_args(FAAD2 DEFAULT_MSG FAAD2_INCLUDE_DIR FAAD2_LIBRARY)
if(FAAD2_FOUND)
set(FAAD2_LIBRARIES ${FAAD2_LIBRARY})
set(FAAD2_INCLUDE_DIRS ${FAAD2_INCLUDE_DIR})
endif()
mark_as_advanced(FAAD2_INCLUDE_DIR FAAD2_LIBRARY)

View File

@ -13,7 +13,8 @@ include(FindPkgConfig)
if (PKG_CONFIG_FOUND) if (PKG_CONFIG_FOUND)
pkg_check_modules(AVCODEC libavcodec) pkg_check_modules(AVCODEC libavcodec)
pkg_check_modules(AVUTIL libavutil) pkg_check_modules(AVUTIL libavutil)
endif() pkg_check_modules(AVRESAMPLE libavresample)
endif(PKG_CONFIG_FOUND)
# avcodec # avcodec
find_path(AVCODEC_INCLUDE_DIR libavcodec/avcodec.h PATHS ${AVCODEC_INCLUDE_DIRS}) find_path(AVCODEC_INCLUDE_DIR libavcodec/avcodec.h PATHS ${AVCODEC_INCLUDE_DIRS})
@ -23,29 +24,34 @@ find_library(AVCODEC_LIBRARY avcodec PATHS ${AVCODEC_LIBRARY_DIRS})
find_path(AVUTIL_INCLUDE_DIR libavutil/avutil.h PATHS ${AVUTIL_INCLUDE_DIRS}) find_path(AVUTIL_INCLUDE_DIR libavutil/avutil.h PATHS ${AVUTIL_INCLUDE_DIRS})
find_library(AVUTIL_LIBRARY avutil PATHS ${AVUTIL_LIBRARY_DIRS}) find_library(AVUTIL_LIBRARY avutil PATHS ${AVUTIL_LIBRARY_DIRS})
if(AVCODEC_INCLUDE_DIR AND AVCODEC_LIBRARY) # avresample
find_path(AVRESAMPLE_INCLUDE_DIR libavresample/avresample.h PATHS ${AVRESAMPLE_INCLUDE_DIRS})
find_library(AVRESAMPLE_LIBRARY avresample PATHS ${AVRESAMPLE_LIBRARY_DIRS})
if (AVCODEC_INCLUDE_DIR AND AVCODEC_LIBRARY AND AVRESAMPLE_LIBRARY)
set(AVCODEC_FOUND TRUE) set(AVCODEC_FOUND TRUE)
endif() endif(AVCODEC_INCLUDE_DIR AND AVCODEC_LIBRARY AND AVRESAMPLE_LIBRARY)
if (AVUTIL_INCLUDE_DIR AND AVUTIL_LIBRARY) if (AVUTIL_INCLUDE_DIR AND AVUTIL_LIBRARY)
set(AVUTIL_FOUND TRUE) set(AVUTIL_FOUND TRUE)
endif() endif(AVUTIL_INCLUDE_DIR AND AVUTIL_LIBRARY)
include(FindPackageHandleStandardArgs) include(FindPackageHandleStandardArgs)
FIND_PACKAGE_HANDLE_STANDARD_ARGS(FFmpeg DEFAULT_MSG AVUTIL_FOUND AVCODEC_FOUND) FIND_PACKAGE_HANDLE_STANDARD_ARGS(FFmpeg DEFAULT_MSG AVUTIL_FOUND AVCODEC_FOUND AVRESAMPLE_FOUND)
if (AVCODEC_VERSION) if (AVCODEC_VERSION)
if (${AVCODEC_VERSION} VERSION_LESS ${REQUIRED_AVCODEC_API_VERSION}) if (${AVCODEC_VERSION} VERSION_LESS ${REQUIRED_AVCODEC_API_VERSION})
message(FATAL_ERROR "libavcodec version >= ${REQUIRED_AVCODEC_VERSION} (API >= ${REQUIRED_AVCODEC_API_VERSION}) is required") message(FATAL_ERROR
"libavcodec version >= ${REQUIRED_AVCODEC_VERSION} (API >= ${REQUIRED_AVCODEC_API_VERSION}) is required")
endif() endif()
else() else(AVCODEC_VERSION)
message("Note: To build libavcodec version >= ${REQUIRED_AVCODEC_VERSION} (API >= ${REQUIRED_AVCODEC_API_VERSION}) is required") message("Note: To build libavcodec version >= ${REQUIRED_AVCODEC_VERSION} (API >= ${REQUIRED_AVCODEC_API_VERSION}) is required")
endif() endif(AVCODEC_VERSION)
if (FFMPEG_FOUND) if (FFMPEG_FOUND)
set(FFMPEG_INCLUDE_DIRS ${AVCODEC_INCLUDE_DIR} ${AVUTIL_INCLUDE_DIR}) set(FFMPEG_INCLUDE_DIRS ${AVCODEC_INCLUDE_DIR} ${AVUTIL_INCLUDE_DIR} ${AVRESAMPLE_INCLUDE_DIR})
set(FFMPEG_LIBRARIES ${AVCODEC_LIBRARY} ${AVUTIL_LIBRARY}) set(FFMPEG_LIBRARIES ${AVCODEC_LIBRARY} ${AVUTIL_LIBRARY} ${AVRESAMPLE_LIBRARY})
endif() endif(FFMPEG_FOUND)
mark_as_advanced(FFMPEG_INCLUDE_DIRS FFMPEG_LIBRARIES) mark_as_advanced(FFMPEG_INCLUDE_DIRS FFMPEG_LIBRARIES)

13
cmake/FindLAME.cmake Normal file
View File

@ -0,0 +1,13 @@
find_path(LAME_INCLUDE_DIR lame/lame.h)
find_library(LAME_LIBRARY NAMES lame mp3lame)
find_package_handle_standard_args(LAME DEFAULT_MSG LAME_INCLUDE_DIR LAME_LIBRARY)
if (LAME_FOUND)
set(LAME_LIBRARIES ${LAME_LIBRARY})
set(LAME_INCLUDE_DIRS ${LAME_INCLUDE_DIR})
endif()
mark_as_advanced(LAME_INCLUDE_DIR LAME_LIBRARY)

View File

@ -52,9 +52,14 @@
#cmakedefine WITH_IOSAUDIO #cmakedefine WITH_IOSAUDIO
#cmakedefine WITH_OPENSLES #cmakedefine WITH_OPENSLES
#cmakedefine WITH_GSM #cmakedefine WITH_GSM
#cmakedefine WITH_LAME
#cmakedefine WITH_FAAD2
#cmakedefine WITH_FAAC
#cmakedefine WITH_GFX_H264 #cmakedefine WITH_GFX_H264
#cmakedefine WITH_OPENH264 #cmakedefine WITH_OPENH264
#cmakedefine WITH_FFMPEG #cmakedefine WITH_FFMPEG
#cmakedefine WITH_DSP_EXPERIMENTAL
#cmakedefine WITH_DSP_FFMPEG
#cmakedefine WITH_X264 #cmakedefine WITH_X264
#cmakedefine WITH_MEDIA_FOUNDATION #cmakedefine WITH_MEDIA_FOUNDATION

View File

@ -24,31 +24,21 @@
#include <freerdp/channels/audin.h> #include <freerdp/channels/audin.h>
#include <freerdp/freerdp.h> #include <freerdp/freerdp.h>
#include <freerdp/codec/audio.h>
/** /**
* Subsystem Interface * Subsystem Interface
*/ */
typedef UINT (*AudinReceive) (const BYTE* data, int size, void* userData); typedef UINT (*AudinReceive)(const AUDIO_FORMAT* format,
const BYTE* data, size_t size, void* userData);
typedef struct audin_format audinFormat;
struct audin_format
{
UINT16 wFormatTag;
UINT16 nChannels;
UINT32 nSamplesPerSec;
UINT16 nBlockAlign;
UINT16 wBitsPerSample;
UINT16 cbSize;
BYTE* data;
};
typedef struct _IAudinDevice IAudinDevice; typedef struct _IAudinDevice IAudinDevice;
struct _IAudinDevice struct _IAudinDevice
{ {
UINT (*Open)(IAudinDevice* devplugin, AudinReceive receive, void* userData); UINT (*Open)(IAudinDevice* devplugin, AudinReceive receive, void* userData);
BOOL (*FormatSupported) (IAudinDevice* devplugin, audinFormat* format); BOOL (*FormatSupported)(IAudinDevice* devplugin, const AUDIO_FORMAT* format);
UINT (*SetFormat) (IAudinDevice* devplugin, audinFormat* format, UINT32 FramesPerPacket); UINT (*SetFormat)(IAudinDevice* devplugin, const AUDIO_FORMAT* format, UINT32 FramesPerPacket);
UINT (*Close)(IAudinDevice* devplugin); UINT (*Close)(IAudinDevice* devplugin);
UINT (*Free)(IAudinDevice* devplugin); UINT (*Free)(IAudinDevice* devplugin);
}; };

View File

@ -26,63 +26,31 @@
/** /**
* Subsystem Interface * Subsystem Interface
*/ */
struct _RDPSND_WAVE
{
BYTE* data;
int length;
BYTE cBlockNo;
UINT16 wFormatNo;
UINT16 wTimeStampA;
UINT16 wTimeStampB;
UINT16 wAudioLength;
UINT32 wLocalTimeA;
UINT32 wLocalTimeB;
BOOL AutoConfirm;
};
typedef struct _RDPSND_WAVE RDPSND_WAVE;
typedef struct rdpsnd_plugin rdpsndPlugin; typedef struct rdpsnd_plugin rdpsndPlugin;
typedef struct rdpsnd_device_plugin rdpsndDevicePlugin; typedef struct rdpsnd_device_plugin rdpsndDevicePlugin;
typedef BOOL (*pcFormatSupported) (rdpsndDevicePlugin* device, AUDIO_FORMAT* format); typedef BOOL (*pcFormatSupported)(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format);
typedef BOOL (*pcOpen) (rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency); typedef BOOL (*pcOpen)(rdpsndDevicePlugin* device, const AUDIO_FORMAT* format, UINT32 latency);
typedef BOOL (*pcSetFormat) (rdpsndDevicePlugin* device, AUDIO_FORMAT* format, int latency);
typedef UINT32 (*pcGetVolume)(rdpsndDevicePlugin* device); typedef UINT32 (*pcGetVolume)(rdpsndDevicePlugin* device);
typedef BOOL (*pcSetVolume)(rdpsndDevicePlugin* device, UINT32 value); typedef BOOL (*pcSetVolume)(rdpsndDevicePlugin* device, UINT32 value);
typedef void (*pcPlay) (rdpsndDevicePlugin* device, BYTE* data, int size); typedef UINT (*pcPlay)(rdpsndDevicePlugin* device, const BYTE* data, size_t size);
typedef void (*pcStart)(rdpsndDevicePlugin* device); typedef void (*pcStart)(rdpsndDevicePlugin* device);
typedef void (*pcClose)(rdpsndDevicePlugin* device); typedef void (*pcClose)(rdpsndDevicePlugin* device);
typedef void (*pcFree)(rdpsndDevicePlugin* device); typedef void (*pcFree)(rdpsndDevicePlugin* device);
typedef BOOL (*pcWaveDecode) (rdpsndDevicePlugin* device, RDPSND_WAVE* wave);
typedef void (*pcWavePlay) (rdpsndDevicePlugin* device, RDPSND_WAVE* wave);
typedef UINT (*pcWaveConfirm) (rdpsndDevicePlugin* device, RDPSND_WAVE* wave);
struct rdpsnd_device_plugin struct rdpsnd_device_plugin
{ {
rdpsndPlugin* rdpsnd; rdpsndPlugin* rdpsnd;
pcFormatSupported FormatSupported; pcFormatSupported FormatSupported;
pcOpen Open; pcOpen Open;
pcSetFormat SetFormat;
pcGetVolume GetVolume; pcGetVolume GetVolume;
pcSetVolume SetVolume; pcSetVolume SetVolume;
pcPlay Play; pcPlay Play;
pcStart Start; pcStart Start;
pcClose Close; pcClose Close;
pcFree Free; pcFree Free;
pcWaveDecode WaveDecode;
pcWavePlay WavePlay;
pcWaveConfirm WaveConfirm;
BOOL DisableConfirmThread;
}; };
#define RDPSND_DEVICE_EXPORT_FUNC_NAME "freerdp_rdpsnd_client_subsystem_entry" #define RDPSND_DEVICE_EXPORT_FUNC_NAME "freerdp_rdpsnd_client_subsystem_entry"

View File

@ -191,12 +191,12 @@ typedef struct AUDIO_FORMAT AUDIO_FORMAT;
extern "C" { extern "C" {
#endif #endif
FREERDP_API UINT32 rdpsnd_compute_audio_time_length(AUDIO_FORMAT* format, int size); FREERDP_API UINT32 rdpsnd_compute_audio_time_length(const AUDIO_FORMAT* format, size_t size);
FREERDP_API char* rdpsnd_get_audio_tag_string(UINT16 wFormatTag); FREERDP_API char* rdpsnd_get_audio_tag_string(UINT16 wFormatTag);
FREERDP_API void rdpsnd_print_audio_format(AUDIO_FORMAT* format); FREERDP_API void rdpsnd_print_audio_format(const AUDIO_FORMAT* format);
FREERDP_API void rdpsnd_print_audio_formats(AUDIO_FORMAT* formats, UINT16 count); FREERDP_API void rdpsnd_print_audio_formats(const AUDIO_FORMAT* formats, UINT16 count);
FREERDP_API void rdpsnd_free_audio_formats(AUDIO_FORMAT* formats, UINT16 count); FREERDP_API void rdpsnd_free_audio_formats(AUDIO_FORMAT* formats, UINT16 count);

View File

@ -20,63 +20,30 @@
#ifndef FREERDP_CODEC_DSP_H #ifndef FREERDP_CODEC_DSP_H
#define FREERDP_CODEC_DSP_H #define FREERDP_CODEC_DSP_H
#include <freerdp/api.h> #include <winpr/stream.h>
union _ADPCM #include <freerdp/api.h>
{ #include <freerdp/codec/audio.h>
struct
{
INT16 last_sample[2];
INT16 last_step[2];
} ima;
struct
{
BYTE predictor[2];
INT32 delta[2];
INT32 sample1[2];
INT32 sample2[2];
} ms;
};
typedef union _ADPCM ADPCM;
typedef struct _FREERDP_DSP_CONTEXT FREERDP_DSP_CONTEXT; typedef struct _FREERDP_DSP_CONTEXT FREERDP_DSP_CONTEXT;
struct _FREERDP_DSP_CONTEXT
{
BYTE* resampled_buffer;
UINT32 resampled_size;
UINT32 resampled_frames;
UINT32 resampled_maxlength;
BYTE* adpcm_buffer;
UINT32 adpcm_size;
UINT32 adpcm_maxlength;
ADPCM adpcm;
BOOL (*resample)(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int bytes_per_sample,
UINT32 schan, UINT32 srate, int sframes,
UINT32 rchan, UINT32 rrate);
BOOL (*decode_ima_adpcm)(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size);
BOOL (*encode_ima_adpcm)(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size);
BOOL (*decode_ms_adpcm)(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size);
BOOL (*encode_ms_adpcm)(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size);
};
#ifdef __cplusplus #ifdef __cplusplus
extern "C" { extern "C" {
#endif #endif
FREERDP_API FREERDP_DSP_CONTEXT* freerdp_dsp_context_new(void); FREERDP_API FREERDP_DSP_CONTEXT* freerdp_dsp_context_new(BOOL encoder);
FREERDP_API BOOL freerdp_dsp_supports_format(const AUDIO_FORMAT* format, BOOL encode);
FREERDP_API BOOL freerdp_dsp_encode(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length,
wStream* out);
FREERDP_API BOOL freerdp_dsp_decode(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length,
wStream* out);
FREERDP_API void freerdp_dsp_context_free(FREERDP_DSP_CONTEXT* context); FREERDP_API void freerdp_dsp_context_free(FREERDP_DSP_CONTEXT* context);
#define freerdp_dsp_context_reset_adpcm(_c) memset(&_c->adpcm, 0, sizeof(ADPCM)) FREERDP_API BOOL freerdp_dsp_context_reset(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* targetFormat);
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@ -155,6 +155,33 @@ if(WITH_SSE2)
endif() endif()
endif() endif()
if (WITH_DSP_FFMPEG)
set(CODEC_SRCS
${CODEC_SRCS}
codec/dsp_ffmpeg.c
codec/dsp_ffmpeg.h)
endif (WITH_DSP_FFMPEG)
if(GSM_FOUND)
freerdp_library_add(${GSM_LIBRARIES})
include_directories(${GSM_INCLUDE_DIRS})
endif()
if(LAME_FOUND)
freerdp_library_add(${LAME_LIBRARIES})
include_directories(${LAME_INCLUDE_DIRS})
endif()
if(FAAD2_FOUND)
freerdp_library_add(${FAAD2_LIBRARIES})
include_directories(${FAAD2_INCLUDE_DIRS})
endif()
if(FAAC_FOUND)
freerdp_library_add(${FAAC_LIBRARIES})
include_directories(${FAAC_INCLUDE_DIRS})
endif()
if(WITH_NEON) if(WITH_NEON)
set_source_files_properties(${CODEC_NEON_SRCS} PROPERTIES COMPILE_FLAGS "-mfpu=neon -Wno-unused-variable" ) set_source_files_properties(${CODEC_NEON_SRCS} PROPERTIES COMPILE_FLAGS "-mfpu=neon -Wno-unused-variable" )
set(CODEC_SRCS ${CODEC_SRCS} ${CODEC_NEON_SRCS}) set(CODEC_SRCS ${CODEC_SRCS} ${CODEC_NEON_SRCS})
@ -324,6 +351,7 @@ endif()
target_link_libraries(${MODULE_NAME} ${PRIVATE_KEYWORD} ${LIBFREERDP_LIBS} winpr) target_link_libraries(${MODULE_NAME} ${PRIVATE_KEYWORD} ${LIBFREERDP_LIBS} winpr)
install(TARGETS ${MODULE_NAME} DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT libraries EXPORT FreeRDPTargets) install(TARGETS ${MODULE_NAME} DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT libraries EXPORT FreeRDPTargets)
if (WITH_DEBUG_SYMBOLS AND MSVC AND BUILD_SHARED_LIBS) if (WITH_DEBUG_SYMBOLS AND MSVC AND BUILD_SHARED_LIBS)
get_target_property(OUTPUT_FILENAME ${MODULE_NAME} OUTPUT_NAME) get_target_property(OUTPUT_FILENAME ${MODULE_NAME} OUTPUT_NAME)

View File

@ -28,7 +28,7 @@
#define TAG FREERDP_TAG("codec") #define TAG FREERDP_TAG("codec")
UINT32 rdpsnd_compute_audio_time_length(AUDIO_FORMAT* format, int size) UINT32 rdpsnd_compute_audio_time_length(const AUDIO_FORMAT* format, size_t size)
{ {
UINT32 mstime; UINT32 mstime;
UINT32 wSamples; UINT32 wSamples;
@ -54,7 +54,6 @@ UINT32 rdpsnd_compute_audio_time_length(AUDIO_FORMAT* format, int size)
if ((format->cbSize == 2) && (format->data)) if ((format->cbSize == 2) && (format->data))
{ {
nSamplesPerBlock = *((UINT16*) format->data); nSamplesPerBlock = *((UINT16*) format->data);
wSamples = (size / format->nBlockAlign) * nSamplesPerBlock; wSamples = (size / format->nBlockAlign) * nSamplesPerBlock;
mstime = (((wSamples * 1000) / format->nSamplesPerSec) / format->nChannels); mstime = (((wSamples * 1000) / format->nSamplesPerSec) / format->nChannels);
} }
@ -113,7 +112,7 @@ char* rdpsnd_get_audio_tag_string(UINT16 wFormatTag)
return "WAVE_FORMAT_UNKNOWN"; return "WAVE_FORMAT_UNKNOWN";
} }
void rdpsnd_print_audio_format(AUDIO_FORMAT* format) void rdpsnd_print_audio_format(const AUDIO_FORMAT* format)
{ {
WLog_INFO(TAG, "%s:\t wFormatTag: 0x%04"PRIX16" nChannels: %"PRIu16" nSamplesPerSec: %"PRIu32" " WLog_INFO(TAG, "%s:\t wFormatTag: 0x%04"PRIX16" nChannels: %"PRIu16" nSamplesPerSec: %"PRIu32" "
"nAvgBytesPerSec: %"PRIu32" nBlockAlign: %"PRIu16" wBitsPerSample: %"PRIu16" cbSize: %"PRIu16"", "nAvgBytesPerSec: %"PRIu32" nBlockAlign: %"PRIu16" wBitsPerSample: %"PRIu16" cbSize: %"PRIu16"",
@ -122,16 +121,16 @@ void rdpsnd_print_audio_format(AUDIO_FORMAT* format)
format->nBlockAlign, format->wBitsPerSample, format->cbSize); format->nBlockAlign, format->wBitsPerSample, format->cbSize);
} }
void rdpsnd_print_audio_formats(AUDIO_FORMAT* formats, UINT16 count) void rdpsnd_print_audio_formats(const AUDIO_FORMAT* formats, UINT16 count)
{ {
int index; UINT16 index;
AUDIO_FORMAT* format; const AUDIO_FORMAT* format;
if (formats) if (formats)
{ {
WLog_INFO(TAG, "AUDIO_FORMATS (%"PRIu16") ={", count); WLog_INFO(TAG, "AUDIO_FORMATS (%"PRIu16") ={", count);
for (index = 0; index < (int) count; index++) for (index = 0; index < count; index++)
{ {
format = &formats[index]; format = &formats[index];
WLog_ERR(TAG, "\t"); WLog_ERR(TAG, "\t");
@ -144,16 +143,13 @@ void rdpsnd_print_audio_formats(AUDIO_FORMAT* formats, UINT16 count)
void rdpsnd_free_audio_formats(AUDIO_FORMAT* formats, UINT16 count) void rdpsnd_free_audio_formats(AUDIO_FORMAT* formats, UINT16 count)
{ {
int index; UINT16 index;
AUDIO_FORMAT* format;
if (formats) if (formats)
{ {
for (index = 0; index < (int) count; index++) for (index = 0; index < count; index++)
{ {
format = &formats[index]; AUDIO_FORMAT* format = &formats[index];
if (format->cbSize)
free(format->data); free(format->data);
} }

View File

@ -28,44 +28,101 @@
#include <winpr/crt.h> #include <winpr/crt.h>
#include <freerdp/types.h> #include <freerdp/types.h>
#include <freerdp/codec/dsp.h> #include <freerdp/codec/dsp.h>
#if defined(WITH_GSM)
#include <gsm/gsm.h>
#endif
#if defined(WITH_LAME)
#include <lame/lame.h>
#endif
#if defined(WITH_DSP_FFMPEG)
#include "dsp_ffmpeg.h"
#endif
#if defined(WITH_FAAD2)
#include <neaacdec.h>
#endif
#if defined(WITH_FAAC)
#include <faac.h>
#endif
#define TAG FREERDP_TAG("dsp")
union _ADPCM
{
struct
{
INT16 last_sample[2];
INT16 last_step[2];
} ima;
struct
{
BYTE predictor[2];
INT32 delta[2];
INT32 sample1[2];
INT32 sample2[2];
} ms;
};
typedef union _ADPCM ADPCM;
struct _FREERDP_DSP_CONTEXT
{
BOOL encoder;
ADPCM adpcm;
AUDIO_FORMAT format;
wStream* buffer;
wStream* resample;
#if defined(WITH_GSM)
gsm gsm;
#endif
#if defined(WITH_LAME)
lame_t lame;
hip_t hip;
#endif
#if defined(WITH_FAAD2)
NeAACDecHandle faad;
BOOL faadSetup;
#endif
#if defined(WITH_FAAC)
faacEncHandle faac;
unsigned long faacInputSamples;
unsigned long faacMaxOutputBytes;
#endif
};
/** /**
* Microsoft Multimedia Standards Update * Microsoft Multimedia Standards Update
* http://download.microsoft.com/download/9/8/6/9863C72A-A3AA-4DDB-B1BA-CA8D17EFD2D4/RIFFNEW.pdf * http://download.microsoft.com/download/9/8/6/9863C72A-A3AA-4DDB-B1BA-CA8D17EFD2D4/RIFFNEW.pdf
*/ */
static BOOL freerdp_dsp_resample(FREERDP_DSP_CONTEXT* context, static BOOL freerdp_dsp_resample(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int bytes_per_sample, const BYTE* src, size_t bytes_per_sample,
UINT32 schan, UINT32 srate, int sframes, UINT32 schan, UINT32 srate, size_t sframes,
UINT32 rchan, UINT32 rrate) UINT32 rchan, UINT32 rrate)
{ {
BYTE* dst;
BYTE* p; BYTE* p;
int rframes; int rframes;
int rsize; int rsize;
int i, j; int i, j;
int n1, n2; int n1, n2;
int sbytes, rbytes; int sbytes, rbytes;
sbytes = bytes_per_sample * schan; sbytes = bytes_per_sample * schan;
rbytes = bytes_per_sample * rchan; rbytes = bytes_per_sample * rchan;
rframes = sframes * rrate / srate; rframes = sframes * rrate / srate;
rsize = rbytes * rframes; rsize = rbytes * rframes;
if (rsize > (int) context->resampled_maxlength) if (!Stream_EnsureCapacity(context->resample, rsize + 1024))
{
BYTE *newBuffer = (BYTE*) realloc(context->resampled_buffer, rsize + 1024);
if (!newBuffer)
return FALSE; return FALSE;
context->resampled_maxlength = rsize + 1024; p = Stream_Buffer(context->resample);
context->resampled_buffer = newBuffer;
}
dst = context->resampled_buffer;
p = dst;
for (i = 0; i < rframes; i++) for (i = 0; i < rframes; i++)
{ {
@ -85,8 +142,8 @@ static BOOL freerdp_dsp_resample(FREERDP_DSP_CONTEXT* context,
} }
} }
context->resampled_frames = rframes; Stream_SetPointer(context->resample, p);
context->resampled_size = rsize; Stream_SealLength(context->resample);
return TRUE; return TRUE;
} }
@ -121,17 +178,18 @@ static UINT16 dsp_decode_ima_adpcm_sample(ADPCM* adpcm,
{ {
INT32 ss; INT32 ss;
INT32 d; INT32 d;
ss = ima_step_size_table[adpcm->ima.last_step[channel]]; ss = ima_step_size_table[adpcm->ima.last_step[channel]];
d = (ss >> 3); d = (ss >> 3);
if (sample & 1) if (sample & 1)
d += (ss >> 2); d += (ss >> 2);
if (sample & 2) if (sample & 2)
d += (ss >> 1); d += (ss >> 1);
if (sample & 4) if (sample & 4)
d += ss; d += ss;
if (sample & 8) if (sample & 8)
d = -d; d = -d;
@ -143,7 +201,6 @@ static UINT16 dsp_decode_ima_adpcm_sample(ADPCM* adpcm,
d = 32767; d = 32767;
adpcm->ima.last_sample[channel] = (INT16) d; adpcm->ima.last_sample[channel] = (INT16) d;
adpcm->ima.last_step[channel] += ima_step_index_table[sample]; adpcm->ima.last_step[channel] += ima_step_index_table[sample];
if (adpcm->ima.last_step[channel] < 0) if (adpcm->ima.last_step[channel] < 0)
@ -155,28 +212,21 @@ static UINT16 dsp_decode_ima_adpcm_sample(ADPCM* adpcm,
} }
static BOOL freerdp_dsp_decode_ima_adpcm(FREERDP_DSP_CONTEXT* context, static BOOL freerdp_dsp_decode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size) const BYTE* src, size_t size, wStream* out)
{ {
BYTE* dst; BYTE* dst;
BYTE sample; BYTE sample;
UINT16 decoded; UINT16 decoded;
UINT32 out_size; UINT32 out_size = size * 4;
int channel; UINT32 channel;
const UINT32 block_size = context->format.nBlockAlign;
const UINT32 channels = context->format.nChannels;
int i; int i;
out_size = size * 4; if (!Stream_EnsureCapacity(out, out_size))
if (out_size > context->adpcm_maxlength)
{
BYTE *newBuffer = realloc(context->adpcm_buffer, out_size + 1024);
if (!newBuffer)
return FALSE; return FALSE;
context->adpcm_maxlength = out_size + 1024; dst = Stream_Buffer(out);
context->adpcm_buffer = newBuffer;
}
dst = context->adpcm_buffer;
while (size > 0) while (size > 0)
{ {
@ -204,12 +254,10 @@ static BOOL freerdp_dsp_decode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
{ {
channel = (i < 4 ? 0 : 1); channel = (i < 4 ? 0 : 1);
sample = ((*src) & 0x0f); sample = ((*src) & 0x0f);
decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, channel, sample); decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, channel, sample);
dst[((i & 3) << 3) + (channel << 1)] = (decoded & 0xFF); dst[((i & 3) << 3) + (channel << 1)] = (decoded & 0xFF);
dst[((i & 3) << 3) + (channel << 1) + 1] = (decoded >> 8); dst[((i & 3) << 3) + (channel << 1) + 1] = (decoded >> 8);
sample = ((*src) >> 4); sample = ((*src) >> 4);
decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, channel, sample); decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, channel, sample);
dst[((i & 3) << 3) + (channel << 1) + 4] = (decoded & 0xFF); dst[((i & 3) << 3) + (channel << 1) + 4] = (decoded & 0xFF);
dst[((i & 3) << 3) + (channel << 1) + 5] = (decoded >> 8); dst[((i & 3) << 3) + (channel << 1) + 5] = (decoded >> 8);
@ -222,12 +270,10 @@ static BOOL freerdp_dsp_decode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
else else
{ {
sample = ((*src) & 0x0f); sample = ((*src) & 0x0f);
decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, 0, sample); decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, 0, sample);
*dst++ = (decoded & 0xFF); *dst++ = (decoded & 0xFF);
*dst++ = (decoded >> 8); *dst++ = (decoded >> 8);
sample = ((*src) >> 4); sample = ((*src) >> 4);
decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, 0, sample); decoded = dsp_decode_ima_adpcm_sample(&context->adpcm, 0, sample);
*dst++ = (decoded & 0xFF); *dst++ = (decoded & 0xFF);
*dst++ = (decoded >> 8); *dst++ = (decoded >> 8);
@ -236,10 +282,227 @@ static BOOL freerdp_dsp_decode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
} }
} }
context->adpcm_size = dst - context->adpcm_buffer; Stream_SetPointer(out, dst);
return TRUE; return TRUE;
} }
#if defined(WITH_GSM)
static BOOL freerdp_dsp_decode_gsm610(FREERDP_DSP_CONTEXT* context,
const BYTE* src, size_t size, wStream* out)
{
size_t offset = 0;
while (offset < size)
{
int rc;
gsm_signal gsmBlockBuffer[160] = { 0 };
rc = gsm_decode(context->gsm, (gsm_byte*) &src[offset], gsmBlockBuffer);
if (rc < 0)
return FALSE;
if ((offset % 65) == 0)
offset += 33;
else
offset += 32;
if (!Stream_EnsureRemainingCapacity(out, sizeof(gsmBlockBuffer)))
return FALSE;
Stream_Write(out, (void*) gsmBlockBuffer, sizeof(gsmBlockBuffer));
}
return TRUE;
}
static BOOL freerdp_dsp_encode_gsm610(FREERDP_DSP_CONTEXT* context,
const BYTE* src, size_t size, wStream* out)
{
size_t offset = 0;
while (offset < size)
{
gsm_signal* signal = (gsm_signal*)&src[offset];
if (!Stream_EnsureRemainingCapacity(out, sizeof(gsm_frame)))
return FALSE;
gsm_encode(context->gsm, signal, Stream_Pointer(out));
if ((offset % 65) == 0)
Stream_Seek(out, 33);
else
Stream_Seek(out, 32);
offset += 160;
}
return TRUE;
}
#endif
#if defined(WITH_LAME)
static BOOL freerdp_dsp_decode_mp3(FREERDP_DSP_CONTEXT* context,
const BYTE* src, size_t size, wStream* out)
{
int rc, x;
short* pcm_l;
short* pcm_r;
size_t buffer_size;
if (!context || !src || !out)
return FALSE;
buffer_size = 2 * context->format.nChannels * context->format.nSamplesPerSec;
if (!Stream_EnsureCapacity(context->buffer, 2 * buffer_size))
return FALSE;
pcm_l = (short*)Stream_Buffer(context->buffer);
pcm_r = (short*)Stream_Buffer(context->buffer) + buffer_size;
rc = hip_decode(context->hip, (unsigned char*)/* API is not modifying content */src,
size, pcm_l, pcm_r);
if (rc <= 0)
return FALSE;
if (!Stream_EnsureRemainingCapacity(out, rc * context->format.nChannels * 2))
return FALSE;
for (x = 0; x < rc; x++)
{
Stream_Write_UINT16(out, pcm_l[x]);
Stream_Write_UINT16(out, pcm_r[x]);
}
return TRUE;
}
static BOOL freerdp_dsp_encode_mp3(FREERDP_DSP_CONTEXT* context,
const BYTE* src, size_t size, wStream* out)
{
size_t samples_per_channel;
int rc;
if (!context || !src || !out)
return FALSE;
samples_per_channel = size / context->format.nChannels / context->format.wBitsPerSample / 8;
/* Ensure worst case buffer size for mp3 stream taken from LAME header */
if (!Stream_EnsureRemainingCapacity(out, 1.25 * samples_per_channel + 7200))
return FALSE;
samples_per_channel = size / 2 /* size of a sample */ / context->format.nChannels;
rc = lame_encode_buffer_interleaved(context->lame, (short*)src, samples_per_channel,
Stream_Pointer(out), Stream_GetRemainingCapacity(out));
if (rc < 0)
return FALSE;
Stream_Seek(out, rc);
return TRUE;
}
#endif
#if defined(WITH_FAAC)
static BOOL freerdp_dsp_encode_faac(FREERDP_DSP_CONTEXT* context,
const BYTE* src, size_t size, wStream* out)
{
int16_t* inSamples = (int16_t*)src;
int32_t* outSamples;
unsigned int nrSamples, x;
int rc;
if (!context || !src || !out)
return FALSE;
nrSamples = size / context->format.nChannels / context->format.wBitsPerSample / 8;
if (!Stream_EnsureCapacity(context->buffer,
context->faacInputSamples * sizeof(int32_t) * context->format.nChannels))
return FALSE;
if (!Stream_EnsureRemainingCapacity(out, context->faacMaxOutputBytes))
return FALSE;
outSamples = Stream_Buffer(context->buffer);
for (x = 0; x < nrSamples * context->format.nChannels; x++)
outSamples[x] = inSamples[x];
rc = faacEncEncode(context->faac, outSamples, nrSamples * context->format.nChannels,
Stream_Pointer(out), Stream_GetRemainingCapacity(out));
if (rc < 0)
return FALSE;
else if (rc > 0)
Stream_Seek(out, rc);
return TRUE;
}
#endif
#if defined(WITH_FAAD2)
static BOOL freerdp_dsp_decode_faad(FREERDP_DSP_CONTEXT* context,
const BYTE* src, size_t size, wStream* out)
{
NeAACDecFrameInfo info;
void* output;
size_t offset = 0;
if (!context || !src || !out)
return FALSE;
if (!context->faadSetup)
{
unsigned long samplerate;
unsigned char channels;
char err = NeAACDecInit(context->faad, /* API is not modifying content */(unsigned char*)src,
size, &samplerate, &channels);
if (err != 0)
return FALSE;
if (channels != context->format.nChannels)
return FALSE;
if (samplerate != context->format.nSamplesPerSec)
return FALSE;
context->faadSetup = TRUE;
}
while (offset < size)
{
size_t outSize;
void* sample_buffer;
outSize = context->format.nSamplesPerSec * context->format.nChannels *
context->format.wBitsPerSample / 8;
if (!Stream_EnsureRemainingCapacity(out, outSize))
return FALSE;
sample_buffer = Stream_Pointer(out);
output = NeAACDecDecode2(context->faad, &info, (unsigned char*)&src[offset], size - offset,
&sample_buffer, Stream_GetRemainingCapacity(out));
if (info.error != 0)
return FALSE;
offset += info.bytesconsumed;
if (info.samples == 0)
continue;
Stream_Seek(out, info.samples * context->format.wBitsPerSample / 8);
}
return TRUE;
}
#endif
/** /**
* 0 1 2 3 * 0 1 2 3
* 2 0 6 4 10 8 14 12 <left> * 2 0 6 4 10 8 14 12 <left>
@ -278,7 +541,6 @@ static BYTE dsp_encode_ima_adpcm_sample(ADPCM* adpcm, int channel, INT16 sample)
INT32 ss; INT32 ss;
BYTE enc; BYTE enc;
INT32 diff; INT32 diff;
ss = ima_step_size_table[adpcm->ima.last_step[channel]]; ss = ima_step_size_table[adpcm->ima.last_step[channel]];
d = e = sample - adpcm->ima.last_sample[channel]; d = e = sample - adpcm->ima.last_sample[channel];
diff = ss >> 3; diff = ss >> 3;
@ -325,7 +587,6 @@ static BYTE dsp_encode_ima_adpcm_sample(ADPCM* adpcm, int channel, INT16 sample)
diff = 32767; diff = 32767;
adpcm->ima.last_sample[channel] = (INT16) diff; adpcm->ima.last_sample[channel] = (INT16) diff;
adpcm->ima.last_step[channel] += ima_step_index_table[enc]; adpcm->ima.last_step[channel] += ima_step_index_table[enc];
if (adpcm->ima.last_step[channel] < 0) if (adpcm->ima.last_step[channel] < 0)
@ -337,38 +598,30 @@ static BYTE dsp_encode_ima_adpcm_sample(ADPCM* adpcm, int channel, INT16 sample)
} }
static BOOL freerdp_dsp_encode_ima_adpcm(FREERDP_DSP_CONTEXT* context, static BOOL freerdp_dsp_encode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size) const BYTE* src, size_t size, wStream* out)
{ {
int i; int i;
BYTE* dst; BYTE* dst;
INT16 sample; INT16 sample;
BYTE encoded; BYTE encoded;
UINT32 out_size; UINT32 out_size;
out_size = size / 2; out_size = size / 2;
if (out_size > context->adpcm_maxlength) if (!Stream_EnsureRemainingCapacity(out, out_size))
{
BYTE *newBuffer = realloc(context->adpcm_buffer, out_size + 1024);
if (!newBuffer)
return FALSE; return FALSE;
context->adpcm_maxlength = out_size + 1024; dst = Stream_Buffer(out);
context->adpcm_buffer = newBuffer;
}
dst = context->adpcm_buffer;
while (size > 0) while (size > 0)
{ {
if ((dst - context->adpcm_buffer) % block_size == 0) if ((dst - Stream_Buffer(out)) % context->format.nBlockAlign == 0)
{ {
*dst++ = context->adpcm.ima.last_sample[0] & 0xFF; *dst++ = context->adpcm.ima.last_sample[0] & 0xFF;
*dst++ = (context->adpcm.ima.last_sample[0] >> 8) & 0xFF; *dst++ = (context->adpcm.ima.last_sample[0] >> 8) & 0xFF;
*dst++ = (BYTE) context->adpcm.ima.last_step[0]; *dst++ = (BYTE) context->adpcm.ima.last_step[0];
*dst++ = 0; *dst++ = 0;
if (channels > 1) if (context->format.nChannels > 1)
{ {
*dst++ = context->adpcm.ima.last_sample[1] & 0xFF; *dst++ = context->adpcm.ima.last_sample[1] & 0xFF;
*dst++ = (context->adpcm.ima.last_sample[1] >> 8) & 0xFF; *dst++ = (context->adpcm.ima.last_sample[1] >> 8) & 0xFF;
@ -377,7 +630,7 @@ static BOOL freerdp_dsp_encode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
} }
} }
if (channels > 1) if (context->format.nChannels > 1)
{ {
ZeroMemory(dst, 8); ZeroMemory(dst, 8);
@ -405,7 +658,7 @@ static BOOL freerdp_dsp_encode_ima_adpcm(FREERDP_DSP_CONTEXT* context,
} }
} }
context->adpcm_size = dst - context->adpcm_buffer; Stream_SetPointer(out, dst);
return TRUE; return TRUE;
} }
@ -435,12 +688,9 @@ static INLINE INT16 freerdp_dsp_decode_ms_adpcm_sample(ADPCM* adpcm, BYTE sample
{ {
INT8 nibble; INT8 nibble;
INT32 presample; INT32 presample;
nibble = (sample & 0x08 ? (INT8) sample - 16 : sample); nibble = (sample & 0x08 ? (INT8) sample - 16 : sample);
presample = ((adpcm->ms.sample1[channel] * ms_adpcm_coeffs1[adpcm->ms.predictor[channel]]) + presample = ((adpcm->ms.sample1[channel] * ms_adpcm_coeffs1[adpcm->ms.predictor[channel]]) +
(adpcm->ms.sample2[channel] * ms_adpcm_coeffs2[adpcm->ms.predictor[channel]])) / 256; (adpcm->ms.sample2[channel] * ms_adpcm_coeffs2[adpcm->ms.predictor[channel]])) / 256;
presample += nibble * adpcm->ms.delta[channel]; presample += nibble * adpcm->ms.delta[channel];
if (presample > 32767) if (presample > 32767)
@ -459,25 +709,18 @@ static INLINE INT16 freerdp_dsp_decode_ms_adpcm_sample(ADPCM* adpcm, BYTE sample
} }
static BOOL freerdp_dsp_decode_ms_adpcm(FREERDP_DSP_CONTEXT* context, static BOOL freerdp_dsp_decode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
const BYTE* src, int size, int channels, int block_size) const BYTE* src, size_t size, wStream* out)
{ {
BYTE* dst; BYTE* dst;
BYTE sample; BYTE sample;
UINT32 out_size; const UINT32 out_size = size * 4;
const UINT32 channels = context->format.nChannels;
const UINT32 block_size = context->format.nBlockAlign;
out_size = size * 4; if (!Stream_EnsureCapacity(out, out_size))
if (out_size > context->adpcm_maxlength)
{
BYTE *newBuffer = realloc(context->adpcm_buffer, out_size + 1024);
if (!newBuffer)
return FALSE; return FALSE;
context->adpcm_maxlength = out_size + 1024; dst = Stream_Buffer(out);
context->adpcm_buffer = newBuffer;
}
dst = context->adpcm_buffer;
while (size > 0) while (size > 0)
{ {
@ -500,7 +743,6 @@ static BOOL freerdp_dsp_decode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
context->adpcm.ms.sample2[1] = *((INT16*) src); context->adpcm.ms.sample2[1] = *((INT16*) src);
src += 2; src += 2;
size -= 14; size -= 14;
*((INT16*) dst) = context->adpcm.ms.sample2[0]; *((INT16*) dst) = context->adpcm.ms.sample2[0];
dst += 2; dst += 2;
*((INT16*) dst) = context->adpcm.ms.sample2[1]; *((INT16*) dst) = context->adpcm.ms.sample2[1];
@ -520,7 +762,6 @@ static BOOL freerdp_dsp_decode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
context->adpcm.ms.sample2[0] = *((INT16*) src); context->adpcm.ms.sample2[0] = *((INT16*) src);
src += 2; src += 2;
size -= 7; size -= 7;
*((INT16*) dst) = context->adpcm.ms.sample2[0]; *((INT16*) dst) = context->adpcm.ms.sample2[0];
dst += 2; dst += 2;
*((INT16*) dst) = context->adpcm.ms.sample1[0]; *((INT16*) dst) = context->adpcm.ms.sample1[0];
@ -536,7 +777,6 @@ static BOOL freerdp_dsp_decode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
dst += 2; dst += 2;
*((INT16*) dst) = freerdp_dsp_decode_ms_adpcm_sample(&context->adpcm, sample & 0x0F, 1); *((INT16*) dst) = freerdp_dsp_decode_ms_adpcm_sample(&context->adpcm, sample & 0x0F, 1);
dst += 2; dst += 2;
sample = *src++; sample = *src++;
size--; size--;
*((INT16*) dst) = freerdp_dsp_decode_ms_adpcm_sample(&context->adpcm, sample >> 4, 0); *((INT16*) dst) = freerdp_dsp_decode_ms_adpcm_sample(&context->adpcm, sample >> 4, 0);
@ -555,7 +795,7 @@ static BOOL freerdp_dsp_decode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
} }
} }
context->adpcm_size = dst - context->adpcm_buffer; Stream_SetPointer(out, dst);
return TRUE; return TRUE;
} }
@ -563,7 +803,6 @@ static BYTE freerdp_dsp_encode_ms_adpcm_sample(ADPCM* adpcm, INT32 sample, int c
{ {
INT32 presample; INT32 presample;
INT32 errordelta; INT32 errordelta;
presample = ((adpcm->ms.sample1[channel] * ms_adpcm_coeffs1[adpcm->ms.predictor[channel]]) + presample = ((adpcm->ms.sample1[channel] * ms_adpcm_coeffs1[adpcm->ms.predictor[channel]]) +
(adpcm->ms.sample2[channel] * ms_adpcm_coeffs2[adpcm->ms.predictor[channel]])) / 256; (adpcm->ms.sample2[channel] * ms_adpcm_coeffs2[adpcm->ms.predictor[channel]])) / 256;
errordelta = (sample - presample) / adpcm->ms.delta[channel]; errordelta = (sample - presample) / adpcm->ms.delta[channel];
@ -585,7 +824,8 @@ static BYTE freerdp_dsp_encode_ms_adpcm_sample(ADPCM* adpcm, INT32 sample, int c
adpcm->ms.sample2[channel] = adpcm->ms.sample1[channel]; adpcm->ms.sample2[channel] = adpcm->ms.sample1[channel];
adpcm->ms.sample1[channel] = presample; adpcm->ms.sample1[channel] = presample;
adpcm->ms.delta[channel] = adpcm->ms.delta[channel] * ms_adpcm_adaptation_table[(((BYTE) errordelta) & 0x0F)] / 256; adpcm->ms.delta[channel] = adpcm->ms.delta[channel] * ms_adpcm_adaptation_table[(((
BYTE) errordelta) & 0x0F)] / 256;
if (adpcm->ms.delta[channel] < 16) if (adpcm->ms.delta[channel] < 16)
adpcm->ms.delta[channel] = 16; adpcm->ms.delta[channel] = 16;
@ -593,26 +833,18 @@ static BYTE freerdp_dsp_encode_ms_adpcm_sample(ADPCM* adpcm, INT32 sample, int c
return ((BYTE) errordelta) & 0x0F; return ((BYTE) errordelta) & 0x0F;
} }
static BOOL freerdp_dsp_encode_ms_adpcm(FREERDP_DSP_CONTEXT* context, static BOOL freerdp_dsp_encode_ms_adpcm(FREERDP_DSP_CONTEXT* context, const BYTE* src, size_t size,
const BYTE* src, int size, int channels, int block_size) wStream* out)
{ {
BYTE* dst; BYTE* dst;
INT32 sample; INT32 sample;
UINT32 out_size; UINT32 out_size;
out_size = size / 2; out_size = size / 2;
if (out_size > context->adpcm_maxlength) if (!Stream_EnsureRemainingCapacity(out, out_size))
{
BYTE *newBuffer = realloc(context->adpcm_buffer, out_size + 1024);
if (!newBuffer)
return FALSE; return FALSE;
context->adpcm_maxlength = out_size + 1024; dst = Stream_Buffer(out);
context->adpcm_buffer = newBuffer;
}
dst = context->adpcm_buffer;
if (context->adpcm.ms.delta[0] < 16) if (context->adpcm.ms.delta[0] < 16)
context->adpcm.ms.delta[0] = 16; context->adpcm.ms.delta[0] = 16;
@ -622,9 +854,9 @@ static BOOL freerdp_dsp_encode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
while (size > 0) while (size > 0)
{ {
if ((dst - context->adpcm_buffer) % block_size == 0) if ((dst - Stream_Buffer(out)) % context->format.nBlockAlign == 0)
{ {
if (channels > 1) if (context->format.nChannels > 1)
{ {
*dst++ = context->adpcm.ms.predictor[0]; *dst++ = context->adpcm.ms.predictor[0];
*dst++ = context->adpcm.ms.predictor[1]; *dst++ = context->adpcm.ms.predictor[1];
@ -664,38 +896,312 @@ static BOOL freerdp_dsp_encode_ms_adpcm(FREERDP_DSP_CONTEXT* context,
*dst = freerdp_dsp_encode_ms_adpcm_sample(&context->adpcm, sample, 0) << 4; *dst = freerdp_dsp_encode_ms_adpcm_sample(&context->adpcm, sample, 0) << 4;
sample = *((INT16*) src); sample = *((INT16*) src);
src += 2; src += 2;
*dst += freerdp_dsp_encode_ms_adpcm_sample(&context->adpcm, sample, channels > 1 ? 1 : 0); *dst += freerdp_dsp_encode_ms_adpcm_sample(&context->adpcm, sample,
context->format.nChannels > 1 ? 1 : 0);
dst++; dst++;
size -= 4; size -= 4;
} }
context->adpcm_size = dst - context->adpcm_buffer; Stream_SetPointer(out, dst);
return TRUE; return TRUE;
} }
FREERDP_DSP_CONTEXT* freerdp_dsp_context_new(void) FREERDP_DSP_CONTEXT* freerdp_dsp_context_new(BOOL encoder)
{ {
FREERDP_DSP_CONTEXT* context; #if defined(WITH_DSP_FFMPEG)
return freerdp_dsp_ffmpeg_context_new(encoder);
#else
FREERDP_DSP_CONTEXT* context = calloc(1, sizeof(FREERDP_DSP_CONTEXT));
context = (FREERDP_DSP_CONTEXT*) calloc(1, sizeof(FREERDP_DSP_CONTEXT));
if (!context) if (!context)
return NULL; return NULL;
context->resample = freerdp_dsp_resample; context->buffer = Stream_New(NULL, 4096);
context->decode_ima_adpcm = freerdp_dsp_decode_ima_adpcm;
context->encode_ima_adpcm = freerdp_dsp_encode_ima_adpcm;
context->decode_ms_adpcm = freerdp_dsp_decode_ms_adpcm;
context->encode_ms_adpcm = freerdp_dsp_encode_ms_adpcm;
if (!context->buffer)
goto fail;
context->resample = Stream_New(NULL, 4096);
if (!context->resample)
goto fail;
context->encoder = encoder;
#if defined(WITH_GSM)
context->gsm = gsm_create();
if (!context->gsm)
goto fail;
{
int rc;
int val = 1;
rc = gsm_option(context->gsm, GSM_OPT_WAV49, &val);
if (rc < 0)
goto fail;
}
#endif
#if defined(WITH_LAME)
if (encoder)
{
context->lame = lame_init();
if (!context->lame)
goto fail;
}
else
{
context->hip = hip_decode_init();
if (!context->hip)
goto fail;
}
#endif
#if defined(WITH_FAAD2)
if (!encoder)
{
context->faad = NeAACDecOpen();
if (!context->faad)
goto fail;
}
#endif
return context; return context;
fail:
freerdp_dsp_context_free(context);
return NULL;
#endif
} }
void freerdp_dsp_context_free(FREERDP_DSP_CONTEXT* context) void freerdp_dsp_context_free(FREERDP_DSP_CONTEXT* context)
{ {
#if defined(WITH_DSP_FFMPEG)
freerdp_dsp_ffmpeg_context_free(context);
#else
if (context) if (context)
{ {
free(context->resampled_buffer); Stream_Free(context->buffer, TRUE);
free(context->adpcm_buffer); Stream_Free(context->resample, TRUE);
#if defined(WITH_GSM)
gsm_destroy(context->gsm);
#endif
#if defined(WITH_LAME)
if (context->encoder)
lame_close(context->lame);
else
hip_decode_exit(context->hip);
#endif
#if defined(WITH_FAAD2)
if (!context->encoder)
NeAACDecClose(context->faad);
#endif
#if defined (WITH_FAAC)
if (context->faac)
faacEncClose(context->faac);
#endif
free(context); free(context);
} }
#endif
}
BOOL freerdp_dsp_encode(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length, wStream* out)
{
#if defined(WITH_DSP_FFMPEG)
return freerdp_dsp_ffmpeg_encode(context, srcFormat, data, length, out);
#else
if (!context || !context->encoder || !srcFormat || !data || !out)
return FALSE;
// TODO: Resample
switch (context->format.wFormatTag)
{
case WAVE_FORMAT_PCM:
if (!Stream_EnsureRemainingCapacity(out, length))
return FALSE;
Stream_Write(out, data, length);
return TRUE;
case WAVE_FORMAT_ADPCM:
return freerdp_dsp_encode_ms_adpcm(context, data, length, out);
case WAVE_FORMAT_DVI_ADPCM:
return freerdp_dsp_encode_ima_adpcm(context, data, length, out);
#if defined(WITH_GSM)
case WAVE_FORMAT_GSM610:
return freerdp_dsp_encode_gsm610(context, data, length, out);
#endif
#if defined(WITH_LAME)
case WAVE_FORMAT_MPEGLAYER3:
return freerdp_dsp_encode_mp3(context, data, length, out);
#endif
#if defined(WITH_FAAC)
case WAVE_FORMAT_AAC_MS:
return freerdp_dsp_encode_faac(context, data, length, out);
#endif
default:
return FALSE;
}
return FALSE;
#endif
}
BOOL freerdp_dsp_decode(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length, wStream* out)
{
#if defined(WITH_DSP_FFMPEG)
return freerdp_dsp_ffmpeg_decode(context, srcFormat, data, length, out);
#else
if (!context || context->encoder || !srcFormat || !data || !out)
return FALSE;
switch (context->format.wFormatTag)
{
case WAVE_FORMAT_PCM:
if (!Stream_EnsureRemainingCapacity(out, length))
return FALSE;
Stream_Write(out, data, length);
return TRUE;
case WAVE_FORMAT_ADPCM:
return freerdp_dsp_decode_ms_adpcm(context, data, length, out);
case WAVE_FORMAT_DVI_ADPCM:
return freerdp_dsp_decode_ima_adpcm(context, data, length, out);
#if defined(WITH_GSM)
case WAVE_FORMAT_GSM610:
return freerdp_dsp_decode_gsm610(context, data, length, out);
#endif
#if defined(WITH_LAME)
case WAVE_FORMAT_MPEGLAYER3:
return freerdp_dsp_decode_mp3(context, data, length, out);
#endif
#if defined(WITH_FAAD2)
case WAVE_FORMAT_AAC_MS:
return freerdp_dsp_decode_faad(context, data, length, out);
#endif
default:
return FALSE;
}
return FALSE;
#endif
}
BOOL freerdp_dsp_supports_format(const AUDIO_FORMAT* format, BOOL encode)
{
#if defined(WITH_DSP_FFMPEG)
return freerdp_dsp_ffmpeg_supports_format(format, encode);
#else
switch (format->wFormatTag)
{
case WAVE_FORMAT_PCM:
case WAVE_FORMAT_ADPCM:
case WAVE_FORMAT_DVI_ADPCM:
return TRUE;
#if defined(WITH_GSM)
case WAVE_FORMAT_GSM610:
#if defined(WITH_DSP_EXPERIMENTAL)
return TRUE;
#else
return !encode;
#endif
#endif
#if defined(WITH_LAME)
case WAVE_FORMAT_MPEGLAYER3:
#if defined(WITH_DSP_EXPERIMENTAL)
return TRUE;
#else
return !encode;
#endif
#endif
case WAVE_FORMAT_AAC_MS:
#if defined(WITH_FAAD2)
if (!encode)
return TRUE;
#endif
#if defined(WITH_FAAC) && defined(WITH_DSP_EXPERIMENTAL)
if (encode)
return TRUE;
#endif
default:
return FALSE;
}
return FALSE;
#endif
}
BOOL freerdp_dsp_context_reset(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* targetFormat)
{
#if defined(WITH_DSP_FFMPEG)
return freerdp_dsp_ffmpeg_context_reset(context, targetFormat);
#else
if (!context || !targetFormat)
return FALSE;
context->format = *targetFormat;
#if defined(WITH_FAAD2)
context->faadSetup = FALSE;
#endif
#if defined(WITH_FAAC)
if (context->encoder)
{
faacEncConfigurationPtr cfg;
if (context->faac)
faacEncClose(context->faac);
context->faac = faacEncOpen(targetFormat->nSamplesPerSec, targetFormat->nChannels,
&context->faacInputSamples, &context->faacMaxOutputBytes);
if (!context->faac)
return FALSE;
cfg = faacEncGetCurrentConfiguration(context->faac);
cfg->bitRate = 10000;
faacEncSetConfiguration(context->faac, cfg);
}
#endif
return TRUE;
#endif
} }

34
libfreerdp/codec/dsp.h Normal file
View File

@ -0,0 +1,34 @@
/**
* FreeRDP: A Remote Desktop Protocol Implementation
* Digital Sound Processing - backend
*
* Copyright 2018 Armin Novak <armin.novak@thincast.com>
* Copyright 2018 Thincast Technologies GmbH
*
* 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.
*/
#ifndef FREERDP_LIB_CODEC_DSP_H
#define FREERDP_LIB_CODEC_DSP_H
#include <freerdp/api.h>
#include <freerdp/codec/audio.h>
#include <freerdp/codec/dsp.h>
struct _FREERDP_DSP_COMMON_CONTEXT
{
wStream* buffer;
wStream* resample;
};
#endif /* FREERDP_LIB_CODEC_DSP_H */

View File

@ -0,0 +1,590 @@
/**
* FreeRDP: A Remote Desktop Protocol Implementation
* Digital Sound Processing - FFMPEG backend
*
* Copyright 2018 Armin Novak <armin.novak@thincast.com>
* Copyright 2018 Thincast Technologies GmbH
*
* 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 <freerdp/log.h>
#include <libavcodec/avcodec.h>
#include <libavutil/avutil.h>
#include <libavutil/opt.h>
#include <libavresample/avresample.h>
#include "dsp.h"
#include "dsp_ffmpeg.h"
#define TAG FREERDP_TAG("dsp.ffmpeg")
struct _FREERDP_DSP_CONTEXT
{
AUDIO_FORMAT format;
BOOL isOpen;
BOOL encoder;
UINT32 bufferedSamples;
enum AVCodecID id;
AVCodec* codec;
AVCodecContext* context;
AVFrame* frame;
AVFrame* resampled;
AVFrame* buffered;
AVPacket* packet;
AVAudioResampleContext* rcontext;
};
static BOOL ffmpeg_codec_is_filtered(enum AVCodecID id, BOOL encoder)
{
if (!encoder)
return FALSE;
switch (id)
{
#if !defined(WITH_DSP_EXPERIMENTAL)
case AV_CODEC_ID_MP3:
case AV_CODEC_ID_GSM_MS:
case AV_CODEC_ID_AAC:
return TRUE;
#endif
case AV_CODEC_ID_NONE:
return TRUE;
default:
return FALSE;
}
}
static enum AVCodecID ffmpeg_get_avcodec(const AUDIO_FORMAT* format)
{
const char* id;
if (!format)
return AV_CODEC_ID_NONE;
id = rdpsnd_get_audio_tag_string(format->wFormatTag);
switch (format->wFormatTag)
{
case WAVE_FORMAT_UNKNOWN:
return AV_CODEC_ID_NONE;
case WAVE_FORMAT_PCM:
switch (format->wBitsPerSample)
{
case 16:
return AV_CODEC_ID_PCM_U16LE;
case 8:
return AV_CODEC_ID_PCM_U8;
default:
return AV_CODEC_ID_NONE;
}
case WAVE_FORMAT_DVI_ADPCM:
return AV_CODEC_ID_ADPCM_MS;
case WAVE_FORMAT_ADPCM:
return AV_CODEC_ID_ADPCM_MS;
case WAVE_FORMAT_ALAW:
return AV_CODEC_ID_PCM_ALAW;
case WAVE_FORMAT_MULAW:
return AV_CODEC_ID_PCM_MULAW;
case WAVE_FORMAT_GSM610:
return AV_CODEC_ID_GSM_MS;
case WAVE_FORMAT_AAC_MS:
return AV_CODEC_ID_AAC;
default:
return AV_CODEC_ID_NONE;
}
}
static int ffmpeg_sample_format(const AUDIO_FORMAT* format)
{
switch (format->wFormatTag)
{
case WAVE_FORMAT_PCM:
switch (format->wBitsPerSample)
{
case 8:
return AV_SAMPLE_FMT_U8;
case 16:
return AV_SAMPLE_FMT_S16;
default:
return FALSE;
}
case WAVE_FORMAT_DVI_ADPCM:
case WAVE_FORMAT_ADPCM:
return AV_SAMPLE_FMT_S16P;
case WAVE_FORMAT_MPEGLAYER3:
case WAVE_FORMAT_AAC_MS:
return AV_SAMPLE_FMT_FLTP;
case WAVE_FORMAT_GSM610:
return AV_SAMPLE_FMT_S16P;
case WAVE_FORMAT_ALAW:
return AV_SAMPLE_FMT_S16;
default:
return FALSE;
}
}
static void ffmpeg_close_context(FREERDP_DSP_CONTEXT* context)
{
if (context)
{
if (context->context)
avcodec_free_context(&context->context);
if (context->frame)
av_frame_free(&context->frame);
if (context->resampled)
av_frame_free(&context->resampled);
if (context->buffered)
av_frame_free(&context->buffered);
if (context->packet)
av_packet_free(&context->packet);
if (context->rcontext)
avresample_free(&context->rcontext);
context->id = AV_CODEC_ID_NONE;
context->codec = NULL;
context->isOpen = FALSE;
context->context = NULL;
context->frame = NULL;
context->resampled = NULL;
context->packet = NULL;
context->rcontext = NULL;
}
}
static BOOL ffmpeg_open_context(FREERDP_DSP_CONTEXT* context)
{
int ret;
int layout;
const AUDIO_FORMAT* format;
if (!context || context->isOpen)
return FALSE;
format = &context->format;
if (!format)
return FALSE;
layout = av_get_default_channel_layout(format->nChannels);
context->id = ffmpeg_get_avcodec(format);
if (ffmpeg_codec_is_filtered(context->id, context->encoder))
goto fail;
if (context->encoder)
context->codec = avcodec_find_encoder(context->id);
else
context->codec = avcodec_find_decoder(context->id);
if (!context->codec)
goto fail;
context->context = avcodec_alloc_context3(context->codec);
if (!context->context)
goto fail;
switch (context->id)
{
/* We need support for multichannel and sample rates != 8000 */
case AV_CODEC_ID_GSM_MS:
context->context->strict_std_compliance = FF_COMPLIANCE_UNOFFICIAL;
break;
default:
break;
}
context->context->channels = format->nChannels;
context->context->channel_layout = layout;
context->context->sample_rate = format->nSamplesPerSec;
context->context->block_align = format->nBlockAlign;
context->context->bit_rate = format->nAvgBytesPerSec * 8;
context->context->sample_fmt = ffmpeg_sample_format(format);
context->context->time_base = av_make_q(1, context->context->sample_rate);
if ((ret = avcodec_open2(context->context, context->codec, NULL)) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error avcodec_open2 %s [%d]", err, ret);
goto fail;
}
context->packet = av_packet_alloc();
if (!context->packet)
goto fail;
context->frame = av_frame_alloc();
if (!context->frame)
goto fail;
context->resampled = av_frame_alloc();
if (!context->resampled)
goto fail;
context->buffered = av_frame_alloc();
if (!context->buffered)
goto fail;
context->rcontext = avresample_alloc_context();
if (!context->rcontext)
goto fail;
context->frame->channel_layout = layout;
context->frame->channels = format->nChannels;
context->frame->sample_rate = format->nSamplesPerSec;
context->frame->format = AV_SAMPLE_FMT_S16;
if (context->encoder)
{
context->resampled->format = context->context->sample_fmt;
context->resampled->sample_rate = context->context->sample_rate;
}
else
{
context->resampled->format = AV_SAMPLE_FMT_S16;
context->resampled->sample_rate = format->nSamplesPerSec;
}
context->resampled->channel_layout = layout;
context->resampled->channels = format->nChannels;
if (context->context->frame_size > 0)
{
context->buffered->channel_layout = context->resampled->channel_layout;
context->buffered->channels = context->resampled->channels;
context->buffered->format = context->resampled->format;
context->buffered->nb_samples = context->context->frame_size;
if ((ret = av_frame_get_buffer(context->buffered, 1)) < 0)
goto fail;
}
context->isOpen = TRUE;
return TRUE;
fail:
ffmpeg_close_context(context);
return FALSE;
}
static BOOL ffmpeg_resample_frame(AVAudioResampleContext* context,
AVFrame* in, AVFrame* out)
{
int ret;
if (!avresample_is_open(context))
{
if ((ret = avresample_config(context, out, in)) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
return FALSE;
}
if ((ret = (avresample_open(context))) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
return FALSE;
}
}
if ((ret = avresample_convert_frame(context, out, in)) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
return FALSE;
}
return TRUE;
}
static BOOL ffmpeg_encode_frame(AVCodecContext* context, AVFrame* in,
AVPacket* packet, wStream* out)
{
int ret;
/* send the packet with the compressed data to the encoder */
ret = avcodec_send_frame(context, in);
if (ret < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error submitting the packet to the encoder %s [%d]",
err, ret);
return FALSE;
}
/* read all the output frames (in general there may be any number of them */
while (ret >= 0)
{
ret = avcodec_receive_packet(context, packet);
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
return TRUE;
else if (ret < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during encoding %s [%d]", err, ret);
return FALSE;
}
if (!Stream_EnsureRemainingCapacity(out, packet->size))
return FALSE;
Stream_Write(out, packet->data, packet->size);
av_packet_unref(packet);
}
return TRUE;
}
static BOOL ffmpeg_fill_frame(AVFrame* frame, const AUDIO_FORMAT* inputFormat,
const BYTE* data, size_t size)
{
int ret, bpp;
frame->channels = inputFormat->nChannels;
frame->sample_rate = inputFormat->nSamplesPerSec;
frame->format = ffmpeg_sample_format(inputFormat);
frame->channel_layout = av_get_default_channel_layout(frame->channels);
bpp = av_get_bytes_per_sample(frame->format);
frame->nb_samples = size / inputFormat->nChannels / bpp;
if ((ret = avcodec_fill_audio_frame(frame, frame->channels,
frame->format,
data, size, 1)) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during audio frame fill %s [%d]", err, ret);
return FALSE;
}
return TRUE;
}
static BOOL ffmpeg_decode(AVCodecContext* dec_ctx, AVPacket* pkt,
AVFrame* frame,
AVAudioResampleContext* resampleContext,
AVFrame* resampled, wStream* out)
{
int ret;
/* send the packet with the compressed data to the decoder */
ret = avcodec_send_packet(dec_ctx, pkt);
if (ret < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error submitting the packet to the decoder %s [%d]",
err, ret);
return FALSE;
}
/* read all the output frames (in general there may be any number of them */
while (ret >= 0)
{
ret = avcodec_receive_frame(dec_ctx, frame);
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
return TRUE;
else if (ret < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during decoding %s [%d]", err, ret);
return FALSE;
}
if (!avresample_is_open(resampleContext))
{
if ((ret = avresample_config(resampleContext, resampled, frame)) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
return FALSE;
}
if ((ret = (avresample_open(resampleContext))) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
return FALSE;
}
}
if ((ret = avresample_convert_frame(resampleContext, resampled, frame)) < 0)
{
const char* err = av_err2str(ret);
WLog_ERR(TAG, "Error during resampling %s [%d]", err, ret);
return FALSE;
}
{
const size_t data_size = resampled->channels * resampled->nb_samples * 2;
Stream_EnsureRemainingCapacity(out, data_size);
Stream_Write(out, resampled->data[0], data_size);
}
}
return TRUE;
}
BOOL freerdp_dsp_ffmpeg_supports_format(const AUDIO_FORMAT* format, BOOL encode)
{
enum AVCodecID id = ffmpeg_get_avcodec(format);
if (ffmpeg_codec_is_filtered(id, encode))
return FALSE;
if (encode)
return avcodec_find_encoder(id) != NULL;
else
return avcodec_find_decoder(id) != NULL;
}
FREERDP_DSP_CONTEXT* freerdp_dsp_ffmpeg_context_new(BOOL encode)
{
FREERDP_DSP_CONTEXT* context;
avcodec_register_all();
context = calloc(1, sizeof(FREERDP_DSP_CONTEXT));
if (!context)
return NULL;
context->encoder = encode;
return context;
}
void freerdp_dsp_ffmpeg_context_free(FREERDP_DSP_CONTEXT* context)
{
if (context)
{
ffmpeg_close_context(context);
free(context);
}
}
BOOL freerdp_dsp_ffmpeg_context_reset(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* targetFormat)
{
if (!context || !targetFormat)
return FALSE;
ffmpeg_close_context(context);
context->format = *targetFormat;
return ffmpeg_open_context(context);
}
BOOL freerdp_dsp_ffmpeg_encode(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* format,
const BYTE* data, size_t length, wStream* out)
{
int rc;
int samples, rest;
if (!context || !format || !data || !out || !context->encoder)
return FALSE;
if (!context || !data || !out)
return FALSE;
/* Create input frame */
if (!ffmpeg_fill_frame(context->frame, format, data, length))
return FALSE;
/* Resample to desired format. */
if (!ffmpeg_resample_frame(context->rcontext,
context->frame,
context->resampled))
return FALSE;
if (context->context->frame_size <= 0)
{
return ffmpeg_encode_frame(context->context, context->resampled,
context->packet, out);
}
else
{
rest = samples = context->resampled->nb_samples;
do
{
if (samples + context->bufferedSamples > context->context->frame_size)
samples = context->context->frame_size - context->bufferedSamples;
rc = av_samples_copy(context->buffered->extended_data, context->resampled->extended_data,
context->bufferedSamples, 0, samples,
context->context->channels, context->context->sample_fmt);
rest -= samples;
context->bufferedSamples += samples;
if (context->context->frame_size <= context->bufferedSamples)
{
/* Encode in desired format. */
if (!ffmpeg_encode_frame(context->context, context->buffered,
context->packet, out))
return FALSE;
context->bufferedSamples = 0;
}
}
while (rest > 0);
return TRUE;
}
}
BOOL freerdp_dsp_ffmpeg_decode(FREERDP_DSP_CONTEXT* context, const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length, wStream* out)
{
if (!context || !srcFormat || !data || !out || context->encoder)
return FALSE;
av_init_packet(context->packet);
context->packet->data = data;
context->packet->size = length;
return ffmpeg_decode(context->context, context->packet, context->frame,
context->rcontext, context->resampled, out);
}

View File

@ -0,0 +1,48 @@
/**
* FreeRDP: A Remote Desktop Protocol Implementation
* Digital Sound Processing - FFMPEG backend
*
* Copyright 2018 Armin Novak <armin.novak@thincast.com>
* Copyright 2018 Thincast Technologies GmbH
*
* 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.
*/
#ifndef FREERDP_LIB_CODEC_DSP_FFMPEG_H
#define FREERDP_LIB_CODEC_DSP_FFMPEG_H
#include <freerdp/api.h>
#include <freerdp/codec/audio.h>
#include <freerdp/codec/dsp.h>
#include <libavcodec/version.h>
#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(57, 48, 101)
#error "DSP module requires libavcodec version >= 57.48.101. Upgrade or set WITH_DSP_FFMPEG=OFF to continue"
#endif
FREERDP_DSP_CONTEXT* freerdp_dsp_ffmpeg_context_new(BOOL encode);
BOOL freerdp_dsp_ffmpeg_supports_format(const AUDIO_FORMAT* format, BOOL encode);
BOOL freerdp_dsp_ffmpeg_encode(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length,
wStream* out);
BOOL freerdp_dsp_ffmpeg_decode(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* srcFormat,
const BYTE* data, size_t length,
wStream* out);
void freerdp_dsp_ffmpeg_context_free(FREERDP_DSP_CONTEXT* context);
BOOL freerdp_dsp_ffmpeg_context_reset(FREERDP_DSP_CONTEXT* context,
const AUDIO_FORMAT* targetFormat);
#endif /* FREERDP_LIB_CODEC_DSP_FFMPEG_H */

View File

@ -43,6 +43,7 @@ static UINT AudinServerOpening(audin_server_context* context)
{ {
AUDIO_FORMAT* agreed_format = NULL; AUDIO_FORMAT* agreed_format = NULL;
int i = 0, j = 0; int i = 0, j = 0;
for (i = 0; i < context->num_client_formats; i++) for (i = 0; i < context->num_client_formats; i++)
{ {
for (j = 0; j < context->num_server_formats; j++) for (j = 0; j < context->num_server_formats; j++)
@ -55,9 +56,9 @@ static UINT AudinServerOpening(audin_server_context* context)
break; break;
} }
} }
if (agreed_format != NULL) if (agreed_format != NULL)
break; break;
} }
if (agreed_format == NULL) if (agreed_format == NULL)
@ -94,6 +95,7 @@ static UINT AudinServerReceiveSamples(audin_server_context* context, const void*
if (subsystem->AudinServerReceiveSamples) if (subsystem->AudinServerReceiveSamples)
subsystem->AudinServerReceiveSamples(subsystem, client, buf, nframes); subsystem->AudinServerReceiveSamples(subsystem, client, buf, nframes);
return CHANNEL_RC_OK; return CHANNEL_RC_OK;
} }
@ -101,6 +103,7 @@ int shadow_client_audin_init(rdpShadowClient* client)
{ {
audin_server_context* audin; audin_server_context* audin;
audin = client->audin = audin_server_context_new(client->vcm); audin = client->audin = audin_server_context_new(client->vcm);
if (!audin) if (!audin)
{ {
return 0; return 0;
@ -117,15 +120,14 @@ int shadow_client_audin_init(rdpShadowClient* client)
{ {
/* Set default audio formats. */ /* Set default audio formats. */
audin->server_formats = default_supported_audio_formats; audin->server_formats = default_supported_audio_formats;
audin->num_server_formats = sizeof(default_supported_audio_formats) / sizeof(default_supported_audio_formats[0]); audin->num_server_formats = sizeof(default_supported_audio_formats) / sizeof(
default_supported_audio_formats[0]);
} }
audin->dst_format = audin->server_formats[0]; audin->dst_format = audin->server_formats[0];
audin->Opening = AudinServerOpening; audin->Opening = AudinServerOpening;
audin->OpenResult = AudinServerOpenResult; audin->OpenResult = AudinServerOpenResult;
audin->ReceiveSamples = AudinServerReceiveSamples; audin->ReceiveSamples = AudinServerReceiveSamples;
return 1; return 1;
} }