encoder_aac.cpp 6.4 KB

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