• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "content/renderer/media/cdm_session_adapter.h"
6 
7 #include "base/bind.h"
8 #include "base/logging.h"
9 #include "base/memory/weak_ptr.h"
10 #include "base/stl_util.h"
11 #include "content/renderer/media/crypto/content_decryption_module_factory.h"
12 #include "content/renderer/media/crypto/key_systems.h"
13 #include "content/renderer/media/webcontentdecryptionmodulesession_impl.h"
14 #include "media/base/cdm_promise.h"
15 #include "media/base/media_keys.h"
16 #include "url/gurl.h"
17 
18 namespace content {
19 
20 const char kMediaEME[] = "Media.EME.";
21 const char kDot[] = ".";
22 
CdmSessionAdapter()23 CdmSessionAdapter::CdmSessionAdapter() :
24 #if defined(ENABLE_BROWSER_CDMS)
25     cdm_id_(0),
26 #endif
27     weak_ptr_factory_(this) {}
28 
~CdmSessionAdapter()29 CdmSessionAdapter::~CdmSessionAdapter() {}
30 
Initialize(const CreatePepperCdmCB & create_pepper_cdm_cb,const std::string & key_system,const GURL & security_origin)31 bool CdmSessionAdapter::Initialize(
32 #if defined(ENABLE_PEPPER_CDMS)
33     const CreatePepperCdmCB& create_pepper_cdm_cb,
34 #elif defined(ENABLE_BROWSER_CDMS)
35     RendererCdmManager* manager,
36 #endif  // defined(ENABLE_PEPPER_CDMS)
37     const std::string& key_system,
38     const GURL& security_origin) {
39   key_system_uma_prefix_ = kMediaEME + KeySystemNameForUMA(key_system) + kDot;
40   base::WeakPtr<CdmSessionAdapter> weak_this = weak_ptr_factory_.GetWeakPtr();
41   media_keys_ = ContentDecryptionModuleFactory::Create(
42       key_system,
43       security_origin,
44 #if defined(ENABLE_PEPPER_CDMS)
45       create_pepper_cdm_cb,
46 #elif defined(ENABLE_BROWSER_CDMS)
47       manager,
48       &cdm_id_,
49 #endif  // defined(ENABLE_PEPPER_CDMS)
50       base::Bind(&CdmSessionAdapter::OnSessionMessage, weak_this),
51       base::Bind(&CdmSessionAdapter::OnSessionReady, weak_this),
52       base::Bind(&CdmSessionAdapter::OnSessionClosed, weak_this),
53       base::Bind(&CdmSessionAdapter::OnSessionError, weak_this),
54       base::Bind(&CdmSessionAdapter::OnSessionKeysChange, weak_this),
55       base::Bind(&CdmSessionAdapter::OnSessionExpirationUpdate, weak_this));
56 
57   // Success if |media_keys_| created.
58   return media_keys_;
59 }
60 
SetServerCertificate(const uint8 * server_certificate,int server_certificate_length,scoped_ptr<media::SimpleCdmPromise> promise)61 void CdmSessionAdapter::SetServerCertificate(
62     const uint8* server_certificate,
63     int server_certificate_length,
64     scoped_ptr<media::SimpleCdmPromise> promise) {
65   media_keys_->SetServerCertificate(
66       server_certificate, server_certificate_length, promise.Pass());
67 }
68 
CreateSession()69 WebContentDecryptionModuleSessionImpl* CdmSessionAdapter::CreateSession() {
70   return new WebContentDecryptionModuleSessionImpl(this);
71 }
72 
RegisterSession(const std::string & web_session_id,base::WeakPtr<WebContentDecryptionModuleSessionImpl> session)73 bool CdmSessionAdapter::RegisterSession(
74     const std::string& web_session_id,
75     base::WeakPtr<WebContentDecryptionModuleSessionImpl> session) {
76   // If this session ID is already registered, don't register it again.
77   if (ContainsKey(sessions_, web_session_id))
78     return false;
79 
80   sessions_[web_session_id] = session;
81   return true;
82 }
83 
UnregisterSession(const std::string & web_session_id)84 void CdmSessionAdapter::UnregisterSession(const std::string& web_session_id) {
85   DCHECK(ContainsKey(sessions_, web_session_id));
86   sessions_.erase(web_session_id);
87 }
88 
InitializeNewSession(const std::string & init_data_type,const uint8 * init_data,int init_data_length,media::MediaKeys::SessionType session_type,scoped_ptr<media::NewSessionCdmPromise> promise)89 void CdmSessionAdapter::InitializeNewSession(
90     const std::string& init_data_type,
91     const uint8* init_data,
92     int init_data_length,
93     media::MediaKeys::SessionType session_type,
94     scoped_ptr<media::NewSessionCdmPromise> promise) {
95   media_keys_->CreateSession(init_data_type,
96                              init_data,
97                              init_data_length,
98                              session_type,
99                              promise.Pass());
100 }
101 
UpdateSession(const std::string & web_session_id,const uint8 * response,int response_length,scoped_ptr<media::SimpleCdmPromise> promise)102 void CdmSessionAdapter::UpdateSession(
103     const std::string& web_session_id,
104     const uint8* response,
105     int response_length,
106     scoped_ptr<media::SimpleCdmPromise> promise) {
107   media_keys_->UpdateSession(
108       web_session_id, response, response_length, promise.Pass());
109 }
110 
CloseSession(const std::string & web_session_id,scoped_ptr<media::SimpleCdmPromise> promise)111 void CdmSessionAdapter::CloseSession(
112     const std::string& web_session_id,
113     scoped_ptr<media::SimpleCdmPromise> promise) {
114   media_keys_->CloseSession(web_session_id, promise.Pass());
115 }
116 
RemoveSession(const std::string & web_session_id,scoped_ptr<media::SimpleCdmPromise> promise)117 void CdmSessionAdapter::RemoveSession(
118     const std::string& web_session_id,
119     scoped_ptr<media::SimpleCdmPromise> promise) {
120   media_keys_->RemoveSession(web_session_id, promise.Pass());
121 }
122 
GetUsableKeyIds(const std::string & web_session_id,scoped_ptr<media::KeyIdsPromise> promise)123 void CdmSessionAdapter::GetUsableKeyIds(
124     const std::string& web_session_id,
125     scoped_ptr<media::KeyIdsPromise> promise) {
126   media_keys_->GetUsableKeyIds(web_session_id, promise.Pass());
127 }
128 
GetDecryptor()129 media::Decryptor* CdmSessionAdapter::GetDecryptor() {
130   return media_keys_->GetDecryptor();
131 }
132 
GetKeySystemUMAPrefix() const133 const std::string& CdmSessionAdapter::GetKeySystemUMAPrefix() const {
134   return key_system_uma_prefix_;
135 }
136 
137 #if defined(ENABLE_BROWSER_CDMS)
GetCdmId() const138 int CdmSessionAdapter::GetCdmId() const {
139   return cdm_id_;
140 }
141 #endif  // defined(ENABLE_BROWSER_CDMS)
142 
OnSessionMessage(const std::string & web_session_id,const std::vector<uint8> & message,const GURL & destination_url)143 void CdmSessionAdapter::OnSessionMessage(const std::string& web_session_id,
144                                          const std::vector<uint8>& message,
145                                          const GURL& destination_url) {
146   WebContentDecryptionModuleSessionImpl* session = GetSession(web_session_id);
147   DLOG_IF(WARNING, !session) << __FUNCTION__ << " for unknown session "
148                              << web_session_id;
149   if (session)
150     session->OnSessionMessage(message, destination_url);
151 }
152 
OnSessionKeysChange(const std::string & web_session_id,bool has_additional_usable_key)153 void CdmSessionAdapter::OnSessionKeysChange(const std::string& web_session_id,
154                                             bool has_additional_usable_key) {
155   WebContentDecryptionModuleSessionImpl* session = GetSession(web_session_id);
156   DLOG_IF(WARNING, !session) << __FUNCTION__ << " for unknown session "
157                              << web_session_id;
158   if (session)
159     session->OnSessionKeysChange(has_additional_usable_key);
160 }
161 
OnSessionExpirationUpdate(const std::string & web_session_id,const base::Time & new_expiry_time)162 void CdmSessionAdapter::OnSessionExpirationUpdate(
163     const std::string& web_session_id,
164     const base::Time& new_expiry_time) {
165   WebContentDecryptionModuleSessionImpl* session = GetSession(web_session_id);
166   DLOG_IF(WARNING, !session) << __FUNCTION__ << " for unknown session "
167                              << web_session_id;
168   if (session)
169     session->OnSessionExpirationUpdate(new_expiry_time);
170 }
171 
OnSessionReady(const std::string & web_session_id)172 void CdmSessionAdapter::OnSessionReady(const std::string& web_session_id) {
173   WebContentDecryptionModuleSessionImpl* session = GetSession(web_session_id);
174   DLOG_IF(WARNING, !session) << __FUNCTION__ << " for unknown session "
175                              << web_session_id;
176   if (session)
177     session->OnSessionReady();
178 }
179 
OnSessionClosed(const std::string & web_session_id)180 void CdmSessionAdapter::OnSessionClosed(const std::string& web_session_id) {
181   WebContentDecryptionModuleSessionImpl* session = GetSession(web_session_id);
182   DLOG_IF(WARNING, !session) << __FUNCTION__ << " for unknown session "
183                              << web_session_id;
184   if (session)
185     session->OnSessionClosed();
186 }
187 
OnSessionError(const std::string & web_session_id,media::MediaKeys::Exception exception_code,uint32 system_code,const std::string & error_message)188 void CdmSessionAdapter::OnSessionError(
189     const std::string& web_session_id,
190     media::MediaKeys::Exception exception_code,
191     uint32 system_code,
192     const std::string& error_message) {
193   WebContentDecryptionModuleSessionImpl* session = GetSession(web_session_id);
194   DLOG_IF(WARNING, !session) << __FUNCTION__ << " for unknown session "
195                              << web_session_id;
196   if (session)
197     session->OnSessionError(exception_code, system_code, error_message);
198 }
199 
GetSession(const std::string & web_session_id)200 WebContentDecryptionModuleSessionImpl* CdmSessionAdapter::GetSession(
201     const std::string& web_session_id) {
202   // Since session objects may get garbage collected, it is possible that there
203   // are events coming back from the CDM and the session has been unregistered.
204   // We can not tell if the CDM is firing events at sessions that never existed.
205   SessionMap::iterator session = sessions_.find(web_session_id);
206   return (session != sessions_.end()) ? session->second.get() : NULL;
207 }
208 
209 }  // namespace content
210