Merge pull request #4453 from akallabeth/sound_channel_refactor
Sound channel refactoring
This commit is contained in:
commit
456b0e8934
@ -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})
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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:
|
||||||
|
@ -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;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
|
@ -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!");
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
|
@ -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" "")
|
||||||
|
@ -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:
|
||||||
|
33
channels/rdpsnd/client/fake/CMakeLists.txt
Normal file
33
channels/rdpsnd/client/fake/CMakeLists.txt
Normal 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")
|
175
channels/rdpsnd/client/fake/rdpsnd_fake.c
Normal file
175
channels/rdpsnd/client/fake/rdpsnd_fake.c
Normal 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;
|
||||||
|
}
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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")
|
||||||
|
@ -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
@ -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 */
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
{
|
{
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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 */
|
||||||
|
@ -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;
|
||||||
|
@ -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
13
cmake/FindFAAC.cmake
Normal 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
13
cmake/FindFAAD2.cmake
Normal 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)
|
@ -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
13
cmake/FindLAME.cmake
Normal 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)
|
@ -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
|
||||||
|
|
||||||
|
@ -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);
|
||||||
};
|
};
|
||||||
|
@ -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"
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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)
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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
34
libfreerdp/codec/dsp.h
Normal 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 */
|
590
libfreerdp/codec/dsp_ffmpeg.c
Normal file
590
libfreerdp/codec/dsp_ffmpeg.c
Normal 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);
|
||||||
|
}
|
48
libfreerdp/codec/dsp_ffmpeg.h
Normal file
48
libfreerdp/codec/dsp_ffmpeg.h
Normal 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 */
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user