470 lines
11 KiB
C
470 lines
11 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* Audio Output Virtual Channel
|
|
*
|
|
* Copyright 2013 Armin Novak <armin.novak@gmail.com>
|
|
*
|
|
* 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 <assert.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdbool.h>
|
|
|
|
#include <winpr/crt.h>
|
|
#include <winpr/cmdline.h>
|
|
#include <winpr/sysinfo.h>
|
|
#include <winpr/collections.h>
|
|
|
|
#include <freerdp/types.h>
|
|
#include <freerdp/codec/dsp.h>
|
|
#include <freerdp/utils/debug.h>
|
|
|
|
#include "opensl_io.h"
|
|
#include "rdpsnd_main.h"
|
|
|
|
typedef struct rdpsnd_opensles_plugin rdpsndopenslesPlugin;
|
|
|
|
struct rdpsnd_opensles_plugin
|
|
{
|
|
rdpsndDevicePlugin device;
|
|
|
|
int latency;
|
|
int wformat;
|
|
int block_size;
|
|
char* device_name;
|
|
|
|
OPENSL_STREAM *stream;
|
|
|
|
UINT32 volume;
|
|
|
|
UINT32 rate;
|
|
UINT32 channels;
|
|
int format;
|
|
FREERDP_DSP_CONTEXT* dsp_context;
|
|
};
|
|
|
|
static int rdpsnd_opensles_volume_to_millibel(unsigned short level, int max)
|
|
{
|
|
const int min = SL_MILLIBEL_MIN;
|
|
const int step = max - min;
|
|
const int rc = (level * step / 0xFFFF) + min;
|
|
|
|
DEBUG_SND("level=%d, min=%d, max=%d, step=%d, result=%d",
|
|
level, min, max, step, rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static unsigned short rdpsnd_opensles_millibel_to_volume(int millibel, int max)
|
|
{
|
|
const int min = SL_MILLIBEL_MIN;
|
|
const int range = max - min;
|
|
const int rc = ((millibel - min) * 0xFFFF + range / 2 + 1) / range;
|
|
|
|
DEBUG_SND("millibel=%d, min=%d, max=%d, range=%d, result=%d",
|
|
millibel, min, max, range, rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static bool rdpsnd_opensles_check_handle(const rdpsndopenslesPlugin *hdl)
|
|
{
|
|
bool rc = true;
|
|
|
|
if (!hdl)
|
|
rc = false;
|
|
else
|
|
{
|
|
if (!hdl->dsp_context)
|
|
rc = false;
|
|
if (!hdl->stream)
|
|
rc = false;
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void rdpsnd_opensles_set_volume(rdpsndDevicePlugin* device,
|
|
UINT32 volume);
|
|
|
|
static int rdpsnd_opensles_set_params(rdpsndopenslesPlugin* opensles)
|
|
{
|
|
DEBUG_SND("opensles=%p", opensles);
|
|
if (!rdpsnd_opensles_check_handle(opensles))
|
|
return 0;
|
|
|
|
if (opensles->stream)
|
|
android_CloseAudioDevice(opensles->stream);
|
|
|
|
opensles->stream = android_OpenAudioDevice(
|
|
opensles->rate, opensles->channels, 20);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void rdpsnd_opensles_set_format(rdpsndDevicePlugin* device,
|
|
AUDIO_FORMAT* format, int latency)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
rdpsnd_opensles_check_handle(opensles);
|
|
|
|
DEBUG_SND("opensles=%p format=%p, latency=%d", opensles, format, latency);
|
|
|
|
if (format)
|
|
{
|
|
DEBUG_SND("format=%d, cbsize=%d, samples=%d, bits=%d, channels=%d, align=%d",
|
|
format->wFormatTag, format->cbSize, format->nSamplesPerSec,
|
|
format->wBitsPerSample, format->nChannels, format->nBlockAlign);
|
|
|
|
opensles->rate = format->nSamplesPerSec;
|
|
opensles->channels = format->nChannels;
|
|
|
|
switch (format->wFormatTag)
|
|
{
|
|
case WAVE_FORMAT_PCM:
|
|
switch (format->wBitsPerSample)
|
|
{
|
|
case 4:
|
|
opensles->format = WAVE_FORMAT_ADPCM;
|
|
break;
|
|
|
|
case 8:
|
|
opensles->format = WAVE_FORMAT_PCM;
|
|
break;
|
|
|
|
case 16:
|
|
opensles->format = WAVE_FORMAT_ADPCM;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WAVE_FORMAT_ADPCM:
|
|
case WAVE_FORMAT_DVI_ADPCM:
|
|
opensles->format = format->wFormatTag;
|
|
break;
|
|
}
|
|
|
|
opensles->wformat = format->wFormatTag;
|
|
opensles->block_size = format->nBlockAlign;
|
|
}
|
|
|
|
opensles->latency = latency;
|
|
|
|
rdpsnd_opensles_set_params(opensles);
|
|
}
|
|
|
|
static void rdpsnd_opensles_open(rdpsndDevicePlugin* device,
|
|
AUDIO_FORMAT* format, int latency)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("opensles=%p format=%p, latency=%d, rate=%d",
|
|
opensles, format, latency, opensles->rate);
|
|
|
|
if( rdpsnd_opensles_check_handle(opensles))
|
|
return;
|
|
|
|
opensles->stream = android_OpenAudioDevice(
|
|
opensles->rate, opensles->channels, 20);
|
|
assert(opensles->stream);
|
|
|
|
if (!opensles->stream)
|
|
DEBUG_WARN("android_OpenAudioDevice failed");
|
|
else
|
|
rdpsnd_opensles_set_volume(device, opensles->volume);
|
|
|
|
rdpsnd_opensles_set_format(device, format, latency);
|
|
}
|
|
|
|
static void rdpsnd_opensles_close(rdpsndDevicePlugin* device)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("opensles=%p", opensles);
|
|
if( !rdpsnd_opensles_check_handle(opensles))
|
|
return;
|
|
|
|
android_CloseAudioDevice(opensles->stream);
|
|
opensles->stream = NULL;
|
|
}
|
|
|
|
static void rdpsnd_opensles_free(rdpsndDevicePlugin* device)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("opensles=%p", opensles);
|
|
assert(opensles);
|
|
|
|
assert(opensles->device_name);
|
|
free(opensles->device_name);
|
|
|
|
assert(opensles->dsp_context);
|
|
freerdp_dsp_context_free(opensles->dsp_context);
|
|
|
|
free(opensles);
|
|
}
|
|
|
|
static BOOL rdpsnd_opensles_format_supported(rdpsndDevicePlugin* device,
|
|
AUDIO_FORMAT* format)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("format=%d, cbsize=%d, samples=%d, bits=%d, channels=%d, align=%d",
|
|
format->wFormatTag, format->cbSize, format->nSamplesPerSec,
|
|
format->wBitsPerSample, format->nChannels, format->nBlockAlign);
|
|
|
|
assert(opensles);
|
|
assert(format);
|
|
|
|
switch (format->wFormatTag)
|
|
{
|
|
case WAVE_FORMAT_PCM:
|
|
if (format->cbSize == 0 &&
|
|
format->nSamplesPerSec <= 48000 &&
|
|
(format->wBitsPerSample == 8 || format->wBitsPerSample == 16) &&
|
|
(format->nChannels == 1 || format->nChannels == 2))
|
|
{
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case WAVE_FORMAT_ADPCM:
|
|
case WAVE_FORMAT_DVI_ADPCM:
|
|
if (format->nSamplesPerSec <= 48000 &&
|
|
format->wBitsPerSample == 4 &&
|
|
(format->nChannels == 1 || format->nChannels == 2))
|
|
{
|
|
return TRUE;
|
|
}
|
|
break;
|
|
|
|
case WAVE_FORMAT_ALAW:
|
|
case WAVE_FORMAT_MULAW:
|
|
case WAVE_FORMAT_GSM610:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static UINT32 rdpsnd_opensles_get_volume(rdpsndDevicePlugin* device)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("opensles=%p", opensles);
|
|
assert(opensles);
|
|
|
|
if (opensles->stream)
|
|
{
|
|
const int max = android_GetOutputVolumeMax(opensles->stream);
|
|
const int rc = android_GetOutputVolume(opensles->stream);
|
|
|
|
if (android_GetOutputMute(opensles->stream))
|
|
opensles->volume = 0;
|
|
else
|
|
{
|
|
const unsigned short vol = rdpsnd_opensles_millibel_to_volume(rc, max);
|
|
opensles->volume = (vol << 16) | (vol & 0xFFFF);
|
|
}
|
|
}
|
|
|
|
return opensles->volume;
|
|
}
|
|
|
|
static void rdpsnd_opensles_set_volume(rdpsndDevicePlugin* device,
|
|
UINT32 value)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("opensles=%p, value=%d", opensles, value);
|
|
assert(opensles);
|
|
|
|
opensles->volume = value;
|
|
|
|
if (opensles->stream)
|
|
{
|
|
if (0 == opensles->volume)
|
|
android_SetOutputMute(opensles->stream, true);
|
|
else
|
|
{
|
|
const int max = android_GetOutputVolumeMax(opensles->stream);
|
|
const int vol = rdpsnd_opensles_volume_to_millibel(value & 0xFFFF, max);
|
|
|
|
android_SetOutputMute(opensles->stream, false);
|
|
android_SetOutputVolume(opensles->stream, vol);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void rdpsnd_opensles_play(rdpsndDevicePlugin* device,
|
|
BYTE *data, int size)
|
|
{
|
|
union
|
|
{
|
|
BYTE *b;
|
|
short *s;
|
|
} src;
|
|
int ret;
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
DEBUG_SND("opensles=%p, data=%p, size=%d", opensles, data, size);
|
|
if (!rdpsnd_opensles_check_handle(opensles))
|
|
return;
|
|
|
|
if (opensles->format == WAVE_FORMAT_ADPCM)
|
|
{
|
|
DEBUG_SND("dsp_context=%p, channels=%d, block_size=%d",
|
|
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=%d, block_size=%d",
|
|
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;
|
|
}
|
|
|
|
DEBUG_SND("size=%d, src=%p", size, src.b);
|
|
assert(0 == size % 2);
|
|
assert(size > 0);
|
|
assert(src.b);
|
|
|
|
ret = android_AudioOut(opensles->stream, src.s, size / 2);
|
|
if (ret < 0)
|
|
DEBUG_WARN("android_AudioOut failed (%d)", ret);
|
|
}
|
|
|
|
static void rdpsnd_opensles_start(rdpsndDevicePlugin* device)
|
|
{
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
rdpsnd_opensles_check_handle(opensles);
|
|
|
|
DEBUG_SND("opensles=%p", opensles);
|
|
}
|
|
|
|
static COMMAND_LINE_ARGUMENT_A rdpsnd_opensles_args[] =
|
|
{
|
|
{ "dev", COMMAND_LINE_VALUE_REQUIRED, "<device>",
|
|
NULL, NULL, -1, NULL, "device" },
|
|
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
|
|
};
|
|
|
|
static int rdpsnd_opensles_parse_addin_args(rdpsndDevicePlugin* device,
|
|
ADDIN_ARGV* args)
|
|
{
|
|
int status;
|
|
DWORD flags;
|
|
COMMAND_LINE_ARGUMENT_A* arg;
|
|
rdpsndopenslesPlugin* opensles = (rdpsndopenslesPlugin*) device;
|
|
|
|
assert(opensles);
|
|
assert(args);
|
|
|
|
DEBUG_SND("opensles=%p, args=%p", opensles, args);
|
|
|
|
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON;
|
|
|
|
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv,
|
|
rdpsnd_opensles_args, flags, opensles, NULL, NULL);
|
|
if (status < 0)
|
|
return status;
|
|
|
|
arg = rdpsnd_opensles_args;
|
|
|
|
do
|
|
{
|
|
if (!(arg->Flags & COMMAND_LINE_VALUE_PRESENT))
|
|
continue;
|
|
|
|
CommandLineSwitchStart(arg)
|
|
|
|
CommandLineSwitchCase(arg, "dev")
|
|
{
|
|
opensles->device_name = _strdup(arg->Value);
|
|
}
|
|
|
|
CommandLineSwitchEnd(arg)
|
|
}
|
|
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
|
|
|
|
return status;
|
|
}
|
|
|
|
#ifdef STATIC_CHANNELS
|
|
#define freerdp_rdpsnd_client_subsystem_entry \
|
|
opensles_freerdp_rdpsnd_client_subsystem_entry
|
|
#endif
|
|
|
|
int freerdp_rdpsnd_client_subsystem_entry(
|
|
PFREERDP_RDPSND_DEVICE_ENTRY_POINTS pEntryPoints)
|
|
{
|
|
ADDIN_ARGV* args;
|
|
rdpsndopenslesPlugin* opensles;
|
|
|
|
DEBUG_SND("pEntryPoints=%p", pEntryPoints);
|
|
|
|
opensles = (rdpsndopenslesPlugin*) malloc(sizeof(rdpsndopenslesPlugin));
|
|
ZeroMemory(opensles, sizeof(rdpsndopenslesPlugin));
|
|
|
|
opensles->device.Open = rdpsnd_opensles_open;
|
|
opensles->device.FormatSupported = rdpsnd_opensles_format_supported;
|
|
opensles->device.SetFormat = rdpsnd_opensles_set_format;
|
|
opensles->device.GetVolume = rdpsnd_opensles_get_volume;
|
|
opensles->device.SetVolume = rdpsnd_opensles_set_volume;
|
|
opensles->device.Start = rdpsnd_opensles_start;
|
|
opensles->device.Play = rdpsnd_opensles_play;
|
|
opensles->device.Close = rdpsnd_opensles_close;
|
|
opensles->device.Free = rdpsnd_opensles_free;
|
|
|
|
args = pEntryPoints->args;
|
|
rdpsnd_opensles_parse_addin_args((rdpsndDevicePlugin*) opensles, args);
|
|
|
|
if (!opensles->device_name)
|
|
opensles->device_name = _strdup("default");
|
|
|
|
opensles->rate = 44100;
|
|
opensles->channels = 2;
|
|
opensles->format = WAVE_FORMAT_ADPCM;
|
|
|
|
opensles->dsp_context = freerdp_dsp_context_new();
|
|
|
|
pEntryPoints->pRegisterRdpsndDevice(pEntryPoints->rdpsnd,
|
|
(rdpsndDevicePlugin*) opensles);
|
|
|
|
DEBUG_SND("success");
|
|
return 0;
|
|
}
|