blob: 7cd33b4c60e21d15eaf6d8bb74690be86bbd1a96 [file] [log] [blame]
// Copyright 2013 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/webcontentdecryptionmodulesession_impl.h"
#include "base/bind.h"
#include "base/callback_helpers.h"
#include "base/logging.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "content/renderer/media/cdm_result_promise.h"
#include "content/renderer/media/cdm_session_adapter.h"
#include "media/base/cdm_promise.h"
#include "third_party/WebKit/public/platform/WebURL.h"
namespace content {
const char kCreateSessionUMAName[] = "CreateSession";
typedef base::Callback<blink::WebContentDecryptionModuleResult::SessionStatus(
const std::string& web_session_id)> SessionInitializedCB;
class NewSessionCdmResultPromise : public CdmResultPromise<std::string> {
public:
NewSessionCdmResultPromise(blink::WebContentDecryptionModuleResult result,
std::string uma_name,
const SessionInitializedCB& new_session_created_cb)
: CdmResultPromise<std::string>(result, uma_name),
new_session_created_cb_(new_session_created_cb) {}
protected:
virtual void OnResolve(const std::string& web_session_id) OVERRIDE {
blink::WebContentDecryptionModuleResult::SessionStatus status =
new_session_created_cb_.Run(web_session_id);
web_cdm_result_.completeWithSession(status);
}
private:
SessionInitializedCB new_session_created_cb_;
};
WebContentDecryptionModuleSessionImpl::WebContentDecryptionModuleSessionImpl(
const scoped_refptr<CdmSessionAdapter>& adapter)
: adapter_(adapter),
is_closed_(false),
weak_ptr_factory_(this) {
}
WebContentDecryptionModuleSessionImpl::
~WebContentDecryptionModuleSessionImpl() {
if (!web_session_id_.empty())
adapter_->UnregisterSession(web_session_id_);
}
void WebContentDecryptionModuleSessionImpl::setClientInterface(Client* client) {
client_ = client;
}
blink::WebString WebContentDecryptionModuleSessionImpl::sessionId() const {
return blink::WebString::fromUTF8(web_session_id_);
}
void WebContentDecryptionModuleSessionImpl::initializeNewSession(
const blink::WebString& init_data_type,
const uint8* init_data,
size_t init_data_length) {
// TODO(jrummell): Remove once blink updated.
NOTREACHED();
}
void WebContentDecryptionModuleSessionImpl::update(const uint8* response,
size_t response_length) {
// TODO(jrummell): Remove once blink updated.
NOTREACHED();
}
void WebContentDecryptionModuleSessionImpl::release() {
// TODO(jrummell): Remove once blink updated.
NOTREACHED();
}
void WebContentDecryptionModuleSessionImpl::initializeNewSession(
const blink::WebString& init_data_type,
const uint8* init_data,
size_t init_data_length,
const blink::WebString& session_type,
blink::WebContentDecryptionModuleResult result) {
// TODO(ddorwin): Guard against this in supported types check and remove this.
// Chromium only supports ASCII MIME types.
if (!base::IsStringASCII(init_data_type)) {
NOTREACHED();
std::string message = "The initialization data type " +
init_data_type.utf8() +
" is not supported by the key system.";
result.completeWithError(
blink::WebContentDecryptionModuleExceptionNotSupportedError,
0,
blink::WebString::fromUTF8(message));
return;
}
std::string init_data_type_as_ascii = base::UTF16ToASCII(init_data_type);
DLOG_IF(WARNING, init_data_type_as_ascii.find('/') != std::string::npos)
<< "init_data_type '" << init_data_type_as_ascii
<< "' may be a MIME type";
adapter_->InitializeNewSession(
init_data_type_as_ascii,
init_data,
init_data_length,
media::MediaKeys::TEMPORARY_SESSION,
scoped_ptr<media::NewSessionCdmPromise>(new NewSessionCdmResultPromise(
result,
adapter_->GetKeySystemUMAPrefix() + kCreateSessionUMAName,
base::Bind(
&WebContentDecryptionModuleSessionImpl::OnSessionInitialized,
base::Unretained(this)))));
}
void WebContentDecryptionModuleSessionImpl::update(
const uint8* response,
size_t response_length,
blink::WebContentDecryptionModuleResult result) {
DCHECK(response);
DCHECK(!web_session_id_.empty());
adapter_->UpdateSession(
web_session_id_,
response,
response_length,
scoped_ptr<media::SimpleCdmPromise>(new SimpleCdmResultPromise(result)));
}
void WebContentDecryptionModuleSessionImpl::close(
blink::WebContentDecryptionModuleResult result) {
DCHECK(!web_session_id_.empty());
adapter_->CloseSession(
web_session_id_,
scoped_ptr<media::SimpleCdmPromise>(new SimpleCdmResultPromise(result)));
}
void WebContentDecryptionModuleSessionImpl::remove(
blink::WebContentDecryptionModuleResult result) {
DCHECK(!web_session_id_.empty());
adapter_->RemoveSession(
web_session_id_,
scoped_ptr<media::SimpleCdmPromise>(new SimpleCdmResultPromise(result)));
}
void WebContentDecryptionModuleSessionImpl::getUsableKeyIds(
blink::WebContentDecryptionModuleResult result) {
DCHECK(!web_session_id_.empty());
adapter_->GetUsableKeyIds(
web_session_id_,
scoped_ptr<media::KeyIdsPromise>(
new CdmResultPromise<media::KeyIdsVector>(result)));
}
void WebContentDecryptionModuleSessionImpl::release(
blink::WebContentDecryptionModuleResult result) {
close(result);
}
void WebContentDecryptionModuleSessionImpl::OnSessionMessage(
const std::vector<uint8>& message,
const GURL& destination_url) {
DCHECK(client_) << "Client not set before message event";
client_->message(
message.empty() ? NULL : &message[0], message.size(), destination_url);
}
void WebContentDecryptionModuleSessionImpl::OnSessionKeysChange(
bool has_additional_usable_key) {
// TODO(jrummell): Update this once Blink client supports this.
}
void WebContentDecryptionModuleSessionImpl::OnSessionExpirationUpdate(
const base::Time& new_expiry_time) {
// TODO(jrummell): Update this once Blink client supports this.
// The EME spec has expiration attribute as the time in milliseconds, so use
// InMillisecondsF() to convert.
}
void WebContentDecryptionModuleSessionImpl::OnSessionReady() {
client_->ready();
}
void WebContentDecryptionModuleSessionImpl::OnSessionClosed() {
if (!is_closed_) {
is_closed_ = true;
client_->close();
}
}
void WebContentDecryptionModuleSessionImpl::OnSessionError(
media::MediaKeys::Exception exception_code,
uint32 system_code,
const std::string& error_message) {
// Convert |exception_code| back to MediaKeyErrorCode if possible.
// TODO(jrummell): Update this conversion when promises flow
// back into blink:: (as blink:: will have its own error definition).
switch (exception_code) {
case media::MediaKeys::CLIENT_ERROR:
client_->error(Client::MediaKeyErrorCodeClient, system_code);
break;
default:
// This will include all other CDM4 errors and any error generated
// by CDM5 or later.
client_->error(Client::MediaKeyErrorCodeUnknown, system_code);
break;
}
}
blink::WebContentDecryptionModuleResult::SessionStatus
WebContentDecryptionModuleSessionImpl::OnSessionInitialized(
const std::string& web_session_id) {
// CDM will return NULL if the session to be loaded can't be found.
if (web_session_id.empty())
return blink::WebContentDecryptionModuleResult::SessionNotFound;
DCHECK(web_session_id_.empty()) << "Session ID may not be changed once set.";
web_session_id_ = web_session_id;
return adapter_->RegisterSession(web_session_id_,
weak_ptr_factory_.GetWeakPtr())
? blink::WebContentDecryptionModuleResult::NewSession
: blink::WebContentDecryptionModuleResult::SessionAlreadyExists;
}
} // namespace content