/*
* Copyright (C) 2015 The Android Open Source Project
*
* 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.
*/
#include "jni_native.h"
#include <stdlib.h>
#include <android/log.h>
#include "loopback.h"
#define LOG_TAG "jni_native"
static int nativeEngineFromThreadType(int threadType) {
switch (threadType) {
case AUDIO_THREAD_TYPE_NATIVE_SLES: return NATIVE_ENGINE_SLES;
case AUDIO_THREAD_TYPE_NATIVE_AAUDIO: return NATIVE_ENGINE_AAUDIO;
}
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG,
"unsupported thread type %d", threadType);
return -1;
}
JNIEXPORT jobject JNICALL
Java_org_drrickorang_loopback_NativeAudioThread_nativeComputeDefaultSettings
(JNIEnv *env, jobject obj __unused, jint bytesPerFrame, jint threadType, jint performanceMode) {
int engine = nativeEngineFromThreadType(threadType);
if (engine == -1) return NULL;
int samplingRate, playerBufferFrameCount, recorderBufferFrameCount;
if (sEngines[engine].computeDefaultSettings(performanceMode, &samplingRate,
&playerBufferFrameCount, &recorderBufferFrameCount) == STATUS_SUCCESS) {
jclass cls = (*env)->FindClass(env, "org/drrickorang/loopback/TestSettings");
jmethodID methodID = (*env)->GetMethodID(env, cls, "<init>", "(III)V");
jobject testSettings = (*env)->NewObject(env, cls, methodID,
samplingRate,
playerBufferFrameCount * bytesPerFrame,
recorderBufferFrameCount * bytesPerFrame);
return testSettings;
} else {
return NULL;
}
}
JNIEXPORT jlong JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeInit
(JNIEnv *env, jobject obj __unused, jint threadType, jint samplingRate, jint frameCount,
jint micSource, jint performanceMode,
jint testType, jdouble frequency1, jobject byteBuffer, jshortArray loopbackTone,
jint maxRecordedLateCallbacks, jint ignoreFirstFrames) {
int engine = nativeEngineFromThreadType(threadType);
if (engine == -1) return 0;
native_engine_instance_t *pInstance =
(native_engine_instance_t*) malloc(sizeof(native_engine_instance_t));
if (pInstance == NULL) {
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG,
"failed to allocate a native engine instance");
return 0;
}
void *pContext = NULL;
char *byteBufferPtr = (*env)->GetDirectBufferAddress(env, byteBuffer);
int byteBufferLength = (*env)->GetDirectBufferCapacity(env, byteBuffer);
short *loopbackToneArray = (*env)->GetShortArrayElements(env, loopbackTone, 0);
if (sEngines[engine].init(&pContext, samplingRate, frameCount, micSource,
performanceMode,
testType, frequency1, byteBufferPtr, byteBufferLength,
loopbackToneArray, maxRecordedLateCallbacks, ignoreFirstFrames) != STATUS_FAIL) {
pInstance->context = pContext;
pInstance->methods = &sEngines[engine];
return (long) pInstance;
}
free(pInstance);
return 0;
}
JNIEXPORT jint JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeProcessNext
(JNIEnv *env __unused, jobject obj __unused, jlong handle, jdoubleArray samplesArray,
jlong offset) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
long maxSamples = (*env)->GetArrayLength(env, samplesArray);
double *pSamples = (*env)->GetDoubleArrayElements(env, samplesArray, 0);
long availableSamples = maxSamples-offset;
double *pCurrentSample = pSamples+offset;
__android_log_print(ANDROID_LOG_INFO, LOG_TAG,
"jni nativeProcessNext currentSample %p, availableSamples %ld ",
pCurrentSample, availableSamples);
int samplesRead = pInstance->methods->processNext(
pInstance->context, pCurrentSample, availableSamples);
return samplesRead;
}
JNIEXPORT jint JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeDestroy
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int status = pInstance->methods->destroy(&pInstance->context);
free(pInstance);
return status;
}
JNIEXPORT jintArray JNICALL
Java_org_drrickorang_loopback_NativeAudioThread_nativeGetRecorderBufferPeriod
(JNIEnv *env, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int* recorderBufferPeriod = pInstance->methods->getRecorderBufferPeriod(
pInstance->context);
// get the length = RANGE
jintArray result = (*env)->NewIntArray(env, RANGE);
(*env)->SetIntArrayRegion(env, result, 0, RANGE, recorderBufferPeriod);
return result;
}
JNIEXPORT jint JNICALL
Java_org_drrickorang_loopback_NativeAudioThread_nativeGetRecorderMaxBufferPeriod
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int recorderMaxBufferPeriod = pInstance->methods->getRecorderMaxBufferPeriod(
pInstance->context);
return recorderMaxBufferPeriod;
}
JNIEXPORT jdouble JNICALL
Java_org_drrickorang_loopback_NativeAudioThread_nativeGetRecorderVarianceBufferPeriod
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int64_t result = pInstance->methods->getRecorderVarianceBufferPeriod(pInstance->context);
// variance has units ns^2 so we have to square the conversion factor
double scaled = (double) result / ((double) NANOS_PER_MILLI * (double) NANOS_PER_MILLI);
return scaled;
}
JNIEXPORT jintArray
JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeGetPlayerBufferPeriod
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int* playerBufferPeriod = pInstance->methods->getPlayerBufferPeriod(pInstance->context);
jintArray result = (*env)->NewIntArray(env, RANGE);
(*env)->SetIntArrayRegion(env, result, 0, RANGE, playerBufferPeriod);
return result;
}
JNIEXPORT jint JNICALL
Java_org_drrickorang_loopback_NativeAudioThread_nativeGetPlayerMaxBufferPeriod
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int playerMaxBufferPeriod = pInstance->methods->getPlayerMaxBufferPeriod(pInstance->context);
return playerMaxBufferPeriod;
}
JNIEXPORT jdouble JNICALL
Java_org_drrickorang_loopback_NativeAudioThread_nativeGetPlayerVarianceBufferPeriod
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
int64_t result = pInstance->methods->getPlayerVarianceBufferPeriod(pInstance->context);
// variance has units ns^2 so we have to square the conversion factor
double scaled = (double) result / ((double) NANOS_PER_MILLI * (double) NANOS_PER_MILLI);
return scaled;
}
jobject getCallbackTimes(JNIEnv *env, callbackTimeStamps *callbacks, short expectedBufferPeriod){
jintArray timeStamps = (*env)->NewIntArray(env, callbacks->index);
(*env)->SetIntArrayRegion(env, timeStamps, 0, callbacks->index, callbacks->timeStampsMs);
jshortArray callbackLengths = (*env)->NewShortArray(env, callbacks->index);
(*env)->SetShortArrayRegion(env, callbackLengths, 0, callbacks->index,
callbacks->callbackDurations);
jclass cls = (*env)->FindClass(env, "org/drrickorang/loopback/BufferCallbackTimes");
jmethodID methodID = (*env)->GetMethodID(env, cls, "<init>", "([I[SZS)V");
jobject callbackTimes=(*env)->NewObject(env,cls, methodID, timeStamps, callbackLengths,
callbacks->exceededCapacity, expectedBufferPeriod);
return callbackTimes;
}
JNIEXPORT jobject
JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeGetPlayerCallbackTimeStamps
(JNIEnv *env, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
callbackTimeStamps *pTSs;
int expectedBufferPeriod = pInstance->methods->getPlayerTimeStampsAndExpectedBufferPeriod(
pInstance->context, &pTSs);
return getCallbackTimes(env, pTSs, expectedBufferPeriod);
}
JNIEXPORT jobject
JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeGetRecorderCallbackTimeStamps
(JNIEnv *env, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
callbackTimeStamps *pTSs;
int expectedBufferPeriod = pInstance->methods->getRecorderTimeStampsAndExpectedBufferPeriod(
pInstance->context, &pTSs);
return getCallbackTimes(env, pTSs, expectedBufferPeriod);
}
JNIEXPORT jint
JNICALL Java_org_drrickorang_loopback_NativeAudioThread_nativeGetCaptureRank
(JNIEnv *env __unused, jobject obj __unused, jlong handle) {
native_engine_instance_t *pInstance = (native_engine_instance_t*) handle;
return pInstance->methods->getCaptureRank(pInstance->context);
}