|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "config.h" |
|
|
|
|
|
#include <stdio.h> |
|
|
|
|
|
#include "libavformat/avformat.h" |
|
|
#include "libavformat/avio.h" |
|
|
|
|
|
#include "libavcodec/avcodec.h" |
|
|
|
|
|
#include "libavutil/buffer.h" |
|
|
#include "libavutil/error.h" |
|
|
#include "libavutil/hwcontext.h" |
|
|
#include "libavutil/hwcontext_qsv.h" |
|
|
#include "libavutil/mem.h" |
|
|
|
|
|
static int get_format(AVCodecContext *avctx, const enum AVPixelFormat *pix_fmts) |
|
|
{ |
|
|
while (*pix_fmts != AV_PIX_FMT_NONE) { |
|
|
if (*pix_fmts == AV_PIX_FMT_QSV) { |
|
|
return AV_PIX_FMT_QSV; |
|
|
} |
|
|
|
|
|
pix_fmts++; |
|
|
} |
|
|
|
|
|
fprintf(stderr, "The QSV pixel format not offered in get_format()\n"); |
|
|
|
|
|
return AV_PIX_FMT_NONE; |
|
|
} |
|
|
|
|
|
static int decode_packet(AVCodecContext *decoder_ctx, |
|
|
AVFrame *frame, AVFrame *sw_frame, |
|
|
AVPacket *pkt, AVIOContext *output_ctx) |
|
|
{ |
|
|
int ret = 0; |
|
|
|
|
|
ret = avcodec_send_packet(decoder_ctx, pkt); |
|
|
if (ret < 0) { |
|
|
fprintf(stderr, "Error during decoding\n"); |
|
|
return ret; |
|
|
} |
|
|
|
|
|
while (ret >= 0) { |
|
|
int i, j; |
|
|
|
|
|
ret = avcodec_receive_frame(decoder_ctx, frame); |
|
|
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) |
|
|
break; |
|
|
else if (ret < 0) { |
|
|
fprintf(stderr, "Error during decoding\n"); |
|
|
return ret; |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ret = av_hwframe_transfer_data(sw_frame, frame, 0); |
|
|
if (ret < 0) { |
|
|
fprintf(stderr, "Error transferring the data to system memory\n"); |
|
|
goto fail; |
|
|
} |
|
|
|
|
|
for (i = 0; i < FF_ARRAY_ELEMS(sw_frame->data) && sw_frame->data[i]; i++) |
|
|
for (j = 0; j < (sw_frame->height >> (i > 0)); j++) |
|
|
avio_write(output_ctx, sw_frame->data[i] + j * sw_frame->linesize[i], sw_frame->width); |
|
|
|
|
|
fail: |
|
|
av_frame_unref(sw_frame); |
|
|
av_frame_unref(frame); |
|
|
|
|
|
if (ret < 0) |
|
|
return ret; |
|
|
} |
|
|
|
|
|
return 0; |
|
|
} |
|
|
|
|
|
int main(int argc, char **argv) |
|
|
{ |
|
|
AVFormatContext *input_ctx = NULL; |
|
|
AVStream *video_st = NULL; |
|
|
AVCodecContext *decoder_ctx = NULL; |
|
|
const AVCodec *decoder; |
|
|
|
|
|
AVPacket *pkt = NULL; |
|
|
AVFrame *frame = NULL, *sw_frame = NULL; |
|
|
|
|
|
AVIOContext *output_ctx = NULL; |
|
|
|
|
|
int ret, i; |
|
|
|
|
|
AVBufferRef *device_ref = NULL; |
|
|
|
|
|
if (argc < 3) { |
|
|
fprintf(stderr, "Usage: %s <input file> <output file>\n", argv[0]); |
|
|
return 1; |
|
|
} |
|
|
|
|
|
|
|
|
ret = avformat_open_input(&input_ctx, argv[1], NULL, NULL); |
|
|
if (ret < 0) { |
|
|
fprintf(stderr, "Cannot open input file '%s': ", argv[1]); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
|
|
|
for (i = 0; i < input_ctx->nb_streams; i++) { |
|
|
AVStream *st = input_ctx->streams[i]; |
|
|
|
|
|
if (st->codecpar->codec_id == AV_CODEC_ID_H264 && !video_st) |
|
|
video_st = st; |
|
|
else |
|
|
st->discard = AVDISCARD_ALL; |
|
|
} |
|
|
if (!video_st) { |
|
|
fprintf(stderr, "No H.264 video stream in the input file\n"); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
|
|
|
ret = av_hwdevice_ctx_create(&device_ref, AV_HWDEVICE_TYPE_QSV, |
|
|
"auto", NULL, 0); |
|
|
if (ret < 0) { |
|
|
fprintf(stderr, "Cannot open the hardware device\n"); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
|
|
|
decoder = avcodec_find_decoder_by_name("h264_qsv"); |
|
|
if (!decoder) { |
|
|
fprintf(stderr, "The QSV decoder is not present in libavcodec\n"); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
decoder_ctx = avcodec_alloc_context3(decoder); |
|
|
if (!decoder_ctx) { |
|
|
ret = AVERROR(ENOMEM); |
|
|
goto finish; |
|
|
} |
|
|
decoder_ctx->codec_id = AV_CODEC_ID_H264; |
|
|
if (video_st->codecpar->extradata_size) { |
|
|
decoder_ctx->extradata = av_mallocz(video_st->codecpar->extradata_size + |
|
|
AV_INPUT_BUFFER_PADDING_SIZE); |
|
|
if (!decoder_ctx->extradata) { |
|
|
ret = AVERROR(ENOMEM); |
|
|
goto finish; |
|
|
} |
|
|
memcpy(decoder_ctx->extradata, video_st->codecpar->extradata, |
|
|
video_st->codecpar->extradata_size); |
|
|
decoder_ctx->extradata_size = video_st->codecpar->extradata_size; |
|
|
} |
|
|
|
|
|
|
|
|
decoder_ctx->hw_device_ctx = av_buffer_ref(device_ref); |
|
|
decoder_ctx->get_format = get_format; |
|
|
|
|
|
ret = avcodec_open2(decoder_ctx, NULL, NULL); |
|
|
if (ret < 0) { |
|
|
fprintf(stderr, "Error opening the decoder: "); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
|
|
|
ret = avio_open(&output_ctx, argv[2], AVIO_FLAG_WRITE); |
|
|
if (ret < 0) { |
|
|
fprintf(stderr, "Error opening the output context: "); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
frame = av_frame_alloc(); |
|
|
sw_frame = av_frame_alloc(); |
|
|
pkt = av_packet_alloc(); |
|
|
if (!frame || !sw_frame || !pkt) { |
|
|
ret = AVERROR(ENOMEM); |
|
|
goto finish; |
|
|
} |
|
|
|
|
|
|
|
|
while (ret >= 0) { |
|
|
ret = av_read_frame(input_ctx, pkt); |
|
|
if (ret < 0) |
|
|
break; |
|
|
|
|
|
if (pkt->stream_index == video_st->index) |
|
|
ret = decode_packet(decoder_ctx, frame, sw_frame, pkt, output_ctx); |
|
|
|
|
|
av_packet_unref(pkt); |
|
|
} |
|
|
|
|
|
|
|
|
ret = decode_packet(decoder_ctx, frame, sw_frame, NULL, output_ctx); |
|
|
|
|
|
finish: |
|
|
if (ret < 0) { |
|
|
char buf[1024]; |
|
|
av_strerror(ret, buf, sizeof(buf)); |
|
|
fprintf(stderr, "%s\n", buf); |
|
|
} |
|
|
|
|
|
avformat_close_input(&input_ctx); |
|
|
|
|
|
av_frame_free(&frame); |
|
|
av_frame_free(&sw_frame); |
|
|
av_packet_free(&pkt); |
|
|
|
|
|
avcodec_free_context(&decoder_ctx); |
|
|
|
|
|
av_buffer_unref(&device_ref); |
|
|
|
|
|
avio_close(output_ctx); |
|
|
|
|
|
return ret; |
|
|
} |
|
|
|