/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim:set ts=2 sw=2 sts=2 et cindent: */
/* 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 "ImageBitmapUtils.h"
#include "ImageBitmapColorUtils.h"
#include "ImageContainer.h"
#include "libyuv.h"
#include "mozilla/dom/ImageBitmapBinding.h"
#include "mozilla/Function.h"
#include "mozilla/gfx/2D.h"

using namespace libyuv;
using namespace mozilla::gfx;

namespace mozilla {
namespace dom {
namespace imagebitmapformat {

class Utils;
class Utils_RGBA32;
class Utils_BGRA32;
class Utils_RGB24;
class Utils_BGR24;
class Utils_Gray8;
class Utils_YUV444P;
class Utils_YUV422P;
class Utils_YUV420P;
class Utils_YUV420SP_NV12;
class Utils_YUV420SP_NV21;
class Utils_HSV;
class Utils_Lab;
class Utils_Depth;

static int GetBytesPerPixelValue(ChannelPixelLayoutDataType aDataType)
{
  switch (aDataType)
  {
  case ChannelPixelLayoutDataType::Uint8:
    return sizeof(uint8_t);
  case ChannelPixelLayoutDataType::Int8:
    return sizeof(int8_t);
  case ChannelPixelLayoutDataType::Uint16:
    return sizeof(uint16_t);
  case ChannelPixelLayoutDataType::Int16:
    return sizeof(int16_t);
  case ChannelPixelLayoutDataType::Uint32:
    return sizeof(uint32_t);
  case ChannelPixelLayoutDataType::Int32:
    return sizeof(int32_t);
  case ChannelPixelLayoutDataType::Float32:
    return sizeof(float);
  case ChannelPixelLayoutDataType::Float64:
    return sizeof(double);
  default:
    return 0;
  }
}

/*
 * The UtilsUniquePtr is a UniquePtr to ImageBitmapFormatUtils with a customized
 * deleter which does nothing. This is used as the return type of
 * ImageBitmapFormatUtils::GetUtils to prevent users deleting the returned
 * pointer.
 */
struct DoNotDelete { void operator()(void* p) {} };
using UtilsUniquePtr = UniquePtr<Utils, DoNotDelete>;

/*
 * ImageBitmapFormatUtils is an abstract class which provides interfaces to
 * extract information of each ImageBitmapFormat and interfaces to convert
 * image data between different ImageBitmapFormats. For each kind of
 * ImageBitmapFromat, we derive a subclass from the ImageBitmapFormatUtils to
 * implement functionalities that are subject to the specific ImageBitmapFormat.
 *
 * ImageBitmapFormatUtils is an abstract class and its sub-classes are designed
 * as singletons. The singleton instance of sub-classes could be initialized and
 * accessed via the ImageBitmapFormatUtils::GetUtils() static method. The
 * singleton instance is a static local variable which does not need to be
 * released manually and, with the C++11 static initialization, the
 * initialization is thread-safe.
 *
 * ImageBitmapFormatUtils and its sub-classes are designed to unify operations
 * of ImageBitmap-extensions over different kinds of ImageBitmapFormats; they
 * provide following functionalities:
 *
 * (1) Create default/customized ImagePixelLayout object of each kind of
 *     ImageBitmapFormat.
 * (2) Store the channel counts of each ImageBitmapFormat.
 * (3) Calculate the needed buffer size of each kind of ImageBitmapFormat with
 *     given width, height and stride.
 * (4) Perform color conversion between supported ImageBitmapFormats. We use
 *     _double dispatching_ to identify the source format and destination format
 *     at run time. The _double dispatching_ here is mainly implemented by
 *     overriding the _convertTo_ method over the ImageBitmapFormatUtils class
 *     hierarchy and overloading the _convertFrom_ methods over all sub-classes
 *     of ImageBitmapFormatUtils.
 */
class Utils
{
public:
  // Get the singleton utility instance of the given ImageBitmapFormat.
  static UtilsUniquePtr GetUtils(ImageBitmapFormat aFormat);

  // Get the needed buffer size to store image data in the current
  // ImageBitmapFormat with the given width and height.
  // The current ImageBitmapFormat is the format used to implement the concrete
  // subclass of which the current instance is initialized.
  virtual uint32_t NeededBufferSize(uint32_t width, uint32_t height) = 0;

  // Creates a default ImagePixelLayout object of the current ImageBitmapFormat
  // with the given width, height and stride.
  virtual UniquePtr<ImagePixelLayout>
  CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride) = 0;

  // Convert the source image data (stored in the aSrcBuffer and described by
  // the aSrcLayout) from the current ImageBitmapFormat to the given
  // ImageBitmapFormat, aDstFormat.
  // The converted image data is stored in the aDstBuffer and described by the
  // returned ImagePixelLayout object.
  virtual UniquePtr<ImagePixelLayout>
  ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  // ConvertFrom():
  // Convert the source image data (which is in the aSrcFormat format, the pixel
  // layout is described by the aSrcLayout and the raw data is stored in the
  // aSrcBuffer) to the current ImageBitmapFormat.
  // The converted image data is stored in the aDstBuffer and described by the
  // returned ImagePixelLayout object.
  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_RGB24* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_BGR24* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_YUV444P* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_YUV422P* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_YUV420P* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_YUV420SP_NV12* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_YUV420SP_NV21* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_HSV* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_Lab* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  virtual UniquePtr<ImagePixelLayout>
  ConvertFrom(Utils_Depth* aSrcFormat, const uint8_t* aSrcBuffer, const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer) = 0;

  // Check whether or not the current ImageBitmapFormat can be converted from
  // the given ImageBitmapFormat.
  virtual bool
  CanConvertFrom(ImageBitmapFormat aSrcFormat) = 0;

  // Get the number of channels.
  uint8_t GetChannelCount() const
  {
    return mChannels;
  }

