/* -*- 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 "GMPChild.h" #include #include "mozilla/unused.h" #include "GMPVideoEncodedFrameImpl.h" #include "GMPVideoi420FrameImpl.h" namespace mozilla { namespace gmp { GMPVideoEncoderChild::GMPVideoEncoderChild(GMPChild* aPlugin) : GMPSharedMemManager(aPlugin), mPlugin(aPlugin), mVideoEncoder(nullptr), mVideoHost(MOZ_THIS_IN_INITIALIZER_LIST()) { MOZ_ASSERT(mPlugin); } GMPVideoEncoderChild::~GMPVideoEncoderChild() { } 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) { MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current()); auto ef = static_cast(aEncodedFrame); GMPVideoEncodedFrameData frameData; ef->RelinquishFrameData(frameData); nsTArray codecSpecific; codecSpecific.AppendElements(aCodecSpecificInfo, aCodecSpecificInfoLength); SendEncoded(frameData, codecSpecific); aEncodedFrame->Destroy(); } void GMPVideoEncoderChild::Error(GMPErr aError) { MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current()); SendError(aError); } bool GMPVideoEncoderChild::RecvInitEncode(const GMPVideoCodec& aCodecSettings, const nsTArray& aCodecSpecific, const int32_t& aNumberOfCores, const uint32_t& aMaxPayloadSize) { if (!mVideoEncoder) { return false; } // 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 true; } bool GMPVideoEncoderChild::RecvEncode(const GMPVideoi420FrameData& aInputFrame, const nsTArray& aCodecSpecificInfo, const nsTArray& aFrameTypes) { if (!mVideoEncoder) { return false; } auto f = new GMPVideoi420FrameImpl(aInputFrame, &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 true; } bool GMPVideoEncoderChild::RecvChildShmemForPool(Shmem& aEncodedBuffer) { if (aEncodedBuffer.IsWritable()) { mVideoHost.SharedMemMgr()->MgrDeallocShmem(GMPSharedMem::kGMPEncodedData, aEncodedBuffer); } return true; } bool GMPVideoEncoderChild::RecvSetChannelParameters(const uint32_t& aPacketLoss, const uint32_t& aRTT) { if (!mVideoEncoder) { return false; } // Ignore any return code. It is OK for this to fail without killing the process. mVideoEncoder->SetChannelParameters(aPacketLoss, aRTT); return true; } bool GMPVideoEncoderChild::RecvSetRates(const uint32_t& aNewBitRate, const uint32_t& aFrameRate) { if (!mVideoEncoder) { return false; } // Ignore any return code. It is OK for this to fail without killing the process. mVideoEncoder->SetRates(aNewBitRate, aFrameRate); return true; } bool GMPVideoEncoderChild::RecvSetPeriodicKeyFrames(const bool& aEnable) { if (!mVideoEncoder) { return false; } // Ignore any return code. It is OK for this to fail without killing the process. mVideoEncoder->SetPeriodicKeyFrames(aEnable); return true; } bool GMPVideoEncoderChild::RecvEncodingComplete() { if (!mVideoEncoder) { unused << Send__delete__(this); return false; } // Ignore any return code. It is OK for this to fail without killing the process. mVideoEncoder->EncodingComplete(); mVideoHost.DoneWithAPI(); mPlugin = nullptr; unused << Send__delete__(this); return true; } } // namespace gmp } // namespace mozilla