FFmpeg指定x265编码器线程数

本文详细探讨了FFmpeg中-x265-threads参数的含义,指出其并不直接代表编码器的线程数,而是用于设置x265编码时并行处理的帧数。在HEVC的波前并行处理机制下,实际线程数可能大于该值。正确指定x265编码器线程数的方法是使用-x265-preset,以控制每个NUMA节点上的线程池大小。
摘要由CSDN通过智能技术生成

转载请注明出处:http://cyc.wiki/index.php/2018/07/17/ffmpeg-x265-threads/


FFmpeg的-threads参数

FFmpeg调用编码器时,一般使用-threads参数对编码器使用的线程数进行设置。
比如:

ffmpeg -s 1920x1080 -framerate 25 -i input.yuv -c:v libx264 -threads 4 -y output.h264

对于x264编码器,这个参数的意义是明确的,既是编码器占用的线程数,也是编码器并行处理的帧数。

但对于x265编码器,这个参数的意义是否还具有同样的意义?
比如:

ffmpeg -s 1920x1080 -framerate 25 -i input.yuv -c:v libx265 -threads 4 -y output.hevc

这个答案存在分歧,虽然不清楚FFmpeg作者的用意,但确实会对一般使用者产生误导,所以在这里阐明。

FFmpeg中libx265.c的线程参数透传