protected:
  Utils(uint32_t aChannels,
                         ChannelPixelLayoutDataType aDataType)
  : mChannels(aChannels)
  , mBytesPerPixelValue(GetBytesPerPixelValue(aDataType))
  , mDataType(aDataType)
  {
  }

  virtual ~Utils()
  {
  }

  const uint8_t mChannels;
  const int mBytesPerPixelValue;
  const ChannelPixelLayoutDataType mDataType;
};

#define DECLARE_Utils(NAME)                          \
class Utils_ ## NAME : public Utils \
{                                                                     \
private:                                                              \
  explicit Utils_ ## NAME ();                        \
  ~Utils_ ## NAME () = default;                      \
  Utils_ ## NAME (Utils_ ## NAME const &) = delete;             \
  Utils_ ## NAME (Utils_ ## NAME &&) = delete;                  \
  Utils_ ## NAME & operator=(Utils_ ## NAME const &) = delete;  \
  Utils_ ## NAME & operator=(Utils_ ## NAME &&) = delete;       \
                                                                                                  \
public:                                                     \
  static Utils_ ## NAME & GetInstance();   \
                                                            \
  virtual uint32_t NeededBufferSize(uint32_t aWidth, uint32_t aHeight) override;  \
                                                                                  \
  virtual UniquePtr<ImagePixelLayout>                          \
  CreateDefaultLayout(uint32_t, uint32_t, uint32_t) override;  \
                                                               \
  virtual UniquePtr<ImagePixelLayout>                                                             \
  ConvertTo(Utils*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                  \
  virtual UniquePtr<ImagePixelLayout>                                                                       \
  ConvertFrom(Utils_RGBA32*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override;  \
                                                                                                            \
  virtual UniquePtr<ImagePixelLayout>                                                                       \
  ConvertFrom(Utils_BGRA32*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override;  \
                                                                                                            \
  virtual UniquePtr<ImagePixelLayout>                                                                     \
  ConvertFrom(Utils_RGB24*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                          \
  virtual UniquePtr<ImagePixelLayout>                                                                     \
  ConvertFrom(Utils_BGR24*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                          \
  virtual UniquePtr<ImagePixelLayout>                                                                     \
  ConvertFrom(Utils_Gray8*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                          \
  virtual UniquePtr<ImagePixelLayout>                                                                       \
  ConvertFrom(Utils_YUV444P*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                            \
  virtual UniquePtr<ImagePixelLayout>                                                                       \
  ConvertFrom(Utils_YUV422P*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                            \
  virtual UniquePtr<ImagePixelLayout>                                                                       \
  ConvertFrom(Utils_YUV420P*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                            \
  virtual UniquePtr<ImagePixelLayout>                                                                             \
  ConvertFrom(Utils_YUV420SP_NV12*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                                  \
  virtual UniquePtr<ImagePixelLayout>                                                                             \
  ConvertFrom(Utils_YUV420SP_NV21*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                                  \
  virtual UniquePtr<ImagePixelLayout>                                                                   \
  ConvertFrom(Utils_HSV*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                        \
  virtual UniquePtr<ImagePixelLayout>                                                                   \
  ConvertFrom(Utils_Lab*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                        \
  virtual UniquePtr<ImagePixelLayout>                                                                     \
  ConvertFrom(Utils_Depth*, const uint8_t*, const ImagePixelLayout*, uint8_t*) override; \
                                                                                                          \
  virtual bool                                  \
  CanConvertFrom(ImageBitmapFormat) override;   \
};

DECLARE_Utils(RGBA32)
DECLARE_Utils(BGRA32)
DECLARE_Utils(RGB24)
DECLARE_Utils(BGR24)
DECLARE_Utils(Gray8)
DECLARE_Utils(YUV444P)
DECLARE_Utils(YUV422P)
DECLARE_Utils(YUV420P)
DECLARE_Utils(YUV420SP_NV12)
DECLARE_Utils(YUV420SP_NV21)
DECLARE_Utils(HSV)
DECLARE_Utils(Lab)
DECLARE_Utils(Depth)

#undef DECLARE_Utils

/*
 * ImageBitmapFormatUtils.
 */
/* static */ UtilsUniquePtr
Utils::GetUtils(ImageBitmapFormat aFormat)
{
  switch(aFormat)
  {
  case ImageBitmapFormat::RGBA32:
    return UtilsUniquePtr(&Utils_RGBA32::GetInstance());
  case ImageBitmapFormat::BGRA32:
    return UtilsUniquePtr(&Utils_BGRA32::GetInstance());
  case ImageBitmapFormat::RGB24:
    return UtilsUniquePtr(&Utils_RGB24::GetInstance());
  case ImageBitmapFormat::BGR24:
    return UtilsUniquePtr(&Utils_BGR24::GetInstance());
  case ImageBitmapFormat::GRAY8:
    return UtilsUniquePtr(&Utils_Gray8::GetInstance());
  case ImageBitmapFormat::YUV444P:
    return UtilsUniquePtr(&Utils_YUV444P::GetInstance());
  case ImageBitmapFormat::YUV422P:
    return UtilsUniquePtr(&Utils_YUV422P::GetInstance());
  case ImageBitmapFormat::YUV420P:
    return UtilsUniquePtr(&Utils_YUV420P::GetInstance());
  case ImageBitmapFormat::YUV420SP_NV12:
    return UtilsUniquePtr(&Utils_YUV420SP_NV12::GetInstance());
  case ImageBitmapFormat::YUV420SP_NV21:
    return UtilsUniquePtr(&Utils_YUV420SP_NV21::GetInstance());
  case ImageBitmapFormat::HSV:
    return UtilsUniquePtr(&Utils_HSV::GetInstance());
  case ImageBitmapFormat::Lab:
    return UtilsUniquePtr(&Utils_Lab::GetInstance());
  case ImageBitmapFormat::DEPTH:
    return UtilsUniquePtr(&Utils_Depth::GetInstance());
  default:
    return nullptr;
  }
}

/*
 * Helper functions.
 */
template<typename SrcType, typename DstType>
static UniquePtr<ImagePixelLayout>
CvtSimpleImgToSimpleImg(Utils* aSrcUtils, const SrcType* aSrcBuffer,
                        const ImagePixelLayout* aSrcLayout, DstType* aDstBuffer,
                        ImageBitmapFormat aDstFormat, int aDstChannelCount,
                        mozilla::function<int (const SrcType*, int, DstType*, int, int, int)> converter)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null utility object.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  const nsTArray<ChannelPixelLayout>& channels = *aSrcLayout;
  MOZ_ASSERT(channels.Length() == aSrcUtils->GetChannelCount(),
             "The channel count is wrong.");

  const int dstStride = channels[0].mWidth * aDstChannelCount * sizeof(DstType);
  int rv = converter(aSrcBuffer, channels[0].mStride,
                     aDstBuffer, dstStride,
                     channels[0].mWidth, channels[0].mHeight);

  if (NS_WARN_IF(rv != 0)) {
    return nullptr;
  }

  return CreateDefaultPixelLayout(aDstFormat, channels[0].mWidth,
                                  channels[0].mHeight, dstStride);
}

static UniquePtr<ImagePixelLayout>
CvtYUVImgToSimpleImg(Utils* aSrcUtils, const uint8_t* aSrcBuffer,
                     const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer,
                     ImageBitmapFormat aDstFormat, int aDstChannelCount,
                     mozilla::function<int (const uint8_t*, int, const uint8_t*, int, const uint8_t*, int, uint8_t*, int, int, int)> converter)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null utility object.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  const nsTArray<ChannelPixelLayout>& channels = *aSrcLayout;
  MOZ_ASSERT(channels.Length() == aSrcUtils->GetChannelCount(),
             "The channel count is wrong.");

  const int dstStride = channels[0].mWidth * aDstChannelCount * sizeof(uint8_t);
  int rv = converter(aSrcBuffer + channels[0].mOffset, channels[0].mStride,
                     aSrcBuffer + channels[1].mOffset, channels[1].mStride,
                     aSrcBuffer + channels[2].mOffset, channels[2].mStride,
                     aDstBuffer, dstStride,
                     channels[0].mWidth, channels[0].mHeight);

  if (NS_WARN_IF(rv != 0)) {
    return nullptr;
  }

  return CreateDefaultPixelLayout(aDstFormat, channels[0].mWidth,
                                  channels[0].mHeight, dstStride);
}

static UniquePtr<ImagePixelLayout>
CvtNVImgToSimpleImg(Utils* aSrcUtils, const uint8_t* aSrcBuffer,
                    const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer,
                    ImageBitmapFormat aDstFormat, int aDstChannelCount,
                    mozilla::function<int (const uint8_t*, int, const uint8_t*, int, uint8_t*, int, int, int)> converter)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null utility object.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  const nsTArray<ChannelPixelLayout>& channels = *aSrcLayout;
  MOZ_ASSERT(channels.Length() == aSrcUtils->GetChannelCount(),
             "The channel count is wrong.");

  const int dstStride = channels[0].mWidth * aDstChannelCount * sizeof(uint8_t);
  int rv = converter(aSrcBuffer + channels[0].mOffset, channels[0].mStride,
                     aSrcBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer, dstStride,
                     channels[0].mWidth, channels[0].mHeight);

  if (NS_WARN_IF(rv != 0)) {
    return nullptr;
  }

  return CreateDefaultPixelLayout(aDstFormat, channels[0].mWidth,
                                  channels[0].mHeight, dstStride);
}

static UniquePtr<ImagePixelLayout>
CvtSimpleImgToYUVImg(Utils* aSrcUtils, const uint8_t* aSrcBuffer,
                     const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer,
                     ImageBitmapFormat aDstFormat,
                     mozilla::function<int (const uint8_t*, int, uint8_t*, int, uint8_t*, int, uint8_t*, int, int, int)> converter)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null utility object.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultPixelLayout(aDstFormat, (*aSrcLayout)[0].mWidth,
                             (*aSrcLayout)[0].mHeight, (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout.");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  int rv = converter(aSrcBuffer, (*aSrcLayout)[0].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer + channels[2].mOffset, channels[2].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  if (NS_WARN_IF(rv != 0)) {
    return nullptr;
  }

  return layout;
}

static UniquePtr<ImagePixelLayout>
CvtSimpleImgToNVImg(Utils* aSrcUtils, const uint8_t* aSrcBuffer,
                    const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer,
                    ImageBitmapFormat aDstFormat,
                    mozilla::function<int (const uint8_t*, int, uint8_t*, int, uint8_t*, int, int, int)> converter)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null utility object.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultPixelLayout(aDstFormat, (*aSrcLayout)[0].mWidth,
                             (*aSrcLayout)[0].mHeight, (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout.");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  int rv = converter(aSrcBuffer, (*aSrcLayout)[0].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  if (NS_WARN_IF(rv != 0)) {
    return nullptr;
  }

  return layout;
}

template<class SrcUtilsType, class DstUtilsType>
static UniquePtr<ImagePixelLayout>
TwoPassConversion(SrcUtilsType* aSrcUtils, const uint8_t* aSrcBuffer,
                  const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer,
                  ImageBitmapFormat aMiddleFormat, DstUtilsType* aDstUtils)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null source utility.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  // I444 -> I420 -> I422
  UtilsUniquePtr yuv420PUtils = Utils::GetUtils(aMiddleFormat);
  UniquePtr<uint8_t> yuv420PBuffer(new uint8_t[yuv420PUtils->NeededBufferSize((*aSrcLayout)[0].mWidth, (*aSrcLayout)[0].mHeight)]);
  UniquePtr<ImagePixelLayout> yuv420PLayout = yuv420PUtils->ConvertFrom(aSrcUtils, aSrcBuffer, aSrcLayout, yuv420PBuffer.get());
  return yuv420PUtils->ConvertTo(aDstUtils, yuv420PBuffer.get(), yuv420PLayout.get(), aDstBuffer);
}

static UniquePtr<ImagePixelLayout>
PureCopy(Utils* aSrcUtils, const uint8_t* aSrcBuffer,
         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer,
         ImageBitmapFormat aDstFormat)
{
  MOZ_ASSERT(aSrcUtils, "Convert color from a null utility object.");
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  const nsTArray<ChannelPixelLayout>& channels = *aSrcLayout;
  MOZ_ASSERT(channels.Length() == aSrcUtils->GetChannelCount(),
             "The channel count is wrong.");


  uint32_t length = 0;

  if (aDstFormat == ImageBitmapFormat::RGBA32 ||
      aDstFormat == ImageBitmapFormat::BGRA32 ||
      aDstFormat == ImageBitmapFormat::RGB24 ||
      aDstFormat == ImageBitmapFormat::BGR24 ||
      aDstFormat == ImageBitmapFormat::GRAY8 ||
      aDstFormat == ImageBitmapFormat::HSV ||
      aDstFormat == ImageBitmapFormat::Lab ||
      aDstFormat == ImageBitmapFormat::DEPTH) {
    length = channels[0].mHeight * channels[0].mStride;
  } else if (aDstFormat == ImageBitmapFormat::YUV444P ||
             aDstFormat == ImageBitmapFormat::YUV422P ||
             aDstFormat == ImageBitmapFormat::YUV420P) {
    length = channels[0].mHeight * channels[0].mStride +
             channels[1].mHeight * channels[1].mStride +
             channels[2].mHeight * channels[2].mStride;
  } else if (aDstFormat == ImageBitmapFormat::YUV420SP_NV12 ||
             aDstFormat == ImageBitmapFormat::YUV420SP_NV21) {
    length = channels[0].mHeight * channels[0].mStride +
             channels[1].mHeight * channels[1].mStride;
  }

  memcpy(aDstBuffer, aSrcBuffer, length);

  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(*aSrcLayout));
  return layout;
}

UniquePtr<ImagePixelLayout>
CreateDefaultLayoutForSimpleImage(uint32_t aWidth, uint32_t aHeight,
                                  uint32_t aStride, int aChannels,
                                  int aBytesPerPixelValue,
                                  ChannelPixelLayoutDataType aDataType)
{
  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(aChannels));

  // set mChannels
  for (uint8_t i = 0; i < aChannels; ++i) {
    ChannelPixelLayout* channel = layout->AppendElement();
    channel->mOffset = i * aBytesPerPixelValue;
    channel->mWidth = aWidth;
    channel->mHeight = aHeight;
    channel->mDataType = aDataType; //ChannelPixelLayoutDataType::Uint8;
    channel->mStride = aStride;
    channel->mSkip = aChannels - 1;
  }

  return layout;
}

/*
 * Utils_RGBA32.
 */
/* static */Utils_RGBA32&
Utils_RGBA32::GetInstance()
{
  static Utils_RGBA32 instance;
  return instance;
}

Utils_RGBA32::Utils_RGBA32()
: Utils(4, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_RGBA32::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                        const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &libyuv::ABGRToARGB);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_RGB24* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &RGB24ToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_BGR24* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &BGR24ToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &YUV444PToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &YUV422PToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &libyuv::I420ToABGR);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &NV12ToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &NV21ToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_HSV* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &HSVToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_Lab* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGBA32, 4, &LabToRGBA32);
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::ConvertFrom(Utils_Depth* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_RGBA32::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_RGBA32::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * Utils_BGRA32.
 */
/* static */Utils_BGRA32&
Utils_BGRA32::GetInstance()
{
  static Utils_BGRA32 instance;
  return instance;
}

Utils_BGRA32::Utils_BGRA32()
: Utils(4, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_BGRA32::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                        const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &libyuv::ABGRToARGB);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_RGB24* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &RGB24ToBGRA32);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_BGR24* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &BGR24ToBGRA32);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &YUV444PToBGRA32);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &libyuv::I422ToARGB);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &libyuv::I420ToARGB);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &libyuv::NV12ToARGB);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &libyuv::NV21ToARGB);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_HSV* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &HSVToBGRA32);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_Lab* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGRA32, 4, &LabToBGRA32);
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::ConvertFrom(Utils_Depth* aSrcFormat, const uint8_t* aSrcBuffer,
                          const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_BGRA32::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::RGBA32 ||
      aSrcFormat == ImageBitmapFormat::BGRA32 ||
      aSrcFormat == ImageBitmapFormat::YUV420P) {
    return true;
  }

  return false;
}

