1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
media / cdm / win / test / media_foundation_clear_key_cdm.h [blame]
// Copyright 2023 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef MEDIA_CDM_WIN_TEST_MEDIA_FOUNDATION_CLEAR_KEY_CDM_H_
#define MEDIA_CDM_WIN_TEST_MEDIA_FOUNDATION_CLEAR_KEY_CDM_H_
#include <mfcontentdecryptionmodule.h>
#include <mferror.h>
#include <mfidl.h>
#include <windows.media.protection.h>
#include <wrl/client.h>
#include <wrl/implements.h>
#include "base/memory/ref_counted.h"
#include "base/memory/scoped_refptr.h"
#include "base/memory/weak_ptr.h"
#include "base/synchronization/lock.h"
#include "base/threading/thread_checker.h"
#include "media/cdm/aes_decryptor.h"
#include "media/cdm/win/test/media_foundation_clear_key_session.h"
namespace media {
class MediaFoundationClearKeyCdm final
: public Microsoft::WRL::RuntimeClass<
Microsoft::WRL::RuntimeClassFlags<Microsoft::WRL::ClassicCom>,
IMFContentDecryptionModule,
IMFGetService,
IMFShutdown,
Microsoft::WRL::FtmBase> {
public:
MediaFoundationClearKeyCdm();
MediaFoundationClearKeyCdm(const MediaFoundationClearKeyCdm&) = delete;
MediaFoundationClearKeyCdm& operator=(const MediaFoundationClearKeyCdm&) =
delete;
~MediaFoundationClearKeyCdm() override;
HRESULT RuntimeClassInitialize(_In_ IPropertyStore* properties);
// IMFContentDecryptionModule
STDMETHODIMP SetContentEnabler(_In_ IMFContentEnabler* content_enabler,
_In_ IMFAsyncResult* result) override;
STDMETHODIMP GetSuspendNotify(
_COM_Outptr_ IMFCdmSuspendNotify** notify) override;
STDMETHODIMP SetPMPHostApp(IMFPMPHostApp* pmp_host_app) override;
STDMETHODIMP CreateSession(
MF_MEDIAKEYSESSION_TYPE session_type,
IMFContentDecryptionModuleSessionCallbacks* callbacks,
IMFContentDecryptionModuleSession** session) override;
STDMETHODIMP SetServerCertificate(_In_reads_bytes_opt_(certificate_size)
const BYTE* certificate,
_In_ DWORD certificate_size) override;
STDMETHODIMP CreateTrustedInput(
_In_reads_bytes_(content_init_data_size) const BYTE* content_init_data,
_In_ DWORD content_init_data_size,
_COM_Outptr_ IMFTrustedInput** trusted_input) override;
STDMETHODIMP GetProtectionSystemIds(_Outptr_result_buffer_(*system_ids)
GUID** system_ids,
_Out_ DWORD* count) override;
// IMFGetService
STDMETHODIMP GetService(__RPC__in REFGUID guid_service,
__RPC__in REFIID riid,
__RPC__deref_out_opt LPVOID* ppv_object) override;
// IMFShutdown
STDMETHODIMP Shutdown() override;
STDMETHODIMP GetShutdownStatus(MFSHUTDOWN_STATUS* status) override;
private:
scoped_refptr<AesDecryptor> GetAesDecryptor();
void OnSessionMessage(const std::string& session_id,
CdmMessageType message_type,
const std::vector<uint8_t>& message);
void OnSessionClosed(const std::string& session_id,
CdmSessionClosedReason reason);
void OnSessionKeysChange(const std::string& session_id,
bool has_additional_usable_key,
CdmKeysInfo keys_info);
void OnSessionIdCreated(
const std::string& session_id,
Microsoft::WRL::ComPtr<IMFContentDecryptionModuleSession> session);
void OnSessionIdRemoved(const std::string& session_id);
MediaFoundationClearKeySession* FindSession(const std::string& session_id);
// To use the hardware protection layer, we need to create an in-process PMP
// server.
Microsoft::WRL::ComPtr<
ABI::Windows::Media::Protection::IMediaProtectionPMPServer>
media_protection_pmp_server_;
scoped_refptr<AesDecryptor> aes_decryptor_;
// Session ID to session map.
std::map<std::string,
Microsoft::WRL::ComPtr<IMFContentDecryptionModuleSession>>
sessions_;
HRESULT GetShutdownStatus() {
base::AutoLock lock(lock_);
return (is_shutdown_) ? MF_E_SHUTDOWN : S_OK;
}
// To protect access to `is_shutdown_` from multiple threads.
// `SetContentEnabler`, `CreateTrustedInput` and `GetService` may run from MF
// work queue threads. Note that this `lock_` does not lock the whole
// function, so it is still possible that Shutdown() is called when above
// calls are running.
base::Lock lock_;
// For IMFShutdown
bool is_shutdown_ GUARDED_BY(lock_) = false;
// Thread checker to enforce that this object is used on a specific thread.
THREAD_CHECKER(thread_checker_);
// NOTE: Weak pointers must be invalidated before all other member variables.
base::WeakPtrFactory<MediaFoundationClearKeyCdm> weak_factory_{this};
};
} // namespace media
#endif // MEDIA_CDM_WIN_TEST_MEDIA_FOUNDATION_CLEAR_KEY_CDM_H_