Blame src/enc/picture_enc.c

Packit 9c6abc
// Copyright 2011 Google Inc. All Rights Reserved.
Packit 9c6abc
//
Packit 9c6abc
// Use of this source code is governed by a BSD-style license
Packit 9c6abc
// that can be found in the COPYING file in the root of the source
Packit 9c6abc
// tree. An additional intellectual property rights grant can be found
Packit 9c6abc
// in the file PATENTS. All contributing project authors may
Packit 9c6abc
// be found in the AUTHORS file in the root of the source tree.
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
//
Packit 9c6abc
// WebPPicture class basis
Packit 9c6abc
//
Packit 9c6abc
// Author: Skal (pascal.massimino@gmail.com)
Packit 9c6abc
Packit 9c6abc
#include <assert.h>
Packit 9c6abc
#include <stdlib.h>
Packit 9c6abc
Packit 9c6abc
#include "src/enc/vp8i_enc.h"
Packit 9c6abc
#include "src/dsp/dsp.h"
Packit 9c6abc
#include "src/utils/utils.h"
Packit 9c6abc
Packit 9c6abc
//------------------------------------------------------------------------------
Packit 9c6abc
// WebPPicture
Packit 9c6abc
//------------------------------------------------------------------------------
Packit 9c6abc
Packit 9c6abc
static int DummyWriter(const uint8_t* data, size_t data_size,
Packit 9c6abc
                       const WebPPicture* const picture) {
Packit 9c6abc
  // The following are to prevent 'unused variable' error message.
Packit 9c6abc
  (void)data;
Packit 9c6abc
  (void)data_size;
Packit 9c6abc
  (void)picture;
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int WebPPictureInitInternal(WebPPicture* picture, int version) {
Packit 9c6abc
  if (WEBP_ABI_IS_INCOMPATIBLE(version, WEBP_ENCODER_ABI_VERSION)) {
Packit 9c6abc
    return 0;   // caller/system version mismatch!
Packit 9c6abc
  }
Packit 9c6abc
  if (picture != NULL) {
Packit 9c6abc
    memset(picture, 0, sizeof(*picture));
Packit 9c6abc
    picture->writer = DummyWriter;
Packit 9c6abc
    WebPEncodingSetError(picture, VP8_ENC_OK);
Packit 9c6abc
  }
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
//------------------------------------------------------------------------------
Packit 9c6abc
Packit 9c6abc
static void WebPPictureResetBufferARGB(WebPPicture* const picture) {
Packit 9c6abc
  picture->memory_argb_ = NULL;
Packit 9c6abc
  picture->argb = NULL;
Packit 9c6abc
  picture->argb_stride = 0;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
static void WebPPictureResetBufferYUVA(WebPPicture* const picture) {
Packit 9c6abc
  picture->memory_ = NULL;
Packit 9c6abc
  picture->y = picture->u = picture->v = picture->a = NULL;
Packit 9c6abc
  picture->y_stride = picture->uv_stride = 0;
Packit 9c6abc
  picture->a_stride = 0;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
void WebPPictureResetBuffers(WebPPicture* const picture) {
Packit 9c6abc
  WebPPictureResetBufferARGB(picture);
Packit 9c6abc
  WebPPictureResetBufferYUVA(picture);
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int WebPPictureAllocARGB(WebPPicture* const picture, int width, int height) {
Packit 9c6abc
  void* memory;
Packit 9c6abc
  const uint64_t argb_size = (uint64_t)width * height;
Packit 9c6abc
Packit 9c6abc
  assert(picture != NULL);
Packit 9c6abc
Packit 9c6abc
  WebPSafeFree(picture->memory_argb_);
Packit 9c6abc
  WebPPictureResetBufferARGB(picture);
Packit 9c6abc
Packit 9c6abc
  if (width <= 0 || height <= 0) {
Packit 9c6abc
    return WebPEncodingSetError(picture, VP8_ENC_ERROR_BAD_DIMENSION);
Packit 9c6abc
  }
Packit 9c6abc
  // allocate a new buffer.
Packit 9c6abc
  memory = WebPSafeMalloc(argb_size + WEBP_ALIGN_CST, sizeof(*picture->argb));
Packit 9c6abc
  if (memory == NULL) {
Packit 9c6abc
    return WebPEncodingSetError(picture, VP8_ENC_ERROR_OUT_OF_MEMORY);
Packit 9c6abc
  }
Packit 9c6abc
  picture->memory_argb_ = memory;
Packit 9c6abc
  picture->argb = (uint32_t*)WEBP_ALIGN(memory);
Packit 9c6abc
  picture->argb_stride = width;
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int WebPPictureAllocYUVA(WebPPicture* const picture, int width, int height) {
Packit 9c6abc
  const WebPEncCSP uv_csp =
Packit 9c6abc
      (WebPEncCSP)((int)picture->colorspace & WEBP_CSP_UV_MASK);
Packit 9c6abc
  const int has_alpha = (int)picture->colorspace & WEBP_CSP_ALPHA_BIT;
Packit 9c6abc
  const int y_stride = width;
Packit 9c6abc
  const int uv_width = (int)(((int64_t)width + 1) >> 1);
Packit 9c6abc
  const int uv_height = (int)(((int64_t)height + 1) >> 1);
Packit 9c6abc
  const int uv_stride = uv_width;
Packit 9c6abc
  int a_width, a_stride;
Packit 9c6abc
  uint64_t y_size, uv_size, a_size, total_size;
Packit 9c6abc
  uint8_t* mem;
Packit 9c6abc
Packit 9c6abc
  assert(picture != NULL);
Packit 9c6abc
Packit 9c6abc
  WebPSafeFree(picture->memory_);
Packit 9c6abc
  WebPPictureResetBufferYUVA(picture);
Packit 9c6abc
Packit 9c6abc
  if (uv_csp != WEBP_YUV420) {
Packit 9c6abc
    return WebPEncodingSetError(picture, VP8_ENC_ERROR_INVALID_CONFIGURATION);
Packit 9c6abc
  }
Packit 9c6abc
Packit 9c6abc
  // alpha
Packit 9c6abc
  a_width = has_alpha ? width : 0;
Packit 9c6abc
  a_stride = a_width;
Packit 9c6abc
  y_size = (uint64_t)y_stride * height;
Packit 9c6abc
  uv_size = (uint64_t)uv_stride * uv_height;
Packit 9c6abc
  a_size =  (uint64_t)a_stride * height;
Packit 9c6abc
Packit 9c6abc
  total_size = y_size + a_size + 2 * uv_size;
Packit 9c6abc
Packit 9c6abc
  // Security and validation checks
Packit 9c6abc
  if (width <= 0 || height <= 0 ||           // luma/alpha param error
Packit 9c6abc
      uv_width <= 0 || uv_height <= 0) {     // u/v param error
Packit 9c6abc
    return WebPEncodingSetError(picture, VP8_ENC_ERROR_BAD_DIMENSION);
Packit 9c6abc
  }
Packit 9c6abc
  // allocate a new buffer.
Packit 9c6abc
  mem = (uint8_t*)WebPSafeMalloc(total_size, sizeof(*mem));
Packit 9c6abc
  if (mem == NULL) {
Packit 9c6abc
    return WebPEncodingSetError(picture, VP8_ENC_ERROR_OUT_OF_MEMORY);
Packit 9c6abc
  }
Packit 9c6abc
Packit 9c6abc
  // From now on, we're in the clear, we can no longer fail...
Packit 9c6abc
  picture->memory_ = (void*)mem;
Packit 9c6abc
  picture->y_stride  = y_stride;
Packit 9c6abc
  picture->uv_stride = uv_stride;
Packit 9c6abc
  picture->a_stride  = a_stride;
Packit 9c6abc
Packit 9c6abc
  // TODO(skal): we could align the y/u/v planes and adjust stride.
Packit 9c6abc
  picture->y = mem;
Packit 9c6abc
  mem += y_size;
Packit 9c6abc
Packit 9c6abc
  picture->u = mem;
Packit 9c6abc
  mem += uv_size;
Packit 9c6abc
  picture->v = mem;
Packit 9c6abc
  mem += uv_size;
Packit 9c6abc
Packit 9c6abc
  if (a_size > 0) {
Packit 9c6abc
    picture->a = mem;
Packit 9c6abc
    mem += a_size;
Packit 9c6abc
  }
Packit 9c6abc
  (void)mem;  // makes the static analyzer happy
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int WebPPictureAlloc(WebPPicture* picture) {
Packit 9c6abc
  if (picture != NULL) {
Packit 9c6abc
    const int width = picture->width;
Packit 9c6abc
    const int height = picture->height;
Packit 9c6abc
Packit 9c6abc
    WebPPictureFree(picture);   // erase previous buffer
Packit 9c6abc
Packit 9c6abc
    if (!picture->use_argb) {
Packit 9c6abc
      return WebPPictureAllocYUVA(picture, width, height);
Packit 9c6abc
    } else {
Packit 9c6abc
      return WebPPictureAllocARGB(picture, width, height);
Packit 9c6abc
    }
Packit 9c6abc
  }
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
void WebPPictureFree(WebPPicture* picture) {
Packit 9c6abc
  if (picture != NULL) {
Packit 9c6abc
    WebPSafeFree(picture->memory_);
Packit 9c6abc
    WebPSafeFree(picture->memory_argb_);
Packit 9c6abc
    WebPPictureResetBuffers(picture);
Packit 9c6abc
  }
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
//------------------------------------------------------------------------------
Packit 9c6abc
// WebPMemoryWriter: Write-to-memory
Packit 9c6abc
Packit 9c6abc
void WebPMemoryWriterInit(WebPMemoryWriter* writer) {
Packit 9c6abc
  writer->mem = NULL;
Packit 9c6abc
  writer->size = 0;
Packit 9c6abc
  writer->max_size = 0;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int WebPMemoryWrite(const uint8_t* data, size_t data_size,
Packit 9c6abc
                    const WebPPicture* picture) {
Packit 9c6abc
  WebPMemoryWriter* const w = (WebPMemoryWriter*)picture->custom_ptr;
Packit 9c6abc
  uint64_t next_size;
Packit 9c6abc
  if (w == NULL) {
Packit 9c6abc
    return 1;
Packit 9c6abc
  }
Packit 9c6abc
  next_size = (uint64_t)w->size + data_size;
Packit 9c6abc
  if (next_size > w->max_size) {
Packit 9c6abc
    uint8_t* new_mem;
Packit 9c6abc
    uint64_t next_max_size = 2ULL * w->max_size;
Packit 9c6abc
    if (next_max_size < next_size) next_max_size = next_size;
Packit 9c6abc
    if (next_max_size < 8192ULL) next_max_size = 8192ULL;
Packit 9c6abc
    new_mem = (uint8_t*)WebPSafeMalloc(next_max_size, 1);
Packit 9c6abc
    if (new_mem == NULL) {
Packit 9c6abc
      return 0;
Packit 9c6abc
    }
Packit 9c6abc
    if (w->size > 0) {
Packit 9c6abc
      memcpy(new_mem, w->mem, w->size);
Packit 9c6abc
    }
Packit 9c6abc
    WebPSafeFree(w->mem);
Packit 9c6abc
    w->mem = new_mem;
Packit 9c6abc
    // down-cast is ok, thanks to WebPSafeMalloc
Packit 9c6abc
    w->max_size = (size_t)next_max_size;
Packit 9c6abc
  }
Packit 9c6abc
  if (data_size > 0) {
Packit 9c6abc
    memcpy(w->mem + w->size, data, data_size);
Packit 9c6abc
    w->size += data_size;
Packit 9c6abc
  }
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
void WebPMemoryWriterClear(WebPMemoryWriter* writer) {
Packit 9c6abc
  if (writer != NULL) {
Packit 9c6abc
    WebPSafeFree(writer->mem);
Packit 9c6abc
    writer->mem = NULL;
Packit 9c6abc
    writer->size = 0;
Packit 9c6abc
    writer->max_size = 0;
Packit 9c6abc
  }
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
//------------------------------------------------------------------------------
Packit 9c6abc
// Simplest high-level calls:
Packit 9c6abc
Packit 9c6abc
typedef int (*Importer)(WebPPicture* const, const uint8_t* const, int);
Packit 9c6abc
Packit 9c6abc
static size_t Encode(const uint8_t* rgba, int width, int height, int stride,
Packit 9c6abc
                     Importer import, float quality_factor, int lossless,
Packit 9c6abc
                     uint8_t** output) {
Packit 9c6abc
  WebPPicture pic;
Packit 9c6abc
  WebPConfig config;
Packit 9c6abc
  WebPMemoryWriter wrt;
Packit 9c6abc
  int ok;
Packit 9c6abc
Packit 9c6abc
  if (output == NULL) return 0;
Packit 9c6abc
Packit 9c6abc
  if (!WebPConfigPreset(&config, WEBP_PRESET_DEFAULT, quality_factor) ||
Packit 9c6abc
      !WebPPictureInit(&pic)) {
Packit 9c6abc
    return 0;  // shouldn't happen, except if system installation is broken
Packit 9c6abc
  }
Packit 9c6abc
Packit 9c6abc
  config.lossless = !!lossless;
Packit 9c6abc
  pic.use_argb = !!lossless;
Packit 9c6abc
  pic.width = width;
Packit 9c6abc
  pic.height = height;
Packit 9c6abc
  pic.writer = WebPMemoryWrite;
Packit 9c6abc
  pic.custom_ptr = &wrt;
Packit 9c6abc
  WebPMemoryWriterInit(&wrt;;
Packit 9c6abc
Packit 9c6abc
  ok = import(&pic, rgba, stride) && WebPEncode(&config, &pic;;
Packit 9c6abc
  WebPPictureFree(&pic;;
Packit 9c6abc
  if (!ok) {
Packit 9c6abc
    WebPMemoryWriterClear(&wrt;;
Packit 9c6abc
    *output = NULL;
Packit 9c6abc
    return 0;
Packit 9c6abc
  }
Packit 9c6abc
  *output = wrt.mem;
Packit 9c6abc
  return wrt.size;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
#define ENCODE_FUNC(NAME, IMPORTER)                                     \
Packit 9c6abc
size_t NAME(const uint8_t* in, int w, int h, int bps, float q,          \
Packit 9c6abc
            uint8_t** out) {                                            \
Packit 9c6abc
  return Encode(in, w, h, bps, IMPORTER, q, 0, out);                    \
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
ENCODE_FUNC(WebPEncodeRGB, WebPPictureImportRGB)
Packit 9c6abc
ENCODE_FUNC(WebPEncodeRGBA, WebPPictureImportRGBA)
Packit 9c6abc
#if !defined(WEBP_REDUCE_CSP)
Packit 9c6abc
ENCODE_FUNC(WebPEncodeBGR, WebPPictureImportBGR)
Packit 9c6abc
ENCODE_FUNC(WebPEncodeBGRA, WebPPictureImportBGRA)
Packit 9c6abc
#endif  // WEBP_REDUCE_CSP
Packit 9c6abc
Packit 9c6abc
#undef ENCODE_FUNC
Packit 9c6abc
Packit 9c6abc
#define LOSSLESS_DEFAULT_QUALITY 70.
Packit 9c6abc
#define LOSSLESS_ENCODE_FUNC(NAME, IMPORTER)                                 \
Packit 9c6abc
size_t NAME(const uint8_t* in, int w, int h, int bps, uint8_t** out) {       \
Packit 9c6abc
  return Encode(in, w, h, bps, IMPORTER, LOSSLESS_DEFAULT_QUALITY, 1, out);  \
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
LOSSLESS_ENCODE_FUNC(WebPEncodeLosslessRGB, WebPPictureImportRGB)
Packit 9c6abc
LOSSLESS_ENCODE_FUNC(WebPEncodeLosslessRGBA, WebPPictureImportRGBA)
Packit 9c6abc
#if !defined(WEBP_REDUCE_CSP)
Packit 9c6abc
LOSSLESS_ENCODE_FUNC(WebPEncodeLosslessBGR, WebPPictureImportBGR)
Packit 9c6abc
LOSSLESS_ENCODE_FUNC(WebPEncodeLosslessBGRA, WebPPictureImportBGRA)
Packit 9c6abc
#endif  // WEBP_REDUCE_CSP
Packit 9c6abc
Packit 9c6abc
#undef LOSSLESS_ENCODE_FUNC
Packit 9c6abc
Packit 9c6abc
//------------------------------------------------------------------------------