UniquePtr<ImagePixelLayout>
Utils_BGRA32::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * Utils_RGB24.
 */
/* static */Utils_RGB24&
Utils_RGB24::GetInstance()
{
  static Utils_RGB24 instance;
  return instance;
}

Utils_RGB24::Utils_RGB24()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_RGB24::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &RGBA32ToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &BGRA32ToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &BGR24ToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &YUV444PToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &YUV422PToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &YUV420PToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &NV12ToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &NV21ToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_HSV* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &HSVToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_Lab* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, 3, &LabToRGB24);
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::ConvertFrom(Utils_Depth* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_RGB24::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_RGB24::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * Utils_BGR24.
 */
/* static */Utils_BGR24&
Utils_BGR24::GetInstance()
{
  static Utils_BGR24 instance;
  return instance;
}

Utils_BGR24::Utils_BGR24()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_BGR24::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &RGBA32ToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &BGRA32ToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &RGB24ToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &YUV444PToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &YUV422PToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &YUV420PToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &NV12ToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &NV21ToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_HSV* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &HSVToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_Lab* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<float, uint8_t>(aSrcFormat, (const float*)aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::BGR24, 3, &LabToBGR24);
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::ConvertFrom(Utils_Depth* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_BGR24::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_BGR24::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * Utils_Gray8.
 */
/* static */Utils_Gray8&
Utils_Gray8::GetInstance()
{
  static Utils_Gray8 instance;
  return instance;
}

Utils_Gray8::Utils_Gray8()
: Utils(1, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_Gray8::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &RGBA32ToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcFormat, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &BGRA32ToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &RGB24ToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, uint8_t>(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &BGR24ToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_Gray8* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &YUV444PToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &YUV422PToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtYUVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &YUV420PToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &NV12ToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtNVImgToSimpleImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::GRAY8, 1, &NV21ToGray8);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_Gray8::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_Gray8::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * class Utils_YUV444P.
 */
/* static */Utils_YUV444P&
Utils_YUV444P::GetInstance()
{
  static Utils_YUV444P instance;
  return instance;
}

Utils_YUV444P::Utils_YUV444P()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_YUV444P::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV444P, &RGBA32ToYUV444P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV444P, &libyuv::ARGBToI444);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV444P, &RGB24ToYUV444P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV444P, &BGR24ToYUV444P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV444P);
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_YUV420P*, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV444P");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  int rv = I420ToI444(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                      aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                      aSrcBuffer + srcChannels[2].mOffset, srcChannels[2].mStride,
                      aDstBuffer + channels[0].mOffset, channels[0].mStride,
                      aDstBuffer + channels[1].mOffset, channels[1].mStride,
                      aDstBuffer + channels[2].mOffset, channels[2].mStride,
                      channels[0].mWidth, channels[0].mHeight);

  if (NS_WARN_IF(rv != 0)) {
    return nullptr;
  }

  return layout;
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_YUV444P::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_YUV444P::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(mChannels));

  // set mChannels
  ChannelPixelLayout* ychannel = layout->AppendElement();
  ChannelPixelLayout* uchannel = layout->AppendElement();
  ChannelPixelLayout* vchannel = layout->AppendElement();
  ychannel->mOffset = 0;
  ychannel->mWidth  = aWidth;
  ychannel->mHeight = aHeight;
  ychannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  ychannel->mStride = aStride;
  ychannel->mSkip   = 0; // aYSkip;

  uchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight;
  uchannel->mWidth  = aWidth;
  uchannel->mHeight = aHeight;
  uchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  uchannel->mStride = aStride;
  uchannel->mSkip   = 0; // aUSkip;

  vchannel->mOffset = uchannel->mOffset + uchannel->mStride * uchannel->mHeight;
  vchannel->mWidth  = aWidth;
  vchannel->mHeight = aHeight;
  vchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  vchannel->mStride = aStride;
  vchannel->mSkip   = 0; // aVSkip;

  return layout;
}

