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.

88 lines
2.9KB

  1. /*
  2. * DC STR demuxer
  3. * Copyright (c) 2015 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. #include "avformat.h"
  22. #include "internal.h"
  23. static int dcstr_probe(AVProbeData *p)
  24. {
  25. if (p->buf_size < 224 || memcmp(p->buf + 213, "Sega Stream", 11))
  26. return 0;
  27. return AVPROBE_SCORE_MAX;
  28. }
  29. static int dcstr_read_header(AVFormatContext *s)
  30. {
  31. unsigned codec, align;
  32. int mult;
  33. AVStream *st;
  34. st = avformat_new_stream(s, NULL);
  35. if (!st)
  36. return AVERROR(ENOMEM);
  37. st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
  38. st->codec->channels = avio_rl32(s->pb);
  39. st->codec->sample_rate = avio_rl32(s->pb);
  40. codec = avio_rl32(s->pb);
  41. align = avio_rl32(s->pb);
  42. avio_skip(s->pb, 4);
  43. st->duration = avio_rl32(s->pb);
  44. mult = avio_rl32(s->pb);
  45. if (st->codec->channels <= 0 || mult <= 0 || mult > INT_MAX / st->codec->channels) {
  46. av_log(s, AV_LOG_ERROR, "invalid number of channels %d x %d\n", st->codec->channels, mult);
  47. return AVERROR_INVALIDDATA;
  48. }
  49. st->codec->channels *= mult;
  50. if (!align || align > INT_MAX / st->codec->channels)
  51. return AVERROR_INVALIDDATA;
  52. st->codec->block_align = align * st->codec->channels;
  53. switch (codec) {
  54. case 4: st->codec->codec_id = AV_CODEC_ID_ADPCM_AICA; break;
  55. case 16: st->codec->codec_id = AV_CODEC_ID_PCM_S16LE_PLANAR; break;
  56. default: avpriv_request_sample(s, "codec %X", codec);
  57. return AVERROR_PATCHWELCOME;
  58. }
  59. avio_skip(s->pb, 0x800 - avio_tell(s->pb));
  60. avpriv_set_pts_info(st, 64, 1, st->codec->sample_rate);
  61. return 0;
  62. }
  63. static int dcstr_read_packet(AVFormatContext *s, AVPacket *pkt)
  64. {
  65. AVCodecContext *codec = s->streams[0]->codec;
  66. return av_get_packet(s->pb, pkt, codec->block_align);
  67. }
  68. AVInputFormat ff_dcstr_demuxer = {
  69. .name = "dcstr",
  70. .long_name = NULL_IF_CONFIG_SMALL("Sega DC STR"),
  71. .read_probe = dcstr_probe,
  72. .read_header = dcstr_read_header,
  73. .read_packet = dcstr_read_packet,
  74. .extensions = "str",
  75. .flags = AVFMT_GENERIC_INDEX | AVFMT_NO_BYTE_SEEK | AVFMT_NOBINSEARCH,
  76. };