/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "GMPVideoEncoderChild.h" #include "GMPContentChild.h" #include #include "mozilla/StaticPrefs_media.h" #include "mozilla/Unused.h" #include "GMPPlatform.h" #include "GMPVideoEncodedFrameImpl.h" #include "GMPVideoi420FrameImpl.h" #include "runnable_utils.h" namespace mozilla::gmp { GMPVideoEncoderChild::GMPVideoEncoderChild(GMPContentChild* aPlugin) : mPlugin(aPlugin), mVideoEncoder(nullptr), mVideoHost(this) { MOZ_ASSERT(mPlugin); } GMPVideoEncoderChild::~GMPVideoEncoderChild() = default; bool GMPVideoEncoderChild::MgrIsOnOwningThread() const { return !mPlugin || mPlugin->GMPMessageLoop() == MessageLoop::current(); } void GMPVideoEncoderChild::Init(GMPVideoEncoder* aEncoder) { MOZ_ASSERT(aEncoder, "Cannot initialize video encoder child without a video encoder!"); mVideoEncoder = aEncoder; } GMPVideoHostImpl& GMPVideoEncoderChild::Host() { return mVideoHost; } void GMPVideoEncoderChild::Encoded(GMPVideoEncodedFrame* aEncodedFrame, const uint8_t* aCodecSpecificInfo, uint32_t aCodecSpecificInfoLength) { if (NS_WARN_IF(!mPlugin)) { aEncodedFrame->Destroy(); return; } MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current()); auto ef = static_cast(aEncodedFrame); if (GMPSharedMemManager* memMgr = mVideoHost.SharedMemMgr()) { ipc::Shmem inputShmem; if (memMgr->MgrTakeShmem(GMPSharedMemClass::Decoded, &inputShmem)) { Unused << SendReturnShmem(std::move(inputShmem)); } } nsTArray codecSpecific; codecSpecific.AppendElements(aCodecSpecificInfo, aCodecSpecificInfoLength); GMPVideoEncodedFrameData frameData; ipc::Shmem frameShmem; nsTArray frameArray; if (ef->RelinquishFrameData(frameData, frameShmem)) { Unused << SendEncodedShmem(frameData, std::move(frameShmem), codecSpecific); } else if (ef->RelinquishFrameData(frameData, frameArray)) { Unused << SendEncodedData(frameData, std::move(frameArray), codecSpecific); } else { MOZ_CRASH("Encoded without any frame data!"); } aEncodedFrame->Destroy(); } void GMPVideoEncoderChild::Error(GMPErr aError) { if (NS_WARN_IF(!mPlugin)) { return; } MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current()); SendError(aError); } mozilla::ipc::IPCResult GMPVideoEncoderChild::RecvInitEncode( const GMPVideoCodec& aCodecSettings, nsTArray&& aCodecSpecific, const int32_t& aNumberOfCores, const uint32_t& aMaxPayloadSize) { if (!mVideoEncoder) { return IPC_FAIL(this, "!mVideoDecoder"); } // Ignore any return code. It is OK for this to fail without killing the // process. mVideoEncoder->InitEncode(aCodecSettings, aCodecSpecific.Elements(), aCodecSpecific.Length(), this, aNumberOfCores, aMaxPayloadSize); return IPC_OK(); } mozilla::ipc::IPCResult GMPVideoEncoderChild::RecvGiveShmem( ipc::Shmem&& aOutputShmem) { if (GMPSharedMemManager* memMgr = mVideoHost.SharedMemMgr()) { memMgr->MgrGiveShmem(GMPSharedMemClass::Encoded, std::move(aOutputShmem)); } else { DeallocShmem(aOutputShmem); } return IPC_OK(); } mozilla::ipc::IPCResult GMPVideoEncoderChild::RecvEncode( const GMPVideoi420FrameData& aInputFrame, ipc::Shmem&& aInputShmem, nsTArray&& aCodecSpecificInfo, nsTArray&& aFrameTypes) { if (!mVideoEncoder) { DeallocShmem(aInputShmem); return IPC_FAIL(this, "!mVideoDecoder"); } auto* f = new GMPVideoi420FrameImpl(aInputFrame, std::move(aInputShmem), &mVideoHost); // Ignore any return code. It is OK for this to fail without killing the // process. mVideoEncoder->Encode(f, aCodecSpecificInfo.Elements(), aCodecSpecificInfo.Length(), aFrameTypes.Elements(), aFrameTypes.Length()); return IPC_OK(); } mozilla::ipc::IPCResult GMPVideoEncoderChild::RecvSetChannelParameters( const uint32_t& aPacketLoss, const uint32_t& aRTT) { if (!mVideoEncoder) { return IPC_FAIL(this, "!mVideoDecoder"); } // Ignore any return code. It is OK for this to fail without killing the // process. mVideoEncoder->SetChannelParameters(aPacketLoss, aRTT); return IPC_OK(); } mozilla::ipc::IPCResult GMPVideoEncoderChild::RecvSetRates( const uint32_t& aNewBitRate, const uint32_t& aFrameRate) { if (!mVideoEncoder) { return IPC_FAIL(this, "!mVideoDecoder"); } // Ignore any return code. It is OK for this to fail without killing the // process. mVideoEncoder->SetRates(aNewBitRate, aFrameRate); return IPC_OK(); } mozilla::ipc::IPCResult GMPVideoEncoderChild::RecvSetPeriodicKeyFrames( const bool& aEnable) { if (!mVideoEncoder) { return IPC_FAIL(this, "!mVideoDecoder"); } // Ignore any return code. It is OK for this to fail without killing the // process. mVideoEncoder->SetPeriodicKeyFrames(aEnable); return IPC_OK(); } void GMPVideoEncoderChild::ActorDestroy(ActorDestroyReason why) { // If there are no decoded frames, then we know that OpenH264 has destroyed // any outstanding references to its pending encode frames. This means it // should be safe to destroy the encoder since there should not be any pending // sync callbacks. if (!SpinPendingGmpEventsUntil( [&]() -> bool { return mVideoHost.IsDecodedFramesEmpty(); }, StaticPrefs::media_gmp_coder_shutdown_timeout_ms())) { NS_WARNING("Timed out waiting for synchronous events!"); } if (mVideoEncoder) { // Ignore any return code. It is OK for this to fail without killing the // process. mVideoEncoder->EncodingComplete(); mVideoEncoder = nullptr; } mVideoHost.DoneWithAPI(); mPlugin = nullptr; } } // namespace mozilla::gmp