/*
 * class Utils_YUV422P.
 */
/* static */Utils_YUV422P&
Utils_YUV422P::GetInstance()
{
  static Utils_YUV422P instance;
  return instance;
}

Utils_YUV422P::Utils_YUV422P()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_YUV422P::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * mBytesPerPixelValue +
         2 * ((aWidth + 1) / 2) * aHeight * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV422P, &RGBA32ToYUV422P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV422P, &libyuv::ARGBToI422);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV422P, &RGB24ToYUV422P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV422P, &BGR24ToYUV422P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV422P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_YUV420P*, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV422P");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::I420ToI422(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aSrcBuffer + srcChannels[2].mOffset, srcChannels[2].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer + channels[2].mOffset, channels[2].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_YUV422P::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_YUV422P::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(mChannels));

  // set mChannels
  ChannelPixelLayout* ychannel = layout->AppendElement();
  ChannelPixelLayout* uchannel = layout->AppendElement();
  ChannelPixelLayout* vchannel = layout->AppendElement();
  ychannel->mOffset = 0;
  ychannel->mWidth  = aWidth;
  ychannel->mHeight = aHeight;
  ychannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  ychannel->mStride = aStride;
  ychannel->mSkip   = 0; // aYSkip;

  uchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight;
  uchannel->mWidth  = (aWidth + 1) / 2;
  uchannel->mHeight = aHeight;
  uchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  uchannel->mStride = (aStride + 1) / 2;
  uchannel->mSkip   = 0; // aUSkip;

  vchannel->mOffset = uchannel->mOffset + uchannel->mStride * uchannel->mHeight;
  vchannel->mWidth  = (aWidth + 1) / 2;
  vchannel->mHeight = aHeight;
  vchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  vchannel->mStride = (aStride + 1) / 2;
  vchannel->mSkip   = 0; // aVSkip;

  return layout;
}

