raw.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. /*
  2. * Raw Video Codec
  3. * Copyright (c) 2001 Fabrice Bellard.
  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. /**
  22. * @file raw.c
  23. * Raw Video Codec
  24. */
  25. #include "avcodec.h"
  26. typedef struct RawVideoContext {
  27. unsigned char * buffer; /* block of memory for holding one frame */
  28. int length; /* number of bytes in buffer */
  29. AVFrame pic; ///< AVCodecContext.coded_frame
  30. } RawVideoContext;
  31. typedef struct PixelFormatTag {
  32. int pix_fmt;
  33. unsigned int fourcc;
  34. } PixelFormatTag;
  35. static const PixelFormatTag pixelFormatTags[] = {
  36. { PIX_FMT_YUV420P, MKTAG('I', '4', '2', '0') }, /* Planar formats */
  37. { PIX_FMT_YUV420P, MKTAG('I', 'Y', 'U', 'V') },
  38. { PIX_FMT_YUV420P, MKTAG('Y', 'V', '1', '2') },
  39. { PIX_FMT_YUV410P, MKTAG('Y', 'U', 'V', '9') },
  40. { PIX_FMT_YUV411P, MKTAG('Y', '4', '1', 'B') },
  41. { PIX_FMT_YUV422P, MKTAG('Y', '4', '2', 'B') },
  42. { PIX_FMT_GRAY8, MKTAG('Y', '8', '0', '0') },
  43. { PIX_FMT_GRAY8, MKTAG(' ', ' ', 'Y', '8') },
  44. { PIX_FMT_YUYV422, MKTAG('Y', 'U', 'Y', '2') }, /* Packed formats */
  45. { PIX_FMT_YUYV422, MKTAG('Y', '4', '2', '2') },
  46. { PIX_FMT_UYVY422, MKTAG('U', 'Y', 'V', 'Y') },
  47. { PIX_FMT_GRAY8, MKTAG('G', 'R', 'E', 'Y') },
  48. { PIX_FMT_RGB555, MKTAG('R', 'G', 'B', 15) },
  49. { PIX_FMT_BGR555, MKTAG('B', 'G', 'R', 15) },
  50. { PIX_FMT_RGB565, MKTAG('R', 'G', 'B', 16) },
  51. { PIX_FMT_BGR565, MKTAG('B', 'G', 'R', 16) },
  52. /* quicktime */
  53. { PIX_FMT_UYVY422, MKTAG('2', 'v', 'u', 'y') },
  54. { PIX_FMT_UYVY422, MKTAG('A', 'V', 'U', 'I') }, /* FIXME merge both fields */
  55. { -1, 0 },
  56. };
  57. static const PixelFormatTag pixelFormatBpsAVI[] = {
  58. { PIX_FMT_PAL8, 8 },
  59. { PIX_FMT_RGB555, 15 },
  60. { PIX_FMT_RGB555, 16 },
  61. { PIX_FMT_BGR24, 24 },
  62. { PIX_FMT_RGB32, 32 },
  63. { -1, 0 },
  64. };
  65. static const PixelFormatTag pixelFormatBpsMOV[] = {
  66. /* FIXME fix swscaler to support those */
  67. /* http://developer.apple.com/documentation/QuickTime/QTFF/QTFFChap3/chapter_4_section_2.html */
  68. { PIX_FMT_PAL8, 8 },
  69. { PIX_FMT_BGR555, 16 },
  70. { PIX_FMT_RGB24, 24 },
  71. { PIX_FMT_BGR32_1, 32 },
  72. { -1, 0 },
  73. };
  74. static int findPixelFormat(const PixelFormatTag *tags, unsigned int fourcc)
  75. {
  76. while (tags->pix_fmt >= 0) {
  77. if (tags->fourcc == fourcc)
  78. return tags->pix_fmt;
  79. tags++;
  80. }
  81. return PIX_FMT_YUV420P;
  82. }
  83. unsigned int avcodec_pix_fmt_to_codec_tag(enum PixelFormat fmt)
  84. {
  85. const PixelFormatTag * tags = pixelFormatTags;
  86. while (tags->pix_fmt >= 0) {
  87. if (tags->pix_fmt == fmt)
  88. return tags->fourcc;
  89. tags++;
  90. }
  91. return 0;
  92. }
  93. /* RAW Decoder Implementation */
  94. static int raw_init_decoder(AVCodecContext *avctx)
  95. {
  96. RawVideoContext *context = avctx->priv_data;
  97. if (avctx->codec_tag == MKTAG('r','a','w',' '))
  98. avctx->pix_fmt = findPixelFormat(pixelFormatBpsMOV, avctx->bits_per_sample);
  99. else if (avctx->codec_tag)
  100. avctx->pix_fmt = findPixelFormat(pixelFormatTags, avctx->codec_tag);
  101. else if (avctx->bits_per_sample)
  102. avctx->pix_fmt = findPixelFormat(pixelFormatBpsAVI, avctx->bits_per_sample);
  103. context->length = avpicture_get_size(avctx->pix_fmt, avctx->width, avctx->height);
  104. context->buffer = av_malloc(context->length);
  105. context->pic.pict_type = FF_I_TYPE;
  106. context->pic.key_frame = 1;
  107. avctx->coded_frame= &context->pic;
  108. if (!context->buffer)
  109. return -1;
  110. return 0;
  111. }
  112. static void flip(AVCodecContext *avctx, AVPicture * picture){
  113. if(!avctx->codec_tag && avctx->bits_per_sample && picture->linesize[2]==0){
  114. picture->data[0] += picture->linesize[0] * (avctx->height-1);
  115. picture->linesize[0] *= -1;
  116. }
  117. }
  118. static int raw_decode(AVCodecContext *avctx,
  119. void *data, int *data_size,
  120. uint8_t *buf, int buf_size)
  121. {
  122. RawVideoContext *context = avctx->priv_data;
  123. AVFrame * frame = (AVFrame *) data;
  124. AVPicture * picture = (AVPicture *) data;
  125. frame->interlaced_frame = avctx->coded_frame->interlaced_frame;
  126. frame->top_field_first = avctx->coded_frame->top_field_first;
  127. if(buf_size < context->length - (avctx->pix_fmt==PIX_FMT_PAL8 ? 256*4 : 0))
  128. return -1;
  129. avpicture_fill(picture, buf, avctx->pix_fmt, avctx->width, avctx->height);
  130. if(avctx->pix_fmt==PIX_FMT_PAL8 && buf_size < context->length){
  131. frame->data[1]= context->buffer;
  132. }
  133. if (avctx->palctrl && avctx->palctrl->palette_changed) {
  134. memcpy(frame->data[1], avctx->palctrl->palette, AVPALETTE_SIZE);
  135. avctx->palctrl->palette_changed = 0;
  136. }
  137. flip(avctx, picture);
  138. if (avctx->codec_tag == MKTAG('Y', 'V', '1', '2'))
  139. {
  140. // swap fields
  141. unsigned char *tmp = picture->data[1];
  142. picture->data[1] = picture->data[2];
  143. picture->data[2] = tmp;
  144. }
  145. *data_size = sizeof(AVPicture);
  146. return buf_size;
  147. }
  148. static int raw_close_decoder(AVCodecContext *avctx)
  149. {
  150. RawVideoContext *context = avctx->priv_data;
  151. av_freep(&context->buffer);
  152. return 0;
  153. }
  154. /* RAW Encoder Implementation */
  155. #ifdef CONFIG_RAWVIDEO_ENCODER
  156. static int raw_init_encoder(AVCodecContext *avctx)
  157. {
  158. avctx->coded_frame = (AVFrame *)avctx->priv_data;
  159. avctx->coded_frame->pict_type = FF_I_TYPE;
  160. avctx->coded_frame->key_frame = 1;
  161. if(!avctx->codec_tag)
  162. avctx->codec_tag = avcodec_pix_fmt_to_codec_tag(avctx->pix_fmt);
  163. return 0;
  164. }
  165. static int raw_encode(AVCodecContext *avctx,
  166. unsigned char *frame, int buf_size, void *data)
  167. {
  168. return avpicture_layout((AVPicture *)data, avctx->pix_fmt, avctx->width,
  169. avctx->height, frame, buf_size);
  170. }
  171. AVCodec rawvideo_encoder = {
  172. "rawvideo",
  173. CODEC_TYPE_VIDEO,
  174. CODEC_ID_RAWVIDEO,
  175. sizeof(AVFrame),
  176. raw_init_encoder,
  177. raw_encode,
  178. };
  179. #endif // CONFIG_RAWVIDEO_ENCODER
  180. AVCodec rawvideo_decoder = {
  181. "rawvideo",
  182. CODEC_TYPE_VIDEO,
  183. CODEC_ID_RAWVIDEO,
  184. sizeof(RawVideoContext),
  185. raw_init_decoder,
  186. NULL,
  187. raw_close_decoder,
  188. raw_decode,
  189. };