123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468 |
- /*
- * Copyright 2011 The LibYuv Project Authors. All rights reserved.
- *
- * Use of this source code is governed by a BSD-style license
- * that can be found in the LICENSE file in the root of the source
- * tree. An additional intellectual property rights grant can be found
- * in the file PATENTS. All contributing project authors may
- * be found in the AUTHORS file in the root of the source tree.
- */
- #include "libyuv/convert.h"
- #include "libyuv/convert_argb.h"
- #ifdef HAVE_JPEG
- #include "libyuv/mjpeg_decoder.h"
- #endif
- #ifdef __cplusplus
- namespace libyuv {
- extern "C" {
- #endif
- #ifdef HAVE_JPEG
- struct I420Buffers {
- uint8_t* y;
- int y_stride;
- uint8_t* u;
- int u_stride;
- uint8_t* v;
- int v_stride;
- int w;
- int h;
- };
- static void JpegCopyI420(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- I420Buffers* dest = (I420Buffers*)(opaque);
- I420Copy(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->y, dest->y_stride, dest->u, dest->u_stride, dest->v,
- dest->v_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->u += ((rows + 1) >> 1) * dest->u_stride;
- dest->v += ((rows + 1) >> 1) * dest->v_stride;
- dest->h -= rows;
- }
- static void JpegI422ToI420(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- I420Buffers* dest = (I420Buffers*)(opaque);
- I422ToI420(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->y, dest->y_stride, dest->u, dest->u_stride, dest->v,
- dest->v_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->u += ((rows + 1) >> 1) * dest->u_stride;
- dest->v += ((rows + 1) >> 1) * dest->v_stride;
- dest->h -= rows;
- }
- static void JpegI444ToI420(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- I420Buffers* dest = (I420Buffers*)(opaque);
- I444ToI420(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->y, dest->y_stride, dest->u, dest->u_stride, dest->v,
- dest->v_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->u += ((rows + 1) >> 1) * dest->u_stride;
- dest->v += ((rows + 1) >> 1) * dest->v_stride;
- dest->h -= rows;
- }
- static void JpegI400ToI420(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- I420Buffers* dest = (I420Buffers*)(opaque);
- I400ToI420(data[0], strides[0], dest->y, dest->y_stride, dest->u,
- dest->u_stride, dest->v, dest->v_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->u += ((rows + 1) >> 1) * dest->u_stride;
- dest->v += ((rows + 1) >> 1) * dest->v_stride;
- dest->h -= rows;
- }
- // Query size of MJPG in pixels.
- LIBYUV_API
- int MJPGSize(const uint8_t* src_mjpg,
- size_t src_size_mjpg,
- int* width,
- int* height) {
- MJpegDecoder mjpeg_decoder;
- LIBYUV_BOOL ret = mjpeg_decoder.LoadFrame(src_mjpg, src_size_mjpg);
- if (ret) {
- *width = mjpeg_decoder.GetWidth();
- *height = mjpeg_decoder.GetHeight();
- }
- mjpeg_decoder.UnloadFrame();
- return ret ? 0 : -1; // -1 for runtime failure.
- }
- // MJPG (Motion JPeg) to I420
- // TODO(fbarchard): review src_width and src_height requirement. dst_width and
- // dst_height may be enough.
- LIBYUV_API
- int MJPGToI420(const uint8_t* src_mjpg,
- size_t src_size_mjpg,
- uint8_t* dst_y,
- int dst_stride_y,
- uint8_t* dst_u,
- int dst_stride_u,
- uint8_t* dst_v,
- int dst_stride_v,
- int src_width,
- int src_height,
- int dst_width,
- int dst_height) {
- if (src_size_mjpg == kUnknownDataSize) {
- // ERROR: MJPEG frame size unknown
- return -1;
- }
- // TODO(fbarchard): Port MJpeg to C.
- MJpegDecoder mjpeg_decoder;
- LIBYUV_BOOL ret = mjpeg_decoder.LoadFrame(src_mjpg, src_size_mjpg);
- if (ret && (mjpeg_decoder.GetWidth() != src_width ||
- mjpeg_decoder.GetHeight() != src_height)) {
- // ERROR: MJPEG frame has unexpected dimensions
- mjpeg_decoder.UnloadFrame();
- return 1; // runtime failure
- }
- if (ret) {
- I420Buffers bufs = {dst_y, dst_stride_y, dst_u, dst_stride_u,
- dst_v, dst_stride_v, dst_width, dst_height};
- // YUV420
- if (mjpeg_decoder.GetColorSpace() == MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 2 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 2 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegCopyI420, &bufs, dst_width,
- dst_height);
- // YUV422
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 2 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI422ToI420, &bufs, dst_width,
- dst_height);
- // YUV444
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 1 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI444ToI420, &bufs, dst_width,
- dst_height);
- // YUV400
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceGrayscale &&
- mjpeg_decoder.GetNumComponents() == 1 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI400ToI420, &bufs, dst_width,
- dst_height);
- } else {
- // TODO(fbarchard): Implement conversion for any other
- // colorspace/subsample factors that occur in practice. ERROR: Unable to
- // convert MJPEG frame because format is not supported
- mjpeg_decoder.UnloadFrame();
- return 1;
- }
- }
- return ret ? 0 : 1;
- }
- struct NV21Buffers {
- uint8_t* y;
- int y_stride;
- uint8_t* vu;
- int vu_stride;
- int w;
- int h;
- };
- static void JpegI420ToNV21(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- NV21Buffers* dest = (NV21Buffers*)(opaque);
- I420ToNV21(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->y, dest->y_stride, dest->vu, dest->vu_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->vu += ((rows + 1) >> 1) * dest->vu_stride;
- dest->h -= rows;
- }
- static void JpegI422ToNV21(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- NV21Buffers* dest = (NV21Buffers*)(opaque);
- I422ToNV21(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->y, dest->y_stride, dest->vu, dest->vu_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->vu += ((rows + 1) >> 1) * dest->vu_stride;
- dest->h -= rows;
- }
- static void JpegI444ToNV21(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- NV21Buffers* dest = (NV21Buffers*)(opaque);
- I444ToNV21(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->y, dest->y_stride, dest->vu, dest->vu_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->vu += ((rows + 1) >> 1) * dest->vu_stride;
- dest->h -= rows;
- }
- static void JpegI400ToNV21(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- NV21Buffers* dest = (NV21Buffers*)(opaque);
- I400ToNV21(data[0], strides[0], dest->y, dest->y_stride, dest->vu,
- dest->vu_stride, dest->w, rows);
- dest->y += rows * dest->y_stride;
- dest->vu += ((rows + 1) >> 1) * dest->vu_stride;
- dest->h -= rows;
- }
- // MJPG (Motion JPeg) to NV21
- LIBYUV_API
- int MJPGToNV21(const uint8_t* src_mjpg,
- size_t src_size_mjpg,
- uint8_t* dst_y,
- int dst_stride_y,
- uint8_t* dst_vu,
- int dst_stride_vu,
- int src_width,
- int src_height,
- int dst_width,
- int dst_height) {
- if (src_size_mjpg == kUnknownDataSize) {
- // ERROR: MJPEG frame size unknown
- return -1;
- }
- // TODO(fbarchard): Port MJpeg to C.
- MJpegDecoder mjpeg_decoder;
- LIBYUV_BOOL ret = mjpeg_decoder.LoadFrame(src_mjpg, src_size_mjpg);
- if (ret && (mjpeg_decoder.GetWidth() != src_width ||
- mjpeg_decoder.GetHeight() != src_height)) {
- // ERROR: MJPEG frame has unexpected dimensions
- mjpeg_decoder.UnloadFrame();
- return 1; // runtime failure
- }
- if (ret) {
- NV21Buffers bufs = {dst_y, dst_stride_y, dst_vu,
- dst_stride_vu, dst_width, dst_height};
- // YUV420
- if (mjpeg_decoder.GetColorSpace() == MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 2 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 2 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI420ToNV21, &bufs, dst_width,
- dst_height);
- // YUV422
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 2 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI422ToNV21, &bufs, dst_width,
- dst_height);
- // YUV444
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 1 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI444ToNV21, &bufs, dst_width,
- dst_height);
- // YUV400
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceGrayscale &&
- mjpeg_decoder.GetNumComponents() == 1 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI400ToNV21, &bufs, dst_width,
- dst_height);
- } else {
- // Unknown colorspace.
- mjpeg_decoder.UnloadFrame();
- return 1;
- }
- }
- return ret ? 0 : 1;
- }
- struct ARGBBuffers {
- uint8_t* argb;
- int argb_stride;
- int w;
- int h;
- };
- static void JpegI420ToARGB(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- ARGBBuffers* dest = (ARGBBuffers*)(opaque);
- I420ToARGB(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->argb, dest->argb_stride, dest->w, rows);
- dest->argb += rows * dest->argb_stride;
- dest->h -= rows;
- }
- static void JpegI422ToARGB(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- ARGBBuffers* dest = (ARGBBuffers*)(opaque);
- I422ToARGB(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->argb, dest->argb_stride, dest->w, rows);
- dest->argb += rows * dest->argb_stride;
- dest->h -= rows;
- }
- static void JpegI444ToARGB(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- ARGBBuffers* dest = (ARGBBuffers*)(opaque);
- I444ToARGB(data[0], strides[0], data[1], strides[1], data[2], strides[2],
- dest->argb, dest->argb_stride, dest->w, rows);
- dest->argb += rows * dest->argb_stride;
- dest->h -= rows;
- }
- static void JpegI400ToARGB(void* opaque,
- const uint8_t* const* data,
- const int* strides,
- int rows) {
- ARGBBuffers* dest = (ARGBBuffers*)(opaque);
- I400ToARGB(data[0], strides[0], dest->argb, dest->argb_stride, dest->w, rows);
- dest->argb += rows * dest->argb_stride;
- dest->h -= rows;
- }
- // MJPG (Motion JPeg) to ARGB
- // TODO(fbarchard): review src_width and src_height requirement. dst_width and
- // dst_height may be enough.
- LIBYUV_API
- int MJPGToARGB(const uint8_t* src_mjpg,
- size_t src_size_mjpg,
- uint8_t* dst_argb,
- int dst_stride_argb,
- int src_width,
- int src_height,
- int dst_width,
- int dst_height) {
- if (src_size_mjpg == kUnknownDataSize) {
- // ERROR: MJPEG frame size unknown
- return -1;
- }
- // TODO(fbarchard): Port MJpeg to C.
- MJpegDecoder mjpeg_decoder;
- LIBYUV_BOOL ret = mjpeg_decoder.LoadFrame(src_mjpg, src_size_mjpg);
- if (ret && (mjpeg_decoder.GetWidth() != src_width ||
- mjpeg_decoder.GetHeight() != src_height)) {
- // ERROR: MJPEG frame has unexpected dimensions
- mjpeg_decoder.UnloadFrame();
- return 1; // runtime failure
- }
- if (ret) {
- ARGBBuffers bufs = {dst_argb, dst_stride_argb, dst_width, dst_height};
- // YUV420
- if (mjpeg_decoder.GetColorSpace() == MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 2 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 2 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI420ToARGB, &bufs, dst_width,
- dst_height);
- // YUV422
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 2 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI422ToARGB, &bufs, dst_width,
- dst_height);
- // YUV444
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceYCbCr &&
- mjpeg_decoder.GetNumComponents() == 3 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 1 &&
- mjpeg_decoder.GetVertSampFactor(1) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(1) == 1 &&
- mjpeg_decoder.GetVertSampFactor(2) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(2) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI444ToARGB, &bufs, dst_width,
- dst_height);
- // YUV400
- } else if (mjpeg_decoder.GetColorSpace() ==
- MJpegDecoder::kColorSpaceGrayscale &&
- mjpeg_decoder.GetNumComponents() == 1 &&
- mjpeg_decoder.GetVertSampFactor(0) == 1 &&
- mjpeg_decoder.GetHorizSampFactor(0) == 1) {
- ret = mjpeg_decoder.DecodeToCallback(&JpegI400ToARGB, &bufs, dst_width,
- dst_height);
- } else {
- // TODO(fbarchard): Implement conversion for any other
- // colorspace/subsample factors that occur in practice. ERROR: Unable to
- // convert MJPEG frame because format is not supported
- mjpeg_decoder.UnloadFrame();
- return 1;
- }
- }
- return ret ? 0 : 1;
- }
- #endif // HAVE_JPEG
- #ifdef __cplusplus
- } // extern "C"
- } // namespace libyuv
- #endif
|