/*
 * Utils_YUV420P.
 */
/* static */Utils_YUV420P&
Utils_YUV420P::GetInstance()
{
  static Utils_YUV420P instance;
  return instance;
}

Utils_YUV420P::Utils_YUV420P()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_YUV420P::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * mBytesPerPixelValue +
         2 * ((aWidth + 1) / 2) * ((aHeight + 1) / 2) * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, &libyuv::ABGRToI420);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, &libyuv::ARGBToI420);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, &RGB24ToYUV420P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToYUVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, &BGR24ToYUV420P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_YUV444P*, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV420P");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::I444ToI420(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aSrcBuffer + srcChannels[2].mOffset, srcChannels[2].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer + channels[2].mOffset, channels[2].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_YUV422P*, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV420P");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::I422ToI420(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aSrcBuffer + srcChannels[2].mOffset, srcChannels[2].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer + channels[2].mOffset, channels[2].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_YUV420SP_NV12*, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV420P");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::NV12ToI420(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer + channels[2].mOffset, channels[2].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_YUV420SP_NV21*, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV420P");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::NV21ToI420(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     aDstBuffer + channels[2].mOffset, channels[2].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                           const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_YUV420P::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420P::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(mChannels));

  // set mChannels
  ChannelPixelLayout* ychannel = layout->AppendElement();
  ChannelPixelLayout* uchannel = layout->AppendElement();
  ChannelPixelLayout* vchannel = layout->AppendElement();
  ychannel->mOffset = 0;
  ychannel->mWidth  = aWidth;
  ychannel->mHeight = aHeight;
  ychannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  ychannel->mStride = aStride;
  ychannel->mSkip   = 0; // aYSkip;

  uchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight;
  uchannel->mWidth  = (aWidth + 1) / 2;
  uchannel->mHeight = (aHeight + 1) / 2;
  uchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  uchannel->mStride = (aStride + 1) / 2;
  uchannel->mSkip   = 0; // aUSkip;

  vchannel->mOffset = uchannel->mOffset + uchannel->mStride * uchannel->mHeight;
  vchannel->mWidth  = (aWidth + 1) / 2;
  vchannel->mHeight = (aHeight + 1) / 2;
  vchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  vchannel->mStride = (aStride + 1) / 2;
  vchannel->mSkip   = 0; // aVSkip;

  return layout;
}

