You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

93 lines
2.8KB

  1. /*
  2. * Pinnacle TARGA CineWave YUV16 decoder
  3. * Copyright (c) 2012 Carl Eugen Hoyos
  4. *
  5. * This file is part of FFmpeg.
  6. *
  7. * FFmpeg is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * FFmpeg is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with FFmpeg; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. */
  21. #include "avcodec.h"
  22. #include "internal.h"
  23. static av_cold int y216_decode_init(AVCodecContext *avctx)
  24. {
  25. avctx->pix_fmt = AV_PIX_FMT_YUV422P16;
  26. avctx->bits_per_raw_sample = 14;
  27. return 0;
  28. }
  29. static int y216_decode_frame(AVCodecContext *avctx, void *data,
  30. int *got_frame, AVPacket *avpkt)
  31. {
  32. AVFrame *pic = data;
  33. const uint16_t *src = (uint16_t *)avpkt->data;
  34. uint16_t *y, *u, *v, aligned_width = FFALIGN(avctx->width, 4);
  35. int i, j;
  36. if (avpkt->size < 4 * avctx->height * aligned_width) {
  37. av_log(avctx, AV_LOG_ERROR, "Insufficient input data.\n");
  38. return AVERROR(EINVAL);
  39. }
  40. if (ff_get_buffer(avctx, pic, 0) < 0) {
  41. av_log(avctx, AV_LOG_ERROR, "Could not allocate buffer.\n");
  42. return AVERROR(ENOMEM);
  43. }
  44. pic->key_frame = 1;
  45. pic->pict_type = AV_PICTURE_TYPE_I;
  46. y = (uint16_t *)pic->data[0];
  47. u = (uint16_t *)pic->data[1];
  48. v = (uint16_t *)pic->data[2];
  49. for (i = 0; i < avctx->height; i++) {
  50. for (j = 0; j < avctx->width >> 1; j++) {
  51. u[ j ] = src[4 * j ] << 2 | src[4 * j ] >> 14;
  52. y[2 * j ] = src[4 * j + 1] << 2 | src[4 * j + 1] >> 14;
  53. v[ j ] = src[4 * j + 2] << 2 | src[4 * j + 2] >> 14;
  54. y[2 * j + 1] = src[4 * j + 3] << 2 | src[4 * j + 3] >> 14;
  55. }
  56. y += pic->linesize[0] >> 1;
  57. u += pic->linesize[1] >> 1;
  58. v += pic->linesize[2] >> 1;
  59. src += aligned_width << 1;
  60. }
  61. *got_frame = 1;
  62. return avpkt->size;
  63. }
  64. static av_cold int y216_decode_close(AVCodecContext *avctx)
  65. {
  66. return 0;
  67. }
  68. AVCodec ff_targa_y216_decoder = {
  69. .name = "targa_y216",
  70. .type = AVMEDIA_TYPE_VIDEO,
  71. .id = AV_CODEC_ID_TARGA_Y216,
  72. .init = y216_decode_init,
  73. .decode = y216_decode_frame,
  74. .close = y216_decode_close,
  75. .capabilities = CODEC_CAP_DR1,
  76. .long_name = NULL_IF_CONFIG_SMALL("Pinnacle TARGA CineWave YUV16"),
  77. };