a5daa17187
Test: touch *.cpp, mma, verify no clang-tidy warnings Test: SoundPoolAacTest, SoundPoolHapticTest, Test: SoundPoolMidiTest, SoundPoolOggTest Bug: 182414042 Change-Id: I77af4f0999c46931d6993d917443df1fa8425353
380 lines
12 KiB
C++
380 lines
12 KiB
C++
/*
|
|
* Copyright (C) 2008 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 <stdio.h>
|
|
|
|
//#define LOG_NDEBUG 0
|
|
#define LOG_TAG "SoundPool-JNI"
|
|
|
|
#include <utils/Log.h>
|
|
#include <jni.h>
|
|
#include <nativehelper/JNIPlatformHelp.h>
|
|
#include <nativehelper/ScopedUtfChars.h>
|
|
#include <android_runtime/AndroidRuntime.h>
|
|
#include "SoundPool.h"
|
|
|
|
using namespace android;
|
|
|
|
static struct fields_t {
|
|
jfieldID mNativeContext;
|
|
jmethodID mPostEvent;
|
|
jclass mSoundPoolClass;
|
|
} fields;
|
|
static inline SoundPool* MusterSoundPool(JNIEnv *env, jobject thiz) {
|
|
// NOLINTNEXTLINE(performance-no-int-to-ptr)
|
|
return reinterpret_cast<SoundPool*>(env->GetLongField(thiz, fields.mNativeContext));
|
|
}
|
|
static const char* const kAudioAttributesClassPathName = "android/media/AudioAttributes";
|
|
struct audio_attributes_fields_t {
|
|
jfieldID fieldUsage; // AudioAttributes.mUsage
|
|
jfieldID fieldContentType; // AudioAttributes.mContentType
|
|
jfieldID fieldFlags; // AudioAttributes.mFlags
|
|
jfieldID fieldFormattedTags;// AudioAttributes.mFormattedTags
|
|
};
|
|
static audio_attributes_fields_t javaAudioAttrFields;
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
static jint
|
|
android_media_SoundPool_load_FD(JNIEnv *env, jobject thiz, jobject fileDescriptor,
|
|
jlong offset, jlong length, jint priority)
|
|
{
|
|
ALOGV("android_media_SoundPool_load_FD");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return 0;
|
|
return (jint) ap->load(jniGetFDFromFileDescriptor(env, fileDescriptor),
|
|
int64_t(offset), int64_t(length), int(priority));
|
|
}
|
|
|
|
static jboolean
|
|
android_media_SoundPool_unload(JNIEnv *env, jobject thiz, jint sampleID) {
|
|
ALOGV("android_media_SoundPool_unload\n");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return JNI_FALSE;
|
|
return ap->unload(sampleID) ? JNI_TRUE : JNI_FALSE;
|
|
}
|
|
|
|
static jint
|
|
android_media_SoundPool_play(JNIEnv *env, jobject thiz, jint sampleID,
|
|
jfloat leftVolume, jfloat rightVolume, jint priority, jint loop,
|
|
jfloat rate)
|
|
{
|
|
ALOGV("android_media_SoundPool_play\n");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return 0;
|
|
return (jint) ap->play(sampleID, leftVolume, rightVolume, priority, loop, rate);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_pause(JNIEnv *env, jobject thiz, jint channelID)
|
|
{
|
|
ALOGV("android_media_SoundPool_pause");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->pause(channelID);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_resume(JNIEnv *env, jobject thiz, jint channelID)
|
|
{
|
|
ALOGV("android_media_SoundPool_resume");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->resume(channelID);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_autoPause(JNIEnv *env, jobject thiz)
|
|
{
|
|
ALOGV("android_media_SoundPool_autoPause");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->autoPause();
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_autoResume(JNIEnv *env, jobject thiz)
|
|
{
|
|
ALOGV("android_media_SoundPool_autoResume");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->autoResume();
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_stop(JNIEnv *env, jobject thiz, jint channelID)
|
|
{
|
|
ALOGV("android_media_SoundPool_stop");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->stop(channelID);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_setVolume(JNIEnv *env, jobject thiz, jint channelID,
|
|
jfloat leftVolume, jfloat rightVolume)
|
|
{
|
|
ALOGV("android_media_SoundPool_setVolume");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->setVolume(channelID, (float) leftVolume, (float) rightVolume);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_mute(JNIEnv *env, jobject thiz, jboolean muting)
|
|
{
|
|
ALOGV("android_media_SoundPool_mute(%d)", muting);
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->mute(muting == JNI_TRUE);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_setPriority(JNIEnv *env, jobject thiz, jint channelID,
|
|
jint priority)
|
|
{
|
|
ALOGV("android_media_SoundPool_setPriority");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->setPriority(channelID, (int) priority);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_setLoop(JNIEnv *env, jobject thiz, jint channelID,
|
|
int loop)
|
|
{
|
|
ALOGV("android_media_SoundPool_setLoop");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->setLoop(channelID, loop);
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_setRate(JNIEnv *env, jobject thiz, jint channelID,
|
|
jfloat rate)
|
|
{
|
|
ALOGV("android_media_SoundPool_setRate");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap == nullptr) return;
|
|
ap->setRate(channelID, (float) rate);
|
|
}
|
|
|
|
static void android_media_callback(SoundPoolEvent event, SoundPool* soundPool, void* user)
|
|
{
|
|
ALOGV("callback: (%d, %d, %d, %p, %p)", event.mMsg, event.mArg1, event.mArg2, soundPool, user);
|
|
JNIEnv *env = AndroidRuntime::getJNIEnv();
|
|
env->CallStaticVoidMethod(
|
|
fields.mSoundPoolClass, fields.mPostEvent, user, event.mMsg, event.mArg1, event.mArg2,
|
|
nullptr /* object */);
|
|
}
|
|
|
|
static jint
|
|
android_media_SoundPool_native_setup(JNIEnv *env, jobject thiz, jobject weakRef,
|
|
jint maxChannels, jobject jaa, jstring opPackageName)
|
|
{
|
|
if (jaa == nullptr) {
|
|
ALOGE("Error creating SoundPool: invalid audio attributes");
|
|
return -1;
|
|
}
|
|
|
|
audio_attributes_t *paa = nullptr;
|
|
// read the AudioAttributes values
|
|
paa = (audio_attributes_t *) calloc(1, sizeof(audio_attributes_t));
|
|
const auto jtags =
|
|
(jstring) env->GetObjectField(jaa, javaAudioAttrFields.fieldFormattedTags);
|
|
const char* tags = env->GetStringUTFChars(jtags, nullptr);
|
|
// copying array size -1, char array for tags was calloc'd, no need to NULL-terminate it
|
|
strncpy(paa->tags, tags, AUDIO_ATTRIBUTES_TAGS_MAX_SIZE - 1);
|
|
env->ReleaseStringUTFChars(jtags, tags);
|
|
paa->usage = (audio_usage_t) env->GetIntField(jaa, javaAudioAttrFields.fieldUsage);
|
|
paa->content_type =
|
|
(audio_content_type_t) env->GetIntField(jaa, javaAudioAttrFields.fieldContentType);
|
|
paa->flags = (audio_flags_mask_t) env->GetIntField(jaa, javaAudioAttrFields.fieldFlags);
|
|
|
|
ALOGV("android_media_SoundPool_native_setup");
|
|
ScopedUtfChars opPackageNameStr(env, opPackageName);
|
|
auto *ap = new SoundPool(maxChannels, paa, opPackageNameStr.c_str());
|
|
if (ap == nullptr) {
|
|
return -1;
|
|
}
|
|
|
|
// save pointer to SoundPool C++ object in opaque field in Java object
|
|
env->SetLongField(thiz, fields.mNativeContext, (jlong) ap);
|
|
|
|
// set callback with weak reference
|
|
jobject globalWeakRef = env->NewGlobalRef(weakRef);
|
|
ap->setCallback(android_media_callback, globalWeakRef);
|
|
|
|
// audio attributes were copied in SoundPool creation
|
|
free(paa);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
android_media_SoundPool_release(JNIEnv *env, jobject thiz)
|
|
{
|
|
ALOGV("android_media_SoundPool_release");
|
|
SoundPool *ap = MusterSoundPool(env, thiz);
|
|
if (ap != nullptr) {
|
|
|
|
// release weak reference and clear callback
|
|
auto weakRef = (jobject) ap->getUserData();
|
|
ap->setCallback(nullptr /* callback */, nullptr /* user */);
|
|
if (weakRef != nullptr) {
|
|
env->DeleteGlobalRef(weakRef);
|
|
}
|
|
|
|
// clear native context
|
|
env->SetLongField(thiz, fields.mNativeContext, 0);
|
|
delete ap;
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Dalvik VM type signatures
|
|
static JNINativeMethod gMethods[] = {
|
|
{ "_load",
|
|
"(Ljava/io/FileDescriptor;JJI)I",
|
|
(void *)android_media_SoundPool_load_FD
|
|
},
|
|
{ "unload",
|
|
"(I)Z",
|
|
(void *)android_media_SoundPool_unload
|
|
},
|
|
{ "_play",
|
|
"(IFFIIF)I",
|
|
(void *)android_media_SoundPool_play
|
|
},
|
|
{ "pause",
|
|
"(I)V",
|
|
(void *)android_media_SoundPool_pause
|
|
},
|
|
{ "resume",
|
|
"(I)V",
|
|
(void *)android_media_SoundPool_resume
|
|
},
|
|
{ "autoPause",
|
|
"()V",
|
|
(void *)android_media_SoundPool_autoPause
|
|
},
|
|
{ "autoResume",
|
|
"()V",
|
|
(void *)android_media_SoundPool_autoResume
|
|
},
|
|
{ "stop",
|
|
"(I)V",
|
|
(void *)android_media_SoundPool_stop
|
|
},
|
|
{ "_setVolume",
|
|
"(IFF)V",
|
|
(void *)android_media_SoundPool_setVolume
|
|
},
|
|
{ "_mute",
|
|
"(Z)V",
|
|
(void *)android_media_SoundPool_mute
|
|
},
|
|
{ "setPriority",
|
|
"(II)V",
|
|
(void *)android_media_SoundPool_setPriority
|
|
},
|
|
{ "setLoop",
|
|
"(II)V",
|
|
(void *)android_media_SoundPool_setLoop
|
|
},
|
|
{ "setRate",
|
|
"(IF)V",
|
|
(void *)android_media_SoundPool_setRate
|
|
},
|
|
{ "native_setup",
|
|
"(Ljava/lang/Object;ILjava/lang/Object;Ljava/lang/String;)I",
|
|
(void*)android_media_SoundPool_native_setup
|
|
},
|
|
{ "native_release",
|
|
"()V",
|
|
(void*)android_media_SoundPool_release
|
|
}
|
|
};
|
|
|
|
static const char* const kClassPathName = "android/media/SoundPool";
|
|
|
|
jint JNI_OnLoad(JavaVM* vm, void* /* reserved */)
|
|
{
|
|
JNIEnv* env = nullptr;
|
|
jint result = -1;
|
|
jclass clazz;
|
|
|
|
if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {
|
|
ALOGE("ERROR: GetEnv failed\n");
|
|
return result;
|
|
}
|
|
assert(env != nullptr);
|
|
|
|
clazz = env->FindClass(kClassPathName);
|
|
if (clazz == nullptr) {
|
|
ALOGE("Can't find %s", kClassPathName);
|
|
return result;
|
|
}
|
|
|
|
fields.mNativeContext = env->GetFieldID(clazz, "mNativeContext", "J");
|
|
if (fields.mNativeContext == nullptr) {
|
|
ALOGE("Can't find SoundPool.mNativeContext");
|
|
return result;
|
|
}
|
|
|
|
fields.mPostEvent = env->GetStaticMethodID(clazz, "postEventFromNative",
|
|
"(Ljava/lang/Object;IIILjava/lang/Object;)V");
|
|
if (fields.mPostEvent == nullptr) {
|
|
ALOGE("Can't find android/media/SoundPool.postEventFromNative");
|
|
return result;
|
|
}
|
|
|
|
// create a reference to class. Technically, we're leaking this reference
|
|
// since it's a static object.
|
|
fields.mSoundPoolClass = (jclass) env->NewGlobalRef(clazz);
|
|
|
|
if (AndroidRuntime::registerNativeMethods(
|
|
env, kClassPathName, gMethods, NELEM(gMethods)) < 0) {
|
|
return result;
|
|
}
|
|
|
|
// Get the AudioAttributes class and fields
|
|
jclass audioAttrClass = env->FindClass(kAudioAttributesClassPathName);
|
|
if (audioAttrClass == nullptr) {
|
|
ALOGE("Can't find %s", kAudioAttributesClassPathName);
|
|
return result;
|
|
}
|
|
auto audioAttributesClassRef = (jclass)env->NewGlobalRef(audioAttrClass);
|
|
javaAudioAttrFields.fieldUsage = env->GetFieldID(audioAttributesClassRef, "mUsage", "I");
|
|
javaAudioAttrFields.fieldContentType
|
|
= env->GetFieldID(audioAttributesClassRef, "mContentType", "I");
|
|
javaAudioAttrFields.fieldFlags = env->GetFieldID(audioAttributesClassRef, "mFlags", "I");
|
|
javaAudioAttrFields.fieldFormattedTags =
|
|
env->GetFieldID(audioAttributesClassRef, "mFormattedTags", "Ljava/lang/String;");
|
|
env->DeleteGlobalRef(audioAttributesClassRef);
|
|
if (javaAudioAttrFields.fieldUsage == nullptr
|
|
|| javaAudioAttrFields.fieldContentType == nullptr
|
|
|| javaAudioAttrFields.fieldFlags == nullptr
|
|
|| javaAudioAttrFields.fieldFormattedTags == nullptr) {
|
|
ALOGE("Can't initialize AudioAttributes fields");
|
|
return result;
|
|
}
|
|
|
|
/* success -- return valid version number */
|
|
return JNI_VERSION_1_4;
|
|
}
|