/*
 * class Utils_YUV420SP_NV12.
 */
/* static */Utils_YUV420SP_NV12&
Utils_YUV420SP_NV12::GetInstance()
{
  static Utils_YUV420SP_NV12 instance;
  return instance;
}

Utils_YUV420SP_NV12::Utils_YUV420SP_NV12()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_YUV420SP_NV12::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * mBytesPerPixelValue +
         2 * ((aWidth + 1) / 2) * ((aHeight + 1) / 2) * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                               const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV12, &RGBA32ToNV12);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV12, &libyuv::ARGBToNV12);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV12, &RGB24ToNV12);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV12, &BGR24ToNV12);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_YUV420P*, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV420SP_NV12");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::I420ToNV12(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aSrcBuffer + srcChannels[2].mOffset, srcChannels[2].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV12);
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_YUV420SP_NV12::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV12::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(mChannels));

  // set mChannels
  ChannelPixelLayout* ychannel = layout->AppendElement();
  ChannelPixelLayout* uchannel = layout->AppendElement();
  ChannelPixelLayout* vchannel = layout->AppendElement();
  ychannel->mOffset = 0;
  ychannel->mWidth  = aWidth;
  ychannel->mHeight = aHeight;
  ychannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  ychannel->mStride = aStride;
  ychannel->mSkip   = 0; // aYSkip;

  uchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight;
  uchannel->mWidth  = (aWidth + 1) / 2;
  uchannel->mHeight = (aHeight + 1) / 2;
  uchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  uchannel->mStride = uchannel->mWidth * 2;
  uchannel->mSkip   = 1; // aUSkip;

  vchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight + 1;
  vchannel->mWidth  = (aWidth + 1) / 2;
  vchannel->mHeight = (aHeight + 1) / 2;
  vchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  vchannel->mStride = vchannel->mWidth * 2;
  vchannel->mSkip   = 1; // aVSkip;

  return layout;
}

/*
 * class Utils_YUV420SP_NV21.
 */
/* static */Utils_YUV420SP_NV21&
Utils_YUV420SP_NV21::GetInstance()
{
  static Utils_YUV420SP_NV21 instance;
  return instance;
}

Utils_YUV420SP_NV21::Utils_YUV420SP_NV21()
: Utils(3, ChannelPixelLayoutDataType::Uint8)
{
}

