2013-09-25 11:47:14 +00:00
|
|
|
/* vim:set ts=2 sw=2 sts=2 et: */
|
|
|
|
/* Any copyright is dedicated to the Public Domain.
|
|
|
|
* http://creativecommons.org/publicdomain/zero/1.0/
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
|
|
|
|
#include "mozilla/layers/TextureClient.h"
|
|
|
|
#include "mozilla/layers/TextureHost.h"
|
|
|
|
#include "gfx2DGlue.h"
|
|
|
|
#include "gfxImageSurface.h"
|
|
|
|
#include "gfxTypes.h"
|
2013-10-16 00:42:03 +00:00
|
|
|
#include "ImageContainer.h"
|
|
|
|
#include "mozilla/layers/YCbCrImageDataSerializer.h"
|
2013-09-25 11:47:14 +00:00
|
|
|
|
|
|
|
using namespace mozilla;
|
|
|
|
using namespace mozilla::layers;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This test performs the following actions:
|
|
|
|
* - creates a surface
|
|
|
|
* - initialize a texture client with it
|
|
|
|
* - serilaizes the texture client
|
|
|
|
* - deserializes the data into a texture host
|
|
|
|
* - reads the surface from the texture host.
|
|
|
|
*
|
|
|
|
* The surface in the end should be equal to the inital one.
|
|
|
|
* This test is run for different combinations of texture types and
|
|
|
|
* image formats.
|
|
|
|
*/
|
|
|
|
|
2013-10-09 22:37:07 +00:00
|
|
|
|
|
|
|
// fills the surface with values betwee 0 and 100.
|
|
|
|
void SetupSurface(gfxImageSurface* surface) {
|
|
|
|
int bpp = gfxASurface::BytePerPixelFromFormat(surface->Format());
|
|
|
|
int stride = surface->Stride();
|
|
|
|
uint8_t val = 0;
|
|
|
|
uint8_t* data = surface->Data();
|
|
|
|
for (int y = 0; y < surface->Height(); ++y) {
|
|
|
|
for (int x = 0; x < surface->Height(); ++x) {
|
|
|
|
for (int b = 0; b < bpp; ++b) {
|
|
|
|
data[y*stride + x*bpp + b] = val;
|
|
|
|
if (val == 100) {
|
|
|
|
val = 0;
|
|
|
|
} else {
|
|
|
|
++val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return true if two surfaces contain the same data
|
|
|
|
void AssertSurfacesEqual(gfxImageSurface* surface1,
|
|
|
|
gfxImageSurface* surface2)
|
|
|
|
{
|
|
|
|
ASSERT_EQ(surface1->GetSize(), surface2->GetSize());
|
|
|
|
ASSERT_EQ(surface1->Format(), surface2->Format());
|
|
|
|
|
|
|
|
uint8_t* data1 = surface1->Data();
|
|
|
|
uint8_t* data2 = surface2->Data();
|
|
|
|
int stride1 = surface1->Stride();
|
|
|
|
int stride2 = surface2->Stride();
|
|
|
|
int bpp = gfxASurface::BytePerPixelFromFormat(surface1->Format());
|
|
|
|
|
|
|
|
for (int y = 0; y < surface1->Height(); ++y) {
|
|
|
|
for (int x = 0; x < surface1->Width(); ++x) {
|
|
|
|
for (int b = 0; b < bpp; ++b) {
|
|
|
|
ASSERT_EQ(data1[y*stride1 + x*bpp + b],
|
|
|
|
data2[y*stride2 + x*bpp + b]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-16 00:42:03 +00:00
|
|
|
// Same as above, for YCbCr surfaces
|
|
|
|
void AssertYCbCrSurfacesEqual(PlanarYCbCrData* surface1,
|
|
|
|
PlanarYCbCrData* surface2)
|
|
|
|
{
|
|
|
|
ASSERT_EQ(surface1->mYSize, surface2->mYSize);
|
|
|
|
ASSERT_EQ(surface1->mCbCrSize, surface2->mCbCrSize);
|
|
|
|
ASSERT_EQ(surface1->mStereoMode, surface2->mStereoMode);
|
|
|
|
ASSERT_EQ(surface1->mPicSize, surface2->mPicSize);
|
|
|
|
|
|
|
|
for (int y = 0; y < surface1->mYSize.height; ++y) {
|
|
|
|
for (int x = 0; x < surface1->mYSize.width; ++x) {
|
|
|
|
ASSERT_EQ(surface1->mYChannel[y*surface1->mYStride + x*(1+surface1->mYSkip)],
|
|
|
|
surface2->mYChannel[y*surface2->mYStride + x*(1+surface2->mYSkip)]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int y = 0; y < surface1->mCbCrSize.height; ++y) {
|
|
|
|
for (int x = 0; x < surface1->mCbCrSize.width; ++x) {
|
|
|
|
ASSERT_EQ(surface1->mCbChannel[y*surface1->mCbCrStride + x*(1+surface1->mCbSkip)],
|
|
|
|
surface2->mCbChannel[y*surface2->mCbCrStride + x*(1+surface2->mCbSkip)]);
|
|
|
|
ASSERT_EQ(surface1->mCrChannel[y*surface1->mCbCrStride + x*(1+surface1->mCrSkip)],
|
|
|
|
surface2->mCrChannel[y*surface2->mCbCrStride + x*(1+surface2->mCrSkip)]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-25 11:47:14 +00:00
|
|
|
// Run the test for a texture client and a surface
|
|
|
|
void TestTextureClientSurface(TextureClient* texture, gfxImageSurface* surface) {
|
|
|
|
|
|
|
|
// client allocation
|
|
|
|
ASSERT_TRUE(texture->AsTextureClientSurface() != nullptr);
|
|
|
|
TextureClientSurface* client = texture->AsTextureClientSurface();
|
|
|
|
client->AllocateForSurface(ToIntSize(surface->GetSize()));
|
|
|
|
ASSERT_TRUE(texture->IsAllocated());
|
|
|
|
|
|
|
|
// client painting
|
|
|
|
client->UpdateSurface(surface);
|
|
|
|
|
|
|
|
nsRefPtr<gfxASurface> aSurface = client->GetAsSurface();
|
|
|
|
nsRefPtr<gfxImageSurface> clientSurface = aSurface->GetAsImageSurface();
|
|
|
|
|
|
|
|
ASSERT_TRUE(texture->Lock(OPEN_READ_ONLY));
|
|
|
|
AssertSurfacesEqual(surface, clientSurface);
|
|
|
|
texture->Unlock();
|
|
|
|
|
|
|
|
// client serialization
|
|
|
|
SurfaceDescriptor descriptor;
|
|
|
|
ASSERT_TRUE(texture->ToSurfaceDescriptor(descriptor));
|
|
|
|
|
|
|
|
ASSERT_NE(descriptor.type(), SurfaceDescriptor::Tnull_t);
|
|
|
|
|
|
|
|
// host deserialization
|
2013-12-12 01:44:47 +00:00
|
|
|
RefPtr<TextureHost> host = CreateBackendIndependentTextureHost(descriptor, nullptr,
|
2013-09-25 11:47:14 +00:00
|
|
|
texture->GetFlags());
|
|
|
|
|
|
|
|
ASSERT_TRUE(host.get() != nullptr);
|
|
|
|
ASSERT_EQ(host->GetFlags(), texture->GetFlags());
|
|
|
|
|
|
|
|
// host read
|
|
|
|
ASSERT_TRUE(host->Lock());
|
2013-11-27 09:59:01 +00:00
|
|
|
RefPtr<mozilla::gfx::DataSourceSurface> hostDataSurface = host->GetAsSurface();
|
2013-09-25 11:47:14 +00:00
|
|
|
host->Unlock();
|
|
|
|
|
2013-11-27 09:59:01 +00:00
|
|
|
nsRefPtr<gfxImageSurface> hostSurface =
|
|
|
|
new gfxImageSurface(hostDataSurface->GetData(),
|
|
|
|
ThebesIntSize(hostDataSurface->GetSize()),
|
|
|
|
hostDataSurface->Stride(),
|
|
|
|
SurfaceFormatToImageFormat(hostDataSurface->GetFormat()));
|
2013-09-25 11:47:14 +00:00
|
|
|
AssertSurfacesEqual(surface, hostSurface.get());
|
|
|
|
}
|
|
|
|
|
2013-10-16 00:42:03 +00:00
|
|
|
// Same as above, for YCbCr surfaces
|
|
|
|
void TestTextureClientYCbCr(TextureClient* client, PlanarYCbCrData& ycbcrData) {
|
|
|
|
|
|
|
|
// client allocation
|
|
|
|
ASSERT_TRUE(client->AsTextureClientYCbCr() != nullptr);
|
|
|
|
TextureClientYCbCr* texture = client->AsTextureClientYCbCr();
|
|
|
|
texture->AllocateForYCbCr(ToIntSize(ycbcrData.mYSize),
|
|
|
|
ToIntSize(ycbcrData.mCbCrSize),
|
|
|
|
ycbcrData.mStereoMode);
|
|
|
|
ASSERT_TRUE(client->IsAllocated());
|
|
|
|
|
|
|
|
// client painting
|
|
|
|
texture->UpdateYCbCr(ycbcrData);
|
|
|
|
|
|
|
|
ASSERT_TRUE(client->Lock(OPEN_READ_ONLY));
|
|
|
|
client->Unlock();
|
|
|
|
|
|
|
|
// client serialization
|
|
|
|
SurfaceDescriptor descriptor;
|
|
|
|
ASSERT_TRUE(client->ToSurfaceDescriptor(descriptor));
|
|
|
|
|
|
|
|
ASSERT_NE(descriptor.type(), SurfaceDescriptor::Tnull_t);
|
|
|
|
|
|
|
|
// host deserialization
|
2013-12-12 01:44:47 +00:00
|
|
|
RefPtr<TextureHost> textureHost = CreateBackendIndependentTextureHost(descriptor, nullptr,
|
2013-10-16 00:42:03 +00:00
|
|
|
client->GetFlags());
|
|
|
|
|
|
|
|
RefPtr<BufferTextureHost> host = static_cast<BufferTextureHost*>(textureHost.get());
|
|
|
|
|
|
|
|
ASSERT_TRUE(host.get() != nullptr);
|
|
|
|
ASSERT_EQ(host->GetFlags(), client->GetFlags());
|
|
|
|
|
|
|
|
// This will work iff the compositor is not BasicCompositor
|
|
|
|
ASSERT_EQ(host->GetFormat(), mozilla::gfx::FORMAT_YUV);
|
|
|
|
|
|
|
|
// host read
|
|
|
|
ASSERT_TRUE(host->Lock());
|
|
|
|
|
|
|
|
ASSERT_TRUE(host->GetFormat() == mozilla::gfx::FORMAT_YUV);
|
|
|
|
|
|
|
|
YCbCrImageDataDeserializer yuvDeserializer(host->GetBuffer());
|
|
|
|
ASSERT_TRUE(yuvDeserializer.IsValid());
|
|
|
|
PlanarYCbCrData data;
|
|
|
|
data.mYChannel = yuvDeserializer.GetYData();
|
|
|
|
data.mCbChannel = yuvDeserializer.GetCbData();
|
|
|
|
data.mCrChannel = yuvDeserializer.GetCrData();
|
|
|
|
data.mYStride = yuvDeserializer.GetYStride();
|
|
|
|
data.mCbCrStride = yuvDeserializer.GetCbCrStride();
|
|
|
|
data.mStereoMode = yuvDeserializer.GetStereoMode();
|
|
|
|
data.mYSize = yuvDeserializer.GetYSize();
|
|
|
|
data.mCbCrSize = yuvDeserializer.GetCbCrSize();
|
|
|
|
data.mYSkip = 0;
|
|
|
|
data.mCbSkip = 0;
|
|
|
|
data.mCrSkip = 0;
|
|
|
|
data.mPicSize = data.mYSize;
|
|
|
|
data.mPicX = 0;
|
|
|
|
data.mPicY = 0;
|
|
|
|
|
|
|
|
AssertYCbCrSurfacesEqual(&ycbcrData, &data);
|
|
|
|
host->Unlock();
|
|
|
|
}
|
|
|
|
|
2013-09-25 11:47:14 +00:00
|
|
|
TEST(Layers, TextureSerialization) {
|
|
|
|
// the test is run on all the following image formats
|
|
|
|
gfxImageFormat formats[3] = {
|
|
|
|
gfxImageFormatARGB32,
|
|
|
|
gfxImageFormatRGB24,
|
|
|
|
gfxImageFormatA8,
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int f = 0; f < 3; ++f) {
|
|
|
|
RefPtr<gfxImageSurface> surface = new gfxImageSurface(gfxIntSize(400,300), formats[f]);
|
|
|
|
SetupSurface(surface.get());
|
|
|
|
AssertSurfacesEqual(surface, surface);
|
|
|
|
|
|
|
|
RefPtr<TextureClient> client
|
|
|
|
= new MemoryTextureClient(nullptr,
|
|
|
|
mozilla::gfx::ImageFormatToSurfaceFormat(surface->Format()),
|
2013-12-12 01:44:51 +00:00
|
|
|
TEXTURE_DEALLOCATE_CLIENT);
|
2013-09-25 11:47:14 +00:00
|
|
|
|
|
|
|
TestTextureClientSurface(client, surface);
|
|
|
|
|
|
|
|
// XXX - Test more texture client types.
|
|
|
|
}
|
|
|
|
}
|
2013-10-16 00:42:03 +00:00
|
|
|
|
|
|
|
TEST(Layers, TextureYCbCrSerialization) {
|
|
|
|
RefPtr<gfxImageSurface> ySurface = new gfxImageSurface(gfxIntSize(400,300), gfxImageFormatA8);
|
|
|
|
RefPtr<gfxImageSurface> cbSurface = new gfxImageSurface(gfxIntSize(200,150), gfxImageFormatA8);
|
|
|
|
RefPtr<gfxImageSurface> crSurface = new gfxImageSurface(gfxIntSize(200,150), gfxImageFormatA8);
|
|
|
|
SetupSurface(ySurface.get());
|
|
|
|
SetupSurface(cbSurface.get());
|
|
|
|
SetupSurface(crSurface.get());
|
|
|
|
|
|
|
|
PlanarYCbCrData clientData;
|
|
|
|
clientData.mYChannel = ySurface->Data();
|
|
|
|
clientData.mCbChannel = cbSurface->Data();
|
|
|
|
clientData.mCrChannel = crSurface->Data();
|
|
|
|
clientData.mYSize = ySurface->GetSize();
|
|
|
|
clientData.mPicSize = ySurface->GetSize();
|
|
|
|
clientData.mCbCrSize = cbSurface->GetSize();
|
|
|
|
clientData.mYStride = ySurface->Stride();
|
|
|
|
clientData.mCbCrStride = cbSurface->Stride();
|
|
|
|
clientData.mStereoMode = STEREO_MODE_MONO;
|
|
|
|
clientData.mYSkip = 0;
|
|
|
|
clientData.mCbSkip = 0;
|
|
|
|
clientData.mCrSkip = 0;
|
|
|
|
clientData.mCrSkip = 0;
|
|
|
|
clientData.mPicX = 0;
|
|
|
|
clientData.mPicX = 0;
|
|
|
|
|
|
|
|
RefPtr<TextureClient> client
|
|
|
|
= new MemoryTextureClient(nullptr,
|
|
|
|
mozilla::gfx::FORMAT_YUV,
|
2013-12-12 01:44:51 +00:00
|
|
|
TEXTURE_DEALLOCATE_CLIENT);
|
2013-10-16 00:42:03 +00:00
|
|
|
|
|
|
|
TestTextureClientYCbCr(client, clientData);
|
|
|
|
|
|
|
|
// XXX - Test more texture client types.
|
|
|
|
}
|