You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

197 lines
7.5KB

  1. /*
  2. * Interface to libtwolame for mp2 encoding
  3. * Copyright (c) 2012 Paul B Mahol
  4. *
  5. * This file is part of FFmpeg.
  6. *
  7. * FFmpeg is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * FFmpeg is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with FFmpeg; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. */
  21. /**
  22. * @file
  23. * Interface to libtwolame for mp2 encoding.
  24. */
  25. #include <twolame.h>
  26. #include "libavutil/opt.h"
  27. #include "avcodec.h"
  28. #include "internal.h"
  29. #include "mpegaudio.h"
  30. typedef struct TWOLAMEContext {
  31. AVClass *class;
  32. int mode;
  33. int psymodel;
  34. int energy;
  35. int error_protection;
  36. int copyright;
  37. int original;
  38. twolame_options *glopts;
  39. int64_t next_pts;
  40. } TWOLAMEContext;
  41. static av_cold int twolame_encode_close(AVCodecContext *avctx)
  42. {
  43. TWOLAMEContext *s = avctx->priv_data;
  44. twolame_close(&s->glopts);
  45. return 0;
  46. }
  47. static av_cold int twolame_encode_init(AVCodecContext *avctx)
  48. {
  49. TWOLAMEContext *s = avctx->priv_data;
  50. int ret;
  51. avctx->frame_size = TWOLAME_SAMPLES_PER_FRAME;
  52. s->glopts = twolame_init();
  53. if (!s->glopts)
  54. return AVERROR(ENOMEM);
  55. twolame_set_verbosity(s->glopts, 0);
  56. twolame_set_mode(s->glopts, s->mode);
  57. twolame_set_psymodel(s->glopts, s->psymodel);
  58. twolame_set_energy_levels(s->glopts, s->energy);
  59. twolame_set_error_protection(s->glopts, s->error_protection);
  60. twolame_set_num_channels(s->glopts, avctx->channels);
  61. twolame_set_in_samplerate(s->glopts, avctx->sample_rate);
  62. twolame_set_out_samplerate(s->glopts, avctx->sample_rate);
  63. if (avctx->flags & CODEC_FLAG_QSCALE || !avctx->bit_rate) {
  64. twolame_set_VBR(s->glopts, TRUE);
  65. twolame_set_VBR_level(s->glopts, avctx->global_quality);
  66. av_log(avctx, AV_LOG_WARNING, "VBR mode is experimental!\n");
  67. } else {
  68. twolame_set_bitrate(s->glopts, avctx->bit_rate / 1000);
  69. }
  70. if ((ret = twolame_init_params(s->glopts)))
  71. goto error;
  72. return 0;
  73. error:
  74. twolame_encode_close(avctx);
  75. return ret;
  76. }
  77. static int twolame_encode_frame(AVCodecContext *avctx, AVPacket *avpkt,
  78. const AVFrame *frame, int *got_packet_ptr)
  79. {
  80. TWOLAMEContext *s = avctx->priv_data;
  81. int ret;
  82. if ((ret = ff_alloc_packet2(avctx, avpkt, MPA_MAX_CODED_FRAME_SIZE)))
  83. return ret;
  84. if (frame) {
  85. switch (avctx->sample_fmt) {
  86. case AV_SAMPLE_FMT_FLT:
  87. ret = twolame_encode_buffer_float32_interleaved(s->glopts,
  88. frame->data[0],
  89. frame->nb_samples,
  90. avpkt->data, avpkt->size);
  91. break;
  92. case AV_SAMPLE_FMT_FLTP:
  93. ret = twolame_encode_buffer_float32(s->glopts,
  94. frame->data[0], frame->data[1],
  95. frame->nb_samples,
  96. avpkt->data, avpkt->size);
  97. break;
  98. case AV_SAMPLE_FMT_S16:
  99. ret = twolame_encode_buffer_interleaved(s->glopts,
  100. frame->data[0],
  101. frame->nb_samples,
  102. avpkt->data, avpkt->size);
  103. break;
  104. case AV_SAMPLE_FMT_S16P:
  105. ret = twolame_encode_buffer(s->glopts,
  106. frame->data[0], frame->data[1],
  107. frame->nb_samples,
  108. avpkt->data, avpkt->size);
  109. break;
  110. default:
  111. return AVERROR_BUG;
  112. }
  113. } else {
  114. ret = twolame_encode_flush(s->glopts, avpkt->data, avpkt->size);
  115. }
  116. if (ret > 0) {
  117. avpkt->duration = ff_samples_to_time_base(avctx, avctx->frame_size);
  118. if (frame) {
  119. if (frame->pts != AV_NOPTS_VALUE)
  120. avpkt->pts = frame->pts;
  121. } else {
  122. avpkt->pts = s->next_pts;
  123. }
  124. if (avpkt->pts != AV_NOPTS_VALUE)
  125. s->next_pts = avpkt->pts + avpkt->duration;
  126. avpkt->size = ret;
  127. *got_packet_ptr = 1;
  128. return 0;
  129. }
  130. return ret;
  131. }
  132. #define OFFSET(x) offsetof(TWOLAMEContext, x)
  133. #define AE AV_OPT_FLAG_AUDIO_PARAM | AV_OPT_FLAG_ENCODING_PARAM
  134. static const AVOption options[] = {
  135. { "mode", "Mpeg Mode", OFFSET(mode), AV_OPT_TYPE_INT, { TWOLAME_AUTO_MODE }, TWOLAME_AUTO_MODE, TWOLAME_MONO, AE, "mode"},
  136. { "auto", NULL, 0, AV_OPT_TYPE_CONST, { TWOLAME_AUTO_MODE }, 0, 0, AE, "mode" },
  137. { "stereo", NULL, 0, AV_OPT_TYPE_CONST, { TWOLAME_STEREO }, 0, 0, AE, "mode" },
  138. { "joint_stereo", NULL, 0, AV_OPT_TYPE_CONST, { TWOLAME_JOINT_STEREO }, 0, 0, AE, "mode" },
  139. { "dual_channel", NULL, 0, AV_OPT_TYPE_CONST, { TWOLAME_DUAL_CHANNEL }, 0, 0, AE, "mode" },
  140. { "mono", NULL, 0, AV_OPT_TYPE_CONST, { TWOLAME_MONO }, 0, 0, AE, "mode" },
  141. { "psymodel", "Psychoacoustic Model", OFFSET(psymodel), AV_OPT_TYPE_INT, { 3 }, -1, 4, AE},
  142. { "energy_levels","enable energy levels", OFFSET(energy), AV_OPT_TYPE_INT, { 0 }, 0, 1, AE},
  143. { "error_protection","enable CRC error protection", OFFSET(error_protection), AV_OPT_TYPE_INT, { 0 }, 0, 1, AE},
  144. { "copyright", "set MPEG Audio Copyright flag", OFFSET(copyright), AV_OPT_TYPE_INT, { 0 }, 0, 1, AE},
  145. { "original", "set MPEG Audio Original flag", OFFSET(original), AV_OPT_TYPE_INT, { 0 }, 0, 1, AE},
  146. { NULL },
  147. };
  148. static const AVClass libtwolame_class = {
  149. .class_name = "libtwolame encoder",
  150. .item_name = av_default_item_name,
  151. .option = options,
  152. .version = LIBAVUTIL_VERSION_INT,
  153. };
  154. AVCodec ff_libtwolame_encoder = {
  155. .name = "libtwolame",
  156. .type = AVMEDIA_TYPE_AUDIO,
  157. .id = AV_CODEC_ID_MP2,
  158. .priv_data_size = sizeof(TWOLAMEContext),
  159. .init = twolame_encode_init,
  160. .encode2 = twolame_encode_frame,
  161. .close = twolame_encode_close,
  162. .capabilities = CODEC_CAP_DELAY,
  163. .sample_fmts = (const enum AVSampleFormat[]) { AV_SAMPLE_FMT_FLT,
  164. AV_SAMPLE_FMT_FLTP,
  165. AV_SAMPLE_FMT_S16,
  166. AV_SAMPLE_FMT_S16P,
  167. AV_SAMPLE_FMT_NONE },
  168. .channel_layouts = (const uint64_t[]) { AV_CH_LAYOUT_MONO,
  169. AV_CH_LAYOUT_STEREO,
  170. 0 },
  171. .supported_samplerates = (const int[]){ 16000, 22050, 24000, 32000, 44100, 48000, 0 },
  172. .long_name = NULL_IF_CONFIG_SMALL("libtwolame MP2 (MPEG audio layer 2)"),
  173. .priv_class = &libtwolame_class,
  174. };