blob: 67402da7a84b2a18f4f0b2bdeeedb4532648526e [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/renderer/media/cdm/renderer_cdm_manager.h"
#include <stddef.h>
#include <utility>
#include "base/stl_util.h"
#include "content/common/media/cdm_messages.h"
#include "content/renderer/media/cdm/proxy_media_keys.h"
#include "media/base/cdm_context.h"
#include "media/base/limits.h"
namespace content {
using media::MediaKeys;
// Maximum sizes for various EME API parameters. These are checks to prevent
// unnecessarily large messages from being passed around, and the sizes
// are somewhat arbitrary as the EME spec doesn't specify any limits.
const size_t kMaxSessionMessageLength = 10240; // 10 KB
RendererCdmManager::RendererCdmManager(RenderFrame* render_frame)
: RenderFrameObserver(render_frame),
next_cdm_id_(media::CdmContext::kInvalidCdmId + 1) {}
RendererCdmManager::~RendererCdmManager() {
DCHECK(proxy_media_keys_map_.empty())
<< "RendererCdmManager is owned by RenderFrameImpl and is destroyed only "
"after all ProxyMediaKeys are destroyed and unregistered.";
}
bool RendererCdmManager::OnMessageReceived(const IPC::Message& msg) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(RendererCdmManager, msg)
IPC_MESSAGE_HANDLER(CdmMsg_SessionMessage, OnSessionMessage)
IPC_MESSAGE_HANDLER(CdmMsg_SessionClosed, OnSessionClosed)
IPC_MESSAGE_HANDLER(CdmMsg_LegacySessionError, OnLegacySessionError)
IPC_MESSAGE_HANDLER(CdmMsg_SessionKeysChange, OnSessionKeysChange)
IPC_MESSAGE_HANDLER(CdmMsg_SessionExpirationUpdate,
OnSessionExpirationUpdate)
IPC_MESSAGE_HANDLER(CdmMsg_ResolvePromise, OnPromiseResolved)
IPC_MESSAGE_HANDLER(CdmMsg_ResolvePromiseWithSession,
OnPromiseResolvedWithSession)
IPC_MESSAGE_HANDLER(CdmMsg_RejectPromise, OnPromiseRejected)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
void RendererCdmManager::InitializeCdm(int cdm_id,
uint32_t promise_id,
ProxyMediaKeys* media_keys,
const std::string& key_system,
const GURL& security_origin,
bool use_hw_secure_codecs) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
CdmHostMsg_InitializeCdm_Params params;
params.key_system = key_system;
params.security_origin = security_origin;
params.use_hw_secure_codecs = use_hw_secure_codecs;
Send(new CdmHostMsg_InitializeCdm(routing_id(), cdm_id, promise_id, params));
}
void RendererCdmManager::SetServerCertificate(
int cdm_id,
uint32_t promise_id,
const std::vector<uint8_t>& certificate) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
Send(new CdmHostMsg_SetServerCertificate(routing_id(), cdm_id, promise_id,
certificate));
}
void RendererCdmManager::CreateSessionAndGenerateRequest(
int cdm_id,
uint32_t promise_id,
media::MediaKeys::SessionType session_type,
CdmHostMsg_CreateSession_InitDataType init_data_type,
const std::vector<uint8_t>& init_data) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
CdmHostMsg_CreateSessionAndGenerateRequest_Params params;
params.render_frame_id = routing_id();
params.cdm_id = cdm_id;
params.promise_id = promise_id;
params.session_type = session_type;
params.init_data_type = init_data_type;
params.init_data = init_data;
Send(new CdmHostMsg_CreateSessionAndGenerateRequest(params));
}
void RendererCdmManager::LoadSession(int cdm_id,
uint32_t promise_id,
media::MediaKeys::SessionType session_type,
const std::string& session_id) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
Send(new CdmHostMsg_LoadSession(routing_id(), cdm_id, promise_id,
session_type, session_id));
}
void RendererCdmManager::UpdateSession(int cdm_id,
uint32_t promise_id,
const std::string& session_id,
const std::vector<uint8_t>& response) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
Send(new CdmHostMsg_UpdateSession(routing_id(), cdm_id, promise_id,
session_id, response));
}
void RendererCdmManager::CloseSession(int cdm_id,
uint32_t promise_id,
const std::string& session_id) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
Send(new CdmHostMsg_CloseSession(routing_id(), cdm_id, promise_id,
session_id));
}
void RendererCdmManager::RemoveSession(int cdm_id,
uint32_t promise_id,
const std::string& session_id) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
Send(new CdmHostMsg_RemoveSession(routing_id(), cdm_id, promise_id,
session_id));
}
void RendererCdmManager::DestroyCdm(int cdm_id) {
DCHECK(GetMediaKeys(cdm_id)) << "|cdm_id| not registered.";
Send(new CdmHostMsg_DestroyCdm(routing_id(), cdm_id));
}
void RendererCdmManager::OnSessionMessage(
int cdm_id,
const std::string& session_id,
media::MediaKeys::MessageType message_type,
const std::vector<uint8_t>& message,
const GURL& legacy_destination_url) {
if (message.size() > kMaxSessionMessageLength) {
NOTREACHED();
LOG(ERROR) << "Message is too long and dropped.";
return;
}
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnSessionMessage(session_id, message_type, message,
legacy_destination_url);
}
void RendererCdmManager::OnSessionClosed(int cdm_id,
const std::string& session_id) {
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnSessionClosed(session_id);
}
void RendererCdmManager::OnLegacySessionError(
int cdm_id,
const std::string& session_id,
MediaKeys::Exception exception,
uint32_t system_code,
const std::string& error_message) {
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnLegacySessionError(session_id, exception, system_code,
error_message);
}
void RendererCdmManager::OnSessionKeysChange(
int cdm_id,
const std::string& session_id,
bool has_additional_usable_key,
const std::vector<media::CdmKeyInformation>& key_info_vector) {
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (!media_keys)
return;
media::CdmKeysInfo keys_info;
keys_info.reserve(key_info_vector.size());
for (const auto& key_info : key_info_vector)
keys_info.push_back(new media::CdmKeyInformation(key_info));
media_keys->OnSessionKeysChange(session_id, has_additional_usable_key,
std::move(keys_info));
}
void RendererCdmManager::OnSessionExpirationUpdate(
int cdm_id,
const std::string& session_id,
const base::Time& new_expiry_time) {
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnSessionExpirationUpdate(session_id, new_expiry_time);
}
void RendererCdmManager::OnPromiseResolved(int cdm_id, uint32_t promise_id) {
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnPromiseResolved(promise_id);
}
void RendererCdmManager::OnPromiseResolvedWithSession(
int cdm_id,
uint32_t promise_id,
const std::string& session_id) {
if (session_id.length() > media::limits::kMaxSessionIdLength) {
NOTREACHED();
OnPromiseRejected(cdm_id, promise_id, MediaKeys::INVALID_ACCESS_ERROR, 0,
"Session ID is too long");
return;
}
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnPromiseResolvedWithSession(promise_id, session_id);
}
void RendererCdmManager::OnPromiseRejected(int cdm_id,
uint32_t promise_id,
MediaKeys::Exception exception,
uint32_t system_code,
const std::string& error_message) {
ProxyMediaKeys* media_keys = GetMediaKeys(cdm_id);
if (media_keys)
media_keys->OnPromiseRejected(promise_id, exception, system_code,
error_message);
}
int RendererCdmManager::RegisterMediaKeys(ProxyMediaKeys* media_keys) {
int cdm_id = next_cdm_id_++;
DCHECK_NE(cdm_id, media::CdmContext::kInvalidCdmId);
DCHECK(!ContainsKey(proxy_media_keys_map_, cdm_id));
proxy_media_keys_map_[cdm_id] = media_keys;
return cdm_id;
}
void RendererCdmManager::UnregisterMediaKeys(int cdm_id) {
DCHECK(ContainsKey(proxy_media_keys_map_, cdm_id));
proxy_media_keys_map_.erase(cdm_id);
}
ProxyMediaKeys* RendererCdmManager::GetMediaKeys(int cdm_id) {
std::map<int, ProxyMediaKeys*>::iterator iter =
proxy_media_keys_map_.find(cdm_id);
return (iter != proxy_media_keys_map_.end()) ? iter->second : NULL;
}
} // namespace content