| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 | // Copyright aXiuShen. All Rights Reserved.#pragma once#include "CoreTypes.h"#include "CoreGlobals.h"#include "Containers/Queue.h"#include "Math/IntPoint.h"#include "Math/Range.h"#include "MediaObjectPool.h"#include "RHI.h"#include "RHIUtilities.h"#include "Templates/SharedPointer.h"/*** Texture sample generated by the WebBrowser.*/class FWebViewTextureSample	: public IMediaPoolable{public:	/** Default constructor. */	FWebViewTextureSample()		: Buffer(nullptr)		, BufferSize(0)		, Dim(FIntPoint::ZeroValue)		, ScaleRotation(FLinearColor(1.0f, 0.0f, 0.0f, 1.0f))		, Offset(FLinearColor(0.0f, 0.0f, 0.0f, 0.0f))	{ }	/** Virtual destructor. */	virtual ~FWebViewTextureSample()	{		if (BufferSize > 0)		{			FMemory::Free(Buffer);		}	}public:	/**	* Get a writable pointer to the sample buffer.	*	* @return Sample buffer.	*/	void* GetMutableBuffer()	{		return Buffer;	}	/**	* Initialize the sample.	*	* @param InDim The sample buffer's width and height (in pixels).	* @param InDuration The duration for which the sample is valid.	* @return true on success, false otherwise.	*/	bool Initialize(const FIntPoint& InDim)	{		if (InDim.GetMin() <= 0)		{ 			return false;		}		Dim = InDim;		return true;	}	/**	* Initialize the sample with a memory buffer.	*	* @param InBuffer The buffer containing the sample data.	* @param Copy Whether the buffer should be copied (true) or referenced (false).	* @see InitializeTexture	*/	void InitializeBuffer(void* InBuffer, bool Copy)	{		if (Copy)		{			SIZE_T RequiredBufferSize = Dim.X * Dim.Y * sizeof(int32);			if (BufferSize < RequiredBufferSize)			{				if (BufferSize == 0)				{					Buffer = FMemory::Malloc(RequiredBufferSize);				}				else				{					Buffer = FMemory::Realloc(Buffer, RequiredBufferSize);				}				BufferSize = RequiredBufferSize;			}			FMemory::Memcpy(Buffer, InBuffer, RequiredBufferSize);		}		else		{			if (BufferSize > 0)			{				FMemory::Free(Buffer);				BufferSize = 0;			}			Buffer = InBuffer;		}	}	/**	* Initialize the sample for copy externally.	*	* @return The size of the buffer	* @see InitializeTexture	*/	int32 InitializeBufferForCopy()	{		SIZE_T RequiredBufferSize = Dim.X * Dim.Y * sizeof(int32);		if (BufferSize < RequiredBufferSize)		{			if (BufferSize == 0)			{				Buffer = FMemory::Malloc(RequiredBufferSize);			}			else			{				Buffer = FMemory::Realloc(Buffer, RequiredBufferSize);			}			BufferSize = RequiredBufferSize;		}		return BufferSize;	}	/**	* Initialize the sample with a texture resource.	*	* @return The texture resource object that will hold the sample data.	* @note This method must be called on the render thread.	* @see InitializeBuffer	*/	FRHITexture2D* InitializeTexture()	{		check(IsInRenderingThread());		if (Texture.IsValid() && (Texture->GetSizeXY() == Dim))		{			return Texture;		}#if WEBVIEW_ENGINE_VERSION>=50100		const FRHITextureCreateDesc Desc =			FRHITextureCreateDesc::Create2D(TEXT("FWebViewTextureSample"))			.SetExtent(Dim)			.SetFormat(PF_B8G8R8A8)			.SetFlags(ETextureCreateFlags::Dynamic | ETextureCreateFlags::SRGB | ETextureCreateFlags::RenderTargetable | ETextureCreateFlags::ShaderResource)			.SetInitialState(ERHIAccess::SRVMask);		Texture = RHICreateTexture(Desc);#else		const ETextureCreateFlags CreateFlags = TexCreate_Dynamic | TexCreate_SRGB;		TRefCountPtr<FRHITexture2D> DummyTexture2DRHI;#if WEBVIEW_ENGINE_VERSION>=50000		FRHIResourceCreateInfo CreateInfo(TEXT("FWebViewTextureSample"));#else		FRHIResourceCreateInfo CreateInfo;#endif		RHICreateTargetableShaderResource2D(			Dim.X,			Dim.Y,			PF_B8G8R8A8,			1,			CreateFlags,			TexCreate_RenderTargetable,			false,			CreateInfo,			Texture,			DummyTexture2DRHI		);#endif		return Texture;	}	/**	* Set the sample Scale, Rotation, Offset.	*	* @param InScaleRotation The sample scale and rotation transform (2x2).	* @param InOffset The sample offset.	*/	void SetScaleRotationOffset(FVector4& InScaleRotation, FVector4& InOffset)	{		ScaleRotation = FLinearColor(InScaleRotation.X, InScaleRotation.Y, InScaleRotation.Z, InScaleRotation.W);		Offset = FLinearColor(InOffset.X, InOffset.Y, InOffset.Z, InOffset.W);	}public:	//~ IMediaTextureSample interface	virtual const void* GetBuffer() 	{		return Buffer;	}	virtual FIntPoint GetDim() const	{		return Dim;	}	virtual uint32 GetStride() const	{		return Dim.X * sizeof(int32);	}#if WITH_ENGINE	virtual FRHITexture* GetTexture() const	{		return Texture.GetReference();	}#endif //WITH_ENGINE	virtual FLinearColor GetScaleRotation() const	{		return ScaleRotation;	}	virtual FLinearColor GetOffset() const	{		return Offset;	}private:	/** The sample's data buffer. */	void* Buffer;	/** Current allocation size of Buffer. */	SIZE_T BufferSize;	/** Width and height of the texture sample. */	FIntPoint Dim;	/** ScaleRotation for the sample. */	FLinearColor ScaleRotation;	/** Offset for the sample. */	FLinearColor Offset;#if WITH_ENGINE	/** Texture resource. */	TRefCountPtr<FRHITexture2D> Texture;#endif //WITH_ENGINE};class FWebViewTextureSampleQueue{public:	/** Default constructor. */	FWebViewTextureSampleQueue()		: NumSamples(0)		, PendingFlushes(0)	{ }	/** Virtual destructor. */	virtual ~FWebViewTextureSampleQueue() { }public:	/**	* Get the number of samples in the queue.	*	* @return Number of samples.	* @see Enqueue, Dequeue, Peek	*/	int32 Num() const	{		return NumSamples;	}public:	//~ TMediaSampleSource interface (to be called only from consumer thread)	virtual bool Dequeue(TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe>& OutSample)	{		DoPendingFlushes();		TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe> Sample;		if (!Samples.Peek(Sample))		{			return false; // empty queue		}		if (!Sample.IsValid())		{			return false; // pending flush		}		Samples.Pop();		FPlatformAtomics::InterlockedDecrement(&NumSamples);		check(NumSamples >= 0);		OutSample = Sample;		return true;	}	virtual bool Peek(TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe>& OutSample)	{		DoPendingFlushes();		TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe> Sample;		if (!Samples.Peek(Sample))		{			return false; // empty queue		}		if (!Sample.IsValid())		{			return false; // pending flush		}		OutSample = Sample;		return true;	}	virtual bool Pop()	{		TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe> Sample;		if (!Samples.Peek(Sample))		{			return false; // empty queue		}		if (!Sample.IsValid())		{			return false; // pending flush		}		Samples.Pop();		FPlatformAtomics::InterlockedDecrement(&NumSamples);		check(NumSamples >= 0);		return true;	}public:	//~ TMediaSampleSink interface (to be called only from producer thread)	virtual bool Enqueue(const TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe>& Sample)	{		if (Sample.IsValid())		{			FPlatformAtomics::InterlockedIncrement(&NumSamples);		}		if (!Samples.Enqueue(Sample))		{			if (Sample.IsValid())			{				FPlatformAtomics::InterlockedDecrement(&NumSamples);			}			return false;		}		return true;	}	virtual void RequestFlush()	{		Samples.Enqueue(nullptr); // insert flush marker		FPlatformAtomics::InterlockedIncrement(&PendingFlushes);	}protected:	/** Perform any pending flushes. */	void DoPendingFlushes()	{		TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe> Sample;		while ((PendingFlushes > 0) && Samples.Dequeue(Sample))		{			if (Sample.IsValid())			{				FPlatformAtomics::InterlockedDecrement(&NumSamples);				check(NumSamples >= 0);			}			else			{				FPlatformAtomics::InterlockedDecrement(&PendingFlushes);			}		}	}private:	/** Number of samples in the queue. */	int32 NumSamples;	/** Number of pending flushes. */	int32 PendingFlushes;	/** Audio sample queue. */	TQueue<TSharedPtr<FWebViewTextureSample, ESPMode::ThreadSafe>, EQueueMode::Mpsc> Samples;};/** Implements a pool for WebBrowser's texture sample objects. */class FFWebViewTextureSamplePool : public TMediaObjectPool<FWebViewTextureSample> { };
 |