From 564aae4e2d4aff797a66ec15184db94413ee6f50 Mon Sep 17 00:00:00 2001 From: Stefano D'Angelo Date: Tue, 23 Jan 2024 18:38:52 +0100 Subject: [PATCH] android jni progress --- templates/android-make/Makefile | 2 +- templates/android-make/vars.mk | 3 + templates/android/src/data.h | 22 ++++++++ templates/android/src/jni.cpp | 93 ++++++++++++++++++++++++++++++- templates/web-demo/src/index.html | 13 +++-- test/android-make.json | 1 + test/plugin.h | 2 +- test/run.sh | 1 + 8 files changed, 126 insertions(+), 11 deletions(-) diff --git a/templates/android-make/Makefile b/templates/android-make/Makefile index 4c7d736..4ef4f51 100644 --- a/templates/android-make/Makefile +++ b/templates/android-make/Makefile @@ -68,7 +68,7 @@ build/obj/${CLASSES_PATH}/MainActivity.class: src/MainActivity.java | build/obj ${JAVAC} -classpath "$(subst $() $(),:,$(JARS))" -d build/obj $^ build/apk/lib/armeabi-v7a/lib${BUNDLE_NAME}.so: src/jni.cpp | build/apk/lib/armeabi-v7a - ${CXX} $^ ${CXXFLAGS} ${LDFLAGS} -o $@ + ${CXX} $^ ${CXXFLAGS} ${CXXFLAGS_EXTRA} ${LDFLAGS} ${LDFLAGS_EXTRA} -o $@ build/assets/index.html: src/index.html | build/assets cp $^ $@ diff --git a/templates/android-make/vars.mk b/templates/android-make/vars.mk index 42ec826..baf4e27 100644 --- a/templates/android-make/vars.mk +++ b/templates/android-make/vars.mk @@ -1,6 +1,9 @@ BUNDLE_NAME := {{=it.product.bundleName}} JAVA_PACKAGE_NAME := {{=it.android.javaPackageName}} +CXXFLAGS_EXTRA := {{=it.make && it.make.cxxflags ? it.make.cxxflags : ""}} {{=it.android_make && it.android_make.cxxflags ? it.android_make.cxxflags : ""}} +LDFLAGS_EXTRA := {{=it.make && it.make.ldflags ? it.make.ldflags : ""}} {{=it.android_make && it.android_make.ldflags ? it.android_make.ldflags : ""}} + KEY_STORE := {{=it.android_make.keyStore}} KEY_ALIAS := {{=it.android_make.keyAlias}} STORE_PASS := {{=it.android_make.storePass}} diff --git a/templates/android/src/data.h b/templates/android/src/data.h index af5aba5..d58fb19 100644 --- a/templates/android/src/data.h +++ b/templates/android/src/data.h @@ -1 +1,23 @@ +#define AUDIO_BUS_IN {{=it.product.buses.findIndex(x => x.type == "audio" && x.direction == "input" && !x.cv && !x.sidechain)}} +#define AUDIO_BUS_OUT {{=it.product.buses.findIndex(x => x.type == "audio" && x.direction == "output" && !x.cv && !x.sidechain)}} + +#define NUM_CHANNELS_IN {{=it.product.buses.filter(x => x.type == "audio" && x.direction == "input" && !x.cv && !x.sidechain) ? (it.product.buses.filter(x => x.type == "audio" && x.direction == "input" && !x.cv && !x.sidechain)[0].channels == "mono" ? 1 : 2) : 0}} +#define NUM_CHANNELS_OUT {{=it.product.buses.filter(x => x.type == "audio" && x.direction == "output" && !x.cv && !x.sidechain) ? (it.product.buses.filter(x => x.type == "audio" && x.direction == "output" && !x.cv && !x.sidechain)[0].channels == "mono" ? 1 : 2) : 0}} + +#define PARAMETERS_N {{=it.product.parameters.length}} + +#if PARAMETERS_N > 0 +static struct { + char out; + float def; +} param_data[PARAMETERS_N] = { +{{~it.product.parameters :p}} + { + /* .out = */ {{=p.direction == "output" ? 1 : 0}}, + /* .def = */ {{=p.defaultValue.toExponential()}} + }, +{{~}} +}; +#endif + #define JNI_FUNC(x) Java_{{=it.android.javaPackageName.replaceAll("_", "_1").replaceAll(".", "_")}}_MainActivity_##x diff --git a/templates/android/src/jni.cpp b/templates/android/src/jni.cpp index 690ad33..f46621f 100644 --- a/templates/android/src/jni.cpp +++ b/templates/android/src/jni.cpp @@ -6,10 +6,7 @@ #include #include #include -*/ #include -#include "data.h" -/* #define MINIAUDIO_IMPLEMENTATION #define MA_ENABLE_ONLY_SPECIFIC_BACKENDS #define MA_ENABLE_AAUDIO @@ -266,15 +263,105 @@ Java_com_orastron_@JNI_NAME@_MainActivity_removeMidiPort(JNIEnv* env, jobject th #endif */ +#include +#include + +#include "data.h" +#include "plugin.h" + +#include +#if NUM_CHANNELS_IN + NUM_CHANNELS_OUT > 0 +# define MINIAUDIO_IMPLEMENTATION +# define MA_ENABLE_ONLY_SPECIFIC_BACKENDS +# define MA_ENABLE_AAUDIO +# include + +# define BLOCK_SIZE 32 +# define NUM_BUFS (NUM_CHANNELS_IN > NUM_CHANNELS_OUT ? NUM_CHANNELS_IN : NUM_CHANNELS_OUT) + +static ma_device device; +#endif +static plugin instance; +static void * mem; + +static void data_callback(ma_device* pDevice, void* pOutput, const void* pInput, ma_uint32 frameCount) { +} + extern "C" JNIEXPORT jboolean JNICALL JNI_FUNC(nativeAudioStart)(JNIEnv* env, jobject thiz) { + (void)env; + (void)thiz; + +#if NUM_CHANNELS_IN + NUM_CHANNELS_OUT > 0 +# if NUM_CHANNELS_IN == 0 + ma_device_config deviceConfig = ma_device_config_init(ma_device_type_playback); +# elif NUM_CHANNELS_OUT == 0 + ma_device_config deviceConfig = ma_device_config_init(ma_device_type_capture); +# else + ma_device_config deviceConfig = ma_device_config_init(ma_device_type_duplex); +# endif + deviceConfig.periodSizeInFrames = BLOCK_SIZE; + deviceConfig.periods = 1; + deviceConfig.performanceProfile = ma_performance_profile_low_latency; + deviceConfig.noPreSilencedOutputBuffer = 1; + deviceConfig.noClip = 0; + deviceConfig.noDisableDenormals = 0; + deviceConfig.noFixedSizedCallback = 1; + deviceConfig.dataCallback = data_callback; + deviceConfig.capture.pDeviceID = NULL; + deviceConfig.capture.format = ma_format_f32; + deviceConfig.capture.channels = NUM_CHANNELS_IN; + deviceConfig.capture.shareMode = ma_share_mode_shared; + deviceConfig.playback.pDeviceID = NULL; + deviceConfig.playback.format = ma_format_f32; + deviceConfig.playback.channels = NUM_CHANNELS_OUT; + deviceConfig.playback.shareMode = ma_share_mode_shared; + + if (ma_device_init(NULL, &deviceConfig, &device) != MA_SUCCESS) + return false; +#endif + + plugin_init(&instance); + +#if PARAMETERS_N > 0 + for (size_t i = 0; i < PARAMETERS_N; i++) + if (!param_data[i].out) + plugin_set_parameter(&instance, i, param_data[i].def); +#endif + + plugin_set_sample_rate(&instance, (float)device.sampleRate); + size_t req = plugin_mem_req(&instance); + if (req != 0) { + mem = malloc(req); + if (mem == NULL) { + plugin_fini(&instance); +#if NUM_CHANNELS_IN + NUM_CHANNELS_OUT > 0 + ma_device_uninit(&device); +#endif + return false; + } + plugin_mem_set(&instance, mem); + } else + mem = NULL; + + plugin_reset(&instance); + return true; } extern "C" JNIEXPORT void JNICALL JNI_FUNC(nativeAudioStop)(JNIEnv* env, jobject thiz) { + (void)env; + (void)thiz; + + plugin_fini(&instance); + +#if NUM_CHANNELS_IN + NUM_CHANNELS_OUT > 0 + ma_device_stop(&device); + ma_device_uninit(&device); +#endif } extern "C" diff --git a/templates/web-demo/src/index.html b/templates/web-demo/src/index.html index cc01d37..20e0ab0 100644 --- a/templates/web-demo/src/index.html +++ b/templates/web-demo/src/index.html @@ -9,7 +9,7 @@ window.demo = demo;