Added basic input module for android microphone redirection.
This commit is contained in:
parent
0d9ab7ce6a
commit
e092852765
@ -47,3 +47,7 @@ endif()
|
||||
if(WITH_PULSE)
|
||||
add_channel_client_subsystem(${MODULE_PREFIX} ${CHANNEL_NAME} "pulse" "")
|
||||
endif()
|
||||
|
||||
if(WITH_OPENSL_ES)
|
||||
add_channel_client_subsystem(${MODULE_PREFIX} ${CHANNEL_NAME} "opensl_es" "")
|
||||
endif()
|
||||
|
@ -591,6 +591,13 @@ int DVCPluginEntry(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
|
||||
audin_load_device_plugin((IWTSPlugin*) audin, audin->subsystem, args);
|
||||
}
|
||||
|
||||
if (!audin->device)
|
||||
{
|
||||
audin_set_subsystem(audin, "opensl_es");
|
||||
audin_set_device_name(audin, "default");
|
||||
audin_load_device_plugin((IWTSPlugin*) audin, audin->subsystem, args);
|
||||
}
|
||||
|
||||
if (audin->device == NULL)
|
||||
{
|
||||
DEBUG_WARN("no sound device.");
|
||||
|
41
channels/audin/client/opensl_es/CMakeLists.txt
Normal file
41
channels/audin/client/opensl_es/CMakeLists.txt
Normal file
@ -0,0 +1,41 @@
|
||||
# FreeRDP: A Remote Desktop Protocol Implementation
|
||||
# FreeRDP cmake build script
|
||||
#
|
||||
# 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.
|
||||
|
||||
define_channel_client_subsystem("audin" "opensl_es" "")
|
||||
|
||||
set(${MODULE_PREFIX}_SRCS
|
||||
audin_opensl_es.c)
|
||||
|
||||
include_directories(..)
|
||||
include_directories(${OPENSL_ES_INCLUDE_DIRS})
|
||||
|
||||
add_channel_client_subsystem_library(${MODULE_PREFIX} ${MODULE_NAME} ${CHANNEL_NAME} "" TRUE "")
|
||||
|
||||
set_target_properties(${MODULE_NAME} PROPERTIES PREFIX "")
|
||||
|
||||
set_complex_link_libraries(VARIABLE ${MODULE_PREFIX}_LIBS
|
||||
MONOLITHIC ${MONOLITHIC_BUILD}
|
||||
MODULE freerdp
|
||||
MODULES freerdp-codec freerdp-utils)
|
||||
|
||||
set(${MODULE_PREFIX}_LIBS ${${MODULE_PREFIX}_LIBS} ${OPENSL_ES_LIBRARIES})
|
||||
|
||||
target_link_libraries(${MODULE_NAME} ${${MODULE_PREFIX}_LIBS})
|
||||
|
||||
if(NOT STATIC_CHANNELS)
|
||||
install(TARGETS ${MODULE_NAME} DESTINATION ${FREERDP_ADDIN_PATH})
|
||||
endif()
|
419
channels/audin/client/opensl_es/audin_opensl_es.c
Normal file
419
channels/audin/client/opensl_es/audin_opensl_es.c
Normal file
@ -0,0 +1,419 @@
|
||||
/**
|
||||
* FreeRDP: A Remote Desktop Protocol Implementation
|
||||
* Audio Input Redirection Virtual Channel - OpenSL ES implementation
|
||||
*
|
||||
* 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 <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <winpr/crt.h>
|
||||
#include <winpr/synch.h>
|
||||
#include <winpr/thread.h>
|
||||
#include <winpr/cmdline.h>
|
||||
|
||||
#include <freerdp/addin.h>
|
||||
#include <freerdp/codec/dsp.h>
|
||||
#include <freerdp/channels/rdpsnd.h>
|
||||
|
||||
#include <SLES/OpenSLES.h>
|
||||
|
||||
#include "opensl_io.h"
|
||||
#include "audin_main.h"
|
||||
|
||||
typedef struct _AudinOpenSL_ESDevice
|
||||
{
|
||||
IAudinDevice iface;
|
||||
|
||||
char* device_name;
|
||||
OPENSL_STREAM *stream;
|
||||
|
||||
UINT32 frames_per_packet;
|
||||
UINT32 target_rate;
|
||||
UINT32 actual_rate;
|
||||
UINT32 target_channels;
|
||||
UINT32 actual_channels;
|
||||
int bytes_per_channel;
|
||||
int wformat;
|
||||
int block_size;
|
||||
|
||||
FREERDP_DSP_CONTEXT* dsp_context;
|
||||
|
||||
HANDLE thread;
|
||||
HANDLE stopEvent;
|
||||
|
||||
void* user_data;
|
||||
} AudinOpenSL_ESDevice;
|
||||
|
||||
static BOOL audin_opensl_es_thread_receive(AudinOpenSL_ESDevice* opensl_es,
|
||||
void* src, int count)
|
||||
{
|
||||
int frames;
|
||||
int cframes;
|
||||
int ret = 0;
|
||||
int encoded_size;
|
||||
BYTE* encoded_data;
|
||||
int rbytes_per_frame;
|
||||
int tbytes_per_frame;
|
||||
|
||||
rbytes_per_frame = opensl_es->actual_channels * opensl_es->bytes_per_channel;
|
||||
tbytes_per_frame = opensl_es->target_channels * opensl_es->bytes_per_channel;
|
||||
|
||||
if ((opensl_es->target_rate == opensl_es->actual_rate) &&
|
||||
(opensl_es->target_channels == opensl_es->actual_channels))
|
||||
{
|
||||
frames = size / rbytes_per_frame;
|
||||
}
|
||||
else
|
||||
{
|
||||
opensl_es->dsp_context->resample(opensl_es->dsp_context, src,
|
||||
opensl_es->bytes_per_channel, opensl_es->actual_channels,
|
||||
opensl_es->actual_rate, size / rbytes_per_frame,
|
||||
opensl_es->target_channels, opensl_es->target_rate);
|
||||
frames = opensl_es->dsp_context->resampled_frames;
|
||||
|
||||
DEBUG_DVC("resampled %d frames at %d to %d frames at %d",
|
||||
size / rbytes_per_frame, opensl_es->actual_rate,
|
||||
frames, opensl_es->target_rate);
|
||||
|
||||
size = frames * tbytes_per_frame;
|
||||
src = opensl_es->dsp_context->resampled_buffer;
|
||||
}
|
||||
|
||||
while (frames > 0)
|
||||
{
|
||||
if (WaitForSingleObject(opensl_es->stopEvent, 0) == WAIT_OBJECT_0)
|
||||
break;
|
||||
|
||||
cframes = opensl_es->frames_per_packet - opensl_es->buffer_frames;
|
||||
|
||||
if (cframes > frames)
|
||||
cframes = frames;
|
||||
|
||||
CopyMemory(opensl_es->buffer + opensl_es->buffer_frames * tbytes_per_frame,
|
||||
src, cframes * tbytes_per_frame);
|
||||
|
||||
opensl_es->buffer_frames += cframes;
|
||||
|
||||
if (opensl_es->buffer_frames >= opensl_es->frames_per_packet)
|
||||
{
|
||||
if (opensl_es->wformat == WAVE_FORMAT_DVI_ADPCM)
|
||||
{
|
||||
opensl_es->dsp_context->encode_ima_adpcm(opensl_es->dsp_context,
|
||||
opensl_es->buffer, opensl_es->buffer_frames * tbytes_per_frame,
|
||||
opensl_es->target_channels, opensl_es->block_size);
|
||||
|
||||
encoded_data = opensl_es->dsp_context->adpcm_buffer;
|
||||
encoded_size = opensl_es->dsp_context->adpcm_size;
|
||||
|
||||
DEBUG_DVC("encoded %d to %d",
|
||||
opensl_es->buffer_frames * tbytes_per_frame, encoded_size);
|
||||
}
|
||||
else
|
||||
{
|
||||
encoded_data = opensl_es->buffer;
|
||||
encoded_size = opensl_es->buffer_frames * tbytes_per_frame;
|
||||
}
|
||||
|
||||
if (WaitForSingleObject(opensl_es->stopEvent, 0) == WAIT_OBJECT_0)
|
||||
break;
|
||||
else
|
||||
ret = opensl_es->receive(encoded_data, encoded_size,
|
||||
opensl_es->user_data);
|
||||
|
||||
opensl_es->buffer_frames = 0;
|
||||
|
||||
if (!ret)
|
||||
break;
|
||||
}
|
||||
|
||||
src += cframes * tbytes_per_frame;
|
||||
frames -= cframes;
|
||||
}
|
||||
|
||||
return (ret) ? TRUE : FALSE;
|
||||
}
|
||||
|
||||
static void* audin_opensl_es_thread_func(void* arg)
|
||||
{
|
||||
float* buffer;
|
||||
int rbytes_per_frame;
|
||||
int tbytes_per_frame;
|
||||
snd_pcm_t* capture_handle = NULL;
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) arg;
|
||||
|
||||
DEBUG_DVC("opensl_es=%p", opensl_es);
|
||||
|
||||
buffer = (BYTE*) calloc(sizeof(float), opensl_es->frames_per_packet);
|
||||
ZeroMemory(buffer, opensl_es->frames_per_packet);
|
||||
freerdp_dsp_context_reset_adpcm(opensl_es->dsp_context);
|
||||
|
||||
while (!(WaitForSingleObject(opensl_es->stopEvent, 0) == WAIT_OBJECT_0))
|
||||
{
|
||||
int rc = android_AudioIn(opensl_es->stream, buffer,
|
||||
opensl_es->frames_per_packet);
|
||||
if (rc < 0)
|
||||
{
|
||||
DEBUG_WARN("snd_pcm_readi (%s)", snd_strerror(error));
|
||||
break;
|
||||
}
|
||||
|
||||
if (!audin_opensl_es_thread_receive(opensl_es, buffer, rc * sizeof(float)))
|
||||
break;
|
||||
}
|
||||
|
||||
free(buffer);
|
||||
|
||||
DEBUG_DVC("thread shutdown.");
|
||||
|
||||
ExitThread(0);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void audin_opensl_es_free(IAudinDevice* device)
|
||||
{
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) device;
|
||||
|
||||
DEBUG_DVC("device=%p", device);
|
||||
|
||||
freerdp_dsp_context_free(opensl_es->dsp_context);
|
||||
|
||||
free(opensl_es->device_name);
|
||||
free(opensl_es);
|
||||
}
|
||||
|
||||
static BOOL audin_opensl_es_format_supported(IAudinDevice* device, audinFormat* format)
|
||||
{
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) device;
|
||||
SLResult rc;
|
||||
|
||||
DEBUG_DVC("device=%p, format=%p", device, 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_DVI_ADPCM:
|
||||
if ((format->nSamplesPerSec <= 48000) &&
|
||||
(format->wBitsPerSample == 4) &&
|
||||
(format->nChannels == 1 || format->nChannels == 2))
|
||||
{
|
||||
return TRUE;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
static void audin_opensl_es_set_format(IAudinDevice* device,
|
||||
audinFormat* format, UINT32 FramesPerPacket)
|
||||
{
|
||||
int bs;
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) device;
|
||||
|
||||
DEBUG_DVC("device=%p, format=%p, FramesPerPacket=%d",
|
||||
device, format, FramesPerPacket);
|
||||
|
||||
opensl_es->target_rate = format->nSamplesPerSec;
|
||||
opensl_es->actual_rate = format->nSamplesPerSec;
|
||||
opensl_es->target_channels = format->nChannels;
|
||||
opensl_es->actual_channels = format->nChannels;
|
||||
|
||||
switch (format->wFormatTag)
|
||||
{
|
||||
case WAVE_FORMAT_PCM:
|
||||
switch (format->wBitsPerSample)
|
||||
{
|
||||
case 8:
|
||||
opensl_es->format = SND_PCM_FORMAT_S8;
|
||||
opensl_es->bytes_per_channel = 1;
|
||||
break;
|
||||
case 16:
|
||||
opensl_es->format = SND_PCM_FORMAT_S16_LE;
|
||||
opensl_es->bytes_per_channel = 2;
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case WAVE_FORMAT_DVI_ADPCM:
|
||||
opensl_es->format = SND_PCM_FORMAT_S16_LE;
|
||||
opensl_es->bytes_per_channel = 2;
|
||||
bs = (format->nBlockAlign - 4 * format->nChannels) * 4;
|
||||
opensl_es->frames_per_packet =
|
||||
(opensl_es->frames_per_packet * format->nChannels * 2 /
|
||||
bs + 1) * bs / (format->nChannels * 2);
|
||||
DEBUG_DVC("aligned FramesPerPacket=%d",
|
||||
opensl_es->frames_per_packet);
|
||||
break;
|
||||
}
|
||||
|
||||
opensl_es->wformat = format->wFormatTag;
|
||||
opensl_es->block_size = format->nBlockAlign;
|
||||
}
|
||||
|
||||
static int audin_opensl_es_open(IAudinDevice* device, AudinReceive receive,
|
||||
void* user_data)
|
||||
{
|
||||
int status = 0;
|
||||
int rbytes_per_frame;
|
||||
int tbytes_per_frame;
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) device;
|
||||
|
||||
DEBUG_DVC("device=%p, receive=%d, user_data=%p", device, receive, user_data);
|
||||
|
||||
opensl_es->stream = android_OpenAudioDevice(opensl_es->target_rate,
|
||||
opensl_es->target_channels, 0, opensl_es->frames_per_packet);
|
||||
|
||||
opensl_es->receive = receive;
|
||||
opensl_es->user_data = user_data;
|
||||
|
||||
rbytes_per_frame = opensl_es->actual_channels * opensl_es->bytes_per_channel;
|
||||
tbytes_per_frame = opensl_es->target_channels * opensl_es->bytes_per_channel;
|
||||
opensl_es->buffer =
|
||||
(BYTE*) malloc(tbytes_per_frame * opensl_es->frames_per_packet);
|
||||
ZeroMemory(opensl_es->buffer,
|
||||
tbytes_per_frame * opensl_es->frames_per_packet);
|
||||
opensl_es->buffer_frames = 0;
|
||||
|
||||
opensl_es->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
|
||||
opensl_es->thread = CreateThread(NULL, 0,
|
||||
(LPTHREAD_START_ROUTINE) audin_opensl_es_thread_func,
|
||||
opensl_es, 0, NULL);
|
||||
}
|
||||
|
||||
static void audin_opensl_es_close(IAudinDevice* device)
|
||||
{
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) device;
|
||||
|
||||
DEBUG_DVC("device=%p", device);
|
||||
|
||||
SetEvent(opensl_es->stopEvent);
|
||||
WaitForSingleObject(opensl_es->thread, INFINITE);
|
||||
CloseHandle(opensl_es->stopEvent);
|
||||
CloseHandle(opensl_es->thread);
|
||||
|
||||
android_CloseAudioDevice(opensl_es->stream);
|
||||
|
||||
opensl_es->stopEvent = NULL;
|
||||
opensl_es->thread = NULL;
|
||||
opensl_es->receive = NULL;
|
||||
opensl_es->user_data = NULL;
|
||||
opsnsl_es->stream = NULL;
|
||||
}
|
||||
|
||||
static const COMMAND_LINE_ARGUMENT_A audin_opensl_es_args[] =
|
||||
{
|
||||
{ "audio-dev", COMMAND_LINE_VALUE_REQUIRED, "<device>",
|
||||
NULL, NULL, -1, NULL, "audio device name" },
|
||||
{ NULL, 0, NULL, NULL, NULL, -1, NULL, NULL }
|
||||
};
|
||||
|
||||
static int audin_opensl_es_parse_addin_args(AudinOpenSL_ESDevice* device,
|
||||
ADDIN_ARGV* args)
|
||||
{
|
||||
int status;
|
||||
DWORD flags;
|
||||
COMMAND_LINE_ARGUMENT_A* arg;
|
||||
AudinOpenSL_ESDevice* opensl_es = (AudinOpenSL_ESDevice*) device;
|
||||
|
||||
DEBUG_DVC("device=%p, args=%p", device, args);
|
||||
|
||||
flags = COMMAND_LINE_SIGIL_NONE | COMMAND_LINE_SEPARATOR_COLON;
|
||||
|
||||
status = CommandLineParseArgumentsA(args->argc, (const char**) args->argv,
|
||||
audin_opensl_es_args, flags, opensl_es, NULL, NULL);
|
||||
if (status < 0)
|
||||
return status;
|
||||
|
||||
arg = audin_opensl_es_args;
|
||||
|
||||
do
|
||||
{
|
||||
if (!(arg->Flags & COMMAND_LINE_VALUE_PRESENT))
|
||||
continue;
|
||||
|
||||
CommandLineSwitchStart(arg)
|
||||
|
||||
CommandLineSwitchCase(arg, "audio-dev")
|
||||
{
|
||||
opensl_es->device_name = _strdup(arg->Value);
|
||||
}
|
||||
|
||||
CommandLineSwitchEnd(arg)
|
||||
}
|
||||
while ((arg = CommandLineFindNextArgumentA(arg)) != NULL);
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
#ifdef STATIC_CHANNELS
|
||||
#define freerdp_audin_client_subsystem_entry \
|
||||
opensl_es_freerdp_audin_client_subsystem_entry
|
||||
#endif
|
||||
|
||||
int freerdp_audin_client_subsystem_entry(
|
||||
PFREERDP_AUDIN_DEVICE_ENTRY_POINTS pEntryPoints)
|
||||
{
|
||||
ADDIN_ARGV* args;
|
||||
AudinOpenSL_ESDevice* opensl_es;
|
||||
|
||||
DEBUG_DVC("pEntryPoints=%p", pEntryPoints);
|
||||
|
||||
opensl_es = (AudinOpenSL_ESDevice*) malloc(sizeof(AudinOpenSL_ESDevice));
|
||||
ZeroMemory(opensl_es, sizeof(AudinOpenSL_ESDevice));
|
||||
|
||||
opensl_es->iface.Open = audin_opensl_es_open;
|
||||
opensl_es->iface.FormatSupported = audin_opensl_es_format_supported;
|
||||
opensl_es->iface.SetFormat = audin_opensl_es_set_format;
|
||||
opensl_es->iface.Close = audin_opensl_es_close;
|
||||
opensl_es->iface.Free = audin_opensl_es_free;
|
||||
|
||||
args = pEntryPoints->args;
|
||||
|
||||
audin_opensl_es_parse_addin_args(opensl_es, args);
|
||||
|
||||
if (!opensl_es->device_name)
|
||||
opensl_es->device_name = _strdup("default");
|
||||
|
||||
opensl_es->frames_per_packet = 128;
|
||||
opensl_es->target_rate = 22050;
|
||||
opensl_es->actual_rate = 22050;
|
||||
opensl_es->format = SND_PCM_FORMAT_S16_LE;
|
||||
opensl_es->target_channels = 2;
|
||||
opensl_es->actual_channels = 2;
|
||||
opensl_es->bytes_per_channel = 2;
|
||||
|
||||
opensl_es->dsp_context = freerdp_dsp_context_new();
|
||||
|
||||
pEntryPoints->pRegisterAudinDevice(pEntryPoints->plugin,
|
||||
(IAudinDevice*) opensl_es);
|
||||
|
||||
return 0;
|
||||
}
|
529
channels/audin/client/opensl_es/opensl_io.c
Normal file
529
channels/audin/client/opensl_es/opensl_io.c
Normal file
@ -0,0 +1,529 @@
|
||||
/*
|
||||
opensl_io.c:
|
||||
Android OpenSL input/output module
|
||||
Copyright (c) 2012, Victor Lazzarini
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
* Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
* Redistributions in binary form must reproduce the above copyright
|
||||
notice, this list of conditions and the following disclaimer in the
|
||||
documentation and/or other materials provided with the distribution.
|
||||
* Neither the name of the <organization> nor the
|
||||
names of its contributors may be used to endorse or promote products
|
||||
derived from this software without specific prior written permission.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
|
||||
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "opensl_io.h"
|
||||
#define CONV16BIT 32768
|
||||
#define CONVMYFLT (1./32768.)
|
||||
|
||||
static void* createThreadLock(void);
|
||||
static int waitThreadLock(void *lock);
|
||||
static void notifyThreadLock(void *lock);
|
||||
static void destroyThreadLock(void *lock);
|
||||
static void bqPlayerCallback(SLAndroidSimpleBufferQueueItf bq, void *context);
|
||||
static void bqRecorderCallback(SLAndroidSimpleBufferQueueItf bq, void *context);
|
||||
|
||||
// creates the OpenSL ES audio engine
|
||||
static SLresult openSLCreateEngine(OPENSL_STREAM *p)
|
||||
{
|
||||
SLresult result;
|
||||
// create engine
|
||||
result = slCreateEngine(&(p->engineObject), 0, NULL, 0, NULL, NULL);
|
||||
if(result != SL_RESULT_SUCCESS) goto engine_end;
|
||||
|
||||
// realize the engine
|
||||
result = (*p->engineObject)->Realize(p->engineObject, SL_BOOLEAN_FALSE);
|
||||
if(result != SL_RESULT_SUCCESS) goto engine_end;
|
||||
|
||||
// get the engine interface, which is needed in order to create other objects
|
||||
result = (*p->engineObject)->GetInterface(p->engineObject, SL_IID_ENGINE, &(p->engineEngine));
|
||||
if(result != SL_RESULT_SUCCESS) goto engine_end;
|
||||
|
||||
engine_end:
|
||||
return result;
|
||||
}
|
||||
|
||||
// opens the OpenSL ES device for output
|
||||
static SLresult openSLPlayOpen(OPENSL_STREAM *p)
|
||||
{
|
||||
SLresult result;
|
||||
SLuint32 sr = p->sr;
|
||||
SLuint32 channels = p->outchannels;
|
||||
|
||||
if(channels){
|
||||
// configure audio source
|
||||
SLDataLocator_AndroidSimpleBufferQueue loc_bufq = {SL_DATALOCATOR_ANDROIDSIMPLEBUFFERQUEUE, 2};
|
||||
|
||||
switch(sr){
|
||||
|
||||
case 8000:
|
||||
sr = SL_SAMPLINGRATE_8;
|
||||
break;
|
||||
case 11025:
|
||||
sr = SL_SAMPLINGRATE_11_025;
|
||||
break;
|
||||
case 16000:
|
||||
sr = SL_SAMPLINGRATE_16;
|
||||
break;
|
||||
case 22050:
|
||||
sr = SL_SAMPLINGRATE_22_05;
|
||||
break;
|
||||
case 24000:
|
||||
sr = SL_SAMPLINGRATE_24;
|
||||
break;
|
||||
case 32000:
|
||||
sr = SL_SAMPLINGRATE_32;
|
||||
break;
|
||||
case 44100:
|
||||
sr = SL_SAMPLINGRATE_44_1;
|
||||
break;
|
||||
case 48000:
|
||||
sr = SL_SAMPLINGRATE_48;
|
||||
break;
|
||||
case 64000:
|
||||
sr = SL_SAMPLINGRATE_64;
|
||||
break;
|
||||
case 88200:
|
||||
sr = SL_SAMPLINGRATE_88_2;
|
||||
break;
|
||||
case 96000:
|
||||
sr = SL_SAMPLINGRATE_96;
|
||||
break;
|
||||
case 192000:
|
||||
sr = SL_SAMPLINGRATE_192;
|
||||
break;
|
||||
default:
|
||||
return -1;
|
||||
}
|
||||
|
||||
const SLInterfaceID ids[] = {SL_IID_VOLUME};
|
||||
const SLboolean req[] = {SL_BOOLEAN_FALSE};
|
||||
result = (*p->engineEngine)->CreateOutputMix(p->engineEngine, &(p->outputMixObject), 1, ids, req);
|
||||
if(result != SL_RESULT_SUCCESS) goto end_openaudio;
|
||||
|
||||
// realize the output mix
|
||||
result = (*p->outputMixObject)->Realize(p->outputMixObject, SL_BOOLEAN_FALSE);
|
||||
|
||||
int speakers;
|
||||
if(channels > 1)
|
||||
speakers = SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT;
|
||||
else speakers = SL_SPEAKER_FRONT_CENTER;
|
||||
SLDataFormat_PCM format_pcm = {SL_DATAFORMAT_PCM,channels, sr,
|
||||
SL_PCMSAMPLEFORMAT_FIXED_16, SL_PCMSAMPLEFORMAT_FIXED_16,
|
||||
speakers, SL_BYTEORDER_LITTLEENDIAN};
|
||||
|
||||
SLDataSource audioSrc = {&loc_bufq, &format_pcm};
|
||||
|
||||
// configure audio sink
|
||||
SLDataLocator_OutputMix loc_outmix = {SL_DATALOCATOR_OUTPUTMIX, p->outputMixObject};
|
||||
SLDataSink audioSnk = {&loc_outmix, NULL};
|
||||
|
||||
// create audio player
|
||||
const SLInterfaceID ids1[] = {SL_IID_ANDROIDSIMPLEBUFFERQUEUE};
|
||||
const SLboolean req1[] = {SL_BOOLEAN_TRUE};
|
||||
result = (*p->engineEngine)->CreateAudioPlayer(p->engineEngine, &(p->bqPlayerObject), &audioSrc, &audioSnk,
|
||||
1, ids1, req1);
|
||||
if(result != SL_RESULT_SUCCESS) goto end_openaudio;
|
||||
|
||||
// realize the player
|
||||
result = (*p->bqPlayerObject)->Realize(p->bqPlayerObject, SL_BOOLEAN_FALSE);
|
||||
if(result != SL_RESULT_SUCCESS) goto end_openaudio;
|
||||
|
||||
// get the play interface
|
||||
result = (*p->bqPlayerObject)->GetInterface(p->bqPlayerObject, SL_IID_PLAY, &(p->bqPlayerPlay));
|
||||
if(result != SL_RESULT_SUCCESS) goto end_openaudio;
|
||||
|
||||
// get the buffer queue interface
|
||||
result = (*p->bqPlayerObject)->GetInterface(p->bqPlayerObject, SL_IID_ANDROIDSIMPLEBUFFERQUEUE,
|
||||
&(p->bqPlayerBufferQueue));
|
||||
if(result != SL_RESULT_SUCCESS) goto end_openaudio;
|
||||
|
||||
// register callback on the buffer queue
|
||||
result = (*p->bqPlayerBufferQueue)->RegisterCallback(p->bqPlayerBufferQueue, bqPlayerCallback, p);
|
||||
if(result != SL_RESULT_SUCCESS) goto end_openaudio;
|
||||
|
||||
// set the player's state to playing
|
||||
result = (*p->bqPlayerPlay)->SetPlayState(p->bqPlayerPlay, SL_PLAYSTATE_PLAYING);
|
||||
|
||||
end_openaudio:
|
||||
return result;
|
||||
}
|
||||
return SL_RESULT_SUCCESS;
|
||||
}
|
||||
|
||||
// Open the OpenSL ES device for input
|
||||
static SLresult openSLRecOpen(OPENSL_STREAM *p){
|
||||
|
||||
SLresult result;
|
||||
SLuint32 sr = p->sr;
|
||||
SLuint32 channels = p->inchannels;
|
||||
|
||||
if(channels){
|
||||
|
||||
switch(sr){
|
||||
|
||||
case 8000:
|
||||
sr = SL_SAMPLINGRATE_8;
|
||||
break;
|
||||
case 11025:
|
||||
sr = SL_SAMPLINGRATE_11_025;
|
||||
break;
|
||||
case 16000:
|
||||
sr = SL_SAMPLINGRATE_16;
|
||||
break;
|
||||
case 22050:
|
||||
sr = SL_SAMPLINGRATE_22_05;
|
||||
break;
|
||||
case 24000:
|
||||
sr = SL_SAMPLINGRATE_24;
|
||||
break;
|
||||
case 32000:
|
||||
sr = SL_SAMPLINGRATE_32;
|
||||
break;
|
||||
case 44100:
|
||||
sr = SL_SAMPLINGRATE_44_1;
|
||||
break;
|
||||
case 48000:
|
||||
sr = SL_SAMPLINGRATE_48;
|
||||
break;
|
||||
case 64000:
|
||||
sr = SL_SAMPLINGRATE_64;
|
||||
break;
|
||||
case 88200:
|
||||
sr = SL_SAMPLINGRATE_88_2;
|
||||
break;
|
||||
case 96000:
|
||||
sr = SL_SAMPLINGRATE_96;
|
||||
break;
|
||||
case 192000:
|
||||
sr = SL_SAMPLINGRATE_192;
|
||||
break;
|
||||
default:
|
||||
return -1;
|
||||
}
|
||||
|
||||
// configure audio source
|
||||
SLDataLocator_IODevice loc_dev = {SL_DATALOCATOR_IODEVICE, SL_IODEVICE_AUDIOINPUT,
|
||||
SL_DEFAULTDEVICEID_AUDIOINPUT, NULL};
|
||||
SLDataSource audioSrc = {&loc_dev, NULL};
|
||||
|
||||
// configure audio sink
|
||||
int speakers;
|
||||
if(channels > 1)
|
||||
speakers = SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT;
|
||||
else speakers = SL_SPEAKER_FRONT_CENTER;
|
||||
SLDataLocator_AndroidSimpleBufferQueue loc_bq = {SL_DATALOCATOR_ANDROIDSIMPLEBUFFERQUEUE, 2};
|
||||
SLDataFormat_PCM format_pcm = {SL_DATAFORMAT_PCM, channels, sr,
|
||||
SL_PCMSAMPLEFORMAT_FIXED_16, SL_PCMSAMPLEFORMAT_FIXED_16,
|
||||
speakers, SL_BYTEORDER_LITTLEENDIAN};
|
||||
SLDataSink audioSnk = {&loc_bq, &format_pcm};
|
||||
|
||||
// create audio recorder
|
||||
// (requires the RECORD_AUDIO permission)
|
||||
const SLInterfaceID id[1] = {SL_IID_ANDROIDSIMPLEBUFFERQUEUE};
|
||||
const SLboolean req[1] = {SL_BOOLEAN_TRUE};
|
||||
result = (*p->engineEngine)->CreateAudioRecorder(p->engineEngine, &(p->recorderObject), &audioSrc,
|
||||
&audioSnk, 1, id, req);
|
||||
if (SL_RESULT_SUCCESS != result) goto end_recopen;
|
||||
|
||||
// realize the audio recorder
|
||||
result = (*p->recorderObject)->Realize(p->recorderObject, SL_BOOLEAN_FALSE);
|
||||
if (SL_RESULT_SUCCESS != result) goto end_recopen;
|
||||
|
||||
// get the record interface
|
||||
result = (*p->recorderObject)->GetInterface(p->recorderObject, SL_IID_RECORD, &(p->recorderRecord));
|
||||
if (SL_RESULT_SUCCESS != result) goto end_recopen;
|
||||
|
||||
// get the buffer queue interface
|
||||
result = (*p->recorderObject)->GetInterface(p->recorderObject, SL_IID_ANDROIDSIMPLEBUFFERQUEUE,
|
||||
&(p->recorderBufferQueue));
|
||||
if (SL_RESULT_SUCCESS != result) goto end_recopen;
|
||||
|
||||
// register callback on the buffer queue
|
||||
result = (*p->recorderBufferQueue)->RegisterCallback(p->recorderBufferQueue, bqRecorderCallback,
|
||||
p);
|
||||
if (SL_RESULT_SUCCESS != result) goto end_recopen;
|
||||
result = (*p->recorderRecord)->SetRecordState(p->recorderRecord, SL_RECORDSTATE_RECORDING);
|
||||
|
||||
end_recopen:
|
||||
return result;
|
||||
}
|
||||
else return SL_RESULT_SUCCESS;
|
||||
|
||||
|
||||
}
|
||||
|
||||
// close the OpenSL IO and destroy the audio engine
|
||||
static void openSLDestroyEngine(OPENSL_STREAM *p){
|
||||
|
||||
// destroy buffer queue audio player object, and invalidate all associated interfaces
|
||||
if (p->bqPlayerObject != NULL) {
|
||||
(*p->bqPlayerObject)->Destroy(p->bqPlayerObject);
|
||||
p->bqPlayerObject = NULL;
|
||||
p->bqPlayerPlay = NULL;
|
||||
p->bqPlayerBufferQueue = NULL;
|
||||
p->bqPlayerEffectSend = NULL;
|
||||
}
|
||||
|
||||
// destroy audio recorder object, and invalidate all associated interfaces
|
||||
if (p->recorderObject != NULL) {
|
||||
(*p->recorderObject)->Destroy(p->recorderObject);
|
||||
p->recorderObject = NULL;
|
||||
p->recorderRecord = NULL;
|
||||
p->recorderBufferQueue = NULL;
|
||||
}
|
||||
|
||||
// destroy output mix object, and invalidate all associated interfaces
|
||||
if (p->outputMixObject != NULL) {
|
||||
(*p->outputMixObject)->Destroy(p->outputMixObject);
|
||||
p->outputMixObject = NULL;
|
||||
}
|
||||
|
||||
// destroy engine object, and invalidate all associated interfaces
|
||||
if (p->engineObject != NULL) {
|
||||
(*p->engineObject)->Destroy(p->engineObject);
|
||||
p->engineObject = NULL;
|
||||
p->engineEngine = NULL;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
// open the android audio device for input and/or output
|
||||
OPENSL_STREAM *android_OpenAudioDevice(int sr, int inchannels, int outchannels, int bufferframes){
|
||||
|
||||
OPENSL_STREAM *p;
|
||||
p = (OPENSL_STREAM *) calloc(sizeof(OPENSL_STREAM),1);
|
||||
|
||||
p->inchannels = inchannels;
|
||||
p->outchannels = outchannels;
|
||||
p->sr = sr;
|
||||
p->inlock = createThreadLock();
|
||||
p->outlock = createThreadLock();
|
||||
|
||||
if((p->outBufSamples = bufferframes*outchannels) != 0) {
|
||||
if((p->outputBuffer[0] = (short *) calloc(p->outBufSamples, sizeof(short))) == NULL ||
|
||||
(p->outputBuffer[1] = (short *) calloc(p->outBufSamples, sizeof(short))) == NULL) {
|
||||
android_CloseAudioDevice(p);
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
if((p->inBufSamples = bufferframes*inchannels) != 0){
|
||||
if((p->inputBuffer[0] = (short *) calloc(p->inBufSamples, sizeof(short))) == NULL ||
|
||||
(p->inputBuffer[1] = (short *) calloc(p->inBufSamples, sizeof(short))) == NULL){
|
||||
android_CloseAudioDevice(p);
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
p->currentInputIndex = 0;
|
||||
p->currentOutputBuffer = 0;
|
||||
p->currentInputIndex = p->inBufSamples;
|
||||
p->currentInputBuffer = 0;
|
||||
|
||||
if(openSLCreateEngine(p) != SL_RESULT_SUCCESS) {
|
||||
android_CloseAudioDevice(p);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if(openSLRecOpen(p) != SL_RESULT_SUCCESS) {
|
||||
android_CloseAudioDevice(p);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if(openSLPlayOpen(p) != SL_RESULT_SUCCESS) {
|
||||
android_CloseAudioDevice(p);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
notifyThreadLock(p->outlock);
|
||||
notifyThreadLock(p->inlock);
|
||||
|
||||
p->time = 0.;
|
||||
return p;
|
||||
}
|
||||
|
||||
// close the android audio device
|
||||
void android_CloseAudioDevice(OPENSL_STREAM *p){
|
||||
|
||||
if (p == NULL)
|
||||
return;
|
||||
|
||||
openSLDestroyEngine(p);
|
||||
|
||||
if (p->inlock != NULL) {
|
||||
notifyThreadLock(p->inlock);
|
||||
destroyThreadLock(p->inlock);
|
||||
p->inlock = NULL;
|
||||
}
|
||||
|
||||
if (p->outlock != NULL) {
|
||||
notifyThreadLock(p->outlock);
|
||||
destroyThreadLock(p->outlock);
|
||||
p->inlock = NULL;
|
||||
}
|
||||
|
||||
if (p->outputBuffer[0] != NULL) {
|
||||
free(p->outputBuffer[0]);
|
||||
p->outputBuffer[0] = NULL;
|
||||
}
|
||||
|
||||
if (p->outputBuffer[1] != NULL) {
|
||||
free(p->outputBuffer[1]);
|
||||
p->outputBuffer[1] = NULL;
|
||||
}
|
||||
|
||||
if (p->inputBuffer[0] != NULL) {
|
||||
free(p->inputBuffer[0]);
|
||||
p->inputBuffer[0] = NULL;
|
||||
}
|
||||
|
||||
if (p->inputBuffer[1] != NULL) {
|
||||
free(p->inputBuffer[1]);
|
||||
p->inputBuffer[1] = NULL;
|
||||
}
|
||||
|
||||
free(p);
|
||||
}
|
||||
|
||||
// returns timestamp of the processed stream
|
||||
double android_GetTimestamp(OPENSL_STREAM *p){
|
||||
return p->time;
|
||||
}
|
||||
|
||||
|
||||
// this callback handler is called every time a buffer finishes recording
|
||||
void bqRecorderCallback(SLAndroidSimpleBufferQueueItf bq, void *context)
|
||||
{
|
||||
OPENSL_STREAM *p = (OPENSL_STREAM *) context;
|
||||
notifyThreadLock(p->inlock);
|
||||
}
|
||||
|
||||
// gets a buffer of size samples from the device
|
||||
int android_AudioIn(OPENSL_STREAM *p,float *buffer,int size){
|
||||
short *inBuffer;
|
||||
int i, bufsamps = p->inBufSamples, index = p->currentInputIndex;
|
||||
if(p == NULL || bufsamps == 0) return 0;
|
||||
|
||||
inBuffer = p->inputBuffer[p->currentInputBuffer];
|
||||
for(i=0; i < size; i++){
|
||||
if (index >= bufsamps) {
|
||||
waitThreadLock(p->inlock);
|
||||
(*p->recorderBufferQueue)->Enqueue(p->recorderBufferQueue,
|
||||
inBuffer,bufsamps*sizeof(short));
|
||||
p->currentInputBuffer = (p->currentInputBuffer ? 0 : 1);
|
||||
index = 0;
|
||||
inBuffer = p->inputBuffer[p->currentInputBuffer];
|
||||
}
|
||||
buffer[i] = (float) inBuffer[index++]*CONVMYFLT;
|
||||
}
|
||||
p->currentInputIndex = index;
|
||||
if(p->outchannels == 0) p->time += (double) size/(p->sr*p->inchannels);
|
||||
return i;
|
||||
}
|
||||
|
||||
// this callback handler is called every time a buffer finishes playing
|
||||
void bqPlayerCallback(SLAndroidSimpleBufferQueueItf bq, void *context)
|
||||
{
|
||||
OPENSL_STREAM *p = (OPENSL_STREAM *) context;
|
||||
notifyThreadLock(p->outlock);
|
||||
}
|
||||
|
||||
// puts a buffer of size samples to the device
|
||||
int android_AudioOut(OPENSL_STREAM *p, float *buffer,int size){
|
||||
|
||||
short *outBuffer;
|
||||
int i, bufsamps = p->outBufSamples, index = p->currentOutputIndex;
|
||||
if(p == NULL || bufsamps == 0) return 0;
|
||||
outBuffer = p->outputBuffer[p->currentOutputBuffer];
|
||||
|
||||
for(i=0; i < size; i++){
|
||||
outBuffer[index++] = (short) (buffer[i]*CONV16BIT);
|
||||
if (index >= p->outBufSamples) {
|
||||
waitThreadLock(p->outlock);
|
||||
(*p->bqPlayerBufferQueue)->Enqueue(p->bqPlayerBufferQueue,
|
||||
outBuffer,bufsamps*sizeof(short));
|
||||
p->currentOutputBuffer = (p->currentOutputBuffer ? 0 : 1);
|
||||
index = 0;
|
||||
outBuffer = p->outputBuffer[p->currentOutputBuffer];
|
||||
}
|
||||
}
|
||||
p->currentOutputIndex = index;
|
||||
p->time += (double) size/(p->sr*p->outchannels);
|
||||
return i;
|
||||
}
|
||||
|
||||
//----------------------------------------------------------------------
|
||||
// thread Locks
|
||||
// to ensure synchronisation between callbacks and processing code
|
||||
void* createThreadLock(void)
|
||||
{
|
||||
threadLock *p;
|
||||
p = (threadLock*) malloc(sizeof(threadLock));
|
||||
if (p == NULL)
|
||||
return NULL;
|
||||
memset(p, 0, sizeof(threadLock));
|
||||
if (pthread_mutex_init(&(p->m), (pthread_mutexattr_t*) NULL) != 0) {
|
||||
free((void*) p);
|
||||
return NULL;
|
||||
}
|
||||
if (pthread_cond_init(&(p->c), (pthread_condattr_t*) NULL) != 0) {
|
||||
pthread_mutex_destroy(&(p->m));
|
||||
free((void*) p);
|
||||
return NULL;
|
||||
}
|
||||
p->s = (unsigned char) 1;
|
||||
|
||||
return p;
|
||||
}
|
||||
|
||||
int waitThreadLock(void *lock)
|
||||
{
|
||||
threadLock *p;
|
||||
int retval = 0;
|
||||
p = (threadLock*) lock;
|
||||
pthread_mutex_lock(&(p->m));
|
||||
while (!p->s) {
|
||||
pthread_cond_wait(&(p->c), &(p->m));
|
||||
}
|
||||
p->s = (unsigned char) 0;
|
||||
pthread_mutex_unlock(&(p->m));
|
||||
}
|
||||
|
||||
void notifyThreadLock(void *lock)
|
||||
{
|
||||
threadLock *p;
|
||||
p = (threadLock*) lock;
|
||||
pthread_mutex_lock(&(p->m));
|
||||
p->s = (unsigned char) 1;
|
||||
pthread_cond_signal(&(p->c));
|
||||
pthread_mutex_unlock(&(p->m));
|
||||
}
|
||||
|
||||
void destroyThreadLock(void *lock)
|
||||
{
|
||||
threadLock *p;
|
||||
p = (threadLock*) lock;
|
||||
if (p == NULL)
|
||||
return;
|
||||
notifyThreadLock(p);
|
||||
pthread_cond_destroy(&(p->c));
|
||||
pthread_mutex_destroy(&(p->m));
|
||||
free(p);
|
||||
}
|
121
channels/audin/client/opensl_es/opensl_io.h
Normal file
121
channels/audin/client/opensl_es/opensl_io.h
Normal file
@ -0,0 +1,121 @@
|
||||
/*
|
||||
opensl_io.c:
|
||||
Android OpenSL input/output module header
|
||||
Copyright (c) 2012, Victor Lazzarini
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
* Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
* Redistributions in binary form must reproduce the above copyright
|
||||
notice, this list of conditions and the following disclaimer in the
|
||||
documentation and/or other materials provided with the distribution.
|
||||
* Neither the name of the <organization> nor the
|
||||
names of its contributors may be used to endorse or promote products
|
||||
derived from this software without specific prior written permission.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
|
||||
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#ifndef OPENSL_IO
|
||||
#define OPENSL_IO
|
||||
|
||||
#include <SLES/OpenSLES.h>
|
||||
#include <SLES/OpenSLES_Android.h>
|
||||
#include <pthread.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
typedef struct threadLock_{
|
||||
pthread_mutex_t m;
|
||||
pthread_cond_t c;
|
||||
unsigned char s;
|
||||
} threadLock;
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct opensl_stream {
|
||||
|
||||
// engine interfaces
|
||||
SLObjectItf engineObject;
|
||||
SLEngineItf engineEngine;
|
||||
|
||||
// output mix interfaces
|
||||
SLObjectItf outputMixObject;
|
||||
|
||||
// buffer queue player interfaces
|
||||
SLObjectItf bqPlayerObject;
|
||||
SLPlayItf bqPlayerPlay;
|
||||
SLAndroidSimpleBufferQueueItf bqPlayerBufferQueue;
|
||||
SLEffectSendItf bqPlayerEffectSend;
|
||||
|
||||
// recorder interfaces
|
||||
SLObjectItf recorderObject;
|
||||
SLRecordItf recorderRecord;
|
||||
SLAndroidSimpleBufferQueueItf recorderBufferQueue;
|
||||
|
||||
// buffer indexes
|
||||
int currentInputIndex;
|
||||
int currentOutputIndex;
|
||||
|
||||
// current buffer half (0, 1)
|
||||
int currentOutputBuffer;
|
||||
int currentInputBuffer;
|
||||
|
||||
// buffers
|
||||
short *outputBuffer[2];
|
||||
short *inputBuffer[2];
|
||||
|
||||
// size of buffers
|
||||
int outBufSamples;
|
||||
int inBufSamples;
|
||||
|
||||
// locks
|
||||
void* inlock;
|
||||
void* outlock;
|
||||
|
||||
double time;
|
||||
int inchannels;
|
||||
int outchannels;
|
||||
int sr;
|
||||
|
||||
} OPENSL_STREAM;
|
||||
|
||||
/*
|
||||
Open the audio device with a given sampling rate (sr), input and output channels and IO buffer size
|
||||
in frames. Returns a handle to the OpenSL stream
|
||||
*/
|
||||
OPENSL_STREAM* android_OpenAudioDevice(int sr, int inchannels, int outchannels, int bufferframes);
|
||||
/*
|
||||
Close the audio device
|
||||
*/
|
||||
void android_CloseAudioDevice(OPENSL_STREAM *p);
|
||||
/*
|
||||
Read a buffer from the OpenSL stream *p, of size samples. Returns the number of samples read.
|
||||
*/
|
||||
int android_AudioIn(OPENSL_STREAM *p, float *buffer,int size);
|
||||
/*
|
||||
Write a buffer to the OpenSL stream *p, of size samples. Returns the number of samples written.
|
||||
*/
|
||||
int android_AudioOut(OPENSL_STREAM *p, float *buffer,int size);
|
||||
/*
|
||||
Get the current IO block time in seconds
|
||||
*/
|
||||
double android_GetTimestamp(OPENSL_STREAM *p);
|
||||
|
||||
#ifdef __cplusplus
|
||||
};
|
||||
#endif
|
||||
|
||||
#endif // #ifndef OPENSL_IO
|
Loading…
Reference in New Issue
Block a user