FFmpeg 7.1.1
Loading...
Searching...
No Matches
qsv_decode.c
Go to the documentation of this file.
1/*
2 * Copyright (c) 2015 Anton Khirnov
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 * @file Intel QSV-accelerated H.264 decoding API usage example
25 * @example qsv_decode.c
26 *
27 * Perform QSV-accelerated H.264 decoding with output frames in the
28 * GPU video surfaces, write the decoded frames to an output file.
29 */
30
31#include <stdio.h>
32
34#include <libavformat/avio.h>
35
36#include <libavcodec/avcodec.h>
37
38#include <libavutil/buffer.h>
39#include <libavutil/error.h>
40#include <libavutil/hwcontext.h>
42#include <libavutil/mem.h>
43
44static int get_format(AVCodecContext *avctx, const enum AVPixelFormat *pix_fmts)
45{
46 while (*pix_fmts != AV_PIX_FMT_NONE) {
47 if (*pix_fmts == AV_PIX_FMT_QSV) {
48 return AV_PIX_FMT_QSV;
49 }
50
51 pix_fmts++;
52 }
53
54 fprintf(stderr, "The QSV pixel format not offered in get_format()\n");
55
56 return AV_PIX_FMT_NONE;
57}
58
60 AVFrame *frame, AVFrame *sw_frame,
61 AVPacket *pkt, AVIOContext *output_ctx)
62{
63 int ret = 0;
64
66 if (ret < 0) {
67 fprintf(stderr, "Error during decoding\n");
68 return ret;
69 }
70
71 while (ret >= 0) {
72 int i, j;
73
75 if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
76 break;
77 else if (ret < 0) {
78 fprintf(stderr, "Error during decoding\n");
79 return ret;
80 }
81
82 /* A real program would do something useful with the decoded frame here.
83 * We just retrieve the raw data and write it to a file, which is rather
84 * useless but pedagogic. */
85 ret = av_hwframe_transfer_data(sw_frame, frame, 0);
86 if (ret < 0) {
87 fprintf(stderr, "Error transferring the data to system memory\n");
88 goto fail;
89 }
90
91 for (i = 0; i < FF_ARRAY_ELEMS(sw_frame->data) && sw_frame->data[i]; i++)
92 for (j = 0; j < (sw_frame->height >> (i > 0)); j++)
93 avio_write(output_ctx, sw_frame->data[i] + j * sw_frame->linesize[i], sw_frame->width);
94
95fail:
96 av_frame_unref(sw_frame);
98
99 if (ret < 0)
100 return ret;
101 }
102
103 return 0;
104}
105
106int main(int argc, char **argv)
107{
108 AVFormatContext *input_ctx = NULL;
109 AVStream *video_st = NULL;
111 const AVCodec *decoder;
112
113 AVPacket *pkt = NULL;
114 AVFrame *frame = NULL, *sw_frame = NULL;
115
116 AVIOContext *output_ctx = NULL;
117
118 int ret, i;
119
120 AVBufferRef *device_ref = NULL;
121
122 if (argc < 3) {
123 fprintf(stderr, "Usage: %s <input file> <output file>\n", argv[0]);
124 return 1;
125 }
126
127 /* open the input file */
128 ret = avformat_open_input(&input_ctx, argv[1], NULL, NULL);
129 if (ret < 0) {
130 fprintf(stderr, "Cannot open input file '%s': ", argv[1]);
131 goto finish;
132 }
133
134 /* find the first H.264 video stream */
135 for (i = 0; i < input_ctx->nb_streams; i++) {
136 AVStream *st = input_ctx->streams[i];
137
138 if (st->codecpar->codec_id == AV_CODEC_ID_H264 && !video_st)
139 video_st = st;
140 else
142 }
143 if (!video_st) {
144 fprintf(stderr, "No H.264 video stream in the input file\n");
145 goto finish;
146 }
147
148 /* open the hardware device */
150 "auto", NULL, 0);
151 if (ret < 0) {
152 fprintf(stderr, "Cannot open the hardware device\n");
153 goto finish;
154 }
155
156 /* initialize the decoder */
157 decoder = avcodec_find_decoder_by_name("h264_qsv");
158 if (!decoder) {
159 fprintf(stderr, "The QSV decoder is not present in libavcodec\n");
160 goto finish;
161 }
162
164 if (!decoder_ctx) {
165 ret = AVERROR(ENOMEM);
166 goto finish;
167 }
169 if (video_st->codecpar->extradata_size) {
172 if (!decoder_ctx->extradata) {
173 ret = AVERROR(ENOMEM);
174 goto finish;
175 }
176 memcpy(decoder_ctx->extradata, video_st->codecpar->extradata,
177 video_st->codecpar->extradata_size);
179 }
180
181
184
185 ret = avcodec_open2(decoder_ctx, NULL, NULL);
186 if (ret < 0) {
187 fprintf(stderr, "Error opening the decoder: ");
188 goto finish;
189 }
190
191 /* open the output stream */
192 ret = avio_open(&output_ctx, argv[2], AVIO_FLAG_WRITE);
193 if (ret < 0) {
194 fprintf(stderr, "Error opening the output context: ");
195 goto finish;
196 }
197
199 sw_frame = av_frame_alloc();
201 if (!frame || !sw_frame || !pkt) {
202 ret = AVERROR(ENOMEM);
203 goto finish;
204 }
205
206 /* actual decoding */
207 while (ret >= 0) {
208 ret = av_read_frame(input_ctx, pkt);
209 if (ret < 0)
210 break;
211
212 if (pkt->stream_index == video_st->index)
213 ret = decode_packet(decoder_ctx, frame, sw_frame, pkt, output_ctx);
214
216 }
217
218 /* flush the decoder */
219 ret = decode_packet(decoder_ctx, frame, sw_frame, NULL, output_ctx);
220
221finish:
222 if (ret < 0) {
223 char buf[1024];
224 av_strerror(ret, buf, sizeof(buf));
225 fprintf(stderr, "%s\n", buf);
226 }
227
228 avformat_close_input(&input_ctx);
229
231 av_frame_free(&sw_frame);
233
235
236 av_buffer_unref(&device_ref);
237
238 avio_close(output_ctx);
239
240 return ret;
241}
Libavcodec external API header.
Main libavformat public API header.
Buffered I/O operations.
int avio_open(AVIOContext **s, const char *url, int flags)
Create and initialize a AVIOContext for accessing the resource indicated by url.
#define AVIO_FLAG_WRITE
write-only
Definition avio.h:618
void avio_write(AVIOContext *s, const unsigned char *buf, int size)
int avio_close(AVIOContext *s)
Close the resource accessed by the AVIOContext s and free it.
int main(int argc, char *argv[])
refcounted data buffer API
static AVPacket * pkt
static int decode_packet(AVCodecContext *dec, const AVPacket *pkt)
static AVFrame * frame
error code definitions
int avcodec_open2(AVCodecContext *avctx, const AVCodec *codec, AVDictionary **options)
Initialize the AVCodecContext to use the given AVCodec.
AVCodecContext * avcodec_alloc_context3(const AVCodec *codec)
Allocate an AVCodecContext and set its fields to default values.
const AVCodec * avcodec_find_decoder_by_name(const char *name)
Find a registered decoder with the specified name.
void avcodec_free_context(AVCodecContext **avctx)
Free the codec context and everything associated with it and write NULL to the provided pointer.
@ AV_CODEC_ID_H264
Definition codec_id.h:79
int avcodec_receive_frame(AVCodecContext *avctx, AVFrame *frame)
Return decoded output data from a decoder or encoder (when the AV_CODEC_FLAG_RECON_FRAME flag is used...
int avcodec_send_packet(AVCodecContext *avctx, const AVPacket *avpkt)
Supply raw packet data as input to a decoder.
#define AV_INPUT_BUFFER_PADDING_SIZE
Required number of additionally allocated bytes at the end of the input bitstream for decoding.
Definition defs.h:40
@ AVDISCARD_ALL
discard all
Definition defs.h:221
void av_packet_free(AVPacket **pkt)
Free the packet, if the packet is reference counted, it will be unreferenced first.
void av_packet_unref(AVPacket *pkt)
Wipe the packet.
AVPacket * av_packet_alloc(void)
Allocate an AVPacket and set its fields to default values.
int av_read_frame(AVFormatContext *s, AVPacket *pkt)
Return the next frame of a stream.
int avformat_open_input(AVFormatContext **ps, const char *url, const AVInputFormat *fmt, AVDictionary **options)
Open an input stream and read the header.
void avformat_close_input(AVFormatContext **s)
Close an opened input AVFormatContext.
void av_buffer_unref(AVBufferRef **buf)
Free a given reference and automatically free the buffer if there are no more references to it.
AVBufferRef * av_buffer_ref(const AVBufferRef *buf)
Create a new reference to an AVBuffer.
int av_strerror(int errnum, char *errbuf, size_t errbuf_size)
Put a description of the AVERROR code errnum in errbuf.
#define AVERROR_EOF
End of file.
Definition error.h:57
#define AVERROR(e)
Definition error.h:45
void av_frame_unref(AVFrame *frame)
Unreference all the buffers referenced by frame and reset the frame fields.
void av_frame_free(AVFrame **frame)
Free the frame and any dynamically allocated objects in it, e.g.
AVFrame * av_frame_alloc(void)
Allocate an AVFrame and set its fields to default values.
void * av_mallocz(size_t size) av_malloc_attrib
Allocate a memory block with alignment suitable for all memory accesses (including vectors if availab...
int av_hwframe_transfer_data(AVFrame *dst, const AVFrame *src, int flags)
Copy data to or from a hw surface.
int av_hwdevice_ctx_create(AVBufferRef **device_ctx, enum AVHWDeviceType type, const char *device, AVDictionary *opts, int flags)
Open a device of the specified type and create an AVHWDeviceContext for it.
@ AV_HWDEVICE_TYPE_QSV
Definition hwcontext.h:33
An API-specific header for AV_HWDEVICE_TYPE_QSV.
#define FF_ARRAY_ELEMS(a)
Definition macros.h:53
Memory handling functions.
AVPixelFormat
Pixel format.
Definition pixfmt.h:71
@ AV_PIX_FMT_NONE
Definition pixfmt.h:72
@ AV_PIX_FMT_QSV
HW acceleration through QSV, data[3] contains a pointer to the mfxFrameSurface1 structure.
Definition pixfmt.h:247
static int get_format(AVCodecContext *avctx, const enum AVPixelFormat *pix_fmts)
Definition qsv_decode.c:44
A reference to a data buffer.
Definition buffer.h:82
main external API structure.
Definition avcodec.h:451
enum AVPixelFormat(* get_format)(struct AVCodecContext *s, const enum AVPixelFormat *fmt)
Callback to negotiate the pixel format.
Definition avcodec.h:793
uint8_t * extradata
some codecs need / can use extradata like Huffman tables.
Definition avcodec.h:529
AVBufferRef * hw_device_ctx
A reference to the AVHWDeviceContext describing the device which will be used by a hardware encoder/d...
Definition avcodec.h:1507
enum AVCodecID codec_id
Definition avcodec.h:461
int extradata_size
Definition avcodec.h:530
int extradata_size
Size of the extradata content in bytes.
Definition codec_par.h:73
uint8_t * extradata
Extra binary data needed for initializing the decoder, codec-dependent.
Definition codec_par.h:69
enum AVCodecID codec_id
Specific type of the encoded data (the codec used).
Definition codec_par.h:55
AVCodec.
Definition codec.h:187
Format I/O context.
Definition avformat.h:1287
unsigned int nb_streams
Number of elements in AVFormatContext.streams.
Definition avformat.h:1343
AVStream ** streams
A list of all streams in the file.
Definition avformat.h:1355
This structure describes decoded (raw) audio or video data.
Definition frame.h:389
uint8_t * data[AV_NUM_DATA_POINTERS]
pointer to the picture/channel planes.
Definition frame.h:410
int width
Definition frame.h:461
int height
Definition frame.h:461
int linesize[AV_NUM_DATA_POINTERS]
For video, a positive or negative value, which is typically indicating the size in bytes of each pict...
Definition frame.h:434
Bytestream IO Context.
Definition avio.h:160
This structure stores compressed data.
Definition packet.h:516
int stream_index
Definition packet.h:541
Stream structure.
Definition avformat.h:748
AVCodecParameters * codecpar
Codec parameters associated with this stream.
Definition avformat.h:771
enum AVDiscard discard
Selects which packets can be discarded at will and do not need to be demuxed.
Definition avformat.h:819
int index
stream index in AVFormatContext
Definition avformat.h:754
static AVCodecContext * decoder_ctx