// 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/webcontentdecryptionmodule_impl.h"
#include <map>
#include <vector>
#include "base/basictypes.h"
#include "base/bind.h"
#include "base/callback_helpers.h"
#include "base/logging.h"
#include "base/memory/weak_ptr.h"
#include "base/strings/string_util.h"
#include "content/renderer/media/crypto/content_decryption_module_factory.h"
#include "content/renderer/media/webcontentdecryptionmodulesession_impl.h"
#include "media/base/media_keys.h"
#include "url/gurl.h"
namespace content {
// Forwards the session ID-based callbacks of the MediaKeys interface to the
// appropriate session object.
class SessionIdAdapter {
public:
SessionIdAdapter();
~SessionIdAdapter();
// On success, creates a MediaKeys, returns it in |media_keys|, returns true.
bool Initialize(const std::string& key_system,
scoped_ptr<media::MediaKeys>* media_keys);
// Generates a unique internal session id.
uint32 GenerateSessionId();
// Adds a session to the internal map. Does not take ownership of the session.
void AddSession(uint32 session_id,
WebContentDecryptionModuleSessionImpl* session);
// Removes a session from the internal map.
void RemoveSession(uint32 session_id);
private:
typedef std::map<uint32, WebContentDecryptionModuleSessionImpl*> SessionMap;
// Callbacks for firing session events.
void OnSessionCreated(uint32 session_id, const std::string& web_session_id);
void OnSessionMessage(uint32 session_id,
const std::vector<uint8>& message,
const std::string& destination_url);
void OnSessionReady(uint32 session_id);
void OnSessionClosed(uint32 session_id);
void OnSessionError(uint32 session_id,
media::MediaKeys::KeyError error_code,
int system_code);
// Helper function of the callbacks.
WebContentDecryptionModuleSessionImpl* GetSession(uint32 session_id);
base::WeakPtrFactory<SessionIdAdapter> weak_ptr_factory_;
SessionMap sessions_;
// Session ID should be unique per renderer process for debugging purposes.
static uint32 next_session_id_;
DISALLOW_COPY_AND_ASSIGN(SessionIdAdapter);
};
const uint32 kStartingSessionId = 1;
uint32 SessionIdAdapter::next_session_id_ = kStartingSessionId;
COMPILE_ASSERT(kStartingSessionId > media::MediaKeys::kInvalidSessionId,
invalid_starting_value);
SessionIdAdapter::SessionIdAdapter()
: weak_ptr_factory_(this) {
}
SessionIdAdapter::~SessionIdAdapter() {
}
bool SessionIdAdapter::Initialize(const std::string& key_system,
scoped_ptr<media::MediaKeys>* media_keys) {
DCHECK(media_keys);
DCHECK(!*media_keys);
base::WeakPtr<SessionIdAdapter> weak_this = weak_ptr_factory_.GetWeakPtr();
scoped_ptr<media::MediaKeys> created_media_keys =
ContentDecryptionModuleFactory::Create(
// TODO(ddorwin): Address lower in the stack: http://crbug.com/252065
"webkit-" + key_system,
#if defined(ENABLE_PEPPER_CDMS)
// TODO(ddorwin): Support Pepper-based CDMs: http://crbug.com/250049
NULL,
NULL,
base::Closure(),
#elif defined(OS_ANDROID)
// TODO(xhwang): Support Android.
NULL,
0,
// TODO(ddorwin): Get the URL for the frame containing the MediaKeys.
GURL(),
#endif // defined(ENABLE_PEPPER_CDMS)
base::Bind(&SessionIdAdapter::OnSessionCreated, weak_this),
base::Bind(&SessionIdAdapter::OnSessionMessage, weak_this),
base::Bind(&SessionIdAdapter::OnSessionReady, weak_this),
base::Bind(&SessionIdAdapter::OnSessionClosed, weak_this),
base::Bind(&SessionIdAdapter::OnSessionError, weak_this));
if (!created_media_keys)
return false;
*media_keys = created_media_keys.Pass();
return true;
}
uint32 SessionIdAdapter::GenerateSessionId() {
return next_session_id_++;
}
void SessionIdAdapter::AddSession(
uint32 session_id,
WebContentDecryptionModuleSessionImpl* session) {
DCHECK(sessions_.find(session_id) == sessions_.end());
sessions_[session_id] = session;
}
void SessionIdAdapter::RemoveSession(uint32 session_id) {
DCHECK(sessions_.find(session_id) != sessions_.end());
sessions_.erase(session_id);
}
void SessionIdAdapter::OnSessionCreated(uint32 session_id,
const std::string& web_session_id) {
GetSession(session_id)->OnSessionCreated(web_session_id);
}
void SessionIdAdapter::OnSessionMessage(uint32 session_id,
const std::vector<uint8>& message,
const std::string& destination_url) {
GetSession(session_id)->OnSessionMessage(message, destination_url);
}
void SessionIdAdapter::OnSessionReady(uint32 session_id) {
GetSession(session_id)->OnSessionReady();
}
void SessionIdAdapter::OnSessionClosed(uint32 session_id) {
GetSession(session_id)->OnSessionClosed();
}
void SessionIdAdapter::OnSessionError(uint32 session_id,
media::MediaKeys::KeyError error_code,
int system_code) {
GetSession(session_id)->OnSessionError(error_code, system_code);
}
WebContentDecryptionModuleSessionImpl* SessionIdAdapter::GetSession(
uint32 session_id) {
DCHECK(sessions_.find(session_id) != sessions_.end());
return sessions_[session_id];
}
//------------------------------------------------------------------------------
WebContentDecryptionModuleImpl*
WebContentDecryptionModuleImpl::Create(const base::string16& key_system) {
// TODO(ddorwin): Guard against this in supported types check and remove this.
// Chromium only supports ASCII key systems.
if (!IsStringASCII(key_system)) {
NOTREACHED();
return NULL;
}
// SessionIdAdapter creates the MediaKeys so it can provide its callbacks to
// during creation of the MediaKeys.
scoped_ptr<media::MediaKeys> media_keys;
scoped_ptr<SessionIdAdapter> adapter(new SessionIdAdapter());
if (!adapter->Initialize(UTF16ToASCII(key_system), &media_keys))
return NULL;
return new WebContentDecryptionModuleImpl(media_keys.Pass(), adapter.Pass());
}
WebContentDecryptionModuleImpl::WebContentDecryptionModuleImpl(
scoped_ptr<media::MediaKeys> media_keys,
scoped_ptr<SessionIdAdapter> adapter)
: media_keys_(media_keys.Pass()),
adapter_(adapter.Pass()) {
}
WebContentDecryptionModuleImpl::~WebContentDecryptionModuleImpl() {
}
// The caller owns the created session.
blink::WebContentDecryptionModuleSession*
WebContentDecryptionModuleImpl::createSession(
blink::WebContentDecryptionModuleSession::Client* client) {
DCHECK(media_keys_);
uint32 session_id = adapter_->GenerateSessionId();
WebContentDecryptionModuleSessionImpl* session =
new WebContentDecryptionModuleSessionImpl(
session_id,
media_keys_.get(),
client,
base::Bind(&WebContentDecryptionModuleImpl::OnSessionClosed,
base::Unretained(this)));
adapter_->AddSession(session_id, session);
return session;
}
void WebContentDecryptionModuleImpl::OnSessionClosed(uint32 session_id) {
adapter_->RemoveSession(session_id);
}
} // namespace content