uint32_t
Utils_YUV420SP_NV21::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * mBytesPerPixelValue +
         2 * ((aWidth + 1) / 2) * ((aHeight + 1) / 2) * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                               const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV21, &RGBA32ToNV21);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV21, &libyuv::ARGBToNV21);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV21, &RGB24ToNV21);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToNVImg(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV21, &BGR24ToNV21);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_YUV420P*, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UniquePtr<ImagePixelLayout> layout =
    CreateDefaultLayout((*aSrcLayout)[0].mWidth,
                        (*aSrcLayout)[0].mHeight,
                        (*aSrcLayout)[0].mWidth);

  MOZ_ASSERT(layout, "Cannot create a ImagePixelLayout of YUV420SP_NV21");

  const nsTArray<ChannelPixelLayout>& channels = *layout;

  const nsTArray<ChannelPixelLayout>& srcChannels = *aSrcLayout;

  libyuv::I420ToNV21(aSrcBuffer + srcChannels[0].mOffset, srcChannels[0].mStride,
                     aSrcBuffer + srcChannels[1].mOffset, srcChannels[1].mStride,
                     aSrcBuffer + srcChannels[2].mOffset, srcChannels[2].mStride,
                     aDstBuffer + channels[0].mOffset, channels[0].mStride,
                     aDstBuffer + channels[1].mOffset, channels[1].mStride,
                     channels[0].mWidth, channels[0].mHeight);

  return layout;
}

// TODO: optimize me.
UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420P, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::YUV420SP_NV21);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                                 const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_YUV420SP_NV21::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_YUV420SP_NV21::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(mChannels));

  // set mChannels
  ChannelPixelLayout* ychannel = layout->AppendElement();
  ChannelPixelLayout* vchannel = layout->AppendElement(); // v is the 2nd channel.
  ChannelPixelLayout* uchannel = layout->AppendElement(); // u is the 3rd channel.
  ychannel->mOffset = 0;
  ychannel->mWidth  = aWidth;
  ychannel->mHeight = aHeight;
  ychannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  ychannel->mStride = aStride;
  ychannel->mSkip   = 0; // aYSkip;

  vchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight;
  vchannel->mWidth  = (aWidth + 1) / 2;
  vchannel->mHeight = (aHeight + 1) / 2;
  vchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  vchannel->mStride = vchannel->mWidth * 2;
  vchannel->mSkip   = 1; // aVSkip;

  uchannel->mOffset = ychannel->mOffset + ychannel->mStride * ychannel->mHeight + 1;
  uchannel->mWidth  = (aWidth + 1) / 2;
  uchannel->mHeight = (aHeight + 1) / 2;
  uchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  uchannel->mStride = uchannel->mWidth * 2;
  uchannel->mSkip   = 1; // aUSkip;

  return layout;
}

/*
 * Utils_HSV.
 */
/* static */Utils_HSV&
Utils_HSV::GetInstance()
{
  static Utils_HSV instance;
  return instance;
}

Utils_HSV::Utils_HSV()
: Utils(3, ChannelPixelLayoutDataType::Float32)
{
}

uint32_t
Utils_HSV::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                     const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcFormat, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::HSV, 3, &RGBA32ToHSV);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcFormat, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::HSV, 3, &BGRA32ToHSV);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcUtils, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::HSV, 3, &RGB24ToHSV);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcUtils, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::HSV, 3, &BGR24ToHSV);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::HSV);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_HSV::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_HSV::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_HSV::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * Utils_Lab.
 */
/* static */Utils_Lab&
Utils_Lab::GetInstance()
{
  static Utils_Lab instance;
  return instance;
}

Utils_Lab::Utils_Lab()
: Utils(3, ChannelPixelLayoutDataType::Float32)
{
}

uint32_t
Utils_Lab::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                     const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_RGBA32* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcUtils, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::Lab, 3, &RGBA32ToLab);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_BGRA32* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcUtils, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::Lab, 3, &BGRA32ToLab);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcUtils, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::Lab, 3, &RGB24ToLab);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return CvtSimpleImgToSimpleImg<uint8_t, float>(aSrcUtils, aSrcBuffer, aSrcLayout, (float*)aDstBuffer, ImageBitmapFormat::Lab, 3, &BGR24ToLab);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return TwoPassConversion(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::RGB24, this);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::Lab);
}

UniquePtr<ImagePixelLayout>
Utils_Lab::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

bool
Utils_Lab::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::GRAY8 ||
      aSrcFormat == ImageBitmapFormat::DEPTH) {
    return false;
  }

  return true;
}

UniquePtr<ImagePixelLayout>
Utils_Lab::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

/*
 * Utils_Depth.
 */
/* static */Utils_Depth&
Utils_Depth::GetInstance()
{
  static Utils_Depth instance;
  return instance;
}

Utils_Depth::Utils_Depth()
: Utils(1, ChannelPixelLayoutDataType::Uint16)
{
}

