blob: 91eb4997519299d67515f5682dbf1cb37ae18c5f [file] [log] [blame]
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001/*
2 * Copyright 2013, 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//#define LOG_NDEBUG 0
18#define LOG_TAG "MediaDrm-JNI"
19#include <utils/Log.h>
20
21#include "android_media_MediaDrm.h"
22
23#include "android_runtime/AndroidRuntime.h"
Ruben Brunk87eac992013-09-09 17:44:59 -070024#include "android_runtime/Log.h"
Jeff Tinker54cfbd62013-04-02 13:14:59 -070025#include "android_os_Parcel.h"
Jeff Tinker8a0c80f2013-02-08 10:20:44 -080026#include "jni.h"
27#include "JNIHelp.h"
28
29#include <binder/IServiceManager.h>
Jeff Tinker54cfbd62013-04-02 13:14:59 -070030#include <binder/Parcel.h>
Jeff Tinker8a0c80f2013-02-08 10:20:44 -080031#include <media/IDrm.h>
32#include <media/IMediaPlayerService.h>
33#include <media/stagefright/foundation/ADebug.h>
Jeff Tinkerf7568b52013-04-17 14:24:40 -070034#include <media/stagefright/MediaErrors.h>
Jeff Tinker8a0c80f2013-02-08 10:20:44 -080035
36namespace android {
37
38#define FIND_CLASS(var, className) \
39 var = env->FindClass(className); \
40 LOG_FATAL_IF(! var, "Unable to find class " className);
41
42#define GET_FIELD_ID(var, clazz, fieldName, fieldDescriptor) \
43 var = env->GetFieldID(clazz, fieldName, fieldDescriptor); \
44 LOG_FATAL_IF(! var, "Unable to find field " fieldName);
45
46#define GET_METHOD_ID(var, clazz, fieldName, fieldDescriptor) \
47 var = env->GetMethodID(clazz, fieldName, fieldDescriptor); \
48 LOG_FATAL_IF(! var, "Unable to find method " fieldName);
49
Jeff Tinker54cfbd62013-04-02 13:14:59 -070050#define GET_STATIC_FIELD_ID(var, clazz, fieldName, fieldDescriptor) \
51 var = env->GetStaticFieldID(clazz, fieldName, fieldDescriptor); \
52 LOG_FATAL_IF(! var, "Unable to find field " fieldName);
53
54#define GET_STATIC_METHOD_ID(var, clazz, fieldName, fieldDescriptor) \
55 var = env->GetStaticMethodID(clazz, fieldName, fieldDescriptor); \
56 LOG_FATAL_IF(! var, "Unable to find static method " fieldName);
57
58
Jeff Tinker8a0c80f2013-02-08 10:20:44 -080059struct RequestFields {
60 jfieldID data;
61 jfieldID defaultUrl;
62};
63
64struct ArrayListFields {
65 jmethodID init;
66 jmethodID add;
67};
68
69struct HashmapFields {
70 jmethodID init;
71 jmethodID get;
72 jmethodID put;
73 jmethodID entrySet;
74};
75
76struct SetFields {
77 jmethodID iterator;
78};
79
80struct IteratorFields {
81 jmethodID next;
82 jmethodID hasNext;
83};
84
85struct EntryFields {
86 jmethodID getKey;
87 jmethodID getValue;
88};
89
Jeff Tinker54cfbd62013-04-02 13:14:59 -070090struct EventTypes {
Jeff Tinker17b89222013-05-21 12:35:06 -070091 jint kEventProvisionRequired;
92 jint kEventKeyRequired;
93 jint kEventKeyExpired;
94 jint kEventVendorDefined;
Jeff Tinker54cfbd62013-04-02 13:14:59 -070095} gEventTypes;
96
Jeff Tinker17b89222013-05-21 12:35:06 -070097struct KeyTypes {
98 jint kKeyTypeStreaming;
99 jint kKeyTypeOffline;
100 jint kKeyTypeRelease;
101} gKeyTypes;
102
Jeff Tinkere4095a82014-03-04 13:17:11 -0800103struct CertificateTypes {
104 jint kCertificateTypeNone;
105 jint kCertificateTypeX509;
106} gCertificateTypes;
107
108struct CertificateFields {
109 jfieldID wrappedPrivateKey;
110 jfieldID certificateData;
111};
112
Jeff Tinkerd712e1a2014-06-19 13:58:12 -0700113struct StateExceptionFields {
114 jmethodID init;
115 jclass classId;
116};
117
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800118struct fields_t {
119 jfieldID context;
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700120 jmethodID post_event;
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700121 RequestFields keyRequest;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800122 RequestFields provisionRequest;
123 ArrayListFields arraylist;
124 HashmapFields hashmap;
125 SetFields set;
126 IteratorFields iterator;
127 EntryFields entry;
Jeff Tinkere4095a82014-03-04 13:17:11 -0800128 CertificateFields certificate;
Jeff Tinkerd712e1a2014-06-19 13:58:12 -0700129 StateExceptionFields stateException;
Jeff Tinkere4095a82014-03-04 13:17:11 -0800130 jclass certificateClassId;
131 jclass hashmapClassId;
132 jclass arraylistClassId;
133 jclass stringClassId;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800134};
135
136static fields_t gFields;
137
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700138// ----------------------------------------------------------------------------
139// ref-counted object for callbacks
140class JNIDrmListener: public DrmListener
141{
142public:
143 JNIDrmListener(JNIEnv* env, jobject thiz, jobject weak_thiz);
144 ~JNIDrmListener();
145 virtual void notify(DrmPlugin::EventType eventType, int extra, const Parcel *obj = NULL);
146private:
147 JNIDrmListener();
148 jclass mClass; // Reference to MediaDrm class
149 jobject mObject; // Weak ref to MediaDrm Java object to call on
150};
151
152JNIDrmListener::JNIDrmListener(JNIEnv* env, jobject thiz, jobject weak_thiz)
153{
154 // Hold onto the MediaDrm class for use in calling the static method
155 // that posts events to the application thread.
156 jclass clazz = env->GetObjectClass(thiz);
157 if (clazz == NULL) {
158 ALOGE("Can't find android/media/MediaDrm");
Jeff Tinkereada5372013-05-21 12:48:14 -0700159 jniThrowException(env, "java/lang/Exception",
160 "Can't find android/media/MediaDrm");
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700161 return;
162 }
163 mClass = (jclass)env->NewGlobalRef(clazz);
164
165 // We use a weak reference so the MediaDrm object can be garbage collected.
166 // The reference is only used as a proxy for callbacks.
167 mObject = env->NewGlobalRef(weak_thiz);
168}
169
170JNIDrmListener::~JNIDrmListener()
171{
172 // remove global references
173 JNIEnv *env = AndroidRuntime::getJNIEnv();
174 env->DeleteGlobalRef(mObject);
175 env->DeleteGlobalRef(mClass);
176}
177
178void JNIDrmListener::notify(DrmPlugin::EventType eventType, int extra,
179 const Parcel *obj)
180{
181 jint jeventType;
182
183 // translate DrmPlugin event types into their java equivalents
184 switch(eventType) {
185 case DrmPlugin::kDrmPluginEventProvisionRequired:
186 jeventType = gEventTypes.kEventProvisionRequired;
187 break;
188 case DrmPlugin::kDrmPluginEventKeyNeeded:
189 jeventType = gEventTypes.kEventKeyRequired;
190 break;
191 case DrmPlugin::kDrmPluginEventKeyExpired:
192 jeventType = gEventTypes.kEventKeyExpired;
193 break;
194 case DrmPlugin::kDrmPluginEventVendorDefined:
195 jeventType = gEventTypes.kEventVendorDefined;
196 break;
197 default:
198 ALOGE("Invalid event DrmPlugin::EventType %d, ignored", (int)eventType);
199 return;
200 }
201
202 JNIEnv *env = AndroidRuntime::getJNIEnv();
203 if (obj && obj->dataSize() > 0) {
204 jobject jParcel = createJavaParcelObject(env);
205 if (jParcel != NULL) {
206 Parcel* nativeParcel = parcelForJavaObject(env, jParcel);
207 nativeParcel->setData(obj->data(), obj->dataSize());
208 env->CallStaticVoidMethod(mClass, gFields.post_event, mObject,
209 jeventType, extra, jParcel);
Patrik2 Carlsson265551a2013-12-10 14:52:43 +0100210 env->DeleteLocalRef(jParcel);
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700211 }
212 }
213
214 if (env->ExceptionCheck()) {
215 ALOGW("An exception occurred while notifying an event.");
216 LOGW_EX(env);
217 env->ExceptionClear();
218 }
219}
220
Jeff Tinkerd712e1a2014-06-19 13:58:12 -0700221static void throwStateException(JNIEnv *env, const char *msg, status_t err) {
222 ALOGE("Illegal state exception: %s (%d)", msg, err);
223
224 jobject exception = env->NewObject(gFields.stateException.classId,
225 gFields.stateException.init, static_cast<int>(err),
226 env->NewStringUTF(msg));
227 env->Throw(static_cast<jthrowable>(exception));
228}
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700229
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800230static bool throwExceptionAsNecessary(
231 JNIEnv *env, status_t err, const char *msg = NULL) {
232
Jeff Tinkerf7568b52013-04-17 14:24:40 -0700233 const char *drmMessage = NULL;
234
235 switch(err) {
236 case ERROR_DRM_UNKNOWN:
237 drmMessage = "General DRM error";
238 break;
239 case ERROR_DRM_NO_LICENSE:
240 drmMessage = "No license";
241 break;
242 case ERROR_DRM_LICENSE_EXPIRED:
243 drmMessage = "License expired";
244 break;
245 case ERROR_DRM_SESSION_NOT_OPENED:
246 drmMessage = "Session not opened";
247 break;
248 case ERROR_DRM_DECRYPT_UNIT_NOT_INITIALIZED:
249 drmMessage = "Not initialized";
250 break;
251 case ERROR_DRM_DECRYPT:
252 drmMessage = "Decrypt error";
253 break;
254 case ERROR_DRM_CANNOT_HANDLE:
255 drmMessage = "Unsupported scheme or data format";
256 break;
257 case ERROR_DRM_TAMPER_DETECTED:
258 drmMessage = "Invalid state";
259 break;
Jeff Tinkerf7568b52013-04-17 14:24:40 -0700260 default:
261 break;
262 }
263
264 String8 vendorMessage;
265 if (err >= ERROR_DRM_VENDOR_MIN && err <= ERROR_DRM_VENDOR_MAX) {
Jeff Tinker29799862014-08-27 11:05:13 -0700266 vendorMessage = String8::format("DRM vendor-defined error: %d", err);
Jeff Tinkerf7568b52013-04-17 14:24:40 -0700267 drmMessage = vendorMessage.string();
268 }
269
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800270 if (err == BAD_VALUE) {
271 jniThrowException(env, "java/lang/IllegalArgumentException", msg);
272 return true;
Jeff Tinker1d7c2182013-04-26 11:12:43 -0700273 } else if (err == ERROR_DRM_NOT_PROVISIONED) {
274 jniThrowException(env, "android/media/NotProvisionedException", msg);
275 return true;
Jeff Tinker3ed38262013-08-02 23:24:51 -0700276 } else if (err == ERROR_DRM_RESOURCE_BUSY) {
277 jniThrowException(env, "android/media/ResourceBusyException", msg);
278 return true;
Jeff Tinker1d7c2182013-04-26 11:12:43 -0700279 } else if (err == ERROR_DRM_DEVICE_REVOKED) {
280 jniThrowException(env, "android/media/DeniedByServerException", msg);
281 return true;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800282 } else if (err != OK) {
Jeff Tinkerf7568b52013-04-17 14:24:40 -0700283 String8 errbuf;
284 if (drmMessage != NULL) {
285 if (msg == NULL) {
286 msg = drmMessage;
287 } else {
Jeff Tinker29799862014-08-27 11:05:13 -0700288 errbuf = String8::format("%s: %s", msg, drmMessage);
Jeff Tinkerf7568b52013-04-17 14:24:40 -0700289 msg = errbuf.string();
290 }
291 }
Jeff Tinkerd712e1a2014-06-19 13:58:12 -0700292 throwStateException(env, msg, err);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800293 return true;
294 }
295 return false;
296}
297
298static sp<IDrm> GetDrm(JNIEnv *env, jobject thiz) {
Ashok Bhat656fd042013-11-28 10:56:06 +0000299 JDrm *jdrm = (JDrm *)env->GetLongField(thiz, gFields.context);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800300 return jdrm ? jdrm->getDrm() : NULL;
301}
302
303JDrm::JDrm(
304 JNIEnv *env, jobject thiz, const uint8_t uuid[16]) {
305 mObject = env->NewWeakGlobalRef(thiz);
306 mDrm = MakeDrm(uuid);
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700307 if (mDrm != NULL) {
308 mDrm->setListener(this);
309 }
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800310}
311
312JDrm::~JDrm() {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800313 JNIEnv *env = AndroidRuntime::getJNIEnv();
314
315 env->DeleteWeakGlobalRef(mObject);
316 mObject = NULL;
317}
318
319// static
320sp<IDrm> JDrm::MakeDrm() {
321 sp<IServiceManager> sm = defaultServiceManager();
322
323 sp<IBinder> binder =
324 sm->getService(String16("media.player"));
325
326 sp<IMediaPlayerService> service =
327 interface_cast<IMediaPlayerService>(binder);
328
329 if (service == NULL) {
330 return NULL;
331 }
332
333 sp<IDrm> drm = service->makeDrm();
334
335 if (drm == NULL || (drm->initCheck() != OK && drm->initCheck() != NO_INIT)) {
336 return NULL;
337 }
338
339 return drm;
340}
341
342// static
343sp<IDrm> JDrm::MakeDrm(const uint8_t uuid[16]) {
344 sp<IDrm> drm = MakeDrm();
345
346 if (drm == NULL) {
347 return NULL;
348 }
349
350 status_t err = drm->createPlugin(uuid);
351
352 if (err != OK) {
353 return NULL;
354 }
355
356 return drm;
357}
358
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700359status_t JDrm::setListener(const sp<DrmListener>& listener) {
360 Mutex::Autolock lock(mLock);
361 mListener = listener;
362 return OK;
363}
364
365void JDrm::notify(DrmPlugin::EventType eventType, int extra, const Parcel *obj) {
366 sp<DrmListener> listener;
367 mLock.lock();
368 listener = mListener;
369 mLock.unlock();
370
371 if (listener != NULL) {
372 Mutex::Autolock lock(mNotifyLock);
373 listener->notify(eventType, extra, obj);
374 }
375}
376
Jeff Tinker600071c2014-04-11 16:11:15 -0700377void JDrm::disconnect() {
378 if (mDrm != NULL) {
379 mDrm->destroyPlugin();
380 mDrm.clear();
381 }
382}
383
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700384
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800385// static
Jeff Tinker7cda4912013-08-21 11:52:34 -0700386bool JDrm::IsCryptoSchemeSupported(const uint8_t uuid[16], const String8 &mimeType) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800387 sp<IDrm> drm = MakeDrm();
388
389 if (drm == NULL) {
390 return false;
391 }
392
Jeff Tinker7cda4912013-08-21 11:52:34 -0700393 return drm->isCryptoSchemeSupported(uuid, mimeType);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800394}
395
396status_t JDrm::initCheck() const {
397 return mDrm == NULL ? NO_INIT : OK;
398}
399
400// JNI conversion utilities
401static Vector<uint8_t> JByteArrayToVector(JNIEnv *env, jbyteArray const &byteArray) {
402 Vector<uint8_t> vector;
403 size_t length = env->GetArrayLength(byteArray);
404 vector.insertAt((size_t)0, length);
405 env->GetByteArrayRegion(byteArray, 0, length, (jbyte *)vector.editArray());
406 return vector;
407}
408
409static jbyteArray VectorToJByteArray(JNIEnv *env, Vector<uint8_t> const &vector) {
410 size_t length = vector.size();
411 jbyteArray result = env->NewByteArray(length);
412 if (result != NULL) {
413 env->SetByteArrayRegion(result, 0, length, (jbyte *)vector.array());
414 }
415 return result;
416}
417
418static String8 JStringToString8(JNIEnv *env, jstring const &jstr) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800419 String8 result;
420
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700421 const char *s = env->GetStringUTFChars(jstr, NULL);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800422 if (s) {
423 result = s;
424 env->ReleaseStringUTFChars(jstr, s);
425 }
426 return result;
427}
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700428
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800429/*
430 import java.util.HashMap;
431 import java.util.Set;
432 import java.Map.Entry;
433 import jav.util.Iterator;
434
435 HashMap<k, v> hm;
436 Set<Entry<k, v> > s = hm.entrySet();
437 Iterator i = s.iterator();
438 Entry e = s.next();
439*/
440
441static KeyedVector<String8, String8> HashMapToKeyedVector(JNIEnv *env, jobject &hashMap) {
Jeff Tinkere4095a82014-03-04 13:17:11 -0800442 jclass clazz = gFields.stringClassId;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800443 KeyedVector<String8, String8> keyedVector;
444
445 jobject entrySet = env->CallObjectMethod(hashMap, gFields.hashmap.entrySet);
446 if (entrySet) {
447 jobject iterator = env->CallObjectMethod(entrySet, gFields.set.iterator);
448 if (iterator) {
449 jboolean hasNext = env->CallBooleanMethod(iterator, gFields.iterator.hasNext);
450 while (hasNext) {
451 jobject entry = env->CallObjectMethod(iterator, gFields.iterator.next);
452 if (entry) {
453 jobject obj = env->CallObjectMethod(entry, gFields.entry.getKey);
454 if (!env->IsInstanceOf(obj, clazz)) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700455 jniThrowException(env, "java/lang/IllegalArgumentException",
456 "HashMap key is not a String");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800457 }
458 jstring jkey = static_cast<jstring>(obj);
459
460 obj = env->CallObjectMethod(entry, gFields.entry.getValue);
461 if (!env->IsInstanceOf(obj, clazz)) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700462 jniThrowException(env, "java/lang/IllegalArgumentException",
463 "HashMap value is not a String");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800464 }
465 jstring jvalue = static_cast<jstring>(obj);
466
467 String8 key = JStringToString8(env, jkey);
468 String8 value = JStringToString8(env, jvalue);
469 keyedVector.add(key, value);
470
471 env->DeleteLocalRef(jkey);
472 env->DeleteLocalRef(jvalue);
473 hasNext = env->CallBooleanMethod(iterator, gFields.iterator.hasNext);
474 }
475 env->DeleteLocalRef(entry);
476 }
477 env->DeleteLocalRef(iterator);
478 }
479 env->DeleteLocalRef(entrySet);
480 }
481 return keyedVector;
482}
483
484static jobject KeyedVectorToHashMap (JNIEnv *env, KeyedVector<String8, String8> const &map) {
Jeff Tinkere4095a82014-03-04 13:17:11 -0800485 jclass clazz = gFields.hashmapClassId;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800486 jobject hashMap = env->NewObject(clazz, gFields.hashmap.init);
487 for (size_t i = 0; i < map.size(); ++i) {
488 jstring jkey = env->NewStringUTF(map.keyAt(i).string());
489 jstring jvalue = env->NewStringUTF(map.valueAt(i).string());
490 env->CallObjectMethod(hashMap, gFields.hashmap.put, jkey, jvalue);
491 env->DeleteLocalRef(jkey);
492 env->DeleteLocalRef(jvalue);
493 }
494 return hashMap;
495}
496
497static jobject ListOfVectorsToArrayListOfByteArray(JNIEnv *env,
498 List<Vector<uint8_t> > list) {
Jeff Tinkere4095a82014-03-04 13:17:11 -0800499 jclass clazz = gFields.arraylistClassId;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800500 jobject arrayList = env->NewObject(clazz, gFields.arraylist.init);
501 List<Vector<uint8_t> >::iterator iter = list.begin();
502 while (iter != list.end()) {
503 jbyteArray byteArray = VectorToJByteArray(env, *iter);
504 env->CallBooleanMethod(arrayList, gFields.arraylist.add, byteArray);
505 env->DeleteLocalRef(byteArray);
506 iter++;
507 }
508
509 return arrayList;
510}
511
512} // namespace android
513
514using namespace android;
515
516static sp<JDrm> setDrm(
517 JNIEnv *env, jobject thiz, const sp<JDrm> &drm) {
Ashok Bhat656fd042013-11-28 10:56:06 +0000518 sp<JDrm> old = (JDrm *)env->GetLongField(thiz, gFields.context);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800519 if (drm != NULL) {
520 drm->incStrong(thiz);
521 }
522 if (old != NULL) {
523 old->decStrong(thiz);
524 }
Narayan Kamathf11dd632013-12-18 16:53:54 +0000525 env->SetLongField(thiz, gFields.context, reinterpret_cast<jlong>(drm.get()));
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800526
527 return old;
528}
529
530static bool CheckSession(JNIEnv *env, const sp<IDrm> &drm, jbyteArray const &jsessionId)
531{
532 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700533 jniThrowException(env, "java/lang/IllegalStateException", "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800534 return false;
535 }
536
537 if (jsessionId == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700538 jniThrowException(env, "java/lang/IllegalArgumentException", "sessionId is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800539 return false;
540 }
541 return true;
542}
543
544static void android_media_MediaDrm_release(JNIEnv *env, jobject thiz) {
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700545 sp<JDrm> drm = setDrm(env, thiz, NULL);
546 if (drm != NULL) {
547 drm->setListener(NULL);
Jeff Tinker600071c2014-04-11 16:11:15 -0700548 drm->disconnect();
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700549 }
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800550}
551
552static void android_media_MediaDrm_native_init(JNIEnv *env) {
553 jclass clazz;
554 FIND_CLASS(clazz, "android/media/MediaDrm");
Ashok Bhat656fd042013-11-28 10:56:06 +0000555 GET_FIELD_ID(gFields.context, clazz, "mNativeContext", "J");
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700556 GET_STATIC_METHOD_ID(gFields.post_event, clazz, "postEventFromNative",
557 "(Ljava/lang/Object;IILjava/lang/Object;)V");
558
559 jfieldID field;
Jeff Tinkere1c76be2013-04-03 18:11:33 -0700560 GET_STATIC_FIELD_ID(field, clazz, "EVENT_PROVISION_REQUIRED", "I");
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700561 gEventTypes.kEventProvisionRequired = env->GetStaticIntField(clazz, field);
Jeff Tinkere1c76be2013-04-03 18:11:33 -0700562 GET_STATIC_FIELD_ID(field, clazz, "EVENT_KEY_REQUIRED", "I");
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700563 gEventTypes.kEventKeyRequired = env->GetStaticIntField(clazz, field);
Jeff Tinkere1c76be2013-04-03 18:11:33 -0700564 GET_STATIC_FIELD_ID(field, clazz, "EVENT_KEY_EXPIRED", "I");
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700565 gEventTypes.kEventKeyExpired = env->GetStaticIntField(clazz, field);
Jeff Tinkere1c76be2013-04-03 18:11:33 -0700566 GET_STATIC_FIELD_ID(field, clazz, "EVENT_VENDOR_DEFINED", "I");
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700567 gEventTypes.kEventVendorDefined = env->GetStaticIntField(clazz, field);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800568
Jeff Tinker17b89222013-05-21 12:35:06 -0700569 GET_STATIC_FIELD_ID(field, clazz, "KEY_TYPE_STREAMING", "I");
570 gKeyTypes.kKeyTypeStreaming = env->GetStaticIntField(clazz, field);
571 GET_STATIC_FIELD_ID(field, clazz, "KEY_TYPE_OFFLINE", "I");
572 gKeyTypes.kKeyTypeOffline = env->GetStaticIntField(clazz, field);
573 GET_STATIC_FIELD_ID(field, clazz, "KEY_TYPE_RELEASE", "I");
574 gKeyTypes.kKeyTypeRelease = env->GetStaticIntField(clazz, field);
575
Jeff Tinkere4095a82014-03-04 13:17:11 -0800576 GET_STATIC_FIELD_ID(field, clazz, "CERTIFICATE_TYPE_NONE", "I");
577 gCertificateTypes.kCertificateTypeNone = env->GetStaticIntField(clazz, field);
578 GET_STATIC_FIELD_ID(field, clazz, "CERTIFICATE_TYPE_X509", "I");
579 gCertificateTypes.kCertificateTypeX509 = env->GetStaticIntField(clazz, field);
580
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700581 FIND_CLASS(clazz, "android/media/MediaDrm$KeyRequest");
Jeff Tinkere1c76be2013-04-03 18:11:33 -0700582 GET_FIELD_ID(gFields.keyRequest.data, clazz, "mData", "[B");
583 GET_FIELD_ID(gFields.keyRequest.defaultUrl, clazz, "mDefaultUrl", "Ljava/lang/String;");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800584
585 FIND_CLASS(clazz, "android/media/MediaDrm$ProvisionRequest");
Jeff Tinkere1c76be2013-04-03 18:11:33 -0700586 GET_FIELD_ID(gFields.provisionRequest.data, clazz, "mData", "[B");
587 GET_FIELD_ID(gFields.provisionRequest.defaultUrl, clazz, "mDefaultUrl", "Ljava/lang/String;");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800588
Jeff Tinkere4095a82014-03-04 13:17:11 -0800589 FIND_CLASS(clazz, "android/media/MediaDrm$Certificate");
590 GET_FIELD_ID(gFields.certificate.wrappedPrivateKey, clazz, "mWrappedKey", "[B");
591 GET_FIELD_ID(gFields.certificate.certificateData, clazz, "mCertificateData", "[B");
Jeff Tinker65c94e62014-04-02 12:23:56 -0700592 gFields.certificateClassId = static_cast<jclass>(env->NewGlobalRef(clazz));
Jeff Tinkere4095a82014-03-04 13:17:11 -0800593
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800594 FIND_CLASS(clazz, "java/util/ArrayList");
595 GET_METHOD_ID(gFields.arraylist.init, clazz, "<init>", "()V");
596 GET_METHOD_ID(gFields.arraylist.add, clazz, "add", "(Ljava/lang/Object;)Z");
597
598 FIND_CLASS(clazz, "java/util/HashMap");
599 GET_METHOD_ID(gFields.hashmap.init, clazz, "<init>", "()V");
600 GET_METHOD_ID(gFields.hashmap.get, clazz, "get", "(Ljava/lang/Object;)Ljava/lang/Object;");
601 GET_METHOD_ID(gFields.hashmap.put, clazz, "put",
602 "(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;");
603 GET_METHOD_ID(gFields.hashmap.entrySet, clazz, "entrySet", "()Ljava/util/Set;");
604
605 FIND_CLASS(clazz, "java/util/Set");
606 GET_METHOD_ID(gFields.set.iterator, clazz, "iterator", "()Ljava/util/Iterator;");
607
608 FIND_CLASS(clazz, "java/util/Iterator");
609 GET_METHOD_ID(gFields.iterator.next, clazz, "next", "()Ljava/lang/Object;");
610 GET_METHOD_ID(gFields.iterator.hasNext, clazz, "hasNext", "()Z");
611
612 FIND_CLASS(clazz, "java/util/Map$Entry");
613 GET_METHOD_ID(gFields.entry.getKey, clazz, "getKey", "()Ljava/lang/Object;");
614 GET_METHOD_ID(gFields.entry.getValue, clazz, "getValue", "()Ljava/lang/Object;");
Jeff Tinkere4095a82014-03-04 13:17:11 -0800615
616 FIND_CLASS(clazz, "java/util/HashMap");
Jeff Tinker65c94e62014-04-02 12:23:56 -0700617 gFields.hashmapClassId = static_cast<jclass>(env->NewGlobalRef(clazz));
Jeff Tinkere4095a82014-03-04 13:17:11 -0800618
619 FIND_CLASS(clazz, "java/lang/String");
Jeff Tinker65c94e62014-04-02 12:23:56 -0700620 gFields.stringClassId = static_cast<jclass>(env->NewGlobalRef(clazz));
Jeff Tinkere4095a82014-03-04 13:17:11 -0800621
622 FIND_CLASS(clazz, "java/util/ArrayList");
Jeff Tinker65c94e62014-04-02 12:23:56 -0700623 gFields.arraylistClassId = static_cast<jclass>(env->NewGlobalRef(clazz));
Jeff Tinkerd712e1a2014-06-19 13:58:12 -0700624
625 FIND_CLASS(clazz, "android/media/MediaDrm$MediaDrmStateException");
626 GET_METHOD_ID(gFields.stateException.init, clazz, "<init>", "(ILjava/lang/String;)V");
627 gFields.stateException.classId = static_cast<jclass>(env->NewGlobalRef(clazz));
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800628}
629
630static void android_media_MediaDrm_native_setup(
631 JNIEnv *env, jobject thiz,
632 jobject weak_this, jbyteArray uuidObj) {
633
634 if (uuidObj == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700635 jniThrowException(env, "java/lang/IllegalArgumentException", "uuid is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800636 return;
637 }
638
639 Vector<uint8_t> uuid = JByteArrayToVector(env, uuidObj);
640
641 if (uuid.size() != 16) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700642 jniThrowException(env, "java/lang/IllegalArgumentException",
643 "invalid UUID size, expected 16 bytes");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800644 return;
645 }
646
647 sp<JDrm> drm = new JDrm(env, thiz, uuid.array());
648
649 status_t err = drm->initCheck();
650
651 if (err != OK) {
652 jniThrowException(
653 env,
Jeff Tinker1d7c2182013-04-26 11:12:43 -0700654 "android/media/UnsupportedSchemeException",
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800655 "Failed to instantiate drm object.");
656 return;
657 }
658
Jeff Tinker54cfbd62013-04-02 13:14:59 -0700659 sp<JNIDrmListener> listener = new JNIDrmListener(env, thiz, weak_this);
660 drm->setListener(listener);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800661 setDrm(env, thiz, drm);
662}
663
664static void android_media_MediaDrm_native_finalize(
665 JNIEnv *env, jobject thiz) {
666 android_media_MediaDrm_release(env, thiz);
667}
668
669static jboolean android_media_MediaDrm_isCryptoSchemeSupportedNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -0800670 JNIEnv *env, jobject /* thiz */, jbyteArray uuidObj, jstring jmimeType) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800671
672 if (uuidObj == NULL) {
673 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
674 return false;
675 }
676
677 Vector<uint8_t> uuid = JByteArrayToVector(env, uuidObj);
678
679 if (uuid.size() != 16) {
680 jniThrowException(
681 env,
682 "java/lang/IllegalArgumentException",
Jeff Tinkereada5372013-05-21 12:48:14 -0700683 "invalid UUID size, expected 16 bytes");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800684 return false;
685 }
686
Jeff Tinker7cda4912013-08-21 11:52:34 -0700687 String8 mimeType;
688 if (jmimeType != NULL) {
689 mimeType = JStringToString8(env, jmimeType);
690 }
691
692 return JDrm::IsCryptoSchemeSupported(uuid.array(), mimeType);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800693}
694
695static jbyteArray android_media_MediaDrm_openSession(
696 JNIEnv *env, jobject thiz) {
697 sp<IDrm> drm = GetDrm(env, thiz);
698
699 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700700 jniThrowException(env, "java/lang/IllegalStateException",
701 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800702 return NULL;
703 }
704
705 Vector<uint8_t> sessionId;
706 status_t err = drm->openSession(sessionId);
707
708 if (throwExceptionAsNecessary(env, err, "Failed to open session")) {
709 return NULL;
710 }
711
712 return VectorToJByteArray(env, sessionId);
713}
714
715static void android_media_MediaDrm_closeSession(
716 JNIEnv *env, jobject thiz, jbyteArray jsessionId) {
717 sp<IDrm> drm = GetDrm(env, thiz);
718
719 if (!CheckSession(env, drm, jsessionId)) {
720 return;
721 }
722
723 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
724
725 status_t err = drm->closeSession(sessionId);
726
727 throwExceptionAsNecessary(env, err, "Failed to close session");
728}
729
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700730static jobject android_media_MediaDrm_getKeyRequest(
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800731 JNIEnv *env, jobject thiz, jbyteArray jsessionId, jbyteArray jinitData,
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700732 jstring jmimeType, jint jkeyType, jobject joptParams) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800733 sp<IDrm> drm = GetDrm(env, thiz);
734
735 if (!CheckSession(env, drm, jsessionId)) {
736 return NULL;
737 }
738
739 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
740
741 Vector<uint8_t> initData;
742 if (jinitData != NULL) {
743 initData = JByteArrayToVector(env, jinitData);
744 }
745
746 String8 mimeType;
747 if (jmimeType != NULL) {
748 mimeType = JStringToString8(env, jmimeType);
749 }
750
Jeff Tinker17b89222013-05-21 12:35:06 -0700751 DrmPlugin::KeyType keyType;
752 if (jkeyType == gKeyTypes.kKeyTypeStreaming) {
753 keyType = DrmPlugin::kKeyType_Streaming;
754 } else if (jkeyType == gKeyTypes.kKeyTypeOffline) {
755 keyType = DrmPlugin::kKeyType_Offline;
756 } else if (jkeyType == gKeyTypes.kKeyTypeRelease) {
757 keyType = DrmPlugin::kKeyType_Release;
758 } else {
759 jniThrowException(env, "java/lang/IllegalArgumentException",
760 "invalid keyType");
761 return NULL;
762 }
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800763
764 KeyedVector<String8, String8> optParams;
765 if (joptParams != NULL) {
766 optParams = HashMapToKeyedVector(env, joptParams);
767 }
768
769 Vector<uint8_t> request;
770 String8 defaultUrl;
771
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700772 status_t err = drm->getKeyRequest(sessionId, initData, mimeType,
773 keyType, optParams, request, defaultUrl);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800774
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700775 if (throwExceptionAsNecessary(env, err, "Failed to get key request")) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800776 return NULL;
777 }
778
779 // Fill out return obj
780 jclass clazz;
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700781 FIND_CLASS(clazz, "android/media/MediaDrm$KeyRequest");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800782
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700783 jobject keyObj = NULL;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800784
785 if (clazz) {
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700786 keyObj = env->AllocObject(clazz);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800787 jbyteArray jrequest = VectorToJByteArray(env, request);
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700788 env->SetObjectField(keyObj, gFields.keyRequest.data, jrequest);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800789
790 jstring jdefaultUrl = env->NewStringUTF(defaultUrl.string());
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700791 env->SetObjectField(keyObj, gFields.keyRequest.defaultUrl, jdefaultUrl);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800792 }
793
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700794 return keyObj;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800795}
796
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700797static jbyteArray android_media_MediaDrm_provideKeyResponse(
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800798 JNIEnv *env, jobject thiz, jbyteArray jsessionId, jbyteArray jresponse) {
799 sp<IDrm> drm = GetDrm(env, thiz);
800
801 if (!CheckSession(env, drm, jsessionId)) {
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700802 return NULL;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800803 }
804
805 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
806
807 if (jresponse == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700808 jniThrowException(env, "java/lang/IllegalArgumentException",
809 "key response is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700810 return NULL;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800811 }
812 Vector<uint8_t> response(JByteArrayToVector(env, jresponse));
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700813 Vector<uint8_t> keySetId;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800814
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700815 status_t err = drm->provideKeyResponse(sessionId, response, keySetId);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800816
Jeff Tinker8117d8f2013-08-16 13:46:02 -0700817 if (throwExceptionAsNecessary(env, err, "Failed to handle key response")) {
818 return NULL;
819 }
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700820 return VectorToJByteArray(env, keySetId);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800821}
822
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700823static void android_media_MediaDrm_removeKeys(
824 JNIEnv *env, jobject thiz, jbyteArray jkeysetId) {
825 sp<IDrm> drm = GetDrm(env, thiz);
826
827 if (jkeysetId == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700828 jniThrowException(env, "java/lang/IllegalArgumentException",
829 "keySetId is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700830 return;
831 }
832
833 Vector<uint8_t> keySetId(JByteArrayToVector(env, jkeysetId));
834
835 status_t err = drm->removeKeys(keySetId);
836
837 throwExceptionAsNecessary(env, err, "Failed to remove keys");
838}
839
840static void android_media_MediaDrm_restoreKeys(
841 JNIEnv *env, jobject thiz, jbyteArray jsessionId,
842 jbyteArray jkeysetId) {
843
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800844 sp<IDrm> drm = GetDrm(env, thiz);
845
846 if (!CheckSession(env, drm, jsessionId)) {
847 return;
848 }
849
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700850 if (jkeysetId == NULL) {
851 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
852 return;
853 }
854
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800855 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700856 Vector<uint8_t> keySetId(JByteArrayToVector(env, jkeysetId));
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800857
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700858 status_t err = drm->restoreKeys(sessionId, keySetId);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800859
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700860 throwExceptionAsNecessary(env, err, "Failed to restore keys");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800861}
862
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700863static jobject android_media_MediaDrm_queryKeyStatus(
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800864 JNIEnv *env, jobject thiz, jbyteArray jsessionId) {
865 sp<IDrm> drm = GetDrm(env, thiz);
866
867 if (!CheckSession(env, drm, jsessionId)) {
868 return NULL;
869 }
870 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
871
872 KeyedVector<String8, String8> infoMap;
873
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700874 status_t err = drm->queryKeyStatus(sessionId, infoMap);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800875
Jeff Tinker16b8cff2013-03-30 16:26:13 -0700876 if (throwExceptionAsNecessary(env, err, "Failed to query key status")) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800877 return NULL;
878 }
879
880 return KeyedVectorToHashMap(env, infoMap);
881}
882
Jeff Tinkere4095a82014-03-04 13:17:11 -0800883static jobject android_media_MediaDrm_getProvisionRequestNative(
884 JNIEnv *env, jobject thiz, jint jcertType, jstring jcertAuthority) {
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800885 sp<IDrm> drm = GetDrm(env, thiz);
886
887 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700888 jniThrowException(env, "java/lang/IllegalStateException",
889 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800890 return NULL;
891 }
892
893 Vector<uint8_t> request;
894 String8 defaultUrl;
895
Jeff Tinkere4095a82014-03-04 13:17:11 -0800896 String8 certType;
897 if (jcertType == gCertificateTypes.kCertificateTypeX509) {
898 certType = "X.509";
899 } else if (jcertType == gCertificateTypes.kCertificateTypeNone) {
900 certType = "none";
901 } else {
902 certType = "invalid";
903 }
904
905 String8 certAuthority = JStringToString8(env, jcertAuthority);
906 status_t err = drm->getProvisionRequest(certType, certAuthority, request, defaultUrl);
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800907
908 if (throwExceptionAsNecessary(env, err, "Failed to get provision request")) {
909 return NULL;
910 }
911
912 // Fill out return obj
913 jclass clazz;
914 FIND_CLASS(clazz, "android/media/MediaDrm$ProvisionRequest");
915
916 jobject provisionObj = NULL;
917
918 if (clazz) {
919 provisionObj = env->AllocObject(clazz);
920 jbyteArray jrequest = VectorToJByteArray(env, request);
921 env->SetObjectField(provisionObj, gFields.provisionRequest.data, jrequest);
922
923 jstring jdefaultUrl = env->NewStringUTF(defaultUrl.string());
924 env->SetObjectField(provisionObj, gFields.provisionRequest.defaultUrl, jdefaultUrl);
925 }
926
927 return provisionObj;
928}
929
Jeff Tinkere4095a82014-03-04 13:17:11 -0800930static jobject android_media_MediaDrm_provideProvisionResponseNative(
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800931 JNIEnv *env, jobject thiz, jbyteArray jresponse) {
932 sp<IDrm> drm = GetDrm(env, thiz);
933
934 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700935 jniThrowException(env, "java/lang/IllegalStateException",
936 "MediaDrm obj is null");
Jeff Tinkere4095a82014-03-04 13:17:11 -0800937 return NULL;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800938 }
939
940 if (jresponse == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700941 jniThrowException(env, "java/lang/IllegalArgumentException",
942 "provision response is null");
Jeff Tinkere4095a82014-03-04 13:17:11 -0800943 return NULL;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800944 }
945
946 Vector<uint8_t> response(JByteArrayToVector(env, jresponse));
Jeff Tinkere4095a82014-03-04 13:17:11 -0800947 Vector<uint8_t> certificate, wrappedKey;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800948
Jeff Tinkere4095a82014-03-04 13:17:11 -0800949 status_t err = drm->provideProvisionResponse(response, certificate, wrappedKey);
950
951 // Fill out return obj
952 jclass clazz = gFields.certificateClassId;
953
954 jobject certificateObj = NULL;
955
956 if (clazz && certificate.size() && wrappedKey.size()) {
957 certificateObj = env->AllocObject(clazz);
958 jbyteArray jcertificate = VectorToJByteArray(env, certificate);
959 env->SetObjectField(certificateObj, gFields.certificate.certificateData, jcertificate);
960
961 jbyteArray jwrappedKey = VectorToJByteArray(env, wrappedKey);
962 env->SetObjectField(certificateObj, gFields.certificate.wrappedPrivateKey, jwrappedKey);
963 }
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800964
965 throwExceptionAsNecessary(env, err, "Failed to handle provision response");
Jeff Tinkere4095a82014-03-04 13:17:11 -0800966 return certificateObj;
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800967}
968
Jeff Tinker6bf5b602014-04-30 10:02:39 -0700969static void android_media_MediaDrm_unprovisionDeviceNative(
970 JNIEnv *env, jobject thiz) {
971 sp<IDrm> drm = GetDrm(env, thiz);
972
973 if (drm == NULL) {
974 jniThrowException(env, "java/lang/IllegalStateException",
975 "MediaDrm obj is null");
976 return;
977 }
978
979 status_t err = drm->unprovisionDevice();
980
981 throwExceptionAsNecessary(env, err, "Failed to handle provision response");
982 return;
983}
984
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800985static jobject android_media_MediaDrm_getSecureStops(
986 JNIEnv *env, jobject thiz) {
987 sp<IDrm> drm = GetDrm(env, thiz);
988
989 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -0700990 jniThrowException(env, "java/lang/IllegalStateException",
991 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -0800992 return NULL;
993 }
994
995 List<Vector<uint8_t> > secureStops;
996
997 status_t err = drm->getSecureStops(secureStops);
998
999 if (throwExceptionAsNecessary(env, err, "Failed to get secure stops")) {
1000 return NULL;
1001 }
1002
1003 return ListOfVectorsToArrayListOfByteArray(env, secureStops);
1004}
1005
1006static void android_media_MediaDrm_releaseSecureStops(
1007 JNIEnv *env, jobject thiz, jbyteArray jssRelease) {
1008 sp<IDrm> drm = GetDrm(env, thiz);
1009
1010 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001011 jniThrowException(env, "java/lang/IllegalStateException",
1012 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001013 return;
1014 }
1015
1016 Vector<uint8_t> ssRelease(JByteArrayToVector(env, jssRelease));
1017
1018 status_t err = drm->releaseSecureStops(ssRelease);
1019
1020 throwExceptionAsNecessary(env, err, "Failed to release secure stops");
1021}
1022
1023static jstring android_media_MediaDrm_getPropertyString(
1024 JNIEnv *env, jobject thiz, jstring jname) {
1025 sp<IDrm> drm = GetDrm(env, thiz);
1026
1027 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001028 jniThrowException(env, "java/lang/IllegalStateException",
1029 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001030 return NULL;
1031 }
1032
1033 if (jname == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001034 jniThrowException(env, "java/lang/IllegalArgumentException",
1035 "property name String is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001036 return NULL;
1037 }
1038
1039 String8 name = JStringToString8(env, jname);
1040 String8 value;
1041
1042 status_t err = drm->getPropertyString(name, value);
1043
1044 if (throwExceptionAsNecessary(env, err, "Failed to get property")) {
1045 return NULL;
1046 }
1047
1048 return env->NewStringUTF(value.string());
1049}
1050
1051static jbyteArray android_media_MediaDrm_getPropertyByteArray(
1052 JNIEnv *env, jobject thiz, jstring jname) {
1053 sp<IDrm> drm = GetDrm(env, thiz);
1054
1055 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001056 jniThrowException(env, "java/lang/IllegalStateException",
1057 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001058 return NULL;
1059 }
1060
1061 if (jname == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001062 jniThrowException(env, "java/lang/IllegalArgumentException",
1063 "property name String is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001064 return NULL;
1065 }
1066
1067 String8 name = JStringToString8(env, jname);
1068 Vector<uint8_t> value;
1069
1070 status_t err = drm->getPropertyByteArray(name, value);
1071
1072 if (throwExceptionAsNecessary(env, err, "Failed to get property")) {
1073 return NULL;
1074 }
1075
1076 return VectorToJByteArray(env, value);
1077}
1078
1079static void android_media_MediaDrm_setPropertyString(
1080 JNIEnv *env, jobject thiz, jstring jname, jstring jvalue) {
1081 sp<IDrm> drm = GetDrm(env, thiz);
1082
1083 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001084 jniThrowException(env, "java/lang/IllegalStateException",
1085 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001086 return;
1087 }
1088
Jeff Tinkereada5372013-05-21 12:48:14 -07001089 if (jname == NULL) {
1090 jniThrowException(env, "java/lang/IllegalArgumentException",
1091 "property name String is null");
1092 return;
1093 }
1094
1095 if (jvalue == NULL) {
1096 jniThrowException(env, "java/lang/IllegalArgumentException",
1097 "property value String is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001098 return;
1099 }
1100
1101 String8 name = JStringToString8(env, jname);
1102 String8 value = JStringToString8(env, jvalue);
1103
1104 status_t err = drm->setPropertyString(name, value);
1105
1106 throwExceptionAsNecessary(env, err, "Failed to set property");
1107}
1108
1109static void android_media_MediaDrm_setPropertyByteArray(
1110 JNIEnv *env, jobject thiz, jstring jname, jbyteArray jvalue) {
1111 sp<IDrm> drm = GetDrm(env, thiz);
1112
1113 if (drm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001114 jniThrowException(env, "java/lang/IllegalStateException",
1115 "MediaDrm obj is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001116 return;
1117 }
1118
Jeff Tinkereada5372013-05-21 12:48:14 -07001119 if (jname == NULL) {
1120 jniThrowException(env, "java/lang/IllegalArgumentException",
1121 "property name String is null");
1122 return;
1123 }
1124
1125 if (jvalue == NULL) {
1126 jniThrowException(env, "java/lang/IllegalArgumentException",
1127 "property value byte array is null");
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001128 return;
1129 }
1130
1131 String8 name = JStringToString8(env, jname);
1132 Vector<uint8_t> value = JByteArrayToVector(env, jvalue);
1133
1134 status_t err = drm->setPropertyByteArray(name, value);
1135
1136 throwExceptionAsNecessary(env, err, "Failed to set property");
1137}
1138
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001139static void android_media_MediaDrm_setCipherAlgorithmNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001140 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001141 jstring jalgorithm) {
1142
1143 sp<IDrm> drm = GetDrm(env, jdrm);
1144
1145 if (!CheckSession(env, drm, jsessionId)) {
1146 return;
1147 }
1148
1149 if (jalgorithm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001150 jniThrowException(env, "java/lang/IllegalArgumentException",
1151 "algorithm String is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001152 return;
1153 }
1154
1155 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1156 String8 algorithm = JStringToString8(env, jalgorithm);
1157
1158 status_t err = drm->setCipherAlgorithm(sessionId, algorithm);
1159
1160 throwExceptionAsNecessary(env, err, "Failed to set cipher algorithm");
1161}
1162
1163static void android_media_MediaDrm_setMacAlgorithmNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001164 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001165 jstring jalgorithm) {
1166
1167 sp<IDrm> drm = GetDrm(env, jdrm);
1168
1169 if (!CheckSession(env, drm, jsessionId)) {
1170 return;
1171 }
1172
1173 if (jalgorithm == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001174 jniThrowException(env, "java/lang/IllegalArgumentException",
1175 "algorithm String is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001176 return;
1177 }
1178
1179 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1180 String8 algorithm = JStringToString8(env, jalgorithm);
1181
1182 status_t err = drm->setMacAlgorithm(sessionId, algorithm);
1183
1184 throwExceptionAsNecessary(env, err, "Failed to set mac algorithm");
1185}
1186
1187
1188static jbyteArray android_media_MediaDrm_encryptNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001189 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001190 jbyteArray jkeyId, jbyteArray jinput, jbyteArray jiv) {
1191
1192 sp<IDrm> drm = GetDrm(env, jdrm);
1193
1194 if (!CheckSession(env, drm, jsessionId)) {
1195 return NULL;
1196 }
1197
1198 if (jkeyId == NULL || jinput == NULL || jiv == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001199 jniThrowException(env, "java/lang/IllegalArgumentException",
1200 "required argument is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001201 return NULL;
1202 }
1203
1204 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1205 Vector<uint8_t> keyId(JByteArrayToVector(env, jkeyId));
1206 Vector<uint8_t> input(JByteArrayToVector(env, jinput));
1207 Vector<uint8_t> iv(JByteArrayToVector(env, jiv));
1208 Vector<uint8_t> output;
1209
1210 status_t err = drm->encrypt(sessionId, keyId, input, iv, output);
1211
Jeff Tinker8117d8f2013-08-16 13:46:02 -07001212 if (throwExceptionAsNecessary(env, err, "Failed to encrypt")) {
1213 return NULL;
1214 }
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001215
1216 return VectorToJByteArray(env, output);
1217}
1218
1219static jbyteArray android_media_MediaDrm_decryptNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001220 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001221 jbyteArray jkeyId, jbyteArray jinput, jbyteArray jiv) {
1222
1223 sp<IDrm> drm = GetDrm(env, jdrm);
1224
1225 if (!CheckSession(env, drm, jsessionId)) {
1226 return NULL;
1227 }
1228
1229 if (jkeyId == NULL || jinput == NULL || jiv == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001230 jniThrowException(env, "java/lang/IllegalArgumentException",
1231 "required argument is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001232 return NULL;
1233 }
1234
1235 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1236 Vector<uint8_t> keyId(JByteArrayToVector(env, jkeyId));
1237 Vector<uint8_t> input(JByteArrayToVector(env, jinput));
1238 Vector<uint8_t> iv(JByteArrayToVector(env, jiv));
1239 Vector<uint8_t> output;
1240
1241 status_t err = drm->decrypt(sessionId, keyId, input, iv, output);
Jeff Tinker8117d8f2013-08-16 13:46:02 -07001242 if (throwExceptionAsNecessary(env, err, "Failed to decrypt")) {
1243 return NULL;
1244 }
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001245
1246 return VectorToJByteArray(env, output);
1247}
1248
1249static jbyteArray android_media_MediaDrm_signNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001250 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001251 jbyteArray jkeyId, jbyteArray jmessage) {
1252
1253 sp<IDrm> drm = GetDrm(env, jdrm);
1254
1255 if (!CheckSession(env, drm, jsessionId)) {
1256 return NULL;
1257 }
1258
1259 if (jkeyId == NULL || jmessage == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001260 jniThrowException(env, "java/lang/IllegalArgumentException",
1261 "required argument is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001262 return NULL;
1263 }
1264
1265 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1266 Vector<uint8_t> keyId(JByteArrayToVector(env, jkeyId));
1267 Vector<uint8_t> message(JByteArrayToVector(env, jmessage));
1268 Vector<uint8_t> signature;
1269
1270 status_t err = drm->sign(sessionId, keyId, message, signature);
1271
Jeff Tinker8117d8f2013-08-16 13:46:02 -07001272 if (throwExceptionAsNecessary(env, err, "Failed to sign")) {
1273 return NULL;
1274 }
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001275
1276 return VectorToJByteArray(env, signature);
1277}
1278
1279static jboolean android_media_MediaDrm_verifyNative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001280 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001281 jbyteArray jkeyId, jbyteArray jmessage, jbyteArray jsignature) {
1282
1283 sp<IDrm> drm = GetDrm(env, jdrm);
1284
1285 if (!CheckSession(env, drm, jsessionId)) {
1286 return false;
1287 }
1288
1289 if (jkeyId == NULL || jmessage == NULL || jsignature == NULL) {
Jeff Tinkereada5372013-05-21 12:48:14 -07001290 jniThrowException(env, "java/lang/IllegalArgumentException",
1291 "required argument is null");
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001292 return false;
1293 }
1294
1295 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1296 Vector<uint8_t> keyId(JByteArrayToVector(env, jkeyId));
1297 Vector<uint8_t> message(JByteArrayToVector(env, jmessage));
1298 Vector<uint8_t> signature(JByteArrayToVector(env, jsignature));
1299 bool match;
1300
1301 status_t err = drm->verify(sessionId, keyId, message, signature, match);
1302
1303 throwExceptionAsNecessary(env, err, "Failed to verify");
1304 return match;
1305}
1306
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001307
Jeff Tinkere4095a82014-03-04 13:17:11 -08001308static jbyteArray android_media_MediaDrm_signRSANative(
Andreas Gampe5a15d0d2014-11-10 18:19:40 -08001309 JNIEnv *env, jobject /* thiz */, jobject jdrm, jbyteArray jsessionId,
Jeff Tinkere4095a82014-03-04 13:17:11 -08001310 jstring jalgorithm, jbyteArray jwrappedKey, jbyteArray jmessage) {
1311
1312 sp<IDrm> drm = GetDrm(env, jdrm);
1313
1314 if (!CheckSession(env, drm, jsessionId)) {
1315 return NULL;
1316 }
1317
1318 if (jalgorithm == NULL || jwrappedKey == NULL || jmessage == NULL) {
1319 jniThrowException(env, "java/lang/IllegalArgumentException",
1320 "required argument is null");
1321 return NULL;
1322 }
1323
1324 Vector<uint8_t> sessionId(JByteArrayToVector(env, jsessionId));
1325 String8 algorithm = JStringToString8(env, jalgorithm);
1326 Vector<uint8_t> wrappedKey(JByteArrayToVector(env, jwrappedKey));
1327 Vector<uint8_t> message(JByteArrayToVector(env, jmessage));
1328 Vector<uint8_t> signature;
1329
1330 status_t err = drm->signRSA(sessionId, algorithm, message, wrappedKey, signature);
1331
1332 if (throwExceptionAsNecessary(env, err, "Failed to sign")) {
1333 return NULL;
1334 }
1335
1336 return VectorToJByteArray(env, signature);
1337}
1338
1339
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001340static JNINativeMethod gMethods[] = {
1341 { "release", "()V", (void *)android_media_MediaDrm_release },
1342 { "native_init", "()V", (void *)android_media_MediaDrm_native_init },
1343
1344 { "native_setup", "(Ljava/lang/Object;[B)V",
1345 (void *)android_media_MediaDrm_native_setup },
1346
1347 { "native_finalize", "()V",
1348 (void *)android_media_MediaDrm_native_finalize },
1349
Jeff Tinker7cda4912013-08-21 11:52:34 -07001350 { "isCryptoSchemeSupportedNative", "([BLjava/lang/String;)Z",
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001351 (void *)android_media_MediaDrm_isCryptoSchemeSupportedNative },
1352
1353 { "openSession", "()[B",
1354 (void *)android_media_MediaDrm_openSession },
1355
1356 { "closeSession", "([B)V",
1357 (void *)android_media_MediaDrm_closeSession },
1358
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001359 { "getKeyRequest", "([B[BLjava/lang/String;ILjava/util/HashMap;)"
1360 "Landroid/media/MediaDrm$KeyRequest;",
1361 (void *)android_media_MediaDrm_getKeyRequest },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001362
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001363 { "provideKeyResponse", "([B[B)[B",
1364 (void *)android_media_MediaDrm_provideKeyResponse },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001365
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001366 { "removeKeys", "([B)V",
1367 (void *)android_media_MediaDrm_removeKeys },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001368
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001369 { "restoreKeys", "([B[B)V",
1370 (void *)android_media_MediaDrm_restoreKeys },
1371
1372 { "queryKeyStatus", "([B)Ljava/util/HashMap;",
1373 (void *)android_media_MediaDrm_queryKeyStatus },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001374
Jeff Tinkere4095a82014-03-04 13:17:11 -08001375 { "getProvisionRequestNative", "(ILjava/lang/String;)Landroid/media/MediaDrm$ProvisionRequest;",
1376 (void *)android_media_MediaDrm_getProvisionRequestNative },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001377
Jeff Tinkere4095a82014-03-04 13:17:11 -08001378 { "provideProvisionResponseNative", "([B)Landroid/media/MediaDrm$Certificate;",
1379 (void *)android_media_MediaDrm_provideProvisionResponseNative },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001380
Jeff Tinker6bf5b602014-04-30 10:02:39 -07001381 { "unprovisionDevice", "()V",
1382 (void *)android_media_MediaDrm_unprovisionDeviceNative },
1383
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001384 { "getSecureStops", "()Ljava/util/List;",
1385 (void *)android_media_MediaDrm_getSecureStops },
1386
1387 { "releaseSecureStops", "([B)V",
1388 (void *)android_media_MediaDrm_releaseSecureStops },
1389
1390 { "getPropertyString", "(Ljava/lang/String;)Ljava/lang/String;",
1391 (void *)android_media_MediaDrm_getPropertyString },
1392
1393 { "getPropertyByteArray", "(Ljava/lang/String;)[B",
1394 (void *)android_media_MediaDrm_getPropertyByteArray },
1395
1396 { "setPropertyString", "(Ljava/lang/String;Ljava/lang/String;)V",
1397 (void *)android_media_MediaDrm_setPropertyString },
1398
1399 { "setPropertyByteArray", "(Ljava/lang/String;[B)V",
1400 (void *)android_media_MediaDrm_setPropertyByteArray },
Jeff Tinker16b8cff2013-03-30 16:26:13 -07001401
1402 { "setCipherAlgorithmNative",
1403 "(Landroid/media/MediaDrm;[BLjava/lang/String;)V",
1404 (void *)android_media_MediaDrm_setCipherAlgorithmNative },
1405
1406 { "setMacAlgorithmNative",
1407 "(Landroid/media/MediaDrm;[BLjava/lang/String;)V",
1408 (void *)android_media_MediaDrm_setMacAlgorithmNative },
1409
1410 { "encryptNative", "(Landroid/media/MediaDrm;[B[B[B[B)[B",
1411 (void *)android_media_MediaDrm_encryptNative },
1412
1413 { "decryptNative", "(Landroid/media/MediaDrm;[B[B[B[B)[B",
1414 (void *)android_media_MediaDrm_decryptNative },
1415
1416 { "signNative", "(Landroid/media/MediaDrm;[B[B[B)[B",
1417 (void *)android_media_MediaDrm_signNative },
1418
1419 { "verifyNative", "(Landroid/media/MediaDrm;[B[B[B[B)Z",
1420 (void *)android_media_MediaDrm_verifyNative },
Jeff Tinkere4095a82014-03-04 13:17:11 -08001421
1422 { "signRSANative", "(Landroid/media/MediaDrm;[BLjava/lang/String;[B[B)[B",
1423 (void *)android_media_MediaDrm_signRSANative },
Jeff Tinker8a0c80f2013-02-08 10:20:44 -08001424};
1425
1426int register_android_media_Drm(JNIEnv *env) {
1427 return AndroidRuntime::registerNativeMethods(env,
1428 "android/media/MediaDrm", gMethods, NELEM(gMethods));
1429}