no bug - Import translations from android-l10n r=release a=l10n CLOSED TREE
[gecko.git] / ipc / glue / UtilityAudioDecoderChild.cpp
blobcc3560e05b889eba3fc661a616964243901074c7
1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=2 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #include "UtilityAudioDecoderChild.h"
9 #include "base/basictypes.h"
10 #include "mozilla/AppShutdown.h"
11 #include "mozilla/dom/ContentParent.h"
13 #ifdef MOZ_WMF_MEDIA_ENGINE
14 # include "mozilla/StaticPrefs_media.h"
15 # include "mozilla/gfx/GPUProcessManager.h"
16 # include "mozilla/gfx/gfxVars.h"
17 # include "mozilla/ipc/UtilityProcessManager.h"
18 # include "mozilla/layers/PVideoBridge.h"
19 # include "mozilla/layers/VideoBridgeUtils.h"
20 #endif
22 #ifdef MOZ_WMF_CDM
23 # include "mozilla/dom/Promise.h"
24 # include "mozilla/EMEUtils.h"
25 # include "mozilla/PMFCDM.h"
26 #endif
28 namespace mozilla::ipc {
30 NS_IMETHODIMP UtilityAudioDecoderChildShutdownObserver::Observe(
31 nsISupports* aSubject, const char* aTopic, const char16_t* aData) {
32 MOZ_ASSERT(strcmp(aTopic, "ipc:utility-shutdown") == 0);
34 nsCOMPtr<nsIObserverService> observerService = services::GetObserverService();
35 if (observerService) {
36 observerService->RemoveObserver(this, "ipc:utility-shutdown");
39 UtilityAudioDecoderChild::Shutdown(mSandbox);
40 return NS_OK;
43 NS_IMPL_ISUPPORTS(UtilityAudioDecoderChildShutdownObserver, nsIObserver);
45 static EnumeratedArray<SandboxingKind, StaticRefPtr<UtilityAudioDecoderChild>,
46 size_t(SandboxingKind::COUNT)>
47 sAudioDecoderChilds;
49 UtilityAudioDecoderChild::UtilityAudioDecoderChild(SandboxingKind aKind)
50 : mSandbox(aKind), mAudioDecoderChildStart(TimeStamp::Now()) {
51 MOZ_ASSERT(NS_IsMainThread());
52 nsCOMPtr<nsIObserverService> observerService = services::GetObserverService();
53 if (observerService) {
54 auto* obs = new UtilityAudioDecoderChildShutdownObserver(aKind);
55 observerService->AddObserver(obs, "ipc:utility-shutdown", false);
59 nsresult UtilityAudioDecoderChild::BindToUtilityProcess(
60 RefPtr<UtilityProcessParent> aUtilityParent) {
61 Endpoint<PUtilityAudioDecoderChild> utilityAudioDecoderChildEnd;
62 Endpoint<PUtilityAudioDecoderParent> utilityAudioDecoderParentEnd;
63 nsresult rv = PUtilityAudioDecoder::CreateEndpoints(
64 aUtilityParent->OtherPid(), base::GetCurrentProcId(),
65 &utilityAudioDecoderParentEnd, &utilityAudioDecoderChildEnd);
67 if (NS_FAILED(rv)) {
68 MOZ_ASSERT(false, "Protocol endpoints failure");
69 return NS_ERROR_FAILURE;
72 nsTArray<gfx::GfxVarUpdate> updates;
73 #ifdef MOZ_WMF_MEDIA_ENGINE
74 // Only MFCDM process needs gfxVars
75 if (mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM) {
76 updates = gfx::gfxVars::FetchNonDefaultVars();
78 #endif
79 if (!aUtilityParent->SendStartUtilityAudioDecoderService(
80 std::move(utilityAudioDecoderParentEnd), std::move(updates))) {
81 MOZ_ASSERT(false, "StartUtilityAudioDecoder service failure");
82 return NS_ERROR_FAILURE;
85 Bind(std::move(utilityAudioDecoderChildEnd));
87 PROFILER_MARKER_UNTYPED("UtilityAudioDecoderChild::BindToUtilityProcess", IPC,
88 MarkerOptions(MarkerTiming::IntervalUntilNowFrom(
89 mAudioDecoderChildStart)));
90 return NS_OK;
93 void UtilityAudioDecoderChild::ActorDestroy(ActorDestroyReason aReason) {
94 MOZ_ASSERT(NS_IsMainThread());
95 #ifdef MOZ_WMF_MEDIA_ENGINE
96 if (mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM) {
97 gfx::gfxVars::RemoveReceiver(this);
99 #endif
100 Shutdown(mSandbox);
103 void UtilityAudioDecoderChild::Bind(
104 Endpoint<PUtilityAudioDecoderChild>&& aEndpoint) {
105 MOZ_ASSERT(NS_IsMainThread());
106 if (NS_WARN_IF(!aEndpoint.Bind(this))) {
107 MOZ_ASSERT_UNREACHABLE("Failed to bind UtilityAudioDecoderChild!");
108 return;
110 #ifdef MOZ_WMF_MEDIA_ENGINE
111 if (mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM) {
112 gfx::gfxVars::AddReceiver(this);
114 #endif
117 /* static */
118 void UtilityAudioDecoderChild::Shutdown(SandboxingKind aKind) {
119 sAudioDecoderChilds[aKind] = nullptr;
122 /* static */
123 RefPtr<UtilityAudioDecoderChild> UtilityAudioDecoderChild::GetSingleton(
124 SandboxingKind aKind) {
125 MOZ_ASSERT(NS_IsMainThread());
126 bool shutdown = AppShutdown::IsInOrBeyond(ShutdownPhase::XPCOMWillShutdown);
127 if (!sAudioDecoderChilds[aKind] && !shutdown) {
128 sAudioDecoderChilds[aKind] = new UtilityAudioDecoderChild(aKind);
130 return sAudioDecoderChilds[aKind];
133 mozilla::ipc::IPCResult
134 UtilityAudioDecoderChild::RecvUpdateMediaCodecsSupported(
135 const RemoteDecodeIn& aLocation,
136 const media::MediaCodecsSupported& aSupported) {
137 dom::ContentParent::BroadcastMediaCodecsSupportedUpdate(aLocation,
138 aSupported);
139 return IPC_OK();
142 #ifdef MOZ_WMF_MEDIA_ENGINE
143 mozilla::ipc::IPCResult
144 UtilityAudioDecoderChild::RecvCompleteCreatedVideoBridge() {
145 MOZ_ASSERT(NS_IsMainThread());
146 MOZ_ASSERT(mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM);
147 mHasCreatedVideoBridge = State::Created;
148 return IPC_OK();
151 void UtilityAudioDecoderChild::OnVarChanged(const gfx::GfxVarUpdate& aVar) {
152 MOZ_ASSERT(mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM);
153 SendUpdateVar(aVar);
156 void UtilityAudioDecoderChild::OnCompositorUnexpectedShutdown() {
157 MOZ_ASSERT(NS_IsMainThread());
158 MOZ_ASSERT(mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM);
159 mHasCreatedVideoBridge = State::None;
160 CreateVideoBridge();
163 bool UtilityAudioDecoderChild::CreateVideoBridge() {
164 MOZ_ASSERT(NS_IsMainThread());
165 MOZ_ASSERT(mSandbox == SandboxingKind::MF_MEDIA_ENGINE_CDM);
167 // Creating or already created, avoiding reinit a bridge.
168 if (mHasCreatedVideoBridge != State::None) {
169 return true;
171 mHasCreatedVideoBridge = State::Creating;
173 // Build content device data first; this ensure that the GPU process is fully
174 // ready.
175 gfx::ContentDeviceData contentDeviceData;
176 gfxPlatform::GetPlatform()->BuildContentDeviceData(&contentDeviceData);
178 gfx::GPUProcessManager* gpuManager = gfx::GPUProcessManager::Get();
179 if (!gpuManager) {
180 NS_WARNING("Failed to get a gpu mananger!");
181 return false;
184 // The child end is the producer of video frames; the parent end is the
185 // consumer.
186 base::ProcessId childPid = UtilityProcessManager::GetSingleton()
187 ->GetProcessParent(mSandbox)
188 ->OtherPid();
189 base::ProcessId parentPid = gpuManager->GPUProcessPid();
190 if (parentPid == base::kInvalidProcessId) {
191 NS_WARNING("GPU process Id is invald!");
192 return false;
195 ipc::Endpoint<layers::PVideoBridgeParent> parentPipe;
196 ipc::Endpoint<layers::PVideoBridgeChild> childPipe;
197 nsresult rv = layers::PVideoBridge::CreateEndpoints(parentPid, childPid,
198 &parentPipe, &childPipe);
199 if (NS_FAILED(rv)) {
200 NS_WARNING("Failed to create endpoints for video bridge!");
201 return false;
204 gpuManager->InitVideoBridge(
205 std::move(parentPipe),
206 layers::VideoBridgeSource::MFMediaEngineCDMProcess);
207 SendInitVideoBridge(std::move(childPipe), contentDeviceData);
208 return true;
210 #endif
212 #ifdef MOZ_WMF_CDM
213 void UtilityAudioDecoderChild::GetKeySystemCapabilities(
214 dom::Promise* aPromise) {
215 EME_LOG("Ask capabilities for all supported CDMs");
216 SendGetKeySystemCapabilities()->Then(
217 NS_GetCurrentThread(), __func__,
218 [promise = RefPtr<dom::Promise>(aPromise)](
219 CopyableTArray<MFCDMCapabilitiesIPDL>&& result) {
220 FallibleTArray<dom::CDMInformation> cdmInfo;
221 for (const auto& capabilities : result) {
222 EME_LOG("Received capabilities for %s",
223 NS_ConvertUTF16toUTF8(capabilities.keySystem()).get());
224 for (const auto& v : capabilities.videoCapabilities()) {
225 EME_LOG(" capabilities: video=%s",
226 NS_ConvertUTF16toUTF8(v.contentType()).get());
228 for (const auto& a : capabilities.audioCapabilities()) {
229 EME_LOG(" capabilities: audio=%s",
230 NS_ConvertUTF16toUTF8(a.contentType()).get());
232 for (const auto& e : capabilities.encryptionSchemes()) {
233 EME_LOG(" capabilities: encryptionScheme=%s",
234 EncryptionSchemeStr(e));
236 auto* info = cdmInfo.AppendElement(fallible);
237 if (!info) {
238 promise->MaybeReject(NS_ERROR_OUT_OF_MEMORY);
239 return;
241 info->mKeySystemName = capabilities.keySystem();
243 KeySystemConfig config;
244 MFCDMCapabilitiesIPDLToKeySystemConfig(capabilities, config);
245 info->mCapabilities = config.GetDebugInfo();
246 info->mClearlead =
247 DoesKeySystemSupportClearLead(info->mKeySystemName);
248 if (capabilities.isHDCP22Compatible()) {
249 info->mIsHDCP22Compatible = true;
252 promise->MaybeResolve(cdmInfo);
254 [promise = RefPtr<dom::Promise>(aPromise)](
255 const mozilla::ipc::ResponseRejectReason& aReason) {
256 EME_LOG("IPC failure for GetKeySystemCapabilities!");
257 promise->MaybeReject(NS_ERROR_DOM_MEDIA_CDM_ERR);
260 #endif
262 } // namespace mozilla::ipc