1 /*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <stdio.h>
18 #include <unordered_set>
19
20 //#define LOG_NDEBUG 0
21 #define LOG_TAG "visualizers-JNI"
22
23 #include <utils/Log.h>
24 #include <jni.h>
25 #include <nativehelper/JNIHelp.h>
26 #include <android_runtime/AndroidRuntime.h>
27 #include <utils/threads.h>
28 #include "Visualizer.h"
29
30 #include <nativehelper/ScopedUtfChars.h>
31
32 #include <android/content/AttributionSourceState.h>
33 #include <android_os_Parcel.h>
34
35 using namespace android;
36
37 using content::AttributionSourceState;
38
39 #define VISUALIZER_SUCCESS 0
40 #define VISUALIZER_ERROR (-1)
41 #define VISUALIZER_ERROR_ALREADY_EXISTS (-2)
42 #define VISUALIZER_ERROR_NO_INIT (-3)
43 #define VISUALIZER_ERROR_BAD_VALUE (-4)
44 #define VISUALIZER_ERROR_INVALID_OPERATION (-5)
45 #define VISUALIZER_ERROR_NO_MEMORY (-6)
46 #define VISUALIZER_ERROR_DEAD_OBJECT (-7)
47
48 #define NATIVE_EVENT_PCM_CAPTURE 0
49 #define NATIVE_EVENT_FFT_CAPTURE 1
50 #define NATIVE_EVENT_SERVER_DIED 2
51
52 // ----------------------------------------------------------------------------
53 static const char* const kClassPathName = "android/media/audiofx/Visualizer";
54 static const char* const kClassPeakRmsPathName =
55 "android/media/audiofx/Visualizer$MeasurementPeakRms";
56
57 struct fields_t {
58 // these fields provide access from C++ to the...
59 jclass clazzEffect; // Visualizer class
60 jmethodID midPostNativeEvent; // event post callback method
61 jfieldID fidNativeVisualizer; // stores in Java the native Visualizer object
62 jfieldID fidJniData; // stores in Java additional resources used by the native Visualizer
63 jfieldID fidPeak; // to access Visualizer.MeasurementPeakRms.mPeak
64 jfieldID fidRms; // to access Visualizer.MeasurementPeakRms.mRms
65 };
66 static fields_t fields;
67
68 struct visualizer_callback_cookie {
69 jclass visualizer_class; // Visualizer class
70 jobject visualizer_ref; // Visualizer object instance
71
72 // 'busy_count' and 'cond' together with 'sLock' are used to serialize
73 // concurrent access to the callback cookie from 'setup'/'release'
74 // and the callback.
75 int busy_count;
76 Condition cond;
77
78 // Lazily allocated arrays used to hold callback data provided to java
79 // applications. These arrays are allocated during the first callback and
80 // reallocated when the size of the callback data changes. Allocating on
81 // demand and saving the arrays means that applications cannot safely hold a
82 // reference to the provided data (they need to make a copy if they want to
83 // hold onto outside of the callback scope), but it avoids GC thrash caused
84 // by constantly allocating and releasing arrays to hold callback data.
85 // 'callback_data_lock' must never be held at the same time with 'sLock'.
86 Mutex callback_data_lock;
87 jbyteArray waveform_data;
88 jbyteArray fft_data;
89
90 // Assumes use of default initialization by the client.
91
~visualizer_callback_cookievisualizer_callback_cookie92 ~visualizer_callback_cookie() {
93 cleanupBuffers();
94 }
95
cleanupBuffersvisualizer_callback_cookie96 void cleanupBuffers() {
97 AutoMutex lock(&callback_data_lock);
98 if (waveform_data || fft_data) {
99 JNIEnv *env = AndroidRuntime::getJNIEnv();
100
101 if (waveform_data) {
102 env->DeleteGlobalRef(waveform_data);
103 waveform_data = NULL;
104 }
105
106 if (fft_data) {
107 env->DeleteGlobalRef(fft_data);
108 fft_data = NULL;
109 }
110 }
111 }
112 };
113
114 // ----------------------------------------------------------------------------
115 struct VisualizerJniStorage {
116 visualizer_callback_cookie mCallbackData{};
117 };
118
119
translateError(int code)120 static jint translateError(int code) {
121 switch(code) {
122 case NO_ERROR:
123 return VISUALIZER_SUCCESS;
124 case ALREADY_EXISTS:
125 return VISUALIZER_ERROR_ALREADY_EXISTS;
126 case NO_INIT:
127 return VISUALIZER_ERROR_NO_INIT;
128 case BAD_VALUE:
129 return VISUALIZER_ERROR_BAD_VALUE;
130 case INVALID_OPERATION:
131 return VISUALIZER_ERROR_INVALID_OPERATION;
132 case NO_MEMORY:
133 return VISUALIZER_ERROR_NO_MEMORY;
134 case DEAD_OBJECT:
135 return VISUALIZER_ERROR_DEAD_OBJECT;
136 default:
137 return VISUALIZER_ERROR;
138 }
139 }
140
141 static Mutex sLock;
142 static std::unordered_set<visualizer_callback_cookie*> sVisualizerCallBackCookies;
143
144 // ----------------------------------------------------------------------------
ensureArraySize(JNIEnv * env,jbyteArray * array,uint32_t size)145 static void ensureArraySize(JNIEnv *env, jbyteArray *array, uint32_t size) {
146 if (NULL != *array) {
147 uint32_t len = env->GetArrayLength(*array);
148 if (len == size)
149 return;
150
151 env->DeleteGlobalRef(*array);
152 *array = NULL;
153 }
154
155 jbyteArray localRef = env->NewByteArray(size);
156 if (NULL != localRef) {
157 // Promote to global ref.
158 *array = (jbyteArray)env->NewGlobalRef(localRef);
159
160 // Release our (now pointless) local ref.
161 env->DeleteLocalRef(localRef);
162 }
163 }
164
captureCallback(void * user,uint32_t waveformSize,uint8_t * waveform,uint32_t fftSize,uint8_t * fft,uint32_t samplingrate)165 static void captureCallback(void* user,
166 uint32_t waveformSize,
167 uint8_t *waveform,
168 uint32_t fftSize,
169 uint8_t *fft,
170 uint32_t samplingrate) {
171
172 visualizer_callback_cookie *callbackInfo = (visualizer_callback_cookie *)user;
173 JNIEnv *env = AndroidRuntime::getJNIEnv();
174
175 if (!user || !env) {
176 ALOGW("captureCallback error user %p, env %p", user, env);
177 return;
178 }
179
180 {
181 Mutex::Autolock l(sLock);
182 if (sVisualizerCallBackCookies.count(callbackInfo) == 0) {
183 return;
184 }
185 callbackInfo->busy_count++;
186 }
187 ALOGV("captureCallback: callbackInfo %p, visualizer_ref %p visualizer_class %p",
188 callbackInfo,
189 callbackInfo->visualizer_ref,
190 callbackInfo->visualizer_class);
191
192 {
193 AutoMutex lock(&callbackInfo->callback_data_lock);
194
195 if (waveformSize != 0 && waveform != NULL) {
196 jbyteArray jArray;
197
198 ensureArraySize(env, &callbackInfo->waveform_data, waveformSize);
199 jArray = callbackInfo->waveform_data;
200
201 if (jArray != NULL) {
202 jbyte *nArray = env->GetByteArrayElements(jArray, NULL);
203 memcpy(nArray, waveform, waveformSize);
204 env->ReleaseByteArrayElements(jArray, nArray, 0);
205 env->CallStaticVoidMethod(
206 callbackInfo->visualizer_class,
207 fields.midPostNativeEvent,
208 callbackInfo->visualizer_ref,
209 NATIVE_EVENT_PCM_CAPTURE,
210 samplingrate,
211 jArray);
212 }
213 }
214
215 if (fftSize != 0 && fft != NULL) {
216 jbyteArray jArray;
217
218 ensureArraySize(env, &callbackInfo->fft_data, fftSize);
219 jArray = callbackInfo->fft_data;
220
221 if (jArray != NULL) {
222 jbyte *nArray = env->GetByteArrayElements(jArray, NULL);
223 memcpy(nArray, fft, fftSize);
224 env->ReleaseByteArrayElements(jArray, nArray, 0);
225 env->CallStaticVoidMethod(
226 callbackInfo->visualizer_class,
227 fields.midPostNativeEvent,
228 callbackInfo->visualizer_ref,
229 NATIVE_EVENT_FFT_CAPTURE,
230 samplingrate,
231 jArray);
232 }
233 }
234 } // callback_data_lock scope
235
236 if (env->ExceptionCheck()) {
237 env->ExceptionDescribe();
238 env->ExceptionClear();
239 }
240 {
241 Mutex::Autolock l(sLock);
242 callbackInfo->busy_count--;
243 callbackInfo->cond.broadcast();
244 }
245 }
246
247 // ----------------------------------------------------------------------------
248
getVisualizer(JNIEnv * env,jobject thiz)249 static sp<Visualizer> getVisualizer(JNIEnv* env, jobject thiz)
250 {
251 Mutex::Autolock l(sLock);
252 Visualizer* const v =
253 (Visualizer*)env->GetLongField(thiz, fields.fidNativeVisualizer);
254 return sp<Visualizer>::fromExisting(v);
255 }
256
setVisualizer(JNIEnv * env,jobject thiz,const sp<Visualizer> & v)257 static sp<Visualizer> setVisualizer(JNIEnv* env, jobject thiz,
258 const sp<Visualizer>& v)
259 {
260 Mutex::Autolock l(sLock);
261 sp<Visualizer> old = sp<Visualizer>::fromExisting(
262 (Visualizer*)env->GetLongField(thiz, fields.fidNativeVisualizer));
263 if (v.get()) {
264 v->incStrong((void*)setVisualizer);
265 }
266 if (old != 0) {
267 old->decStrong((void*)setVisualizer);
268 }
269 env->SetLongField(thiz, fields.fidNativeVisualizer, (jlong)v.get());
270 return old;
271 }
272
273 // ----------------------------------------------------------------------------
274 // This function gets some field IDs, which in turn causes class initialization.
275 // It is called from a static block in Visualizer, which won't run until the
276 // first time an instance of this class is used.
277 static void
android_media_visualizer_native_init(JNIEnv * env)278 android_media_visualizer_native_init(JNIEnv *env)
279 {
280
281 ALOGV("android_media_visualizer_native_init");
282
283 fields.clazzEffect = NULL;
284
285 // Get the Visualizer class
286 jclass clazz = env->FindClass(kClassPathName);
287 if (clazz == NULL) {
288 ALOGE("Can't find %s", kClassPathName);
289 return;
290 }
291
292 fields.clazzEffect = (jclass)env->NewGlobalRef(clazz);
293
294 // Get the Visualizer.MeasurementPeakRms class
295 clazz = env->FindClass(kClassPeakRmsPathName);
296 if (clazz == NULL) {
297 ALOGE("Can't find %s", kClassPeakRmsPathName);
298 return;
299 }
300 jclass clazzMeasurementPeakRms = (jclass)env->NewGlobalRef(clazz);
301
302 // Get the postEvent method
303 fields.midPostNativeEvent = env->GetStaticMethodID(
304 fields.clazzEffect,
305 "postEventFromNative", "(Ljava/lang/Object;II[B)V");
306 if (fields.midPostNativeEvent == NULL) {
307 ALOGE("Can't find Visualizer.%s", "postEventFromNative");
308 return;
309 }
310
311 // Get the variables fields
312 // nativeTrackInJavaObj
313 fields.fidNativeVisualizer = env->GetFieldID(
314 fields.clazzEffect,
315 "mNativeVisualizer", "J");
316 if (fields.fidNativeVisualizer == NULL) {
317 ALOGE("Can't find Visualizer.%s", "mNativeVisualizer");
318 return;
319 }
320 // fidJniData;
321 fields.fidJniData = env->GetFieldID(
322 fields.clazzEffect,
323 "mJniData", "J");
324 if (fields.fidJniData == NULL) {
325 ALOGE("Can't find Visualizer.%s", "mJniData");
326 return;
327 }
328 // fidPeak
329 fields.fidPeak = env->GetFieldID(
330 clazzMeasurementPeakRms,
331 "mPeak", "I");
332 if (fields.fidPeak == NULL) {
333 ALOGE("Can't find Visualizer.MeasurementPeakRms.%s", "mPeak");
334 return;
335 }
336 // fidRms
337 fields.fidRms = env->GetFieldID(
338 clazzMeasurementPeakRms,
339 "mRms", "I");
340 if (fields.fidRms == NULL) {
341 ALOGE("Can't find Visualizer.MeasurementPeakRms.%s", "mPeak");
342 return;
343 }
344
345 env->DeleteGlobalRef(clazzMeasurementPeakRms);
346 }
347
android_media_visualizer_effect_callback(int32_t event,void * user,void * info)348 static void android_media_visualizer_effect_callback(int32_t event,
349 void *user,
350 void *info) {
351 if ((event == AudioEffect::EVENT_ERROR) &&
352 (*((status_t*)info) == DEAD_OBJECT)) {
353 visualizer_callback_cookie* callbackInfo =
354 (visualizer_callback_cookie *)user;
355 JNIEnv *env = AndroidRuntime::getJNIEnv();
356
357 if (!user || !env) {
358 ALOGW("effectCallback error user %p, env %p", user, env);
359 return;
360 }
361 {
362 Mutex::Autolock l(sLock);
363 if (sVisualizerCallBackCookies.count(callbackInfo) == 0) {
364 return;
365 }
366 callbackInfo->busy_count++;
367 }
368 ALOGV("effectCallback: callbackInfo %p, visualizer_ref %p visualizer_class %p",
369 callbackInfo,
370 callbackInfo->visualizer_ref,
371 callbackInfo->visualizer_class);
372
373 env->CallStaticVoidMethod(
374 callbackInfo->visualizer_class,
375 fields.midPostNativeEvent,
376 callbackInfo->visualizer_ref,
377 NATIVE_EVENT_SERVER_DIED,
378 0, NULL);
379 if (env->ExceptionCheck()) {
380 env->ExceptionDescribe();
381 env->ExceptionClear();
382 }
383 {
384 Mutex::Autolock l(sLock);
385 callbackInfo->busy_count--;
386 callbackInfo->cond.broadcast();
387 }
388 }
389 }
390
391 static jint
android_media_visualizer_native_setup(JNIEnv * env,jobject thiz,jobject weak_this,jint sessionId,jintArray jId,jobject jAttributionSource)392 android_media_visualizer_native_setup(JNIEnv *env, jobject thiz, jobject weak_this,
393 jint sessionId, jintArray jId, jobject jAttributionSource)
394 {
395 ALOGV("android_media_visualizer_native_setup");
396 VisualizerJniStorage* lpJniStorage = NULL;
397 int lStatus = VISUALIZER_ERROR_NO_MEMORY;
398 sp<Visualizer> lpVisualizer;
399 jint* nId = NULL;
400 AttributionSourceState attributionSource;
401 Parcel* parcel = nullptr;
402
403 setVisualizer(env, thiz, 0);
404
405 lpJniStorage = new VisualizerJniStorage();
406 if (lpJniStorage == NULL) {
407 ALOGE("setup: Error creating JNI Storage");
408 goto setup_failure;
409 }
410
411 lpJniStorage->mCallbackData.visualizer_class = (jclass)env->NewGlobalRef(fields.clazzEffect);
412 // we use a weak reference so the Visualizer object can be garbage collected.
413 lpJniStorage->mCallbackData.visualizer_ref = env->NewGlobalRef(weak_this);
414
415 ALOGV("setup: lpJniStorage: %p visualizer_ref %p visualizer_class %p, &mCallbackData %p",
416 lpJniStorage,
417 lpJniStorage->mCallbackData.visualizer_ref,
418 lpJniStorage->mCallbackData.visualizer_class,
419 &lpJniStorage->mCallbackData);
420
421 if (jId == NULL) {
422 ALOGE("setup: NULL java array for id pointer");
423 lStatus = VISUALIZER_ERROR_BAD_VALUE;
424 goto setup_failure;
425 }
426
427 // create the native Visualizer object
428 parcel = parcelForJavaObject(env, jAttributionSource);
429 attributionSource.readFromParcel(parcel);
430 lpVisualizer = sp<Visualizer>::make(attributionSource);
431 if (lpVisualizer == 0) {
432 ALOGE("Error creating Visualizer");
433 goto setup_failure;
434 }
435 lpVisualizer->set(0,
436 android_media_visualizer_effect_callback,
437 &lpJniStorage->mCallbackData,
438 (audio_session_t) sessionId);
439
440 lStatus = translateError(lpVisualizer->initCheck());
441 if (lStatus != VISUALIZER_SUCCESS && lStatus != VISUALIZER_ERROR_ALREADY_EXISTS) {
442 ALOGE("Visualizer initCheck failed %d", lStatus);
443 goto setup_failure;
444 }
445
446 nId = env->GetIntArrayElements(jId, nullptr /* isCopy */);
447 if (nId == NULL) {
448 ALOGE("setup: Error retrieving id pointer");
449 lStatus = VISUALIZER_ERROR_BAD_VALUE;
450 goto setup_failure;
451 }
452 nId[0] = lpVisualizer->id();
453 env->ReleaseIntArrayElements(jId, nId, 0 /* mode */);
454 nId = NULL;
455
456 setVisualizer(env, thiz, lpVisualizer);
457
458 {
459 Mutex::Autolock l(sLock);
460 sVisualizerCallBackCookies.insert(&lpJniStorage->mCallbackData);
461 }
462 env->SetLongField(thiz, fields.fidJniData, (jlong)lpJniStorage);
463
464 return VISUALIZER_SUCCESS;
465
466 // failures:
467 setup_failure:
468
469 if (nId != NULL) {
470 env->ReleaseIntArrayElements(jId, nId, 0 /* mode */);
471 }
472
473 if (lpJniStorage) {
474 env->DeleteGlobalRef(lpJniStorage->mCallbackData.visualizer_class);
475 env->DeleteGlobalRef(lpJniStorage->mCallbackData.visualizer_ref);
476 delete lpJniStorage;
477 }
478 env->SetLongField(thiz, fields.fidJniData, 0);
479
480 return (jint) lStatus;
481 }
482
483 // ----------------------------------------------------------------------------
484 #define CALLBACK_COND_WAIT_TIMEOUT_MS 1000
android_media_visualizer_native_release(JNIEnv * env,jobject thiz)485 static void android_media_visualizer_native_release(JNIEnv *env, jobject thiz) {
486 {
487 sp<Visualizer> lpVisualizer = setVisualizer(env, thiz, 0);
488 if (lpVisualizer == 0) {
489 return;
490 }
491 lpVisualizer->release();
492 // Visualizer can still can be held by AudioEffect::EffectClient
493 }
494 // delete the JNI data
495 VisualizerJniStorage* lpJniStorage =
496 (VisualizerJniStorage *)env->GetLongField(thiz, fields.fidJniData);
497
498 // reset the native resources in the Java object so any attempt to access
499 // them after a call to release fails.
500 env->SetLongField(thiz, fields.fidJniData, 0);
501
502 if (lpJniStorage) {
503 {
504 Mutex::Autolock l(sLock);
505 visualizer_callback_cookie *lpCookie = &lpJniStorage->mCallbackData;
506 ALOGV("deleting pJniStorage: %p\n", lpJniStorage);
507 sVisualizerCallBackCookies.erase(lpCookie);
508 while (lpCookie->busy_count > 0) {
509 if (lpCookie->cond.waitRelative(sLock,
510 milliseconds(CALLBACK_COND_WAIT_TIMEOUT_MS)) !=
511 NO_ERROR) {
512 break;
513 }
514 }
515 ALOG_ASSERT(lpCookie->busy_count == 0, "Unbalanced busy_count inc/dec");
516 env->DeleteGlobalRef(lpJniStorage->mCallbackData.visualizer_class);
517 env->DeleteGlobalRef(lpJniStorage->mCallbackData.visualizer_ref);
518 } // sLock scope
519 delete lpJniStorage;
520 }
521 }
522
android_media_visualizer_native_finalize(JNIEnv * env,jobject thiz)523 static void android_media_visualizer_native_finalize(JNIEnv *env, jobject thiz) {
524 ALOGV("android_media_visualizer_native_finalize jobject: %p\n", thiz);
525 android_media_visualizer_native_release(env, thiz);
526 }
527
528 // ----------------------------------------------------------------------------
529 static jint
android_media_visualizer_native_setEnabled(JNIEnv * env,jobject thiz,jboolean enabled)530 android_media_visualizer_native_setEnabled(JNIEnv *env, jobject thiz, jboolean enabled)
531 {
532 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
533 if (lpVisualizer == 0) {
534 return VISUALIZER_ERROR_NO_INIT;
535 }
536
537 jint retVal = translateError(lpVisualizer->setEnabled(enabled));
538
539 if (!enabled) {
540 VisualizerJniStorage* lpJniStorage = (VisualizerJniStorage *)env->GetLongField(
541 thiz, fields.fidJniData);
542
543 if (NULL != lpJniStorage)
544 lpJniStorage->mCallbackData.cleanupBuffers();
545 }
546
547 return retVal;
548 }
549
550 static jboolean
android_media_visualizer_native_getEnabled(JNIEnv * env,jobject thiz)551 android_media_visualizer_native_getEnabled(JNIEnv *env, jobject thiz)
552 {
553 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
554 if (lpVisualizer == 0) {
555 return JNI_FALSE;
556 }
557
558 if (lpVisualizer->getEnabled()) {
559 return JNI_TRUE;
560 } else {
561 return JNI_FALSE;
562 }
563 }
564
565 static jintArray
android_media_visualizer_native_getCaptureSizeRange(JNIEnv * env,jobject)566 android_media_visualizer_native_getCaptureSizeRange(JNIEnv *env, jobject /* thiz */)
567 {
568 jintArray jRange = env->NewIntArray(2);
569 jint *nRange = env->GetIntArrayElements(jRange, NULL);
570 nRange[0] = Visualizer::getMinCaptureSize();
571 nRange[1] = Visualizer::getMaxCaptureSize();
572 ALOGV("getCaptureSizeRange() min %d max %d", nRange[0], nRange[1]);
573 env->ReleaseIntArrayElements(jRange, nRange, 0);
574 return jRange;
575 }
576
577 static jint
android_media_visualizer_native_getMaxCaptureRate(JNIEnv *,jobject)578 android_media_visualizer_native_getMaxCaptureRate(JNIEnv* /* env */, jobject /* thiz */)
579 {
580 return (jint) Visualizer::getMaxCaptureRate();
581 }
582
583 static jint
android_media_visualizer_native_setCaptureSize(JNIEnv * env,jobject thiz,jint size)584 android_media_visualizer_native_setCaptureSize(JNIEnv *env, jobject thiz, jint size)
585 {
586 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
587 if (lpVisualizer == 0) {
588 return VISUALIZER_ERROR_NO_INIT;
589 }
590
591 return translateError(lpVisualizer->setCaptureSize(size));
592 }
593
594 static jint
android_media_visualizer_native_getCaptureSize(JNIEnv * env,jobject thiz)595 android_media_visualizer_native_getCaptureSize(JNIEnv *env, jobject thiz)
596 {
597 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
598 if (lpVisualizer == 0) {
599 return -1;
600 }
601 return (jint) lpVisualizer->getCaptureSize();
602 }
603
604 static jint
android_media_visualizer_native_setScalingMode(JNIEnv * env,jobject thiz,jint mode)605 android_media_visualizer_native_setScalingMode(JNIEnv *env, jobject thiz, jint mode)
606 {
607 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
608 if (lpVisualizer == 0) {
609 return VISUALIZER_ERROR_NO_INIT;
610 }
611
612 return translateError(lpVisualizer->setScalingMode(mode));
613 }
614
615 static jint
android_media_visualizer_native_getScalingMode(JNIEnv * env,jobject thiz)616 android_media_visualizer_native_getScalingMode(JNIEnv *env, jobject thiz)
617 {
618 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
619 if (lpVisualizer == 0) {
620 return -1;
621 }
622 return (jint)lpVisualizer->getScalingMode();
623 }
624
625 static jint
android_media_visualizer_native_setMeasurementMode(JNIEnv * env,jobject thiz,jint mode)626 android_media_visualizer_native_setMeasurementMode(JNIEnv *env, jobject thiz, jint mode)
627 {
628 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
629 if (lpVisualizer == 0) {
630 return VISUALIZER_ERROR_NO_INIT;
631 }
632 return translateError(lpVisualizer->setMeasurementMode(mode));
633 }
634
635 static jint
android_media_visualizer_native_getMeasurementMode(JNIEnv * env,jobject thiz)636 android_media_visualizer_native_getMeasurementMode(JNIEnv *env, jobject thiz)
637 {
638 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
639 if (lpVisualizer == 0) {
640 return MEASUREMENT_MODE_NONE;
641 }
642 return lpVisualizer->getMeasurementMode();
643 }
644
645 static jint
android_media_visualizer_native_getSamplingRate(JNIEnv * env,jobject thiz)646 android_media_visualizer_native_getSamplingRate(JNIEnv *env, jobject thiz)
647 {
648 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
649 if (lpVisualizer == 0) {
650 return -1;
651 }
652 return (jint) lpVisualizer->getSamplingRate();
653 }
654
655 static jint
android_media_visualizer_native_getWaveForm(JNIEnv * env,jobject thiz,jbyteArray jWaveform)656 android_media_visualizer_native_getWaveForm(JNIEnv *env, jobject thiz, jbyteArray jWaveform)
657 {
658 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
659 if (lpVisualizer == 0) {
660 return VISUALIZER_ERROR_NO_INIT;
661 }
662
663 jbyte* nWaveform = env->GetByteArrayElements(jWaveform, nullptr /* isCopy */);
664 if (nWaveform == NULL) {
665 return VISUALIZER_ERROR_NO_MEMORY;
666 }
667 jint status = translateError(lpVisualizer->getWaveForm((uint8_t *)nWaveform));
668
669 env->ReleaseByteArrayElements(jWaveform, nWaveform, 0 /* mode */);
670 return status;
671 }
672
673 static jint
android_media_visualizer_native_getFft(JNIEnv * env,jobject thiz,jbyteArray jFft)674 android_media_visualizer_native_getFft(JNIEnv *env, jobject thiz, jbyteArray jFft)
675 {
676 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
677 if (lpVisualizer == 0) {
678 return VISUALIZER_ERROR_NO_INIT;
679 }
680
681 jbyte* nFft = env->GetByteArrayElements(jFft, nullptr /* isCopy */);
682 if (nFft == NULL) {
683 return VISUALIZER_ERROR_NO_MEMORY;
684 }
685 jint status = translateError(lpVisualizer->getFft((uint8_t *)nFft));
686
687 env->ReleaseByteArrayElements(jFft, nFft, 0 /* mode */);
688
689 return status;
690 }
691
692 static jint
android_media_visualizer_native_getPeakRms(JNIEnv * env,jobject thiz,jobject jPeakRmsObj)693 android_media_visualizer_native_getPeakRms(JNIEnv *env, jobject thiz, jobject jPeakRmsObj)
694 {
695 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
696 if (lpVisualizer == 0) {
697 return VISUALIZER_ERROR_NO_INIT;
698 }
699 int32_t measurements[2];
700 jint status = translateError(
701 lpVisualizer->getIntMeasurements(MEASUREMENT_MODE_PEAK_RMS,
702 2, measurements));
703 if (status == VISUALIZER_SUCCESS) {
704 // measurement worked, write the values to the java object
705 env->SetIntField(jPeakRmsObj, fields.fidPeak, measurements[MEASUREMENT_IDX_PEAK]);
706 env->SetIntField(jPeakRmsObj, fields.fidRms, measurements[MEASUREMENT_IDX_RMS]);
707 }
708 return status;
709 }
710
711 static jint
android_media_setPeriodicCapture(JNIEnv * env,jobject thiz,jint rate,jboolean jWaveform,jboolean jFft)712 android_media_setPeriodicCapture(JNIEnv *env, jobject thiz, jint rate, jboolean jWaveform, jboolean jFft)
713 {
714 sp<Visualizer> lpVisualizer = getVisualizer(env, thiz);
715 if (lpVisualizer == 0) {
716 return VISUALIZER_ERROR_NO_INIT;
717 }
718 VisualizerJniStorage* lpJniStorage = (VisualizerJniStorage *)env->GetLongField(thiz,
719 fields.fidJniData);
720 if (lpJniStorage == NULL) {
721 return VISUALIZER_ERROR_NO_INIT;
722 }
723
724 ALOGV("setPeriodicCapture: rate %d, jWaveform %d jFft %d",
725 rate,
726 jWaveform,
727 jFft);
728
729 uint32_t flags = Visualizer::CAPTURE_CALL_JAVA;
730 if (jWaveform) flags |= Visualizer::CAPTURE_WAVEFORM;
731 if (jFft) flags |= Visualizer::CAPTURE_FFT;
732 Visualizer::capture_cbk_t cbk = captureCallback;
733 if (!jWaveform && !jFft) cbk = NULL;
734
735 return translateError(lpVisualizer->setCaptureCallBack(cbk,
736 &lpJniStorage->mCallbackData,
737 flags,
738 rate));
739 }
740
741 // ----------------------------------------------------------------------------
742
743 // Dalvik VM type signatures
744 static const JNINativeMethod gMethods[] = {
745 {"native_init", "()V", (void *)android_media_visualizer_native_init},
746 {"native_setup", "(Ljava/lang/Object;I[ILandroid/os/Parcel;)I",
747 (void *)android_media_visualizer_native_setup},
748 {"native_finalize", "()V", (void *)android_media_visualizer_native_finalize},
749 {"native_release", "()V", (void *)android_media_visualizer_native_release},
750 {"native_setEnabled", "(Z)I", (void *)android_media_visualizer_native_setEnabled},
751 {"native_getEnabled", "()Z", (void *)android_media_visualizer_native_getEnabled},
752 {"getCaptureSizeRange", "()[I", (void *)android_media_visualizer_native_getCaptureSizeRange},
753 {"getMaxCaptureRate", "()I", (void *)android_media_visualizer_native_getMaxCaptureRate},
754 {"native_setCaptureSize", "(I)I", (void *)android_media_visualizer_native_setCaptureSize},
755 {"native_getCaptureSize", "()I", (void *)android_media_visualizer_native_getCaptureSize},
756 {"native_setScalingMode", "(I)I", (void *)android_media_visualizer_native_setScalingMode},
757 {"native_getScalingMode", "()I", (void *)android_media_visualizer_native_getScalingMode},
758 {"native_setMeasurementMode","(I)I", (void *)android_media_visualizer_native_setMeasurementMode},
759 {"native_getMeasurementMode","()I", (void *)android_media_visualizer_native_getMeasurementMode},
760 {"native_getSamplingRate", "()I", (void *)android_media_visualizer_native_getSamplingRate},
761 {"native_getWaveForm", "([B)I", (void *)android_media_visualizer_native_getWaveForm},
762 {"native_getFft", "([B)I", (void *)android_media_visualizer_native_getFft},
763 {"native_getPeakRms", "(Landroid/media/audiofx/Visualizer$MeasurementPeakRms;)I",
764 (void *)android_media_visualizer_native_getPeakRms},
765 {"native_setPeriodicCapture","(IZZ)I",(void *)android_media_setPeriodicCapture},
766 };
767
768 // ----------------------------------------------------------------------------
769
register_android_media_visualizer(JNIEnv * env)770 int register_android_media_visualizer(JNIEnv *env)
771 {
772 return AndroidRuntime::registerNativeMethods(env, kClassPathName, gMethods, NELEM(gMethods));
773 }
774