/* * Copyright © 2012 Mozilla Foundation * * This program is made available under an ISC-style license. See the * accompanying file LICENSE for details. */ #undef NDEBUG #include #include #include #include #include #include #include #include #include #if defined(__ANDROID__) #include "android/sles_definitions.h" #include #include #include #include #include #endif #include "android/cubeb-output-latency.h" #include "cubeb-internal.h" #include "cubeb/cubeb.h" #include "cubeb_android.h" #include "cubeb_array_queue.h" #include "cubeb_resampler.h" #define ANDROID_VERSION_GINGERBREAD_MR1 10 #define ANDROID_VERSION_JELLY_BEAN 18 #define ANDROID_VERSION_LOLLIPOP 21 #define ANDROID_VERSION_MARSHMALLOW 23 #define ANDROID_VERSION_N_MR1 25 #define DEFAULT_SAMPLE_RATE 48000 #define DEFAULT_NUM_OF_FRAMES 480 extern cubeb_ops const opensl_ops; struct cubeb { struct cubeb_ops const * ops; void * lib; SLInterfaceID SL_IID_BUFFERQUEUE; SLInterfaceID SL_IID_PLAY; #if defined(__ANDROID__) SLInterfaceID SL_IID_ANDROIDCONFIGURATION; SLInterfaceID SL_IID_ANDROIDSIMPLEBUFFERQUEUE; #endif SLInterfaceID SL_IID_VOLUME; SLInterfaceID SL_IID_RECORD; SLObjectItf engObj; SLEngineItf eng; SLObjectItf outmixObj; output_latency_function * p_output_latency_function; }; #define NELEMS(A) (sizeof(A) / sizeof(A)[0]) #define NBUFS 2 struct cubeb_stream { /* Note: Must match cubeb_stream layout in cubeb.c. */ cubeb * context; void * user_ptr; /**/ pthread_mutex_t mutex; SLObjectItf playerObj; SLPlayItf play; SLBufferQueueItf bufq; SLVolumeItf volume; void ** queuebuf; uint32_t queuebuf_capacity; uint32_t queuebuf_idx; long queuebuf_len; long bytespersec; uint32_t framesize; /* Total number of played frames. * Synchronized by stream::mutex lock. */ long written; /* Flag indicating draining. Synchronized * by stream::mutex lock. */ int draining; /* Flags to determine in/out.*/ uint32_t input_enabled; uint32_t output_enabled; /* Recorder abstract object. */ SLObjectItf recorderObj; /* Recorder Itf for input capture. */ SLRecordItf recorderItf; /* Buffer queue for input capture. */ SLAndroidSimpleBufferQueueItf recorderBufferQueueItf; /* Store input buffers. */ void ** input_buffer_array; /* The capacity of the array. * On capture only can be small (4). * On full duplex is calculated to * store 1 sec of data buffers. */ uint32_t input_array_capacity; /* Current filled index of input buffer array. * It is initiated to -1 indicating buffering * have not started yet. */ int input_buffer_index; /* Length of input buffer.*/ uint32_t input_buffer_length; /* Input frame size */ uint32_t input_frame_size; /* Device sampling rate. If user rate is not * accepted an compatible rate is set. If it is * accepted this is equal to params.rate. */ uint32_t input_device_rate; /* Exchange input buffers between input * and full duplex threads. */ array_queue * input_queue; /* Silent input buffer used on full duplex. */ void * input_silent_buffer; /* Number of input frames from the start of the stream*/ uint32_t input_total_frames; /* Flag to stop the execution of user callback and * close all working threads. Synchronized by * stream::mutex lock. */ uint32_t shutdown; /* Store user callback. */ cubeb_data_callback data_callback; /* Store state callback. */ cubeb_state_callback state_callback; cubeb_resampler * resampler; unsigned int user_output_rate; unsigned int output_configured_rate; unsigned int buffer_size_frames; // Audio output latency used in cubeb_stream_get_position(). unsigned int output_latency_ms; int64_t lastPosition; int64_t lastPositionTimeStamp; int64_t lastCompensativePosition; int voice_input; int voice_output; std::unique_ptr input_params; std::unique_ptr output_params; // A non-empty buffer means that f32 -> int16 conversion need to happen std::vector conversion_buffer_output; std::vector conversion_buffer_input; }; /* Forward declaration. */ static int opensl_stop_player(cubeb_stream * stm); static int opensl_stop_recorder(cubeb_stream * stm); static int opensl_get_draining(cubeb_stream * stm) { #ifdef DEBUG int r = pthread_mutex_trylock(&stm->mutex); assert((r == EDEADLK || r == EBUSY) && "get_draining: mutex should be locked but it's not."); #endif return stm->draining; } static void opensl_set_draining(cubeb_stream * stm, int value) { #ifdef DEBUG int r = pthread_mutex_trylock(&stm->mutex); LOG("set draining try r = %d", r); assert((r == EDEADLK || r == EBUSY) && "set_draining: mutex should be locked but it's not."); #endif assert(value == 0 || value == 1); stm->draining = value; } static void opensl_notify_drained(cubeb_stream * stm) { assert(stm); int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int draining = opensl_get_draining(stm); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (draining) { stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_DRAINED); if (stm->play) { LOG("stop player in play_callback"); r = opensl_stop_player(stm); assert(r == CUBEB_OK); } if (stm->recorderItf) { r = opensl_stop_recorder(stm); assert(r == CUBEB_OK); } } } static uint32_t opensl_get_shutdown(cubeb_stream * stm) { #ifdef DEBUG int r = pthread_mutex_trylock(&stm->mutex); assert((r == EDEADLK || r == EBUSY) && "get_shutdown: mutex should be locked but it's not."); #endif return stm->shutdown; } static void opensl_set_shutdown(cubeb_stream * stm, uint32_t value) { #ifdef DEBUG int r = pthread_mutex_trylock(&stm->mutex); LOG("set shutdown try r = %d", r); assert((r == EDEADLK || r == EBUSY) && "set_shutdown: mutex should be locked but it's not."); #endif assert(value == 0 || value == 1); stm->shutdown = value; } static void play_callback(SLPlayItf caller, void * user_ptr, SLuint32 event) { cubeb_stream * stm = static_cast(user_ptr); assert(stm); switch (event) { case SL_PLAYEVENT_HEADATMARKER: opensl_notify_drained(stm); break; default: break; } } static void recorder_marker_callback(SLRecordItf caller, void * pContext, SLuint32 event) { cubeb_stream * stm = static_cast(pContext); assert(stm); if (event == SL_RECORDEVENT_HEADATMARKER) { int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int draining = opensl_get_draining(stm); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (draining) { stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_DRAINED); if (stm->recorderItf) { r = opensl_stop_recorder(stm); assert(r == CUBEB_OK); } if (stm->play) { r = opensl_stop_player(stm); assert(r == CUBEB_OK); } } } } // Returns a buffer suitable to write output data to. void * get_output_buffer(cubeb_stream * stm, void * output_buffer, uint32_t sample_count) { if (stm->conversion_buffer_output.empty()) { return output_buffer; } if (stm->conversion_buffer_output.size() < sample_count) { stm->conversion_buffer_output.resize(sample_count); } return stm->conversion_buffer_output.data(); } void * release_output_buffer(cubeb_stream * stm, void * original_output_buffer, uint32_t sample_count) { if (stm->conversion_buffer_output.empty()) { return original_output_buffer; } int16_t * int16_buf = reinterpret_cast(original_output_buffer); for (uint32_t i = 0; i < sample_count; i++) { float v = stm->conversion_buffer_output[i] * 32768.0f; float clamped = std::max(-32768.0f, std::min(32767.0f, v)); int16_buf[i] = static_cast(clamped); } return original_output_buffer; } static void bufferqueue_callback(SLBufferQueueItf caller, void * user_ptr) { cubeb_stream * stm = static_cast(user_ptr); assert(stm); SLBufferQueueState state; SLresult res; long written = 0; res = (*stm->bufq)->GetState(stm->bufq, &state); assert(res == SL_RESULT_SUCCESS); if (state.count > 1) { return; } void * buf = stm->queuebuf[stm->queuebuf_idx]; void * buf_original_ptr = buf; uint32_t sample_count = stm->output_params->channels * stm->queuebuf_len / stm->framesize; written = 0; int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int draining = opensl_get_draining(stm); uint32_t shutdown = opensl_get_shutdown(stm); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (!draining && !shutdown) { buf = get_output_buffer(stm, buf, sample_count); written = cubeb_resampler_fill(stm->resampler, nullptr, nullptr, buf, stm->queuebuf_len / stm->framesize); buf = release_output_buffer(stm, buf_original_ptr, sample_count); ALOGV("bufferqueue_callback: resampler fill returned %ld frames", written); if (written < 0 || written * stm->framesize > static_cast(stm->queuebuf_len)) { ALOGV("bufferqueue_callback: error, shutting down", written); r = pthread_mutex_lock(&stm->mutex); assert(r == 0); opensl_set_shutdown(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); opensl_stop_player(stm); stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_ERROR); return; } } // Keep sending silent data even in draining mode to prevent the audio // back-end from being stopped automatically by OpenSL/ES. assert(static_cast(stm->queuebuf_len) >= written * stm->framesize); memset(reinterpret_cast(buf) + written * stm->framesize, 0, stm->queuebuf_len - written * stm->framesize); res = (*stm->bufq)->Enqueue(stm->bufq, buf, stm->queuebuf_len); assert(res == SL_RESULT_SUCCESS); stm->queuebuf_idx = (stm->queuebuf_idx + 1) % stm->queuebuf_capacity; if (written > 0) { pthread_mutex_lock(&stm->mutex); stm->written += written; pthread_mutex_unlock(&stm->mutex); } if (!draining && written * stm->framesize < static_cast(stm->queuebuf_len)) { LOG("bufferqueue_callback draining"); r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int64_t written_duration = INT64_C(1000) * stm->written * stm->framesize / stm->bytespersec; opensl_set_draining(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (written_duration == 0) { // since we didn't write any sample, it's not possible to reach the marker // time and trigger the callback. We should initiative notify drained. opensl_notify_drained(stm); } else { // Use SL_PLAYEVENT_HEADATMARKER event from slPlayCallback of SLPlayItf // to make sure all the data has been processed. (*stm->play) ->SetMarkerPosition(stm->play, (SLmillisecond)written_duration); } return; } } static int opensl_enqueue_recorder(cubeb_stream * stm, void ** last_filled_buffer) { assert(stm); int current_index = stm->input_buffer_index; void * last_buffer = nullptr; if (current_index < 0) { // This is the first enqueue current_index = 0; } else { // The current index hold the last filled buffer get it before advance // index. last_buffer = stm->input_buffer_array[current_index]; // Advance to get next available buffer current_index = static_cast((current_index + 1) % stm->input_array_capacity); } // enqueue next empty buffer to be filled by the recorder SLresult res = (*stm->recorderBufferQueueItf) ->Enqueue(stm->recorderBufferQueueItf, stm->input_buffer_array[current_index], stm->input_buffer_length); if (res != SL_RESULT_SUCCESS) { LOG("Enqueue recorder failed. Error code: %lu", res); return CUBEB_ERROR; } // All good, update buffer and index. stm->input_buffer_index = current_index; if (last_filled_buffer) { *last_filled_buffer = last_buffer; } return CUBEB_OK; } // If necessary, convert and returns an input buffer. // Otherwise, just returns the pointer that has been passed in. void * convert_input_buffer_if_needed(cubeb_stream * stm, void * input_buffer, uint32_t sample_count) { // Perform conversion if needed if (stm->conversion_buffer_input.empty()) { return input_buffer; } if (stm->conversion_buffer_input.size() < sample_count) { stm->conversion_buffer_input.resize(sample_count); } int16_t * int16_buf = reinterpret_cast(input_buffer); for (uint32_t i = 0; i < sample_count; i++) { stm->conversion_buffer_input[i] = static_cast(int16_buf[i]) / 32768.f; } return stm->conversion_buffer_input.data(); } // input data callback void recorder_callback(SLAndroidSimpleBufferQueueItf bq, void * context) { assert(context); cubeb_stream * stm = static_cast(context); assert(stm->recorderBufferQueueItf); int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); uint32_t shutdown = opensl_get_shutdown(stm); int draining = opensl_get_draining(stm); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (shutdown || draining) { // According to the OpenSL ES 1.1 Specification, 8.14 SLBufferQueueItf // page 184, on transition to the SL_RECORDSTATE_STOPPED state, // the application should continue to enqueue buffers onto the queue // to retrieve the residual recorded data in the system. r = opensl_enqueue_recorder(stm, nullptr); assert(r == CUBEB_OK); return; } // Enqueue next available buffer and get the last filled buffer. void * input_buffer = nullptr; r = opensl_enqueue_recorder(stm, &input_buffer); assert(r == CUBEB_OK); assert(input_buffer); long input_frame_count = stm->input_buffer_length / stm->input_frame_size; uint32_t sample_count = input_frame_count * stm->input_params->channels; input_buffer = convert_input_buffer_if_needed(stm, input_buffer, sample_count); // Fill resampler with last input long got = cubeb_resampler_fill(stm->resampler, input_buffer, &input_frame_count, nullptr, 0); // Error case if (got < 0 || got > input_frame_count) { r = pthread_mutex_lock(&stm->mutex); assert(r == 0); opensl_set_shutdown(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); r = opensl_stop_recorder(stm); assert(r == CUBEB_OK); stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_ERROR); } // Advance total stream frames stm->input_total_frames += got; if (got < input_frame_count) { r = pthread_mutex_lock(&stm->mutex); assert(r == 0); opensl_set_draining(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); int64_t duration = INT64_C(1000) * stm->input_total_frames / stm->input_device_rate; (*stm->recorderItf) ->SetMarkerPosition(stm->recorderItf, (SLmillisecond)duration); return; } } void recorder_fullduplex_callback(SLAndroidSimpleBufferQueueItf bq, void * context) { assert(context); cubeb_stream * stm = static_cast(context); assert(stm->recorderBufferQueueItf); int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int draining = opensl_get_draining(stm); uint32_t shutdown = opensl_get_shutdown(stm); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (shutdown || draining) { /* On draining and shutdown the recorder should have been stoped from * the one set the flags. Accordint to the doc, on transition to * the SL_RECORDSTATE_STOPPED state, the application should * continue to enqueue buffers onto the queue to retrieve the residual * recorded data in the system. */ LOG("Input shutdown %d or drain %d", shutdown, draining); int r = opensl_enqueue_recorder(stm, nullptr); assert(r == CUBEB_OK); return; } // Enqueue next available buffer and get the last filled buffer. void * input_buffer = nullptr; r = opensl_enqueue_recorder(stm, &input_buffer); assert(r == CUBEB_OK); assert(input_buffer); assert(stm->input_queue); r = array_queue_push(stm->input_queue, input_buffer); if (r == -1) { LOG("Input queue is full, drop input ..."); return; } LOG("Input pushed in the queue, input array %zu", array_queue_get_size(stm->input_queue)); } static void player_fullduplex_callback(SLBufferQueueItf caller, void * user_ptr) { cubeb_stream * stm = static_cast(user_ptr); assert(stm); SLresult res; int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int draining = opensl_get_draining(stm); uint32_t shutdown = opensl_get_shutdown(stm); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); // Get output void * output_buffer = nullptr; r = pthread_mutex_lock(&stm->mutex); assert(r == 0); output_buffer = stm->queuebuf[stm->queuebuf_idx]; void * output_buffer_original_ptr = output_buffer; // Advance the output buffer queue index stm->queuebuf_idx = (stm->queuebuf_idx + 1) % stm->queuebuf_capacity; r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (shutdown || draining) { LOG("Shutdown/draining, send silent"); // Set silent on buffer memset(output_buffer, 0, stm->queuebuf_len); // Enqueue data in player buffer queue res = (*stm->bufq)->Enqueue(stm->bufq, output_buffer, stm->queuebuf_len); assert(res == SL_RESULT_SUCCESS); return; } // Get input. void * input_buffer = array_queue_pop(stm->input_queue); long input_frame_count = stm->input_buffer_length / stm->input_frame_size; long sample_count = input_frame_count * stm->input_params->channels; long frames_needed = stm->queuebuf_len / stm->framesize; uint32_t output_sample_count = stm->output_params->channels * stm->queuebuf_len / stm->framesize; if (!input_buffer) { LOG("Input hole set silent input buffer"); input_buffer = stm->input_silent_buffer; } input_buffer = convert_input_buffer_if_needed(stm, input_buffer, sample_count); output_buffer = get_output_buffer(stm, output_buffer, output_sample_count); long written = 0; // Trigger user callback through resampler written = cubeb_resampler_fill(stm->resampler, input_buffer, &input_frame_count, output_buffer, frames_needed); output_buffer = release_output_buffer(stm, output_buffer_original_ptr, sample_count); LOG("Fill: written %ld, frames_needed %ld, input array size %zu", written, frames_needed, array_queue_get_size(stm->input_queue)); if (written < 0 || written > frames_needed) { // Error case r = pthread_mutex_lock(&stm->mutex); assert(r == 0); opensl_set_shutdown(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); opensl_stop_player(stm); opensl_stop_recorder(stm); stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_ERROR); memset(output_buffer, 0, stm->queuebuf_len); // Enqueue data in player buffer queue res = (*stm->bufq)->Enqueue(stm->bufq, output_buffer, stm->queuebuf_len); assert(res == SL_RESULT_SUCCESS); return; } // Advance total out written frames counter r = pthread_mutex_lock(&stm->mutex); assert(r == 0); stm->written += written; r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (written < frames_needed) { r = pthread_mutex_lock(&stm->mutex); assert(r == 0); int64_t written_duration = INT64_C(1000) * stm->written * stm->framesize / stm->bytespersec; opensl_set_draining(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); // Use SL_PLAYEVENT_HEADATMARKER event from slPlayCallback of SLPlayItf // to make sure all the data has been processed. (*stm->play)->SetMarkerPosition(stm->play, (SLmillisecond)written_duration); } // Keep sending silent data even in draining mode to prevent the audio // back-end from being stopped automatically by OpenSL/ES. memset((uint8_t *)output_buffer + written * stm->framesize, 0, stm->queuebuf_len - written * stm->framesize); // Enqueue data in player buffer queue res = (*stm->bufq)->Enqueue(stm->bufq, output_buffer, stm->queuebuf_len); assert(res == SL_RESULT_SUCCESS); } static void opensl_destroy(cubeb * ctx); #if defined(__ANDROID__) #if (__ANDROID_API__ >= ANDROID_VERSION_LOLLIPOP) using system_property_get = int(const char *, char *); static int wrap_system_property_get(const char * name, char * value) { void * libc = dlopen("libc.so", RTLD_LAZY); if (!libc) { LOG("Failed to open libc.so"); return -1; } system_property_get * func = (system_property_get *)dlsym(libc, "__system_property_get"); int ret = -1; if (func) { ret = func(name, value); } dlclose(libc); return ret; } #endif static int get_android_version(void) { char version_string[PROP_VALUE_MAX]; memset(version_string, 0, PROP_VALUE_MAX); #if (__ANDROID_API__ >= ANDROID_VERSION_LOLLIPOP) int len = wrap_system_property_get("ro.build.version.sdk", version_string); #else int len = __system_property_get("ro.build.version.sdk", version_string); #endif if (len <= 0) { LOG("Failed to get Android version!\n"); return len; } int version = (int)strtol(version_string, nullptr, 10); LOG("Android version %d", version); return version; } #endif extern "C" { int opensl_init(cubeb ** context, char const * context_name) { cubeb * ctx; #if defined(__ANDROID__) int android_version = get_android_version(); if (android_version > 0 && android_version <= ANDROID_VERSION_GINGERBREAD_MR1) { // Don't even attempt to run on Gingerbread and lower LOG("Error: Android version too old, exiting."); return CUBEB_ERROR; } #endif *context = nullptr; ctx = static_cast(calloc(1, sizeof(*ctx))); assert(ctx); ctx->ops = &opensl_ops; ctx->lib = dlopen("libOpenSLES.so", RTLD_LAZY); if (!ctx->lib) { LOG("Error: Couldn't find libOpenSLES.so, exiting"); free(ctx); return CUBEB_ERROR; } typedef SLresult (*slCreateEngine_t)( SLObjectItf *, SLuint32, const SLEngineOption *, SLuint32, const SLInterfaceID *, const SLboolean *); slCreateEngine_t f_slCreateEngine = (slCreateEngine_t)dlsym(ctx->lib, "slCreateEngine"); SLInterfaceID SL_IID_ENGINE = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_ENGINE"); SLInterfaceID SL_IID_OUTPUTMIX = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_OUTPUTMIX"); ctx->SL_IID_VOLUME = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_VOLUME"); ctx->SL_IID_BUFFERQUEUE = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_BUFFERQUEUE"); #if defined(__ANDROID__) ctx->SL_IID_ANDROIDCONFIGURATION = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_ANDROIDCONFIGURATION"); ctx->SL_IID_ANDROIDSIMPLEBUFFERQUEUE = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_ANDROIDSIMPLEBUFFERQUEUE"); #endif ctx->SL_IID_PLAY = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_PLAY"); ctx->SL_IID_RECORD = *(SLInterfaceID *)dlsym(ctx->lib, "SL_IID_RECORD"); if (!f_slCreateEngine || !SL_IID_ENGINE || !SL_IID_OUTPUTMIX || !ctx->SL_IID_BUFFERQUEUE || #if defined(__ANDROID__) !ctx->SL_IID_ANDROIDCONFIGURATION || !ctx->SL_IID_ANDROIDSIMPLEBUFFERQUEUE || #endif !ctx->SL_IID_PLAY || !ctx->SL_IID_RECORD) { LOG("Error: didn't find required symbols, exiting."); opensl_destroy(ctx); return CUBEB_ERROR; } const SLEngineOption opt[] = {{SL_ENGINEOPTION_THREADSAFE, SL_BOOLEAN_TRUE}}; SLresult res; res = f_slCreateEngine(&ctx->engObj, 1, opt, 0, nullptr, nullptr); if (res != SL_RESULT_SUCCESS) { LOG("Error: slCreateEngine failure, exiting."); opensl_destroy(ctx); return CUBEB_ERROR; } res = (*ctx->engObj)->Realize(ctx->engObj, SL_BOOLEAN_FALSE); if (res != SL_RESULT_SUCCESS) { LOG("Error: engine realization failure, exiting."); opensl_destroy(ctx); return CUBEB_ERROR; } res = (*ctx->engObj)->GetInterface(ctx->engObj, SL_IID_ENGINE, &ctx->eng); if (res != SL_RESULT_SUCCESS) { LOG("Error: GetInterface(..., SL_IID_ENGINE, ...), exiting."); opensl_destroy(ctx); return CUBEB_ERROR; } const SLInterfaceID idsom[] = {SL_IID_OUTPUTMIX}; const SLboolean reqom[] = {SL_BOOLEAN_TRUE}; res = (*ctx->eng)->CreateOutputMix(ctx->eng, &ctx->outmixObj, 1, idsom, reqom); if (res != SL_RESULT_SUCCESS) { LOG("Error: CreateOutputMix failure, exiting."); opensl_destroy(ctx); return CUBEB_ERROR; } res = (*ctx->outmixObj)->Realize(ctx->outmixObj, SL_BOOLEAN_FALSE); if (res != SL_RESULT_SUCCESS) { LOG("Error: Output mix object failure, exiting."); opensl_destroy(ctx); return CUBEB_ERROR; } ctx->p_output_latency_function = cubeb_output_latency_load_method(android_version); if (!cubeb_output_latency_method_is_loaded(ctx->p_output_latency_function)) { LOG("Warning: output latency is not available, cubeb_stream_get_position() " "is not supported"); } *context = ctx; LOG("Cubeb init (%p) success", ctx); return CUBEB_OK; } } static char const * opensl_get_backend_id(cubeb * ctx) { return "opensl"; } static int opensl_get_max_channel_count(cubeb * ctx, uint32_t * max_channels) { assert(ctx && max_channels); /* The android mixer handles up to two channels, see http://androidxref.com/4.2.2_r1/xref/frameworks/av/services/audioflinger/AudioFlinger.h#67 */ *max_channels = 2; return CUBEB_OK; } static void opensl_destroy(cubeb * ctx) { if (ctx->outmixObj) { (*ctx->outmixObj)->Destroy(ctx->outmixObj); } if (ctx->engObj) { (*ctx->engObj)->Destroy(ctx->engObj); } dlclose(ctx->lib); if (ctx->p_output_latency_function) { cubeb_output_latency_unload_method(ctx->p_output_latency_function); } free(ctx); } static void opensl_stream_destroy(cubeb_stream * stm); #if defined(__ANDROID__) && (__ANDROID_API__ >= ANDROID_VERSION_LOLLIPOP) static int opensl_set_format_ext(SLAndroidDataFormat_PCM_EX * format, cubeb_stream_params * params) { assert(format); assert(params); format->formatType = SL_ANDROID_DATAFORMAT_PCM_EX; format->numChannels = params->channels; // sampleRate is in milliHertz format->sampleRate = params->rate * 1000; format->channelMask = params->channels == 1 ? SL_SPEAKER_FRONT_CENTER : SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT; switch (params->format) { case CUBEB_SAMPLE_S16LE: format->bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_16; format->containerSize = SL_PCMSAMPLEFORMAT_FIXED_16; format->representation = SL_ANDROID_PCM_REPRESENTATION_SIGNED_INT; format->endianness = SL_BYTEORDER_LITTLEENDIAN; break; case CUBEB_SAMPLE_S16BE: format->bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_16; format->containerSize = SL_PCMSAMPLEFORMAT_FIXED_16; format->representation = SL_ANDROID_PCM_REPRESENTATION_SIGNED_INT; format->endianness = SL_BYTEORDER_BIGENDIAN; break; case CUBEB_SAMPLE_FLOAT32LE: format->bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_32; format->containerSize = SL_PCMSAMPLEFORMAT_FIXED_32; format->representation = SL_ANDROID_PCM_REPRESENTATION_FLOAT; format->endianness = SL_BYTEORDER_LITTLEENDIAN; break; case CUBEB_SAMPLE_FLOAT32BE: format->bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_32; format->containerSize = SL_PCMSAMPLEFORMAT_FIXED_32; format->representation = SL_ANDROID_PCM_REPRESENTATION_FLOAT; format->endianness = SL_BYTEORDER_BIGENDIAN; break; default: return CUBEB_ERROR_INVALID_FORMAT; } return CUBEB_OK; } #endif static int opensl_set_format(SLDataFormat_PCM * format, cubeb_stream_params * params) { assert(format); assert(params); // If this function is called, this backend has been compiled with an older // version of Android, that doesn't support floating point audio IO. // The stream is configured with int16 of the proper endianess, and conversion // will happen during playback. format->formatType = SL_DATAFORMAT_PCM; format->numChannels = params->channels; // samplesPerSec is in milliHertz format->samplesPerSec = params->rate * 1000; format->bitsPerSample = SL_PCMSAMPLEFORMAT_FIXED_16; format->containerSize = SL_PCMSAMPLEFORMAT_FIXED_16; format->channelMask = params->channels == 1 ? SL_SPEAKER_FRONT_CENTER : SL_SPEAKER_FRONT_LEFT | SL_SPEAKER_FRONT_RIGHT; switch (params->format) { case CUBEB_SAMPLE_S16LE: case CUBEB_SAMPLE_FLOAT32LE: format->endianness = SL_BYTEORDER_LITTLEENDIAN; break; case CUBEB_SAMPLE_S16BE: case CUBEB_SAMPLE_FLOAT32BE: format->endianness = SL_BYTEORDER_BIGENDIAN; break; default: assert(false && "unhandled value"); } return CUBEB_OK; } template int initialize_with_format(cubeb_stream * stm, cubeb_stream_params * params, Function func) { void * format = nullptr; bool using_floats = false; uint32_t * format_sample_rate; #if defined(__ANDROID__) && (__ANDROID_API__ >= ANDROID_VERSION_LOLLIPOP) SLAndroidDataFormat_PCM_EX pcm_ext_format; if (get_android_version() >= ANDROID_VERSION_LOLLIPOP) { if (opensl_set_format_ext(&pcm_ext_format, params) != CUBEB_OK) { LOG("opensl_set_format_ext: error, exiting"); return CUBEB_ERROR_INVALID_FORMAT; } format = &pcm_ext_format; format_sample_rate = &pcm_ext_format.sampleRate; using_floats = pcm_ext_format.representation == SL_ANDROID_PCM_REPRESENTATION_FLOAT; } #endif SLDataFormat_PCM pcm_format; if (!format) { if (opensl_set_format(&pcm_format, params) != CUBEB_OK) { LOG("opensl_set_format: error, exiting"); return CUBEB_ERROR_INVALID_FORMAT; } format = &pcm_format; format_sample_rate = &pcm_format.samplesPerSec; } return func(format, format_sample_rate, using_floats); } static int opensl_configure_capture(cubeb_stream * stm, cubeb_stream_params * params) { assert(stm); assert(params); /* For now set device rate to params rate. */ stm->input_device_rate = params->rate; int rv = initialize_with_format( stm, params, [=](void * format, uint32_t * format_sample_rate, bool using_floats) -> int { SLDataLocator_AndroidSimpleBufferQueue lDataLocatorOut; lDataLocatorOut.locatorType = SL_DATALOCATOR_ANDROIDSIMPLEBUFFERQUEUE; lDataLocatorOut.numBuffers = NBUFS; SLDataSink dataSink; dataSink.pLocator = &lDataLocatorOut; dataSink.pFormat = format; SLDataLocator_IODevice dataLocatorIn; dataLocatorIn.locatorType = SL_DATALOCATOR_IODEVICE; dataLocatorIn.deviceType = SL_IODEVICE_AUDIOINPUT; dataLocatorIn.deviceID = SL_DEFAULTDEVICEID_AUDIOINPUT; dataLocatorIn.device = nullptr; SLDataSource dataSource; dataSource.pLocator = &dataLocatorIn; dataSource.pFormat = nullptr; const SLInterfaceID lSoundRecorderIIDs[] = { stm->context->SL_IID_RECORD, stm->context->SL_IID_ANDROIDSIMPLEBUFFERQUEUE, stm->context->SL_IID_ANDROIDCONFIGURATION}; const SLboolean lSoundRecorderReqs[] = { SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE}; // create the audio recorder abstract object SLresult res = (*stm->context->eng) ->CreateAudioRecorder(stm->context->eng, &stm->recorderObj, &dataSource, &dataSink, NELEMS(lSoundRecorderIIDs), lSoundRecorderIIDs, lSoundRecorderReqs); // Sample rate not supported. Try again with default sample rate! if (res == SL_RESULT_CONTENT_UNSUPPORTED) { if (stm->output_enabled && stm->output_configured_rate != 0) { // Set the same with the player. Since there is no // api for input device this is a safe choice. stm->input_device_rate = stm->output_configured_rate; } else { // The output preferred rate is used for an input only scenario. // The default rate expected to be supported from all android // devices. stm->input_device_rate = DEFAULT_SAMPLE_RATE; } *format_sample_rate = stm->input_device_rate * 1000; res = (*stm->context->eng) ->CreateAudioRecorder( stm->context->eng, &stm->recorderObj, &dataSource, &dataSink, NELEMS(lSoundRecorderIIDs), lSoundRecorderIIDs, lSoundRecorderReqs); } if (res != SL_RESULT_SUCCESS) { LOG("Failed to create recorder, not trying other input" " rate. Error code: %lu", res); return CUBEB_ERROR; } // It's always possible to use int16 regardless of the Android version. // However if compiling for older Android version, it's possible to // request f32 audio, but Android only supports int16, in which case a // conversion need to happen. if ((params->format == CUBEB_SAMPLE_FLOAT32NE || params->format == CUBEB_SAMPLE_FLOAT32BE) && !using_floats) { // setup conversion from f32 to int16 LOG("Input stream configured for using float, but not supported: a " "conversion will be performed"); stm->conversion_buffer_input.resize(1); } return CUBEB_OK; }); if (rv != CUBEB_OK) { LOG("Could not initialize recorder."); return rv; } SLresult res; if (get_android_version() > ANDROID_VERSION_JELLY_BEAN) { SLAndroidConfigurationItf recorderConfig; res = (*stm->recorderObj) ->GetInterface(stm->recorderObj, stm->context->SL_IID_ANDROIDCONFIGURATION, &recorderConfig); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get the android configuration interface for recorder. " "Error " "code: %lu", res); return CUBEB_ERROR; } // Voice recognition is the lowest latency, according to the docs. Camcorder // uses a microphone that is in the same direction as the camera. SLint32 streamType = stm->voice_input ? SL_ANDROID_RECORDING_PRESET_VOICE_RECOGNITION : SL_ANDROID_RECORDING_PRESET_CAMCORDER; res = (*recorderConfig) ->SetConfiguration(recorderConfig, SL_ANDROID_KEY_RECORDING_PRESET, &streamType, sizeof(SLint32)); if (res != SL_RESULT_SUCCESS) { LOG("Failed to set the android configuration to VOICE for the recorder. " "Error code: %lu", res); return CUBEB_ERROR; } } // realize the audio recorder res = (*stm->recorderObj)->Realize(stm->recorderObj, SL_BOOLEAN_FALSE); if (res != SL_RESULT_SUCCESS) { LOG("Failed to realize recorder. Error code: %lu", res); return CUBEB_ERROR; } // get the record interface res = (*stm->recorderObj) ->GetInterface(stm->recorderObj, stm->context->SL_IID_RECORD, &stm->recorderItf); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get recorder interface. Error code: %lu", res); return CUBEB_ERROR; } res = (*stm->recorderItf) ->RegisterCallback(stm->recorderItf, recorder_marker_callback, stm); if (res != SL_RESULT_SUCCESS) { LOG("Failed to register recorder marker callback. Error code: %lu", res); return CUBEB_ERROR; } (*stm->recorderItf)->SetMarkerPosition(stm->recorderItf, (SLmillisecond)0); res = (*stm->recorderItf) ->SetCallbackEventsMask(stm->recorderItf, (SLuint32)SL_RECORDEVENT_HEADATMARKER); if (res != SL_RESULT_SUCCESS) { LOG("Failed to set headatmarker event mask. Error code: %lu", res); return CUBEB_ERROR; } // get the simple android buffer queue interface res = (*stm->recorderObj) ->GetInterface(stm->recorderObj, stm->context->SL_IID_ANDROIDSIMPLEBUFFERQUEUE, &stm->recorderBufferQueueItf); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get recorder (android) buffer queue interface. Error code: " "%lu", res); return CUBEB_ERROR; } // register callback on record (input) buffer queue slAndroidSimpleBufferQueueCallback rec_callback = recorder_callback; if (stm->output_enabled) { // Register full duplex callback instead. rec_callback = recorder_fullduplex_callback; } res = (*stm->recorderBufferQueueItf) ->RegisterCallback(stm->recorderBufferQueueItf, rec_callback, stm); if (res != SL_RESULT_SUCCESS) { LOG("Failed to register recorder buffer queue callback. Error code: %lu", res); return CUBEB_ERROR; } // Calculate length of input buffer according to requested latency uint32_t sample_size = 0; if (params->format == CUBEB_SAMPLE_FLOAT32BE || params->format == CUBEB_SAMPLE_FLOAT32NE) { sample_size = sizeof(float); } else { sample_size = sizeof(int16_t); } stm->input_frame_size = params->channels * sample_size; stm->input_buffer_length = (stm->input_frame_size * stm->buffer_size_frames); // Calculate the capacity of input array stm->input_array_capacity = NBUFS; if (stm->output_enabled) { // Full duplex, update capacity to hold 1 sec of data stm->input_array_capacity = 1 * stm->input_device_rate / stm->input_buffer_length; } // Allocate input array stm->input_buffer_array = (void **)calloc(1, sizeof(void *) * stm->input_array_capacity); // Buffering has not started yet. stm->input_buffer_index = -1; // Prepare input buffers for (uint32_t i = 0; i < stm->input_array_capacity; ++i) { stm->input_buffer_array[i] = calloc(1, stm->input_buffer_length); } // On full duplex allocate input queue and silent buffer if (stm->output_enabled) { stm->input_queue = array_queue_create(stm->input_array_capacity); assert(stm->input_queue); stm->input_silent_buffer = calloc(1, stm->input_buffer_length); assert(stm->input_silent_buffer); } // Enqueue buffer to start rolling once recorder started rv = opensl_enqueue_recorder(stm, nullptr); if (rv != CUBEB_OK) { return rv; } LOG("Cubeb stream init recorder success"); return CUBEB_OK; } static int opensl_configure_playback(cubeb_stream * stm, cubeb_stream_params * params) { assert(stm); assert(params); stm->user_output_rate = params->rate; stm->lastPosition = -1; stm->lastPositionTimeStamp = 0; stm->lastCompensativePosition = -1; int rv = initialize_with_format( stm, params, [=](void * format, uint32_t * format_sample_rate, bool using_floats) { SLDataLocator_BufferQueue loc_bufq; loc_bufq.locatorType = SL_DATALOCATOR_BUFFERQUEUE; loc_bufq.numBuffers = NBUFS; SLDataSource source; source.pLocator = &loc_bufq; source.pFormat = format; SLDataLocator_OutputMix loc_outmix; loc_outmix.locatorType = SL_DATALOCATOR_OUTPUTMIX; loc_outmix.outputMix = stm->context->outmixObj; SLDataSink sink; sink.pLocator = &loc_outmix; sink.pFormat = nullptr; #if defined(__ANDROID__) const SLInterfaceID ids[] = {stm->context->SL_IID_BUFFERQUEUE, stm->context->SL_IID_VOLUME, stm->context->SL_IID_ANDROIDCONFIGURATION}; const SLboolean req[] = {SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE}; #else const SLInterfaceID ids[] = {ctx->SL_IID_BUFFERQUEUE, ctx->SL_IID_VOLUME}; const SLboolean req[] = {SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE}; #endif assert(NELEMS(ids) == NELEMS(req)); uint32_t preferred_sampling_rate = stm->user_output_rate; SLresult res = SL_RESULT_CONTENT_UNSUPPORTED; if (preferred_sampling_rate) { res = (*stm->context->eng) ->CreateAudioPlayer(stm->context->eng, &stm->playerObj, &source, &sink, NELEMS(ids), ids, req); } // Sample rate not supported? Try again with primary sample rate! if (res == SL_RESULT_CONTENT_UNSUPPORTED && preferred_sampling_rate != DEFAULT_SAMPLE_RATE) { preferred_sampling_rate = DEFAULT_SAMPLE_RATE; *format_sample_rate = preferred_sampling_rate * 1000; res = (*stm->context->eng) ->CreateAudioPlayer(stm->context->eng, &stm->playerObj, &source, &sink, NELEMS(ids), ids, req); } if (res != SL_RESULT_SUCCESS) { LOG("Failed to create audio player. Error code: %lu", res); return CUBEB_ERROR; } stm->output_configured_rate = preferred_sampling_rate; // It's always possible to use int16 regardless of the Android version. // However if compiling for older Android version, it's possible to // request f32 audio, but Android only supports int16, in which case a // conversion need to happen. if ((params->format == CUBEB_SAMPLE_FLOAT32NE || params->format == CUBEB_SAMPLE_FLOAT32BE) && !using_floats) { // setup conversion from f32 to int16 LOG("Input stream configured for using float, but not supported: a " "conversion will be performed"); stm->conversion_buffer_output.resize(1); } if (!using_floats) { stm->framesize = params->channels * sizeof(int16_t); } else { stm->framesize = params->channels * sizeof(float); } return CUBEB_OK; }); if (rv != CUBEB_OK) { LOG("Couldn't set format on sink or source"); return rv; } stm->bytespersec = stm->output_configured_rate * stm->framesize; stm->queuebuf_len = stm->framesize * stm->buffer_size_frames; // Calculate the capacity of input array stm->queuebuf_capacity = NBUFS; // Allocate input arrays stm->queuebuf = (void **)calloc(1, sizeof(void *) * stm->queuebuf_capacity); for (uint32_t i = 0; i < stm->queuebuf_capacity; ++i) { stm->queuebuf[i] = calloc(1, stm->queuebuf_len); assert(stm->queuebuf[i]); } SLAndroidConfigurationItf playerConfig = nullptr; SLresult res; if (get_android_version() >= ANDROID_VERSION_N_MR1) { res = (*stm->playerObj) ->GetInterface(stm->playerObj, stm->context->SL_IID_ANDROIDCONFIGURATION, &playerConfig); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get Android configuration interface. Error code: %lu", res); return CUBEB_ERROR; } SLint32 streamType = SL_ANDROID_STREAM_MEDIA; if (stm->voice_output) { streamType = SL_ANDROID_STREAM_VOICE; } res = (*playerConfig) ->SetConfiguration(playerConfig, SL_ANDROID_KEY_STREAM_TYPE, &streamType, sizeof(streamType)); if (res != SL_RESULT_SUCCESS) { LOG("Failed to set Android configuration to %d Error code: %lu", streamType, res); } SLuint32 performanceMode = SL_ANDROID_PERFORMANCE_LATENCY; if (stm->buffer_size_frames > POWERSAVE_LATENCY_FRAMES_THRESHOLD) { LOG("Audio stream configured for power saving"); performanceMode = SL_ANDROID_PERFORMANCE_POWER_SAVING; } res = (*playerConfig) ->SetConfiguration(playerConfig, SL_ANDROID_KEY_PERFORMANCE_MODE, &performanceMode, sizeof(performanceMode)); if (res != SL_RESULT_SUCCESS) { LOG("Failed to set Android performance mode to %d Error code: %lu. This " "is not fatal.", performanceMode, res); } } res = (*stm->playerObj)->Realize(stm->playerObj, SL_BOOLEAN_FALSE); if (res != SL_RESULT_SUCCESS) { LOG("Failed to realize player object. Error code: %lu", res); return CUBEB_ERROR; } // There are two ways of getting the audio output latency: // - a configuration value, only available on some devices (notably devices // running FireOS) // - A Java method, that we call using JNI. // // The first method is prefered, if available, because it can account for more // latency causes, and is more precise. // Latency has to be queried after the realization of the interface, when // using SL_IID_ANDROIDCONFIGURATION. SLuint32 audioLatency = 0; SLuint32 paramSize = sizeof(SLuint32); // The reported latency is in milliseconds. if (playerConfig) { res = (*playerConfig) ->GetConfiguration(playerConfig, (const SLchar *)"androidGetAudioLatency", ¶mSize, &audioLatency); if (res == SL_RESULT_SUCCESS) { LOG("Got playback latency using android configuration extension"); stm->output_latency_ms = audioLatency; } } // `playerConfig` is available, but the above failed, or `playerConfig` is not // available. In both cases, we need to acquire the output latency by an other // mean. if ((playerConfig && res != SL_RESULT_SUCCESS) || !playerConfig) { if (cubeb_output_latency_method_is_loaded( stm->context->p_output_latency_function)) { LOG("Got playback latency using JNI"); stm->output_latency_ms = cubeb_get_output_latency(stm->context->p_output_latency_function); } else { LOG("No alternate latency querying method loaded, A/V sync will be off."); stm->output_latency_ms = 0; } } LOG("Audio output latency: %dms", stm->output_latency_ms); res = (*stm->playerObj) ->GetInterface(stm->playerObj, stm->context->SL_IID_PLAY, &stm->play); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get play interface. Error code: %lu", res); return CUBEB_ERROR; } res = (*stm->playerObj) ->GetInterface(stm->playerObj, stm->context->SL_IID_BUFFERQUEUE, &stm->bufq); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get bufferqueue interface. Error code: %lu", res); return CUBEB_ERROR; } res = (*stm->playerObj) ->GetInterface(stm->playerObj, stm->context->SL_IID_VOLUME, &stm->volume); if (res != SL_RESULT_SUCCESS) { LOG("Failed to get volume interface. Error code: %lu", res); return CUBEB_ERROR; } res = (*stm->play)->RegisterCallback(stm->play, play_callback, stm); if (res != SL_RESULT_SUCCESS) { LOG("Failed to register play callback. Error code: %lu", res); return CUBEB_ERROR; } // Work around wilhelm/AudioTrack badness, bug 1221228 (*stm->play)->SetMarkerPosition(stm->play, (SLmillisecond)0); res = (*stm->play) ->SetCallbackEventsMask(stm->play, (SLuint32)SL_PLAYEVENT_HEADATMARKER); if (res != SL_RESULT_SUCCESS) { LOG("Failed to set headatmarker event mask. Error code: %lu", res); return CUBEB_ERROR; } slBufferQueueCallback player_callback = bufferqueue_callback; if (stm->input_enabled) { player_callback = player_fullduplex_callback; } res = (*stm->bufq)->RegisterCallback(stm->bufq, player_callback, stm); if (res != SL_RESULT_SUCCESS) { LOG("Failed to register bufferqueue callback. Error code: %lu", res); return CUBEB_ERROR; } { // Enqueue a silent frame so once the player becomes playing, the frame // will be consumed and kick off the buffer queue callback. // Note the duration of a single frame is less than 1ms. We don't bother // adjusting the playback position. uint8_t * buf = reinterpret_cast(stm->queuebuf[stm->queuebuf_idx++]); memset(buf, 0, stm->framesize); res = (*stm->bufq)->Enqueue(stm->bufq, buf, stm->framesize); assert(res == SL_RESULT_SUCCESS); } LOG("Cubeb stream init playback success"); return CUBEB_OK; } static int opensl_validate_stream_param(cubeb_stream_params * stream_params) { if ((stream_params && (stream_params->channels < 1 || stream_params->channels > 32))) { return CUBEB_ERROR_INVALID_FORMAT; } if ((stream_params && (stream_params->prefs & CUBEB_STREAM_PREF_LOOPBACK))) { LOG("Loopback is not supported"); return CUBEB_ERROR_NOT_SUPPORTED; } return CUBEB_OK; } int has_pref_set(cubeb_stream_params * input_params, cubeb_stream_params * output_params, cubeb_stream_prefs pref) { return (input_params && input_params->prefs & pref) || (output_params && output_params->prefs & pref); } static int opensl_stream_init(cubeb * ctx, cubeb_stream ** stream, char const * stream_name, cubeb_devid input_device, cubeb_stream_params * input_stream_params, cubeb_devid output_device, cubeb_stream_params * output_stream_params, unsigned int latency_frames, cubeb_data_callback data_callback, cubeb_state_callback state_callback, void * user_ptr) { cubeb_stream * stm = nullptr; cubeb_async_log_reset_threads(); assert(ctx); if (input_device || output_device) { LOG("Device selection is not supported in Android. The default will be " "used"); } *stream = nullptr; int r = opensl_validate_stream_param(output_stream_params); if (r != CUBEB_OK) { LOG("Output stream params not valid"); return r; } r = opensl_validate_stream_param(input_stream_params); if (r != CUBEB_OK) { LOG("Input stream params not valid"); return r; } stm = reinterpret_cast(calloc(1, sizeof(*stm))); assert(stm); if (input_stream_params) { stm->input_params = std::make_unique(*input_stream_params); } if (output_stream_params) { stm->output_params = std::make_unique(*output_stream_params); } stm->context = ctx; stm->data_callback = data_callback; stm->state_callback = state_callback; stm->user_ptr = user_ptr; stm->buffer_size_frames = latency_frames ? latency_frames : DEFAULT_NUM_OF_FRAMES; stm->input_enabled = (input_stream_params) ? 1 : 0; stm->output_enabled = (output_stream_params) ? 1 : 0; stm->shutdown = 1; stm->voice_input = has_pref_set(input_stream_params, nullptr, CUBEB_STREAM_PREF_VOICE); stm->voice_output = has_pref_set(nullptr, output_stream_params, CUBEB_STREAM_PREF_VOICE); LOG("cubeb stream prefs: voice_input: %s voice_output: %s", stm->voice_input ? "true" : "false", stm->voice_output ? "true" : "false"); #ifdef DEBUG pthread_mutexattr_t attr; pthread_mutexattr_init(&attr); pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ERRORCHECK); r = pthread_mutex_init(&stm->mutex, &attr); #else r = pthread_mutex_init(&stm->mutex, nullptr); #endif assert(r == 0); if (output_stream_params) { LOG("Playback params: Rate %d, channels %d, format %d, latency in frames " "%d.", output_stream_params->rate, output_stream_params->channels, output_stream_params->format, stm->buffer_size_frames); r = opensl_configure_playback(stm, output_stream_params); if (r != CUBEB_OK) { LOG("Error: playback-side configuration error, exiting."); opensl_stream_destroy(stm); return r; } } if (input_stream_params) { LOG("Capture params: Rate %d, channels %d, format %d, latency in frames " "%d.", input_stream_params->rate, input_stream_params->channels, input_stream_params->format, stm->buffer_size_frames); r = opensl_configure_capture(stm, input_stream_params); if (r != CUBEB_OK) { LOG("Error: record-side configuration error, exiting."); opensl_stream_destroy(stm); return r; } } /* Configure resampler*/ uint32_t target_sample_rate; if (input_stream_params) { target_sample_rate = input_stream_params->rate; } else { assert(output_stream_params); target_sample_rate = output_stream_params->rate; } // Use the actual configured rates for input // and output. cubeb_stream_params input_params; if (input_stream_params) { input_params = *input_stream_params; input_params.rate = stm->input_device_rate; } cubeb_stream_params output_params; if (output_stream_params) { output_params = *output_stream_params; output_params.rate = stm->output_configured_rate; } stm->resampler = cubeb_resampler_create( stm, input_stream_params ? &input_params : nullptr, output_stream_params ? &output_params : nullptr, target_sample_rate, data_callback, user_ptr, CUBEB_RESAMPLER_QUALITY_DEFAULT, CUBEB_RESAMPLER_RECLOCK_NONE); if (!stm->resampler) { LOG("Failed to create resampler"); opensl_stream_destroy(stm); return CUBEB_ERROR; } *stream = stm; LOG("Cubeb stream (%p) init success", stm); return CUBEB_OK; } static int opensl_start_player(cubeb_stream * stm) { assert(stm->playerObj); SLuint32 playerState; (*stm->playerObj)->GetState(stm->playerObj, &playerState); if (playerState == SL_OBJECT_STATE_REALIZED) { SLresult res = (*stm->play)->SetPlayState(stm->play, SL_PLAYSTATE_PLAYING); if (res != SL_RESULT_SUCCESS) { LOG("Failed to start player. Error code: %lu", res); return CUBEB_ERROR; } } return CUBEB_OK; } static int opensl_start_recorder(cubeb_stream * stm) { assert(stm->recorderObj); SLuint32 recorderState; (*stm->recorderObj)->GetState(stm->recorderObj, &recorderState); if (recorderState == SL_OBJECT_STATE_REALIZED) { SLresult res = (*stm->recorderItf) ->SetRecordState(stm->recorderItf, SL_RECORDSTATE_RECORDING); if (res != SL_RESULT_SUCCESS) { LOG("Failed to start recorder. Error code: %lu", res); return CUBEB_ERROR; } } return CUBEB_OK; } static int opensl_stream_start(cubeb_stream * stm) { assert(stm); int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); opensl_set_shutdown(stm, 0); opensl_set_draining(stm, 0); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (stm->playerObj) { r = opensl_start_player(stm); if (r != CUBEB_OK) { return r; } } if (stm->recorderObj) { int r = opensl_start_recorder(stm); if (r != CUBEB_OK) { return r; } } stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_STARTED); LOG("Cubeb stream (%p) started", stm); return CUBEB_OK; } static int opensl_stop_player(cubeb_stream * stm) { assert(stm->playerObj); assert(stm->shutdown || stm->draining); SLresult res = (*stm->play)->SetPlayState(stm->play, SL_PLAYSTATE_PAUSED); if (res != SL_RESULT_SUCCESS) { LOG("Failed to stop player. Error code: %lu", res); return CUBEB_ERROR; } return CUBEB_OK; } static int opensl_stop_recorder(cubeb_stream * stm) { assert(stm->recorderObj); assert(stm->shutdown || stm->draining); SLresult res = (*stm->recorderItf) ->SetRecordState(stm->recorderItf, SL_RECORDSTATE_PAUSED); if (res != SL_RESULT_SUCCESS) { LOG("Failed to stop recorder. Error code: %lu", res); return CUBEB_ERROR; } return CUBEB_OK; } static int opensl_stream_stop(cubeb_stream * stm) { assert(stm); int r = pthread_mutex_lock(&stm->mutex); assert(r == 0); opensl_set_shutdown(stm, 1); r = pthread_mutex_unlock(&stm->mutex); assert(r == 0); if (stm->playerObj) { r = opensl_stop_player(stm); if (r != CUBEB_OK) { return r; } } if (stm->recorderObj) { int r = opensl_stop_recorder(stm); if (r != CUBEB_OK) { return r; } } stm->state_callback(stm, stm->user_ptr, CUBEB_STATE_STOPPED); LOG("Cubeb stream (%p) stopped", stm); return CUBEB_OK; } static int opensl_destroy_recorder(cubeb_stream * stm) { assert(stm); assert(stm->recorderObj); if (stm->recorderBufferQueueItf) { SLresult res = (*stm->recorderBufferQueueItf)->Clear(stm->recorderBufferQueueItf); if (res != SL_RESULT_SUCCESS) { LOG("Failed to clear recorder buffer queue. Error code: %lu", res); return CUBEB_ERROR; } stm->recorderBufferQueueItf = nullptr; for (uint32_t i = 0; i < stm->input_array_capacity; ++i) { free(stm->input_buffer_array[i]); } } (*stm->recorderObj)->Destroy(stm->recorderObj); stm->recorderObj = nullptr; stm->recorderItf = nullptr; if (stm->input_queue) { array_queue_destroy(stm->input_queue); } free(stm->input_silent_buffer); return CUBEB_OK; } static void opensl_stream_destroy(cubeb_stream * stm) { assert(stm->draining || stm->shutdown); // If we're still draining at stream destroy time, pause the streams now so we // can destroy them safely. if (stm->draining) { opensl_stream_stop(stm); } // Sleep for 10ms to give active streams time to pause so that no further // buffer callbacks occur. Inspired by the same workaround (sleepBeforeClose) // in liboboe. usleep(10 * 1000); if (stm->playerObj) { (*stm->playerObj)->Destroy(stm->playerObj); stm->playerObj = nullptr; stm->play = nullptr; stm->bufq = nullptr; for (uint32_t i = 0; i < stm->queuebuf_capacity; ++i) { free(stm->queuebuf[i]); } } if (stm->recorderObj) { int r = opensl_destroy_recorder(stm); assert(r == CUBEB_OK); } if (stm->resampler) { cubeb_resampler_destroy(stm->resampler); } pthread_mutex_destroy(&stm->mutex); LOG("Cubeb stream (%p) destroyed", stm); free(stm); } static int opensl_stream_get_position(cubeb_stream * stm, uint64_t * position) { SLmillisecond msec; uint32_t compensation_msec = 0; SLresult res; res = (*stm->play)->GetPosition(stm->play, &msec); if (res != SL_RESULT_SUCCESS) { return CUBEB_ERROR; } timespec t{}; clock_gettime(CLOCK_MONOTONIC, &t); if (stm->lastPosition == msec) { compensation_msec = (t.tv_sec * 1000000000LL + t.tv_nsec - stm->lastPositionTimeStamp) / 1000000; } else { stm->lastPositionTimeStamp = t.tv_sec * 1000000000LL + t.tv_nsec; stm->lastPosition = msec; } uint64_t samplerate = stm->user_output_rate; uint32_t output_latency = stm->output_latency_ms; pthread_mutex_lock(&stm->mutex); int64_t maximum_position = stm->written * (int64_t)stm->user_output_rate / stm->output_configured_rate; pthread_mutex_unlock(&stm->mutex); assert(maximum_position >= 0); if (msec > output_latency) { int64_t unadjusted_position; if (stm->lastCompensativePosition > msec + compensation_msec) { // Over compensation, use lastCompensativePosition. unadjusted_position = samplerate * (stm->lastCompensativePosition - output_latency) / 1000; } else { unadjusted_position = samplerate * (msec - output_latency + compensation_msec) / 1000; stm->lastCompensativePosition = msec + compensation_msec; } *position = unadjusted_position < maximum_position ? unadjusted_position : maximum_position; } else { *position = 0; } return CUBEB_OK; } static int opensl_stream_get_latency(cubeb_stream * stm, uint32_t * latency) { assert(stm); assert(latency); uint32_t stream_latency_frames = stm->user_output_rate * stm->output_latency_ms / 1000; *latency = static_cast(stream_latency_frames + cubeb_resampler_latency(stm->resampler)); return CUBEB_OK; } int opensl_stream_set_volume(cubeb_stream * stm, float volume) { SLresult res; SLmillibel max_level, millibels; float unclamped_millibels; res = (*stm->volume)->GetMaxVolumeLevel(stm->volume, &max_level); if (res != SL_RESULT_SUCCESS) { return CUBEB_ERROR; } /* millibels are 100*dB, so the conversion from the volume's linear amplitude * is 100 * 20 * log(volume). However we clamp the resulting value before * passing it to lroundf() in order to prevent it from silently returning an * erroneous value when the unclamped value exceeds the size of a long. */ unclamped_millibels = 100.0f * 20.0f * log10f(fmaxf(volume, 0.0f)); unclamped_millibels = fmaxf(unclamped_millibels, SL_MILLIBEL_MIN); unclamped_millibels = fminf(unclamped_millibels, max_level); millibels = static_cast(lroundf(unclamped_millibels)); res = (*stm->volume)->SetVolumeLevel(stm->volume, millibels); if (res != SL_RESULT_SUCCESS) { return CUBEB_ERROR; } return CUBEB_OK; } struct cubeb_ops const opensl_ops = { .init = opensl_init, .get_backend_id = opensl_get_backend_id, .get_max_channel_count = opensl_get_max_channel_count, .get_min_latency = nullptr, .get_preferred_sample_rate = nullptr, .enumerate_devices = nullptr, .device_collection_destroy = nullptr, .destroy = opensl_destroy, .stream_init = opensl_stream_init, .stream_destroy = opensl_stream_destroy, .stream_start = opensl_stream_start, .stream_stop = opensl_stream_stop, .stream_get_position = opensl_stream_get_position, .stream_get_latency = opensl_stream_get_latency, .stream_get_input_latency = nullptr, .stream_set_volume = opensl_stream_set_volume, .stream_set_name = nullptr, .stream_get_current_device = nullptr, .stream_device_destroy = nullptr, .stream_register_device_changed_callback = nullptr, .register_device_collection_changed = nullptr};