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