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.

91 lines
2.7KB

  1. /*
  2. * ADS/SS2 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 "libavutil/channel_layout.h"
  22. #include "avformat.h"
  23. #include "internal.h"
  24. static int ads_probe(const AVProbeData *p)
  25. {
  26. if (memcmp(p->buf, "SShd", 4) ||
  27. memcmp(p->buf+32, "SSbd", 4))
  28. return 0;
  29. return AVPROBE_SCORE_MAX / 3 * 2;
  30. }
  31. static int ads_read_header(AVFormatContext *s)
  32. {
  33. int align, codec;
  34. AVStream *st;
  35. int64_t size;
  36. st = avformat_new_stream(s, NULL);
  37. if (!st)
  38. return AVERROR(ENOMEM);
  39. avio_skip(s->pb, 8);
  40. st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
  41. codec = avio_rl32(s->pb);
  42. st->codecpar->sample_rate = avio_rl32(s->pb);
  43. if (st->codecpar->sample_rate <= 0)
  44. return AVERROR_INVALIDDATA;
  45. st->codecpar->channels = avio_rl32(s->pb);
  46. if (st->codecpar->channels <= 0)
  47. return AVERROR_INVALIDDATA;
  48. align = avio_rl32(s->pb);
  49. if (align <= 0 || align > INT_MAX / st->codecpar->channels)
  50. return AVERROR_INVALIDDATA;
  51. if (codec == 1)
  52. st->codecpar->codec_id = AV_CODEC_ID_PCM_S16LE_PLANAR;
  53. else
  54. st->codecpar->codec_id = AV_CODEC_ID_ADPCM_PSX;
  55. st->codecpar->block_align = st->codecpar->channels * align;
  56. avio_skip(s->pb, 12);
  57. size = avio_rl32(s->pb);
  58. if (st->codecpar->codec_id == AV_CODEC_ID_ADPCM_PSX && size >= 0x40)
  59. st->duration = (size - 0x40) / 16 / st->codecpar->channels * 28;
  60. avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate);
  61. return 0;
  62. }
  63. static int ads_read_packet(AVFormatContext *s, AVPacket *pkt)
  64. {
  65. AVCodecParameters *par = s->streams[0]->codecpar;
  66. int ret;
  67. ret = av_get_packet(s->pb, pkt, par->block_align);
  68. pkt->stream_index = 0;
  69. return ret;
  70. }
  71. AVInputFormat ff_ads_demuxer = {
  72. .name = "ads",
  73. .long_name = NULL_IF_CONFIG_SMALL("Sony PS2 ADS"),
  74. .read_probe = ads_probe,
  75. .read_header = ads_read_header,
  76. .read_packet = ads_read_packet,
  77. .extensions = "ads,ss2",
  78. };