]> git.sesse.net Git - ffmpeg/blob - doc/examples/vaapi_encode.c
Merge commit '508378556631dc18d32247b4a4e35703758e1ca9'
[ffmpeg] / doc / examples / vaapi_encode.c
1 /*
2  * Video Acceleration API (video encoding) encode sample
3  *
4  * This file is part of FFmpeg.
5  *
6  * FFmpeg is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * FFmpeg is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with FFmpeg; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19  */
20
21 /**
22  * @file
23  * Intel VAAPI-accelerated encoding example.
24  *
25  * @example vaapi_encode.c
26  * This example shows how to do VAAPI-accelerated encoding. now only support NV12
27  * raw file, usage like: vaapi_encode 1920 1080 input.yuv output.h264
28  *
29  */
30
31 #include <stdio.h>
32 #include <string.h>
33 #include <errno.h>
34
35 #include <libavcodec/avcodec.h>
36 #include <libavutil/pixdesc.h>
37 #include <libavutil/hwcontext.h>
38
39 static int width, height;
40 static AVBufferRef *hw_device_ctx = NULL;
41
42 static int set_hwframe_ctx(AVCodecContext *ctx, AVBufferRef *hw_device_ctx)
43 {
44     AVBufferRef *hw_frames_ref;
45     AVHWFramesContext *frames_ctx = NULL;
46     int err = 0;
47
48     if (!(hw_frames_ref = av_hwframe_ctx_alloc(hw_device_ctx))) {
49         fprintf(stderr, "Failed to create VAAPI frame context.\n");
50         return -1;
51     }
52     frames_ctx = (AVHWFramesContext *)(hw_frames_ref->data);
53     frames_ctx->format    = AV_PIX_FMT_VAAPI;
54     frames_ctx->sw_format = AV_PIX_FMT_NV12;
55     frames_ctx->width     = width;
56     frames_ctx->height    = height;
57     frames_ctx->initial_pool_size = 20;
58     if ((err = av_hwframe_ctx_init(hw_frames_ref)) < 0) {
59         fprintf(stderr, "Failed to initialize VAAPI frame context."
60                 "Error code: %s\n",av_err2str(err));
61         av_buffer_unref(&hw_frames_ref);
62         return err;
63     }
64     ctx->hw_frames_ctx = av_buffer_ref(hw_frames_ref);
65     if (!ctx->hw_frames_ctx)
66         err = AVERROR(ENOMEM);
67
68     av_buffer_unref(&hw_frames_ref);
69     return err;
70 }
71
72 static int encode_write(AVCodecContext *avctx, AVFrame *frame, FILE *fout)
73 {
74     int ret = 0;
75     AVPacket enc_pkt;
76
77     av_init_packet(&enc_pkt);
78     enc_pkt.data = NULL;
79     enc_pkt.size = 0;
80
81     if ((ret = avcodec_send_frame(avctx, frame)) < 0) {
82         fprintf(stderr, "Error code: %s\n", av_err2str(ret));
83         goto end;
84     }
85     while (1) {
86         ret = avcodec_receive_packet(avctx, &enc_pkt);
87         if (ret)
88             break;
89
90         enc_pkt.stream_index = 0;
91         ret = fwrite(enc_pkt.data, enc_pkt.size, 1, fout);
92         av_packet_unref(&enc_pkt);
93     }
94
95 end:
96     ret = ((ret == AVERROR(EAGAIN)) ? 0 : -1);
97     return ret;
98 }
99
100 int main(int argc, char *argv[])
101 {
102     int size, err;
103     FILE *fin = NULL, *fout = NULL;
104     AVFrame *sw_frame = NULL, *hw_frame = NULL;
105     AVCodecContext *avctx = NULL;
106     AVCodec *codec = NULL;
107     const char *enc_name = "h264_vaapi";
108
109     if (argc < 5) {
110         fprintf(stderr, "Usage: %s <width> <height> <input file> <output file>\n", argv[0]);
111         return -1;
112     }
113
114     width  = atoi(argv[1]);
115     height = atoi(argv[2]);
116     size   = width * height;
117
118     if (!(fin = fopen(argv[3], "r"))) {
119         fprintf(stderr, "Fail to open input file : %s\n", strerror(errno));
120         return -1;
121     }
122     if (!(fout = fopen(argv[4], "w+b"))) {
123         fprintf(stderr, "Fail to open output file : %s\n", strerror(errno));
124         err = -1;
125         goto close;
126     }
127
128     avcodec_register_all();
129
130     err = av_hwdevice_ctx_create(&hw_device_ctx, AV_HWDEVICE_TYPE_VAAPI,
131                                  NULL, NULL, 0);
132     if (err < 0) {
133         fprintf(stderr, "Failed to create a VAAPI device. Error code: %s\n", av_err2str(err));
134         goto close;
135     }
136
137     if (!(codec = avcodec_find_encoder_by_name(enc_name))) {
138         fprintf(stderr, "Could not find encoder.\n");
139         err = -1;
140         goto close;
141     }
142
143     if (!(avctx = avcodec_alloc_context3(codec))) {
144         err = AVERROR(ENOMEM);
145         goto close;
146     }
147
148     avctx->width     = width;
149     avctx->height    = height;
150     avctx->time_base = (AVRational){1, 25};
151     avctx->framerate = (AVRational){25, 1};
152     avctx->sample_aspect_ratio = (AVRational){1, 1};
153     avctx->pix_fmt   = AV_PIX_FMT_VAAPI;
154
155     /* set hw_frames_ctx for encoder's AVCodecContext */
156     if ((err = set_hwframe_ctx(avctx, hw_device_ctx)) < 0) {
157         fprintf(stderr, "Failed to set hwframe context.\n");
158         goto close;
159     }
160
161     if ((err = avcodec_open2(avctx, codec, NULL)) < 0) {
162         fprintf(stderr, "Cannot open video encoder codec. Error code: %s\n", av_err2str(err));
163         goto close;
164     }
165
166     while (1) {
167         if (!(sw_frame = av_frame_alloc())) {
168             err = AVERROR(ENOMEM);
169             goto close;
170         }
171         /* read data into software frame, and transfer them into hw frame */
172         sw_frame->width  = width;
173         sw_frame->height = height;
174         sw_frame->format = AV_PIX_FMT_NV12;
175         if ((err = av_frame_get_buffer(sw_frame, 32)) < 0)
176             goto close;
177         if ((err = fread((uint8_t*)(sw_frame->data[0]), size, 1, fin)) <= 0)
178             break;
179         if ((err = fread((uint8_t*)(sw_frame->data[1]), size/2, 1, fin)) <= 0)
180             break;
181
182         if (!(hw_frame = av_frame_alloc())) {
183             err = AVERROR(ENOMEM);
184             goto close;
185         }
186         if ((err = av_hwframe_get_buffer(avctx->hw_frames_ctx, hw_frame, 0)) < 0) {
187             fprintf(stderr, "Error code: %s.\n", av_err2str(err));
188             goto close;
189         }
190         if (!hw_frame->hw_frames_ctx) {
191             err = AVERROR(ENOMEM);
192             goto close;
193         }
194         if ((err = av_hwframe_transfer_data(hw_frame, sw_frame, 0)) < 0) {
195             fprintf(stderr, "Error while transferring frame data to surface."
196                     "Error code: %s.\n", av_err2str(err));
197             goto close;
198         }
199
200         if ((err = (encode_write(avctx, hw_frame, fout))) < 0) {
201             fprintf(stderr, "Failed to encode.\n");
202             goto close;
203         }
204         av_frame_free(&hw_frame);
205         av_frame_free(&sw_frame);
206     }
207
208     /* flush encoder */
209     err = encode_write(avctx, NULL, fout);
210     if (err == AVERROR_EOF)
211         err = 0;
212
213 close:
214     if (fin)
215         fclose(fin);
216     if (fout)
217         fclose(fout);
218     av_frame_free(&sw_frame);
219     av_frame_free(&hw_frame);
220     if (avctx)
221         avcodec_free_context(&avctx);
222     av_buffer_unref(&hw_device_ctx);
223
224     return err;
225 }