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.

181 lines
7.0KB

  1. /*
  2. * This file is part of FFmpeg.
  3. *
  4. * FFmpeg is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU Lesser General Public
  6. * License as published by the Free Software Foundation; either
  7. * version 2.1 of the License, or (at your option) any later version.
  8. *
  9. * FFmpeg is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. * Lesser General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU Lesser General Public
  15. * License along with FFmpeg; if not, write to the Free Software
  16. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  17. */
  18. /**
  19. * @file
  20. * ModPlug demuxer
  21. * @todo metadata
  22. */
  23. #include <libmodplug/modplug.h>
  24. #include "libavutil/opt.h"
  25. #include "avformat.h"
  26. typedef struct ModPlugContext {
  27. const AVClass *class;
  28. ModPlugFile *f;
  29. uint8_t *buf; ///< input file content
  30. /* options */
  31. int noise_reduction;
  32. int reverb_depth;
  33. int reverb_delay;
  34. int bass_amount;
  35. int bass_range;
  36. int surround_depth;
  37. int surround_delay;
  38. int max_size; ///< max file size to allocate
  39. } ModPlugContext;
  40. #define FF_MODPLUG_MAX_FILE_SIZE (100 * 1<<20) // 100M
  41. #define FF_MODPLUG_DEF_FILE_SIZE ( 5 * 1<<20) // 5M
  42. #define OFFSET(x) offsetof(ModPlugContext, x)
  43. #define D AV_OPT_FLAG_DECODING_PARAM
  44. static const AVOption options[] = {
  45. {"noise_reduction", "Enable noise reduction 0(off)-1(on)", OFFSET(noise_reduction), FF_OPT_TYPE_INT, {.dbl = 0}, 0, 1, D},
  46. {"reverb_depth", "Reverb level 0(quiet)-100(loud)", OFFSET(reverb_depth), FF_OPT_TYPE_INT, {.dbl = 0}, 0, 100, D},
  47. {"reverb_delay", "Reverb delay in ms, usually 40-200ms", OFFSET(reverb_delay), FF_OPT_TYPE_INT, {.dbl = 0}, 0, INT_MAX, D},
  48. {"bass_amount", "XBass level 0(quiet)-100(loud)", OFFSET(bass_amount), FF_OPT_TYPE_INT, {.dbl = 0}, 0, 100, D},
  49. {"bass_range", "XBass cutoff in Hz 10-100", OFFSET(bass_range), FF_OPT_TYPE_INT, {.dbl = 0}, 0, 100, D},
  50. {"surround_depth", "Surround level 0(quiet)-100(heavy)", OFFSET(surround_depth), FF_OPT_TYPE_INT, {.dbl = 0}, 0, 100, D},
  51. {"surround_delay", "Surround delay in ms, usually 5-40ms", OFFSET(surround_delay), FF_OPT_TYPE_INT, {.dbl = 0}, 0, INT_MAX, D},
  52. {"max_size", "Max file size supported (in bytes). Default is 5MB. Set to 0 for no limit (not recommended)",
  53. OFFSET(max_size), FF_OPT_TYPE_INT, {.dbl = FF_MODPLUG_DEF_FILE_SIZE}, 0, FF_MODPLUG_MAX_FILE_SIZE, D},
  54. {NULL},
  55. };
  56. #define SET_OPT_IF_REQUESTED(libopt, opt, flag) do { \
  57. if (modplug->opt) { \
  58. settings.libopt = modplug->opt; \
  59. settings.mFlags |= flag; \
  60. } \
  61. } while (0)
  62. static int modplug_read_header(AVFormatContext *s, AVFormatParameters *ap)
  63. {
  64. AVStream *st;
  65. AVIOContext *pb = s->pb;
  66. ModPlug_Settings settings;
  67. ModPlugContext *modplug = s->priv_data;
  68. int sz = avio_size(pb);
  69. if (sz < 0) {
  70. av_log(s, AV_LOG_WARNING, "Could not determine file size\n");
  71. sz = modplug->max_size;
  72. } else if (modplug->max_size && sz > modplug->max_size) {
  73. sz = modplug->max_size;
  74. av_log(s, AV_LOG_WARNING, "Max file size reach%s, allocating %dB "
  75. "but demuxing is likely to fail due to incomplete buffer\n",
  76. sz == FF_MODPLUG_DEF_FILE_SIZE ? " (see -max_size)" : "", sz);
  77. }
  78. modplug->buf = av_malloc(modplug->max_size);
  79. if (!modplug->buf)
  80. return AVERROR(ENOMEM);
  81. sz = avio_read(pb, modplug->buf, sz);
  82. ModPlug_GetSettings(&settings);
  83. settings.mChannels = 2;
  84. settings.mBits = 16;
  85. settings.mFrequency = 44100;
  86. settings.mResamplingMode = MODPLUG_RESAMPLE_FIR; // best quality
  87. settings.mLoopCount = 0; // prevents looping forever
  88. if (modplug->noise_reduction) settings.mFlags |= MODPLUG_ENABLE_NOISE_REDUCTION;
  89. SET_OPT_IF_REQUESTED(mReverbDepth, reverb_depth, MODPLUG_ENABLE_REVERB);
  90. SET_OPT_IF_REQUESTED(mReverbDelay, reverb_delay, MODPLUG_ENABLE_REVERB);
  91. SET_OPT_IF_REQUESTED(mBassAmount, bass_amount, MODPLUG_ENABLE_MEGABASS);
  92. SET_OPT_IF_REQUESTED(mBassRange, bass_range, MODPLUG_ENABLE_MEGABASS);
  93. SET_OPT_IF_REQUESTED(mSurroundDepth, surround_depth, MODPLUG_ENABLE_SURROUND);
  94. SET_OPT_IF_REQUESTED(mSurroundDelay, surround_delay, MODPLUG_ENABLE_SURROUND);
  95. if (modplug->reverb_depth) settings.mReverbDepth = modplug->reverb_depth;
  96. if (modplug->reverb_delay) settings.mReverbDelay = modplug->reverb_delay;
  97. if (modplug->bass_amount) settings.mBassAmount = modplug->bass_amount;
  98. if (modplug->bass_range) settings.mBassRange = modplug->bass_range;
  99. if (modplug->surround_depth) settings.mSurroundDepth = modplug->surround_depth;
  100. if (modplug->surround_delay) settings.mSurroundDelay = modplug->surround_delay;
  101. ModPlug_SetSettings(&settings);
  102. modplug->f = ModPlug_Load(modplug->buf, sz);
  103. if (!modplug->f)
  104. return AVERROR_INVALIDDATA;
  105. st = av_new_stream(s, 0);
  106. if (!st)
  107. return AVERROR(ENOMEM);
  108. av_set_pts_info(st, 64, 1, 1000);
  109. st->duration = ModPlug_GetLength(modplug->f);
  110. st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
  111. st->codec->codec_id = CODEC_ID_PCM_S16LE;
  112. st->codec->channels = settings.mChannels;
  113. st->codec->sample_rate = settings.mFrequency;
  114. return 0;
  115. }
  116. static int modplug_read_packet(AVFormatContext *s, AVPacket *pkt)
  117. {
  118. ModPlugContext *modplug = s->priv_data;
  119. if (av_new_packet(pkt, 512) < 0)
  120. return AVERROR(ENOMEM);
  121. pkt->size = ModPlug_Read(modplug->f, pkt->data, 512);
  122. if (pkt->size <= 0) {
  123. av_free_packet(pkt);
  124. return pkt->size == 0 ? AVERROR_EOF : AVERROR(EIO);
  125. }
  126. return 0;
  127. }
  128. static int modplug_read_close(AVFormatContext *s)
  129. {
  130. ModPlugContext *modplug = s->priv_data;
  131. ModPlug_Unload(modplug->f);
  132. return 0;
  133. }
  134. static int modplug_read_seek(AVFormatContext *s, int stream_idx, int64_t ts, int flags)
  135. {
  136. const ModPlugContext *modplug = s->priv_data;
  137. ModPlug_Seek(modplug->f, (int)ts);
  138. return 0;
  139. }
  140. static const AVClass modplug_class = {
  141. .class_name = "ModPlug demuxer",
  142. .item_name = av_default_item_name,
  143. .option = options,
  144. .version = LIBAVUTIL_VERSION_INT,
  145. };
  146. AVInputFormat ff_libmodplug_demuxer = {
  147. .name = "libmodplug",
  148. .long_name = NULL_IF_CONFIG_SMALL("ModPlug demuxer"),
  149. .priv_data_size = sizeof(ModPlugContext),
  150. .read_header = modplug_read_header,
  151. .read_packet = modplug_read_packet,
  152. .read_close = modplug_read_close,
  153. .read_seek = modplug_read_seek,
  154. .extensions = "669,abc,amf,ams,dbm,dmf,dsm,far,it,mdl,med,mid,mod,mt2,mtm,okt,psm,ptm,s3m,stm,ult,umx,xm"
  155. ",itgz,itr,itz,mdgz,mdr,mdz,s3gz,s3r,s3z,xmgz,xmr,xmz", // compressed mods
  156. .priv_class = &modplug_class,
  157. };