mirror of
				https://github.com/mozilla/gecko-dev.git
				synced 2025-11-04 10:18:41 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			247 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* -*- 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 "GMPVideoDecoderChild.h"
 | 
						|
#include "GMPVideoi420FrameImpl.h"
 | 
						|
#include "GMPContentChild.h"
 | 
						|
#include <stdio.h>
 | 
						|
#include "mozilla/Unused.h"
 | 
						|
#include "GMPVideoEncodedFrameImpl.h"
 | 
						|
#include "runnable_utils.h"
 | 
						|
 | 
						|
namespace mozilla::gmp {
 | 
						|
 | 
						|
GMPVideoDecoderChild::GMPVideoDecoderChild(GMPContentChild* aPlugin)
 | 
						|
    : GMPSharedMemManager(aPlugin),
 | 
						|
      mPlugin(aPlugin),
 | 
						|
      mVideoDecoder(nullptr),
 | 
						|
      mVideoHost(this),
 | 
						|
      mNeedShmemIntrCount(0),
 | 
						|
      mPendingDecodeComplete(false) {
 | 
						|
  MOZ_ASSERT(mPlugin);
 | 
						|
}
 | 
						|
 | 
						|
GMPVideoDecoderChild::~GMPVideoDecoderChild() {
 | 
						|
  MOZ_ASSERT(!mNeedShmemIntrCount);
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::Init(GMPVideoDecoder* aDecoder) {
 | 
						|
  MOZ_ASSERT(aDecoder,
 | 
						|
             "Cannot initialize video decoder child without a video decoder!");
 | 
						|
  mVideoDecoder = aDecoder;
 | 
						|
}
 | 
						|
 | 
						|
GMPVideoHostImpl& GMPVideoDecoderChild::Host() { return mVideoHost; }
 | 
						|
 | 
						|
void GMPVideoDecoderChild::Decoded(GMPVideoi420Frame* aDecodedFrame) {
 | 
						|
  if (!aDecodedFrame) {
 | 
						|
    MOZ_CRASH("Not given a decoded frame!");
 | 
						|
  }
 | 
						|
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    aDecodedFrame->Destroy();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  auto df = static_cast<GMPVideoi420FrameImpl*>(aDecodedFrame);
 | 
						|
 | 
						|
  GMPVideoi420FrameData frameData;
 | 
						|
  df->InitFrameData(frameData);
 | 
						|
  SendDecoded(frameData);
 | 
						|
 | 
						|
  aDecodedFrame->Destroy();
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::ReceivedDecodedReferenceFrame(
 | 
						|
    const uint64_t aPictureId) {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  SendReceivedDecodedReferenceFrame(aPictureId);
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::ReceivedDecodedFrame(const uint64_t aPictureId) {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  SendReceivedDecodedFrame(aPictureId);
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::InputDataExhausted() {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  SendInputDataExhausted();
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::DrainComplete() {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  SendDrainComplete();
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::ResetComplete() {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  SendResetComplete();
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::Error(GMPErr aError) {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  SendError(aError);
 | 
						|
}
 | 
						|
 | 
						|
mozilla::ipc::IPCResult GMPVideoDecoderChild::RecvInitDecode(
 | 
						|
    const GMPVideoCodec& aCodecSettings, nsTArray<uint8_t>&& aCodecSpecific,
 | 
						|
    const int32_t& aCoreCount) {
 | 
						|
  if (!mVideoDecoder) {
 | 
						|
    return IPC_FAIL(this, "!mVideoDecoder");
 | 
						|
  }
 | 
						|
 | 
						|
  // Ignore any return code. It is OK for this to fail without killing the
 | 
						|
  // process.
 | 
						|
  mVideoDecoder->InitDecode(aCodecSettings, aCodecSpecific.Elements(),
 | 
						|
                            aCodecSpecific.Length(), this, aCoreCount);
 | 
						|
  return IPC_OK();
 | 
						|
}
 | 
						|
 | 
						|
mozilla::ipc::IPCResult GMPVideoDecoderChild::RecvDecode(
 | 
						|
    const GMPVideoEncodedFrameData& aInputFrame, const bool& aMissingFrames,
 | 
						|
    nsTArray<uint8_t>&& aCodecSpecificInfo, const int64_t& aRenderTimeMs) {
 | 
						|
  if (!mVideoDecoder) {
 | 
						|
    return IPC_FAIL(this, "!mVideoDecoder");
 | 
						|
  }
 | 
						|
 | 
						|
  auto f = new GMPVideoEncodedFrameImpl(aInputFrame, &mVideoHost);
 | 
						|
 | 
						|
  // Ignore any return code. It is OK for this to fail without killing the
 | 
						|
  // process.
 | 
						|
  mVideoDecoder->Decode(f, aMissingFrames, aCodecSpecificInfo.Elements(),
 | 
						|
                        aCodecSpecificInfo.Length(), aRenderTimeMs);
 | 
						|
 | 
						|
  return IPC_OK();
 | 
						|
}
 | 
						|
 | 
						|
mozilla::ipc::IPCResult GMPVideoDecoderChild::RecvChildShmemForPool(
 | 
						|
    Shmem&& aFrameBuffer) {
 | 
						|
  GMPSharedMemManager* memMgr = mVideoHost.SharedMemMgr();
 | 
						|
  if (memMgr && aFrameBuffer.IsWritable()) {
 | 
						|
    memMgr->MgrDeallocShmem(GMPSharedMem::kGMPFrameData, aFrameBuffer);
 | 
						|
  }
 | 
						|
  return IPC_OK();
 | 
						|
}
 | 
						|
 | 
						|
mozilla::ipc::IPCResult GMPVideoDecoderChild::RecvReset() {
 | 
						|
  if (!mVideoDecoder) {
 | 
						|
    return IPC_FAIL(this, "!mVideoDecoder");
 | 
						|
  }
 | 
						|
 | 
						|
  // Ignore any return code. It is OK for this to fail without killing the
 | 
						|
  // process.
 | 
						|
  mVideoDecoder->Reset();
 | 
						|
 | 
						|
  return IPC_OK();
 | 
						|
}
 | 
						|
 | 
						|
mozilla::ipc::IPCResult GMPVideoDecoderChild::RecvDrain() {
 | 
						|
  if (!mVideoDecoder) {
 | 
						|
    return IPC_FAIL(this, "!mVideoDecoder");
 | 
						|
  }
 | 
						|
 | 
						|
  // Ignore any return code. It is OK for this to fail without killing the
 | 
						|
  // process.
 | 
						|
  mVideoDecoder->Drain();
 | 
						|
 | 
						|
  return IPC_OK();
 | 
						|
}
 | 
						|
 | 
						|
mozilla::ipc::IPCResult GMPVideoDecoderChild::RecvDecodingComplete() {
 | 
						|
  MOZ_ASSERT(mPlugin);
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  if (mNeedShmemIntrCount) {
 | 
						|
    // There's a GMP blocked in Alloc() waiting for the CallNeedShem() to
 | 
						|
    // return a frame they can use. Don't call the GMP's DecodingComplete()
 | 
						|
    // now and don't delete the GMPVideoDecoderChild, defer processing the
 | 
						|
    // DecodingComplete() until once the Alloc() finishes.
 | 
						|
    mPendingDecodeComplete = true;
 | 
						|
    return IPC_OK();
 | 
						|
  }
 | 
						|
 | 
						|
  // This will call ActorDestroy.
 | 
						|
  Unused << Send__delete__(this);
 | 
						|
  return IPC_OK();
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::ActorDestroy(ActorDestroyReason why) {
 | 
						|
  if (mVideoDecoder) {
 | 
						|
    // Ignore any return code. It is OK for this to fail without killing the
 | 
						|
    // process.
 | 
						|
    mVideoDecoder->DecodingComplete();
 | 
						|
    mVideoDecoder = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  mVideoHost.DoneWithAPI();
 | 
						|
 | 
						|
  mPlugin = nullptr;
 | 
						|
}
 | 
						|
 | 
						|
bool GMPVideoDecoderChild::Alloc(size_t aSize, Shmem* aMem) {
 | 
						|
  if (NS_WARN_IF(!mPlugin)) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(mPlugin->GMPMessageLoop() == MessageLoop::current());
 | 
						|
 | 
						|
  bool rv;
 | 
						|
#ifndef SHMEM_ALLOC_IN_CHILD
 | 
						|
  ++mNeedShmemIntrCount;
 | 
						|
  rv = SendNeedShmem(aSize, aMem);
 | 
						|
  --mNeedShmemIntrCount;
 | 
						|
  if (mPendingDecodeComplete && mNeedShmemIntrCount == 0) {
 | 
						|
    mPendingDecodeComplete = false;
 | 
						|
    mPlugin->GMPMessageLoop()->PostTask(
 | 
						|
        NewRunnableMethod("gmp::GMPVideoDecoderChild::RecvDecodingComplete",
 | 
						|
                          this, &GMPVideoDecoderChild::RecvDecodingComplete));
 | 
						|
  }
 | 
						|
#else
 | 
						|
  rv = AllocShmem(aSize, aType, aMem);
 | 
						|
#endif
 | 
						|
  return rv;
 | 
						|
}
 | 
						|
 | 
						|
void GMPVideoDecoderChild::Dealloc(Shmem&& aMem) {
 | 
						|
#ifndef SHMEM_ALLOC_IN_CHILD
 | 
						|
  SendParentShmemForPool(std::move(aMem));
 | 
						|
#else
 | 
						|
  DeallocShmem(aMem);
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace mozilla::gmp
 |