| /* |
| * Copyright 2017, 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. |
| */ |
| |
| #define LOG_TAG "MediaMetricsJNI" |
| |
| #include <jni.h> |
| #include <nativehelper/JNIHelp.h> |
| |
| #include "android_media_MediaMetricsJNI.h" |
| #include "android_os_Parcel.h" |
| #include <media/MediaAnalyticsItem.h> |
| #include <binder/Parcel.h> |
| |
| |
| // This source file is compiled and linked into: |
| // core/jni/ (libandroid_runtime.so) |
| |
| namespace android { |
| |
| // place the attributes into a java PersistableBundle object |
| jobject MediaMetricsJNI::writeMetricsToBundle(JNIEnv* env, MediaAnalyticsItem *item, jobject mybundle) { |
| |
| jclass clazzBundle = env->FindClass("android/os/PersistableBundle"); |
| if (clazzBundle==NULL) { |
| ALOGE("can't find android/os/PersistableBundle"); |
| return NULL; |
| } |
| // sometimes the caller provides one for us to fill |
| if (mybundle == NULL) { |
| // create the bundle |
| jmethodID constructID = env->GetMethodID(clazzBundle, "<init>", "()V"); |
| mybundle = env->NewObject(clazzBundle, constructID); |
| if (mybundle == NULL) { |
| return NULL; |
| } |
| } |
| |
| // grab methods that we can invoke |
| jmethodID setIntID = env->GetMethodID(clazzBundle, "putInt", "(Ljava/lang/String;I)V"); |
| jmethodID setLongID = env->GetMethodID(clazzBundle, "putLong", "(Ljava/lang/String;J)V"); |
| jmethodID setDoubleID = env->GetMethodID(clazzBundle, "putDouble", "(Ljava/lang/String;D)V"); |
| jmethodID setStringID = env->GetMethodID(clazzBundle, "putString", "(Ljava/lang/String;Ljava/lang/String;)V"); |
| |
| // env, class, method, {parms} |
| //env->CallVoidMethod(env, mybundle, setIntID, jstr, jint); |
| |
| // iterate through my attributes |
| // -- get name, get type, get value |
| // -- insert appropriately into the bundle |
| for (size_t i = 0 ; i < item->mPropCount; i++ ) { |
| MediaAnalyticsItem::Prop *prop = &item->mProps[i]; |
| // build the key parameter from prop->mName |
| jstring keyName = env->NewStringUTF(prop->mName); |
| // invoke the appropriate method to insert |
| switch (prop->mType) { |
| case MediaAnalyticsItem::kTypeInt32: |
| env->CallVoidMethod(mybundle, setIntID, |
| keyName, (jint) prop->u.int32Value); |
| break; |
| case MediaAnalyticsItem::kTypeInt64: |
| env->CallVoidMethod(mybundle, setLongID, |
| keyName, (jlong) prop->u.int64Value); |
| break; |
| case MediaAnalyticsItem::kTypeDouble: |
| env->CallVoidMethod(mybundle, setDoubleID, |
| keyName, (jdouble) prop->u.doubleValue); |
| break; |
| case MediaAnalyticsItem::kTypeCString: |
| env->CallVoidMethod(mybundle, setStringID, keyName, |
| env->NewStringUTF(prop->u.CStringValue)); |
| break; |
| default: |
| ALOGE("to_String bad item type: %d for %s", |
| prop->mType, prop->mName); |
| break; |
| } |
| } |
| |
| return mybundle; |
| } |
| |
| // convert the specified batch metrics attributes to a persistent bundle. |
| // The encoding of the byte array is specified in |
| // frameworks/av/media/libmediametrics/MediaAnalyticsItem.cpp |
| // |
| // type encodings; matches frameworks/av/media/libmediametrics/MediaAnalyticsItem.cpp |
| enum { kInt32 = 0, kInt64, kDouble, kRate, kCString}; |
| |
| jobject MediaMetricsJNI::writeAttributesToBundle(JNIEnv* env, jobject mybundle, char *buffer, size_t length) { |
| ALOGV("writeAttributes()"); |
| |
| if (buffer == NULL || length <= 0) { |
| ALOGW("bad parameters to writeAttributesToBundle()"); |
| return NULL; |
| } |
| |
| jclass clazzBundle = env->FindClass("android/os/PersistableBundle"); |
| if (clazzBundle==NULL) { |
| ALOGE("can't find android/os/PersistableBundle"); |
| return NULL; |
| } |
| // sometimes the caller provides one for us to fill |
| if (mybundle == NULL) { |
| // create the bundle |
| jmethodID constructID = env->GetMethodID(clazzBundle, "<init>", "()V"); |
| mybundle = env->NewObject(clazzBundle, constructID); |
| if (mybundle == NULL) { |
| ALOGD("unable to create mybundle"); |
| return NULL; |
| } |
| } |
| |
| int left = length; |
| char *buf = buffer; |
| |
| // grab methods that we can invoke |
| jmethodID setIntID = env->GetMethodID(clazzBundle, "putInt", "(Ljava/lang/String;I)V"); |
| jmethodID setLongID = env->GetMethodID(clazzBundle, "putLong", "(Ljava/lang/String;J)V"); |
| jmethodID setDoubleID = env->GetMethodID(clazzBundle, "putDouble", "(Ljava/lang/String;D)V"); |
| jmethodID setStringID = env->GetMethodID(clazzBundle, "putString", "(Ljava/lang/String;Ljava/lang/String;)V"); |
| |
| |
| #define _EXTRACT(size, val) \ |
| { if ((size) > left) goto badness; memcpy(&val, buf, (size)); buf += (size); left -= (size);} |
| #define _SKIP(size) \ |
| { if ((size) > left) goto badness; buf += (size); left -= (size);} |
| |
| int32_t bufsize; |
| _EXTRACT(sizeof(int32_t), bufsize); |
| if (bufsize != length) { |
| goto badness; |
| } |
| int32_t proto; |
| _EXTRACT(sizeof(int32_t), proto); |
| if (proto != 0) { |
| ALOGE("unsupported wire protocol %d", proto); |
| goto badness; |
| } |
| |
| int32_t count; |
| _EXTRACT(sizeof(int32_t), count); |
| |
| // iterate through my attributes |
| // -- get name, get type, get value, insert into bundle appropriately. |
| for (int i = 0 ; i < count; i++ ) { |
| // prop name len (int16) |
| int16_t keylen; |
| _EXTRACT(sizeof(int16_t), keylen); |
| if (keylen <= 0) goto badness; |
| // prop name itself |
| char *key = buf; |
| jstring keyName = env->NewStringUTF(buf); |
| _SKIP(keylen); |
| |
| // prop type (int8_t) |
| int8_t attrType; |
| _EXTRACT(sizeof(int8_t), attrType); |
| |
| int16_t attrSize; |
| _EXTRACT(sizeof(int16_t), attrSize); |
| |
| switch (attrType) { |
| case kInt32: |
| { |
| int32_t i32; |
| _EXTRACT(sizeof(int32_t), i32); |
| env->CallVoidMethod(mybundle, setIntID, |
| keyName, (jint) i32); |
| break; |
| } |
| case kInt64: |
| { |
| int64_t i64; |
| _EXTRACT(sizeof(int64_t), i64); |
| env->CallVoidMethod(mybundle, setLongID, |
| keyName, (jlong) i64); |
| break; |
| } |
| case kDouble: |
| { |
| double d64; |
| _EXTRACT(sizeof(double), d64); |
| env->CallVoidMethod(mybundle, setDoubleID, |
| keyName, (jdouble) d64); |
| break; |
| } |
| case kCString: |
| { |
| jstring value = env->NewStringUTF(buf); |
| env->CallVoidMethod(mybundle, setStringID, |
| keyName, value); |
| _SKIP(attrSize); |
| break; |
| } |
| default: |
| ALOGW("ignoring Attribute '%s' unknown type: %d", |
| key, attrType); |
| _SKIP(attrSize); |
| break; |
| } |
| } |
| |
| // should have consumed it all |
| if (left != 0) { |
| ALOGW("did not consume entire buffer; left(%d) != 0", left); |
| goto badness; |
| } |
| |
| return mybundle; |
| |
| badness: |
| return NULL; |
| } |
| |
| // Helper function to convert a native PersistableBundle to a Java |
| // PersistableBundle. |
| jobject MediaMetricsJNI::nativeToJavaPersistableBundle(JNIEnv *env, |
| os::PersistableBundle* nativeBundle) { |
| if (env == NULL || nativeBundle == NULL) { |
| ALOGE("Unexpected NULL parmeter"); |
| return NULL; |
| } |
| |
| // Create a Java parcel with the native parcel data. |
| // Then create a new PersistableBundle with that parcel as a parameter. |
| jobject jParcel = android::createJavaParcelObject(env); |
| if (jParcel == NULL) { |
| ALOGE("Failed to create a Java Parcel."); |
| return NULL; |
| } |
| |
| android::Parcel* nativeParcel = android::parcelForJavaObject(env, jParcel); |
| if (nativeParcel == NULL) { |
| ALOGE("Failed to get the native Parcel."); |
| return NULL; |
| } |
| |
| android::status_t result = nativeBundle->writeToParcel(nativeParcel); |
| nativeParcel->setDataPosition(0); |
| if (result != android::OK) { |
| ALOGE("Failed to write nativeBundle to Parcel: %d.", result); |
| return NULL; |
| } |
| |
| #define STATIC_INIT_JNI(T, obj, method, globalref, ...) \ |
| static T obj{};\ |
| if (obj == NULL) { \ |
| obj = method(__VA_ARGS__); \ |
| if (obj == NULL) { \ |
| ALOGE("%s can't find " #obj, __func__); \ |
| return NULL; \ |
| } else { \ |
| obj = globalref; \ |
| }\ |
| } \ |
| |
| STATIC_INIT_JNI(jclass, clazzBundle, env->FindClass, |
| static_cast<jclass>(env->NewGlobalRef(clazzBundle)), |
| "android/os/PersistableBundle"); |
| STATIC_INIT_JNI(jfieldID, bundleCreatorId, env->GetStaticFieldID, |
| bundleCreatorId, |
| clazzBundle, "CREATOR", "Landroid/os/Parcelable$Creator;"); |
| STATIC_INIT_JNI(jobject, bundleCreator, env->GetStaticObjectField, |
| env->NewGlobalRef(bundleCreator), |
| clazzBundle, bundleCreatorId); |
| STATIC_INIT_JNI(jclass, clazzCreator, env->FindClass, |
| static_cast<jclass>(env->NewGlobalRef(clazzCreator)), |
| "android/os/Parcelable$Creator"); |
| STATIC_INIT_JNI(jmethodID, createFromParcelId, env->GetMethodID, |
| createFromParcelId, |
| clazzCreator, "createFromParcel", "(Landroid/os/Parcel;)Ljava/lang/Object;"); |
| |
| jobject newBundle = env->CallObjectMethod(bundleCreator, createFromParcelId, jParcel); |
| if (newBundle == NULL) { |
| ALOGE("Failed to create a new PersistableBundle " |
| "from the createFromParcel call."); |
| } |
| |
| return newBundle; |
| } |
| |
| }; // namespace android |
| |