FFmpeg
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
api-band-test.c
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2015 Ludmila Glinskih
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a copy
5  * of this software and associated documentation files (the "Software"), to deal
6  * in the Software without restriction, including without limitation the rights
7  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8  * copies of the Software, and to permit persons to whom the Software is
9  * furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice shall be included in
12  * all copies or substantial portions of the Software.
13  *
14  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
20  * THE SOFTWARE.
21  */
22 
23 /**
24  * draw_horiz_band test.
25  */
26 
27 #include "libavutil/adler32.h"
28 #include "libavcodec/avcodec.h"
29 #include "libavformat/avformat.h"
30 #include "libavutil/imgutils.h"
31 
35 
36 static void draw_horiz_band(AVCodecContext *ctx, const AVFrame *fr, int offset[4],
37  int slice_position, int type, int height)
38 {
39  int i;
40  const AVPixFmtDescriptor *pix_fmt_desc;
41  int chroma_w, chroma_h;
42  int shift_slice_position;
43  int shift_height;
44 
46 
47  pix_fmt_desc = av_pix_fmt_desc_get(ctx->pix_fmt);
48  chroma_w = -((-ctx->width) >> pix_fmt_desc->log2_chroma_w);
49  chroma_h = -((-height) >> pix_fmt_desc->log2_chroma_h);
50  shift_slice_position = -((-slice_position) >> pix_fmt_desc->log2_chroma_h);
51  shift_height = -((-ctx->height) >> pix_fmt_desc->log2_chroma_h);
52 
53  for (i = 0; i < height; i++) {
54  memcpy(slice_byte_buffer + ctx->width * slice_position + i * ctx->width,
55  fr->data[0] + offset[0] + i * fr->linesize[0], ctx->width);
56  }
57  for (i = 0; i < chroma_h; i++) {
58  memcpy(slice_byte_buffer + ctx->width * ctx->height + chroma_w * shift_slice_position + i * chroma_w,
59  fr->data[1] + offset[1] + i * fr->linesize[1], chroma_w);
60  }
61  for (i = 0; i < chroma_h; i++) {
62  memcpy(slice_byte_buffer + ctx->width * ctx->height + chroma_w * shift_height + chroma_w * shift_slice_position + i * chroma_w,
63  fr->data[2] + offset[2] + i * fr->linesize[2], chroma_w);
64  }
65 }
66 
67 static int video_decode(const char *input_filename)
68 {
69  AVCodec *codec = NULL;
70  AVCodecContext *origin_ctx = NULL, *ctx= NULL;
71  uint8_t *byte_buffer = NULL;
72  AVFrame *fr = NULL;
73  AVPacket pkt;
75  int number_of_written_bytes;
76  int video_stream;
77  int got_frame = 0;
78  int byte_buffer_size;
79  int result;
80  int end_of_stream = 0;
81 
83 
84  result = avformat_open_input(&fmt_ctx, input_filename, NULL, NULL);
85  if (result < 0) {
86  av_log(NULL, AV_LOG_ERROR, "Can't open file\n");
87  return result;
88  }
89 
90  result = avformat_find_stream_info(fmt_ctx, NULL);
91  if (result < 0) {
92  av_log(NULL, AV_LOG_ERROR, "Can't get stream info\n");
93  return result;
94  }
95 
96  video_stream = av_find_best_stream(fmt_ctx, AVMEDIA_TYPE_VIDEO, -1, -1, NULL, 0);
97  if (video_stream < 0) {
98  av_log(NULL, AV_LOG_ERROR, "Can't find video stream in input file\n");
99  return -1;
100  }
101 
102  origin_ctx = fmt_ctx->streams[video_stream]->codec;
103 
104  codec = avcodec_find_decoder(origin_ctx->codec_id);
105  if (!codec) {
106  av_log(NULL, AV_LOG_ERROR, "Can't find decoder\n");
107  return -1;
108  }
109 
110  ctx = avcodec_alloc_context3(codec);
111  if (!ctx) {
112  av_log(NULL, AV_LOG_ERROR, "Can't allocate decoder context\n");
113  return AVERROR(ENOMEM);
114  }
115 
116  result = avcodec_copy_context(ctx, origin_ctx);
117  if (result) {
118  av_log(NULL, AV_LOG_ERROR, "Can't copy decoder context\n");
119  return result;
120  }
121 
122  ctx->draw_horiz_band = draw_horiz_band;
123  ctx->thread_count = 1;
124 
125  result = avcodec_open2(ctx, codec, NULL);
126  if (result < 0) {
127  av_log(ctx, AV_LOG_ERROR, "Can't open decoder\n");
128  return result;
129  }
130 
131  fr = av_frame_alloc();
132  if (!fr) {
133  av_log(NULL, AV_LOG_ERROR, "Can't allocate frame\n");
134  return AVERROR(ENOMEM);
135  }
136 
137  if (strcmp(codec->name, "flv") && strcmp(codec->name, "mpeg4") && strcmp(codec->name, "huffyuv")) {
138  av_log(NULL, AV_LOG_ERROR, "Wrong codec\n");
139  return -1;
140  }
141 
142  byte_buffer_size = av_image_get_buffer_size(ctx->pix_fmt, ctx->width, ctx->height, 32);
143  byte_buffer = av_malloc(byte_buffer_size);
144  if (!byte_buffer) {
145  av_log(NULL, AV_LOG_ERROR, "Can't allocate buffer\n");
146  return AVERROR(ENOMEM);
147  }
148 
149  slice_byte_buffer = av_malloc(byte_buffer_size);
150  if (!slice_byte_buffer) {
151  av_log(NULL, AV_LOG_ERROR, "Can't allocate buffer\n");
152  return AVERROR(ENOMEM);
153  }
154  memset(slice_byte_buffer, 0, byte_buffer_size);
155  slice_byte_buffer_size = byte_buffer_size;
156 
157  av_init_packet(&pkt);
158  do {
159  if (!end_of_stream) {
160  if (av_read_frame(fmt_ctx, &pkt) < 0) {
161  end_of_stream = 1;
162  }
163  }
164  if (end_of_stream) {
165  pkt.data = NULL;
166  pkt.size = 0;
167  }
168  if (pkt.stream_index == video_stream || end_of_stream) {
169  got_frame = 0;
170  result = avcodec_decode_video2(ctx, fr, &got_frame, &pkt);
171  if (result < 0) {
172  av_log(NULL, AV_LOG_ERROR, "Error decoding frame\n");
173  return result;
174  }
175  if (got_frame) {
176  number_of_written_bytes = av_image_copy_to_buffer(byte_buffer, byte_buffer_size,
177  (const uint8_t* const *)fr->data, (const int*) fr->linesize,
178  ctx->pix_fmt, ctx->width, ctx->height, 1);
179  if (number_of_written_bytes < 0) {
180  av_log(NULL, AV_LOG_ERROR, "Can't copy image to buffer\n");
181  return number_of_written_bytes;
182  }
183  if (draw_horiz_band_called == 0) {
184  av_log(NULL, AV_LOG_ERROR, "draw_horiz_band haven't been called!\n");
185  return -1;
186  }
187  if (av_adler32_update(0, (const uint8_t*)byte_buffer, number_of_written_bytes) !=
188  av_adler32_update(0, (const uint8_t*)slice_byte_buffer, number_of_written_bytes)) {
189  av_log(NULL, AV_LOG_ERROR, "Decoded frames with and without draw_horiz_band are not the same!\n");
190  return -1;
191  }
192  }
193  av_free_packet(&pkt);
194  av_init_packet(&pkt);
195  }
196  } while (!end_of_stream || got_frame);
197 
198  av_free_packet(&pkt);
199  av_frame_free(&fr);
200  avcodec_close(ctx);
201  avformat_close_input(&fmt_ctx);
202  avcodec_free_context(&ctx);
203  av_freep(&byte_buffer);
205  return 0;
206 }
207 
208 int main(int argc, char **argv)
209 {
210  if (argc < 2)
211  {
212  av_log(NULL, AV_LOG_ERROR, "Incorrect input: expected %s <name of a video file>\nNote that test works only for huffyuv, flv and mpeg4 decoders\n", argv[0]);
213  return 1;
214  }
215 
216  av_register_all();
217 
218  if (video_decode(argv[1]) != 0)
219  return 1;
220 
221  return 0;
222 }
#define NULL
Definition: coverity.c:32
void av_free_packet(AVPacket *pkt)
Free a packet.
Definition: avpacket.c:280
const AVPixFmtDescriptor * av_pix_fmt_desc_get(enum AVPixelFormat pix_fmt)
Definition: pixdesc.c:2129
This structure describes decoded (raw) audio or video data.
Definition: frame.h:171
int av_image_copy_to_buffer(uint8_t *dst, int dst_size, const uint8_t *const src_data[4], const int src_linesize[4], enum AVPixelFormat pix_fmt, int width, int height, int align)
Copy image data from an image into a buffer.
Definition: imgutils.c:369
misc image utilities
static void draw_horiz_band(AVCodecContext *ctx, const AVFrame *fr, int offset[4], int slice_position, int type, int height)
Definition: api-band-test.c:36
int avformat_open_input(AVFormatContext **ps, const char *filename, AVInputFormat *fmt, AVDictionary **options)
Open an input stream and read the header.
Definition: utils.c:405
static AVFormatContext * fmt_ctx
int size
Definition: avcodec.h:1424
enum AVPixelFormat pix_fmt
Pixel format, see AV_PIX_FMT_xxx.
Definition: avcodec.h:1722
static AVStream * video_stream
static AVPacket pkt
AVCodec.
Definition: avcodec.h:3472
int avcodec_copy_context(AVCodecContext *dest, const AVCodecContext *src)
Copy the settings of the source AVCodecContext into the destination AVCodecContext.
Definition: options.c:182
uint8_t log2_chroma_w
Amount to shift the luma width right to find the chroma width.
Definition: pixdesc.h:80
static int video_decode(const char *input_filename)
Definition: api-band-test.c:67
Format I/O context.
Definition: avformat.h:1273
uint8_t
#define av_malloc(s)
AVFrame * av_frame_alloc(void)
Allocate an AVFrame and set its fields to default values.
Definition: frame.c:135
unsigned long av_adler32_update(unsigned long adler, const uint8_t *buf, unsigned int len)
Calculate the Adler32 checksum of a buffer.
Definition: adler32.c:44
AVStream ** streams
A list of all streams in the file.
Definition: avformat.h:1341
uint8_t * data
Definition: avcodec.h:1423
#define av_log(a,...)
int av_find_best_stream(AVFormatContext *ic, enum AVMediaType type, int wanted_stream_nb, int related_stream, AVCodec **decoder_ret, int flags)
Find the "best" stream in the file.
Definition: utils.c:3572
int avcodec_close(AVCodecContext *avctx)
Close a given AVCodecContext and free all the data associated with it (but not the AVCodecContext its...
Definition: utils.c:2902
int av_image_get_buffer_size(enum AVPixelFormat pix_fmt, int width, int height, int align)
Return the size in bytes of the amount of data required to store an image with the given parameters...
Definition: imgutils.c:347
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
Definition: log.h:176
uint8_t log2_chroma_h
Amount to shift the luma height right to find the chroma height.
Definition: pixdesc.h:89
int avcodec_decode_video2(AVCodecContext *avctx, AVFrame *picture, int *got_picture_ptr, const AVPacket *avpkt)
Decode the video frame of size avpkt->size from avpkt->data into picture.
Definition: utils.c:2408
#define AVERROR(e)
Definition: error.h:43
void av_frame_free(AVFrame **frame)
Free the frame and any dynamically allocated objects in it, e.g.
Definition: frame.c:148
int main(int argc, char **argv)
const char * name
Name of the codec implementation.
Definition: avcodec.h:3479
static const uint8_t offset[127][2]
Definition: vf_spp.c:92
Libavcodec external API header.
AVCodecContext * codec
Codec context associated with this stream.
Definition: avformat.h:861
uint8_t slice_byte_buffer_size
Definition: api-band-test.c:33
AVCodecContext * avcodec_alloc_context3(const AVCodec *codec)
Allocate an AVCodecContext and set its fields to default values.
Definition: options.c:149
int width
picture width / height.
Definition: avcodec.h:1681
static const char * input_filename
Definition: ffplay.c:309
uint8_t * slice_byte_buffer
draw_horiz_band test.
Definition: api-band-test.c:32
Public header for libavutil Adler32 hasher.
void avcodec_free_context(AVCodecContext **avctx)
Free the codec context and everything associated with it and write NULL to the provided pointer...
Definition: options.c:164
enum AVCodecID codec_id
Definition: avcodec.h:1519
int linesize[AV_NUM_DATA_POINTERS]
For video, size in bytes of each picture line.
Definition: frame.h:199
Descriptor that unambiguously describes how the bits of a pixel are stored in the up to 4 data planes...
Definition: pixdesc.h:69
int draw_horiz_band_called
Definition: api-band-test.c:34
main external API structure.
Definition: avcodec.h:1502
AVCodec * avcodec_find_decoder(enum AVCodecID id)
Find a registered decoder with a matching codec ID.
Definition: utils.c:3016
GLint GLenum type
Definition: opengl_enc.c:105
BYTE int const BYTE int int int height
Definition: avisynth_c.h:676
int avcodec_open2(AVCodecContext *avctx, const AVCodec *codec, AVDictionary **options)
Initialize the AVCodecContext to use the given AVCodec.
Definition: utils.c:1349
int av_read_frame(AVFormatContext *s, AVPacket *pkt)
Return the next frame of a stream.
Definition: utils.c:1478
uint8_t * data[AV_NUM_DATA_POINTERS]
pointer to the picture/channel planes.
Definition: frame.h:182
Main libavformat public API header.
int avformat_find_stream_info(AVFormatContext *ic, AVDictionary **options)
Read packets of a media file to get stream information.
Definition: utils.c:3084
void av_init_packet(AVPacket *pkt)
Initialize optional fields of a packet with default values.
Definition: avpacket.c:49
void avformat_close_input(AVFormatContext **s)
Close an opened input AVFormatContext.
Definition: utils.c:3721
#define av_freep(p)
int stream_index
Definition: avcodec.h:1425
This structure stores compressed data.
Definition: avcodec.h:1400
void av_register_all(void)
Initialize libavformat and register all the muxers, demuxers and protocols.
Definition: allformats.c:51