encoder_video_x264.cpp 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. #include "encoder_video_x264.h"
  2. #include "headers_ffmpeg.h"
  3. #include "error_define.h"
  4. #include "log_helper.h"
  5. namespace am {
  6. encoder_video_x264::encoder_video_x264()
  7. {
  8. ffmpeg_register_all();
  9. _encoder = NULL;
  10. _encoder_ctx = NULL;
  11. _frame = NULL;
  12. _buff = NULL;
  13. _buff_size = 0;
  14. _y_size = 0;
  15. }
  16. encoder_video_x264::~encoder_video_x264()
  17. {
  18. stop();
  19. cleanup();
  20. }
  21. int encoder_video_x264::init(
  22. int pic_width, int pic_height, int frame_rate, int bit_rate, int qb, int key_pic_sec)
  23. {
  24. if (_inited == true)
  25. return AE_NO;
  26. int err = AE_NO;
  27. int ret = 0;
  28. AVDictionary *options = 0;
  29. // 使用ultrafast预设以获得最低延迟
  30. av_dict_set(&options, "preset", "ultrafast", 0);
  31. av_dict_set(&options, "tune", "zerolatency", 0);
  32. // 设置x264参数以匹配低延迟配置
  33. av_dict_set(&options, "x264-params", "keyint=30:min-keyint=30:no-scenecut=1", 0);
  34. av_dict_set(&options, "profile", "high", 0);
  35. av_dict_set(&options, "level", "4.1", 0);
  36. // 与命令行一致,明确设置CRF为18
  37. av_dict_set(&options, "crf", "18", 0);
  38. do {
  39. _encoder = avcodec_find_encoder(AV_CODEC_ID_H264);
  40. if (!_encoder) {
  41. err = AE_FFMPEG_FIND_ENCODER_FAILED;
  42. break;
  43. }
  44. _encoder_ctx = avcodec_alloc_context3(_encoder);
  45. if (!_encoder_ctx) {
  46. err = AE_FFMPEG_ALLOC_CONTEXT_FAILED;
  47. break;
  48. }
  49. _encoder_ctx->codec_id = AV_CODEC_ID_H264;
  50. _encoder_ctx->codec_type = AVMEDIA_TYPE_VIDEO;
  51. _encoder_ctx->pix_fmt = AV_PIX_FMT_YUV420P;
  52. _encoder_ctx->width = pic_width;
  53. _encoder_ctx->height = pic_height;
  54. _encoder_ctx->time_base.num = 1;
  55. _encoder_ctx->time_base.den = frame_rate;
  56. _encoder_ctx->framerate = {frame_rate, 1};
  57. _encoder_ctx->bit_rate = bit_rate;
  58. // 设置GOP大小为30帧,与用户ffmpeg命令一致
  59. _encoder_ctx->gop_size = 30;
  60. // 使用CRF 18以获得高质量和低延迟的平衡
  61. // 注意:FFmpeg的libx264编码器通过crf参数控制质量
  62. // 这里我们设置合理的qmin和qmax值
  63. _encoder_ctx->qmin = 10;
  64. _encoder_ctx->qmax = 25;
  65. // 设置CRF值(通过全局质量参数)
  66. _encoder_ctx->global_quality = 18 * FF_QP2LAMBDA;
  67. _encoder_ctx->max_b_frames = 0; //NO B Frame
  68. _encoder_ctx->flags |= AV_CODEC_FLAG_GLOBAL_HEADER;
  69. ret = avcodec_open2(_encoder_ctx, _encoder, &options);
  70. if (ret != 0) {
  71. err = AE_FFMPEG_OPEN_CODEC_FAILED;
  72. break;
  73. }
  74. _frame = av_frame_alloc();
  75. _buff_size = av_image_get_buffer_size(_encoder_ctx->pix_fmt,
  76. _encoder_ctx->width,
  77. _encoder_ctx->height,
  78. 1);
  79. _buff = (uint8_t *) av_malloc(_buff_size);
  80. if (!_buff) {
  81. break;
  82. }
  83. av_image_fill_arrays(_frame->data,
  84. _frame->linesize,
  85. _buff,
  86. _encoder_ctx->pix_fmt,
  87. _encoder_ctx->width,
  88. _encoder_ctx->height,
  89. 1);
  90. _frame->format = _encoder_ctx->pix_fmt;
  91. _frame->width = _encoder_ctx->width;
  92. _frame->height = _encoder_ctx->height;
  93. _y_size = _encoder_ctx->width * _encoder_ctx->height;
  94. _time_base = _encoder_ctx->time_base;
  95. _inited = true;
  96. } while (0);
  97. if (err != AE_NO) {
  98. al_debug("%s,error:%d %lu", err2str(err), ret, GetLastError());
  99. cleanup();
  100. }
  101. if (options)
  102. av_dict_free(&options);
  103. return err;
  104. }
  105. int encoder_video_x264::get_extradata_size()
  106. {
  107. return _encoder_ctx->extradata_size;
  108. }
  109. const uint8_t *encoder_video_x264::get_extradata()
  110. {
  111. return (const uint8_t *) _encoder_ctx->extradata;
  112. }
  113. AVCodecID encoder_video_x264::get_codec_id()
  114. {
  115. if (_inited == false)
  116. return AV_CODEC_ID_NONE;
  117. return _encoder->id;
  118. }
  119. void encoder_video_x264::cleanup()
  120. {
  121. if (_frame)
  122. av_free(_frame);
  123. _frame = NULL;
  124. if (_buff)
  125. av_free(_buff);
  126. _buff = NULL;
  127. if (_encoder)
  128. avcodec_close(_encoder_ctx);
  129. _encoder = NULL;
  130. if (_encoder_ctx)
  131. avcodec_free_context(&_encoder_ctx);
  132. _encoder_ctx = NULL;
  133. }
  134. int encoder_video_x264::encode(AVFrame *frame, AVPacket *packet)
  135. {
  136. int ret = avcodec_send_frame(_encoder_ctx, frame);
  137. if (ret < 0) {
  138. return AE_FFMPEG_ENCODE_FRAME_FAILED;
  139. }
  140. while (ret >= 0) {
  141. av_init_packet(packet);
  142. ret = avcodec_receive_packet(_encoder_ctx, packet);
  143. if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) {
  144. break;
  145. }
  146. if (ret < 0) {
  147. return AE_FFMPEG_READ_PACKET_FAILED;
  148. }
  149. if (ret == 0 && _on_data)
  150. _on_data(packet);
  151. av_packet_unref(packet);
  152. }
  153. return AE_NO;
  154. }
  155. void encoder_video_x264::encode_loop()
  156. {
  157. AVPacket *packet = av_packet_alloc();
  158. AVFrame yuv_frame;
  159. int error = AE_NO;
  160. while (_running) {
  161. std::unique_lock<std::mutex> lock(_mutex);
  162. // 改为基于条件的等待,避免固定50ms超时带来的延迟抖动
  163. _cond_var.wait(lock, [this] { return _cond_notify || !_running; });
  164. if (!_running)
  165. break;
  166. while (_ring_buffer->get(_buff, _buff_size, yuv_frame)) {
  167. // Normalize incoming frame timestamps to encoder time_base (1/fps)
  168. // Source pts is in AV_TIME_BASE (microseconds) from capturer
  169. if (yuv_frame.pts != AV_NOPTS_VALUE) {
  170. _frame->pts = av_rescale_q(yuv_frame.pts, AVRational{1, AV_TIME_BASE}, _encoder_ctx->time_base);
  171. } else {
  172. _frame->pts = AV_NOPTS_VALUE;
  173. }
  174. // Keep pkt_dts consistent with pts for no-B-frames encoders
  175. _frame->pkt_dts = _frame->pts;
  176. if ((error = encode(_frame, packet)) != AE_NO) {
  177. if (_on_error)
  178. _on_error(error);
  179. al_fatal("encode 264 packet failed:%d", error);
  180. break;
  181. }
  182. }
  183. _cond_notify = false;
  184. }
  185. av_packet_free(&packet);
  186. }
  187. } // namespace am