// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/base/android/media_drm_storage_bridge.h"
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include "base/android/callback_android.h"
#include "base/android/jni_array.h"
#include "base/android/jni_string.h"
#include "base/functional/bind.h"
#include "base/task/single_thread_task_runner.h"
#include "base/unguessable_token.h"
#include "media/base/android/android_util.h"
#include "media/base/android/media_drm_bridge.h"
#include "media/base/media_drm_key_type.h"
// Must come after all headers that specialize FromJniType() / ToJniType().
#include "media/base/android/media_jni_headers/MediaDrmStorageBridge_jni.h"
using base::android::AttachCurrentThread;
using base::android::ConvertJavaStringToUTF8;
using base::android::ConvertUTF8ToJavaString;
using base::android::JavaByteArrayToByteVector;
using base::android::JavaByteArrayToString;
using base::android::JavaParamRef;
using base::android::RunBooleanCallbackAndroid;
using base::android::RunObjectCallbackAndroid;
using base::android::ScopedJavaLocalRef;
using base::android::ToJavaByteArray;
namespace media {
MediaDrmStorageBridge::MediaDrmStorageBridge()
: task_runner_(base::SingleThreadTaskRunner::GetCurrentDefault()) {}
MediaDrmStorageBridge::~MediaDrmStorageBridge() = default;
void MediaDrmStorageBridge::Initialize(const CreateStorageCB& create_storage_cb,
InitCB init_cb) {
DCHECK(create_storage_cb);
impl_ = create_storage_cb.Run();
impl_->Initialize(base::BindOnce(&MediaDrmStorageBridge::OnInitialized,
weak_factory_.GetWeakPtr(),
std::move(init_cb)));
}
void MediaDrmStorageBridge::OnProvisioned(
JNIEnv* env,
const JavaParamRef<jobject>& j_storage,
// Callback<Boolean>
const JavaParamRef<jobject>& j_callback) {
DCHECK(impl_);
task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&MediaDrmStorage::OnProvisioned, impl_->AsWeakPtr(),
base::BindOnce(&MediaDrmStorageBridge::RunAndroidBoolCallback,
// Bind callback to WeakPtr in case callback is called
// after object is deleted.
weak_factory_.GetWeakPtr(),
CreateJavaObjectPtr(j_callback.obj()))));
}
void MediaDrmStorageBridge::OnLoadInfo(
JNIEnv* env,
const JavaParamRef<jobject>& j_storage,
const JavaParamRef<jbyteArray>& j_session_id,
// Callback<PersistentInfo>
const JavaParamRef<jobject>& j_callback) {
DCHECK(impl_);
std::string session_id;
JavaByteArrayToString(env, j_session_id, &session_id);
task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&MediaDrmStorage::LoadPersistentSession, impl_->AsWeakPtr(),
session_id,
base::BindOnce(&MediaDrmStorageBridge::OnSessionDataLoaded,
weak_factory_.GetWeakPtr(),
CreateJavaObjectPtr(j_callback.obj()), session_id)));
}
void MediaDrmStorageBridge::OnSaveInfo(
JNIEnv* env,
const JavaParamRef<jobject>& j_storage,
const JavaParamRef<jobject>& j_persist_info,
// Callback<Boolean>
const JavaParamRef<jobject>& j_callback) {
DCHECK(impl_);
std::vector<uint8_t> key_set_id;
JavaByteArrayToByteVector(
env, Java_PersistentInfo_keySetId(env, j_persist_info), &key_set_id);
std::string mime = ConvertJavaStringToUTF8(
env, Java_PersistentInfo_mimeType(env, j_persist_info));
std::string session_id;
JavaByteArrayToString(env, Java_PersistentInfo_emeId(env, j_persist_info),
&session_id);
// This function should only be called for licenses needs persistent storage
// (e.g. persistent license). STREAMING license doesn't require persistent
// storage support.
auto key_type = static_cast<MediaDrmKeyType>(
Java_PersistentInfo_keyType(env, j_persist_info));
DCHECK(key_type == MediaDrmKeyType::OFFLINE ||
key_type == MediaDrmKeyType::RELEASE);
task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&MediaDrmStorage::SavePersistentSession, impl_->AsWeakPtr(),
session_id,
MediaDrmStorage::SessionData(std::move(key_set_id), std::move(mime),
key_type),
base::BindOnce(&MediaDrmStorageBridge::RunAndroidBoolCallback,
weak_factory_.GetWeakPtr(),
CreateJavaObjectPtr(j_callback.obj()))));
}
void MediaDrmStorageBridge::OnClearInfo(
JNIEnv* env,
const JavaParamRef<jobject>& j_storage,
const JavaParamRef<jbyteArray>& j_session_id,
// Callback<Boolean>
const JavaParamRef<jobject>& j_callback) {
DCHECK(impl_);
std::string session_id;
JavaByteArrayToString(env, j_session_id, &session_id);
task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&MediaDrmStorage::RemovePersistentSession, impl_->AsWeakPtr(),
std::move(session_id),
base::BindOnce(&MediaDrmStorageBridge::RunAndroidBoolCallback,
weak_factory_.GetWeakPtr(),
CreateJavaObjectPtr(j_callback.obj()))));
}
void MediaDrmStorageBridge::RunAndroidBoolCallback(JavaObjectPtr j_callback,
bool success) {
RunBooleanCallbackAndroid(*j_callback, success);
}
void MediaDrmStorageBridge::OnInitialized(
InitCB init_cb,
bool success,
const MediaDrmStorage::MediaDrmOriginId& origin_id) {
if (!success) {
DCHECK(!origin_id);
std::move(init_cb).Run(false);
return;
}
// Note: It's possible that |success| is true but |origin_id| is empty,
// to indicate per-device provisioning. If so, do not set |origin_id_|
// so that it remains the empty string.
if (origin_id && origin_id.value()) {
origin_id_ = origin_id->ToString();
} else {
// |origin_id| is empty. However, if per-application provisioning is
// supported, the empty string is not allowed.
DCHECK(origin_id_.empty());
if (MediaDrmBridge::IsPerApplicationProvisioningSupported()) {
std::move(init_cb).Run(false);
return;
}
}
std::move(init_cb).Run(true);
}
void MediaDrmStorageBridge::OnSessionDataLoaded(
JavaObjectPtr j_callback,
const std::string& session_id,
std::unique_ptr<MediaDrmStorage::SessionData> session_data) {
if (!session_data) {
RunObjectCallbackAndroid(*j_callback, ScopedJavaLocalRef<jobject>());
return;
}
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jbyteArray> j_eme_id = ToJavaByteArray(env, session_id);
ScopedJavaLocalRef<jbyteArray> j_key_set_id =
ToJavaByteArray(env, session_data->key_set_id);
ScopedJavaLocalRef<jstring> j_mime =
ConvertUTF8ToJavaString(env, session_data->mime_type);
RunObjectCallbackAndroid(*j_callback,
Java_PersistentInfo_create(
env, j_eme_id, j_key_set_id, j_mime,
static_cast<uint32_t>(session_data->key_type)));
}
} // namespace media