FFmpeg中封装了调用x265编码器的libx265编码器模块,代码位于libavcodec/libx265.c,其中

    ctx->params->frameNumTh
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
FFmpeg是一个非常强大的开源多媒体处理框架,提供了丰富的编解码器和工具,可以对音视频进行录制、转码、剪辑、播放等操作。在使用FFmpeg进行编码时,可以通过多线程技术提高编码效率。 下面是一个简单的C++多线程编码示例: ```cpp #include <iostream> #include <thread> #include <mutex> #include <condition_variable> #include <queue> #include <chrono> extern "C" { #include <libavcodec/avcodec.h> #include <libavformat/avformat.h> #include <libavutil/opt.h> } using namespace std::chrono_literals; // 用于存储编码据的队列 std::queue<AVPacket*> packet_queue; std::mutex packet_mutex; std::condition_variable packet_cond; // 编码线程 void encode_thread(AVCodecContext* codec_ctx, AVFrame* frame, AVFormatContext* fmt_ctx) { int ret; AVPacket* pkt = av_packet_alloc(); while (true) { // 从队列中取出一帧待编码据 std::unique_lock<std::mutex> lock(packet_mutex); packet_cond.wait(lock, [] { return !packet_queue.empty(); }); pkt = packet_queue.front(); packet_queue.pop(); // 编码该帧据 ret = avcodec_send_frame(codec_ctx, frame); if (ret < 0) { std::cerr << "Error sending frame to encoder: " << av_err2str(ret) << std::endl; break; } while (ret >= 0) { ret = avcodec_receive_packet(codec_ctx, pkt); if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) { break; } else if (ret < 0) { std::cerr << "Error receiving packet from encoder: " << av_err2str(ret) << std::endl; break; } // 将编码后的据写入文件 av_packet_rescale_ts(pkt, codec_ctx->time_base, fmt_ctx->streams[0]->time_base); av_interleaved_write_frame(fmt_ctx, pkt); av_packet_unref(pkt); } av_packet_free(&pkt); } } int main(int argc, char** argv) { // 初始化FFmpeg av_register_all(); avcodec_register_all(); // 打开输入文件并获取视频流信息 AVFormatContext* in_fmt_ctx = nullptr; if (avformat_open_input(&in_fmt_ctx, "input.mp4", nullptr, nullptr) < 0) { std::cerr << "Error opening input file" << std::endl; return 1; } if (avformat_find_stream_info(in_fmt_ctx, nullptr) < 0) { std::cerr << "Error finding stream information" << std::endl; return 1; } int video_stream_index = av_find_best_stream(in_fmt_ctx, AVMEDIA_TYPE_VIDEO, -1, -1, nullptr, 0); if (video_stream_index < 0) { std::cerr << "Error finding video stream" << std::endl; return 1; } AVStream* in_video_stream = in_fmt_ctx->streams[video_stream_index]; // 打开输出文件并初始化视频编码器 AVFormatContext* out_fmt_ctx = nullptr; if (avformat_alloc_output_context2(&out_fmt_ctx, nullptr, nullptr, "output.mp4") < 0) { std::cerr << "Error allocating output context" << std::endl; return 1; } AVStream* out_video_stream = avformat_new_stream(out_fmt_ctx, nullptr); if (!out_video_stream) { std::cerr << "Error creating new video stream" << std::endl; return 1; } AVCodec* codec = avcodec_find_encoder(out_fmt_ctx->oformat->video_codec); if (!codec) { std::cerr << "Error finding video encoder" << std::endl; return 1; } AVCodecContext* codec_ctx = avcodec_alloc_context3(codec); if (!codec_ctx) { std::cerr << "Error allocating video codec context" << std::endl; return 1; } codec_ctx->width = in_video_stream->codecpar->width; codec_ctx->height = in_video_stream->codecpar->height; codec_ctx->pix_fmt = codec->pix_fmts[0]; codec_ctx->time_base = { 1, in_video_stream->codecpar->frame_rate.num }; codec_ctx->framerate = { in_video_stream->codecpar->frame_rate, 1 }; if (avcodec_open2(codec_ctx, codec, nullptr) < 0) { std::cerr << "Error opening video codec" << std::endl; return 1; } if (avcodec_parameters_from_context(out_video_stream->codecpar, codec_ctx) < 0) { std::cerr << "Error copying codec parameters" << std::endl; return 1; } out_video_stream->time_base = codec_ctx->time_base; if (avio_open(&out_fmt_ctx->pb, "output.mp4", AVIO_FLAG_WRITE) < 0) { std::cerr << "Error opening output file" << std::endl; return 1; } if (avformat_write_header(out_fmt_ctx, nullptr) < 0) { std::cerr << "Error writing output file header" << std::endl; return 1; } // 初始化视频帧 AVFrame* frame = av_frame_alloc(); frame->format = codec_ctx->pix_fmt; frame->width = codec_ctx->width; frame->height = codec_ctx->height; if (av_frame_get_buffer(frame, 0) < 0) { std::cerr << "Error allocating frame buffer" << std::endl; return 1; } // 启动编码线程 std::thread encode_thr(encode_thread, codec_ctx, frame, out_fmt_ctx); // 读取视频帧并送入编码队列 AVPacket* pkt = av_packet_alloc(); AVFrame* in_frame = av_frame_alloc(); while (av_read_frame(in_fmt_ctx, pkt) == 0) { if (pkt->stream_index == video_stream_index) { avcodec_send_packet(codec_ctx, pkt); avcodec_receive_frame(codec_ctx, in_frame); // 将原始帧据转换为目标格式 SwsContext* sws_ctx = sws_getContext( in_frame->width, in_frame->height, static_cast<AVPixelFormat>(in_frame->format), codec_ctx->width, codec_ctx->height, codec_ctx->pix_fmt, SWS_BICUBIC, nullptr, nullptr, nullptr); if (!sws_ctx) { std::cerr << "Error creating sws context" << std::endl; return 1; } sws_scale(sws_ctx, in_frame->data, in_frame->linesize, 0, in_frame->height, frame->data, frame->linesize); sws_freeContext(sws_ctx); // 将转换后的帧据送入编码队列 std::unique_lock<std::mutex> lock(packet_mutex); packet_queue.push(av_packet_clone(pkt)); packet_cond.notify_one(); } av_packet_unref(pkt); } av_packet_free(&pkt); av_frame_free(&in_frame); // 结束编码线程 encode_thr.join(); // 写入文件尾部并释放资源 av_write_trailer(out_fmt_ctx); avformat_close_input(&in_fmt_ctx); if (out_fmt_ctx && !(out_fmt_ctx->oformat->flags & AVFMT_NOFILE)) { avio_closep(&out_fmt_ctx->pb); } avformat_free_context(out_fmt_ctx); avcodec_free_context(&codec_ctx); av_frame_free(&frame); return 0; } ``` 该示例中,主线程读取原始视频帧并将其送入编码队列中,编码线程从队列中取出待编码据,并将编码后的据写入文件。多线程编码可以有效利用多核CPU提高编码效率,但同时也需要注意线程同步和据安全等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值