encoder_aac.cpp 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. #include "encoder_aac.h"
  2. #include "headers_ffmpeg.h"
  3. #include "ring_buffer.h"
  4. #include "error_define.h"
  5. #include "log_helper.h"
  6. namespace am {
  7. encoder_aac::encoder_aac()
  8. {
  9. ffmpeg_register_all();
  10. _ring_buffer = new ring_buffer<AVFrame>(1024 * 1024 * 10);
  11. _inited = false;
  12. _running = false;
  13. _encoder = NULL;
  14. _encoder_ctx = NULL;
  15. _frame = NULL;
  16. _buff = NULL;
  17. _buff_size = 0;
  18. _cond_notify = false;
  19. #ifdef SAVE_AAC
  20. _aac_io_ctx = NULL;
  21. _aac_stream = NULL;
  22. _aac_fmt_ctx = NULL;
  23. #endif
  24. }
  25. encoder_aac::~encoder_aac()
  26. {
  27. stop();
  28. cleanup();
  29. delete _ring_buffer;
  30. }
  31. int encoder_aac::init(int nb_channels, int sample_rate, AVSampleFormat fmt, int bit_rate)
  32. {
  33. int err = AE_NO;
  34. int ret = 0;
  35. if (_inited == true)
  36. return err;
  37. do {
  38. _encoder = (AVCodec*)avcodec_find_encoder(AV_CODEC_ID_AAC);
  39. if (!_encoder) {
  40. err = AE_FFMPEG_FIND_ENCODER_FAILED;
  41. break;
  42. }
  43. _encoder_ctx = avcodec_alloc_context3(_encoder);
  44. if (!_encoder_ctx) {
  45. err = AE_FFMPEG_ALLOC_CONTEXT_FAILED;
  46. break;
  47. }
  48. ffmpeg_set_channels(_encoder_ctx, nb_channels);
  49. ffmpeg_set_channel_layout(_encoder_ctx, ffmpeg_get_default_channel_layout(nb_channels));
  50. _encoder_ctx->sample_rate = sample_rate;
  51. _encoder_ctx->sample_fmt = fmt;
  52. _encoder_ctx->bit_rate = bit_rate;
  53. _encoder_ctx->strict_std_compliance = FF_COMPLIANCE_EXPERIMENTAL;
  54. _encoder_ctx->time_base.den = sample_rate;
  55. _encoder_ctx->time_base.num = 1;
  56. _encoder_ctx->flags |= AV_CODEC_FLAG_GLOBAL_HEADER;
  57. ret = avcodec_open2(_encoder_ctx, _encoder, NULL);
  58. if (ret < 0) {
  59. err = AE_FFMPEG_OPEN_CODEC_FAILED;
  60. break;
  61. }
  62. _buff_size = av_samples_get_buffer_size(NULL, nb_channels, _encoder_ctx->frame_size, fmt, 1);
  63. _buff = (uint8_t *) av_malloc(_buff_size);
  64. _frame = av_frame_alloc();
  65. if (!_frame) {
  66. err = AE_FFMPEG_ALLOC_FRAME_FAILED;
  67. break;
  68. }
  69. ffmpeg_set_frame_channels(_frame, nb_channels);
  70. _frame->nb_samples = _encoder_ctx->frame_size;
  71. ffmpeg_set_frame_channel_layout(_frame, ffmpeg_get_default_channel_layout(nb_channels));
  72. _frame->format = fmt;
  73. _frame->sample_rate = sample_rate;
  74. ret = avcodec_fill_audio_frame(_frame, nb_channels, fmt, _buff, _buff_size, 0);
  75. #ifdef SAVE_AAC
  76. ret = avio_open(&_aac_io_ctx, "save.aac", AVIO_FLAG_READ_WRITE);
  77. if (ret < 0) {
  78. err = AE_FFMPEG_OPEN_IO_FAILED;
  79. break;
  80. }
  81. _aac_fmt_ctx = avformat_alloc_context();
  82. if (!_aac_fmt_ctx) {
  83. err = AE_FFMPEG_ALLOC_CONTEXT_FAILED;
  84. break;
  85. }
  86. _aac_fmt_ctx->pb = _aac_io_ctx;
  87. _aac_fmt_ctx->oformat = av_guess_format(NULL, "save.aac", NULL);
  88. _aac_fmt_ctx->url = av_strdup("save.aac");
  89. _aac_stream = avformat_new_stream(_aac_fmt_ctx, NULL);
  90. if (!_aac_stream) {
  91. err = AE_FFMPEG_CREATE_STREAM_FAILED;
  92. break;
  93. }
  94. ret = avcodec_parameters_from_context(_aac_stream->codecpar, _encoder_ctx);
  95. if (ret < 0) {
  96. err = AE_FFMPEG_COPY_PARAMS_FAILED;
  97. break;
  98. }
  99. if (_aac_fmt_ctx->oformat->flags | AV_CODEC_FLAG_GLOBAL_HEADER) {
  100. _aac_stream->codec->extradata_size
  101. = _encoder_ctx->extradata_size; // +AV_INPUT_BUFFER_PADDING_SIZE;
  102. _aac_stream->codec->extradata = (uint8_t *) av_memdup(_encoder_ctx->extradata,
  103. _encoder_ctx->extradata_size);
  104. }
  105. #endif
  106. _inited = true;
  107. } while (0);
  108. if (err != AE_NO) {
  109. al_debug("%s,error:%d", err2str(err), ret);
  110. cleanup();
  111. }
  112. return err;
  113. }
  114. int encoder_aac::get_extradata_size()
  115. {
  116. return _encoder_ctx->extradata_size;
  117. }
  118. const uint8_t *encoder_aac::get_extradata()
  119. {
  120. return (const uint8_t *) _encoder_ctx->extradata;
  121. }
  122. int encoder_aac::get_nb_samples()
  123. {
  124. return _encoder_ctx->frame_size;
  125. }
  126. int encoder_aac::start()
  127. {
  128. int error = AE_NO;
  129. if (_running == true) {
  130. return error;
  131. }
  132. if (_inited == false) {
  133. return AE_NEED_INIT;
  134. }
  135. _running = true;
  136. _thread = std::thread(std::bind(&encoder_aac::encode_loop, this));
  137. return error;
  138. }
  139. void encoder_aac::stop()
  140. {
  141. _running = false;
  142. _cond_notify = true;
  143. _cond_var.notify_all();
  144. if (_thread.joinable())
  145. _thread.join();
  146. }
  147. int encoder_aac::put(const uint8_t *data, int data_len, AVFrame *frame)
  148. {
  149. std::unique_lock<std::mutex> lock(_mutex);
  150. AVFrame frame_cp;
  151. memcpy(&frame_cp, frame, sizeof(AVFrame));
  152. _ring_buffer->put(data, data_len, frame_cp);
  153. _cond_notify = true;
  154. _cond_var.notify_all();
  155. return 0;
  156. }
  157. const AVRational &encoder_aac::get_time_base()
  158. {
  159. return _encoder_ctx->time_base;
  160. }
  161. AVCodecID encoder_aac::get_codec_id()
  162. {
  163. if (_inited == false)
  164. return AV_CODEC_ID_NONE;
  165. return _encoder->id;
  166. }
  167. int encoder_aac::encode(AVFrame *frame, AVPacket *packet)
  168. {
  169. int ret = avcodec_send_frame(_encoder_ctx, frame);
  170. if (ret < 0) {
  171. return AE_FFMPEG_ENCODE_FRAME_FAILED;
  172. }
  173. while (ret == 0) {
  174. av_init_packet(packet);
  175. ret = avcodec_receive_packet(_encoder_ctx, packet);
  176. if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) {
  177. break;
  178. }
  179. if (ret < 0) {
  180. return AE_FFMPEG_READ_PACKET_FAILED;
  181. }
  182. if (ret == 0 && _on_data)
  183. _on_data(packet);
  184. #ifdef SAVE_AAC
  185. av_packet_rescale_ts(packet, _encoder_ctx->time_base, _aac_stream->time_base);
  186. packet->stream_index = _aac_stream->index;
  187. av_write_frame(_aac_fmt_ctx, packet);
  188. #endif
  189. av_packet_unref(packet);
  190. }
  191. return AE_NO;
  192. }
  193. void encoder_aac::encode_loop()
  194. {
  195. int len = 0;
  196. int error = AE_NO;
  197. AVPacket *packet = av_packet_alloc();
  198. AVFrame pcm_frame;
  199. #ifdef SAVE_AAC
  200. avformat_write_header(_aac_fmt_ctx, NULL);
  201. #endif
  202. while (_running) {
  203. std::unique_lock<std::mutex> lock(_mutex);
  204. while (!_cond_notify && _running)
  205. _cond_var.wait_for(lock, std::chrono::milliseconds(300));
  206. while ((len = _ring_buffer->get(_buff, _buff_size, pcm_frame))) {
  207. _frame->pts = pcm_frame.pts;
  208. #if FFMPEG_VERSION_MAJOR >= 7
  209. // In FFmpeg 7+, pkt_pts is removed, use pts instead
  210. // _frame->pts is already set above
  211. #else
  212. _frame->pkt_pts = pcm_frame.pkt_pts;
  213. #endif
  214. _frame->pkt_dts = pcm_frame.pkt_dts;
  215. if ((error = encode(_frame, packet)) != AE_NO) {
  216. if (_on_error)
  217. _on_error(error);
  218. al_fatal("read aac packet failed:%d", error);
  219. break;
  220. }
  221. }
  222. _cond_notify = false;
  223. }
  224. //flush pcm data in encoder
  225. encode(NULL, packet);
  226. av_packet_free(&packet);
  227. #ifdef SAVE_AAC
  228. av_write_trailer(_aac_fmt_ctx);
  229. #endif
  230. }
  231. void encoder_aac::cleanup()
  232. {
  233. if (_encoder) {
  234. avcodec_close(_encoder_ctx);
  235. }
  236. _encoder = NULL;
  237. if (_encoder_ctx) {
  238. avcodec_free_context(&_encoder_ctx);
  239. }
  240. if (_frame)
  241. av_free(_frame);
  242. _frame = NULL;
  243. if (_buff)
  244. av_free(_buff);
  245. _buff = NULL;
  246. _encoder_ctx = NULL;
  247. #ifdef SAVE_AAC
  248. if (_aac_fmt_ctx) {
  249. avio_closep(&_aac_fmt_ctx->pb);
  250. avformat_free_context(_aac_fmt_ctx);
  251. }
  252. #endif
  253. }
  254. } // namespace am