uint32_t
Utils_Depth::NeededBufferSize(uint32_t aWidth, uint32_t aHeight)
{
  return aWidth * aHeight * (uint32_t)mChannels * mBytesPerPixelValue;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertTo(Utils* aDstFormat, const uint8_t* aSrcBuffer,
                       const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return aDstFormat->ConvertFrom(this, aSrcBuffer, aSrcLayout, aDstBuffer);
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_RGBA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_BGRA32* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_RGB24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_BGR24* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_Gray8* aSrcFormat, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_YUV444P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_YUV422P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_YUV420P* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_YUV420SP_NV12* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_YUV420SP_NV21* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_HSV* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_Lab* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return nullptr;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::ConvertFrom(Utils_Depth* aSrcUtils, const uint8_t* aSrcBuffer,
                         const ImagePixelLayout* aSrcLayout, uint8_t* aDstBuffer)
{
  return PureCopy(aSrcUtils, aSrcBuffer, aSrcLayout, aDstBuffer, ImageBitmapFormat::DEPTH);
}

bool
Utils_Depth::CanConvertFrom(ImageBitmapFormat aSrcFormat)
{
  if (aSrcFormat == ImageBitmapFormat::DEPTH ) {
    return true;
  }

  return false;
}

UniquePtr<ImagePixelLayout>
Utils_Depth::CreateDefaultLayout(uint32_t aWidth, uint32_t aHeight, uint32_t aStride)
{
  return CreateDefaultLayoutForSimpleImage(aWidth, aHeight, aStride, mChannels, mBytesPerPixelValue, mDataType);
}

} // namespace imagebitmapformat

/*
 * Global functions.
 */

using namespace mozilla::dom::imagebitmapformat;

UniquePtr<ImagePixelLayout>
CreateDefaultPixelLayout(ImageBitmapFormat aFormat, uint32_t aWidth,
                         uint32_t aHeight, uint32_t aStride)
{
  UtilsUniquePtr format = Utils::GetUtils(aFormat);
  MOZ_ASSERT(format, "Cannot get a valid ImageBitmapFormatUtils instance.");

  return format->CreateDefaultLayout(aWidth, aHeight, aStride);
}

UniquePtr<ImagePixelLayout>
CreatePixelLayoutFromPlanarYCbCrData(const layers::PlanarYCbCrData* aData)
{
  if (!aData) {
    // something wrong
    return nullptr;
  }

  UniquePtr<ImagePixelLayout> layout(new ImagePixelLayout(3));

  ChannelPixelLayout* ychannel = layout->AppendElement();
  ChannelPixelLayout* uchannel = layout->AppendElement();
  ChannelPixelLayout* vchannel = layout->AppendElement();

  ychannel->mOffset = 0;

  if (aData->mCrChannel - aData->mCbChannel > 0) {
    uchannel->mOffset = ychannel->mOffset + (aData->mCbChannel - aData->mYChannel);
    vchannel->mOffset = uchannel->mOffset + (aData->mCrChannel - aData->mCbChannel);
  } else {
    uchannel->mOffset = ychannel->mOffset + (aData->mCrChannel - aData->mYChannel);
    vchannel->mOffset = uchannel->mOffset + (aData->mCbChannel - aData->mCrChannel);
  }

  ychannel->mWidth = aData->mYSize.width;
  ychannel->mHeight = aData->mYSize.height;
  ychannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  ychannel->mStride = aData->mYStride;
  ychannel->mSkip = aData->mYSkip;

  uchannel->mWidth = aData->mCbCrSize.width;
  uchannel->mHeight = aData->mCbCrSize.height;
  uchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  uchannel->mStride = aData->mCbCrStride;
  uchannel->mSkip = aData->mCbSkip;

  vchannel->mWidth = aData->mCbCrSize.width;
  vchannel->mHeight = aData->mCbCrSize.height;
  vchannel->mDataType = ChannelPixelLayoutDataType::Uint8;
  vchannel->mStride = aData->mCbCrStride;
  vchannel->mSkip = aData->mCrSkip;

  return layout;
}

uint8_t
GetChannelCountOfImageFormat(ImageBitmapFormat aFormat)
{
  UtilsUniquePtr format = Utils::GetUtils(aFormat);
  MOZ_ASSERT(format, "Cannot get a valid ImageBitmapFormatUtils instance.");

  return format->GetChannelCount();
}

uint32_t
CalculateImageBufferSize(ImageBitmapFormat aFormat,
                         uint32_t aWidth, uint32_t aHeight)
{
  UtilsUniquePtr format = Utils::GetUtils(aFormat);
  MOZ_ASSERT(format, "Cannot get a valid ImageBitmapFormatUtils instance.");

  return format->NeededBufferSize(aWidth, aHeight);
}

UniquePtr<ImagePixelLayout>
CopyAndConvertImageData(ImageBitmapFormat aSrcFormat,
                        const uint8_t* aSrcBuffer,
                        const ImagePixelLayout* aSrcLayout,
                        ImageBitmapFormat aDstFormat,
                        uint8_t* aDstBuffer)
{
  MOZ_ASSERT(aSrcBuffer, "Convert color from a null buffer.");
  MOZ_ASSERT(aSrcLayout, "Convert color from a null layout.");
  MOZ_ASSERT(aDstBuffer, "Convert color to a null buffer.");

  UtilsUniquePtr srcFormat = Utils::GetUtils(aSrcFormat);
  UtilsUniquePtr dstFormat = Utils::GetUtils(aDstFormat);
  MOZ_ASSERT(srcFormat, "Cannot get a valid ImageBitmapFormatUtils instance.");
  MOZ_ASSERT(dstFormat, "Cannot get a valid ImageBitmapFormatUtils instance.");

  return srcFormat->ConvertTo(dstFormat.get(), aSrcBuffer, aSrcLayout, aDstBuffer);
}

ImageBitmapFormat
FindBestMatchingFromat(ImageBitmapFormat aSrcFormat,
                       const Sequence<ImageBitmapFormat>& aCandidates) {

  for(auto& candidate : aCandidates) {
    UtilsUniquePtr candidateFormat = Utils::GetUtils(candidate);
    MOZ_ASSERT(candidateFormat, "Cannot get a valid ImageBitmapFormatUtils instance.");

    if (candidateFormat->CanConvertFrom(aSrcFormat)) {
      return candidate;
    }
  }

  return ImageBitmapFormat::EndGuard_;
}

} // namespace dom
} // namespace mozilla