oggparseskeleton.c 3.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. /*
  2. * Copyright (C) 2010 David Conrad
  3. *
  4. * This file is part of FFmpeg.
  5. *
  6. * FFmpeg is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 2.1 of the License, or (at your option) any later version.
  10. *
  11. * FFmpeg is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public
  17. * License along with FFmpeg; if not, write to the Free Software
  18. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  19. */
  20. #include "libavcodec/bytestream.h"
  21. #include "avformat.h"
  22. #include "internal.h"
  23. #include "oggdec.h"
  24. static int skeleton_header(AVFormatContext *s, int idx)
  25. {
  26. struct ogg *ogg = s->priv_data;
  27. struct ogg_stream *os = ogg->streams + idx;
  28. AVStream *st = s->streams[idx];
  29. uint8_t *buf = os->buf + os->pstart;
  30. int version_major, version_minor;
  31. int64_t start_num, start_den;
  32. uint64_t start_granule;
  33. int target_idx, start_time;
  34. strcpy(st->codec->codec_name, "skeleton");
  35. st->codec->codec_type = AVMEDIA_TYPE_DATA;
  36. if (os->psize < 8)
  37. return -1;
  38. if (!strncmp(buf, "fishead", 8)) {
  39. if (os->psize < 64)
  40. return -1;
  41. version_major = AV_RL16(buf+8);
  42. version_minor = AV_RL16(buf+10);
  43. if (version_major != 3 && version_major != 4) {
  44. av_log(s, AV_LOG_WARNING, "Unknown skeleton version %d.%d\n",
  45. version_major, version_minor);
  46. return -1;
  47. }
  48. // This is the overall start time. We use it for the start time of
  49. // of the skeleton stream since if left unset lavf assumes 0,
  50. // which we don't want since skeleton is timeless
  51. // FIXME: the real meaning of this field is "start playback at
  52. // this time which can be in the middle of a packet
  53. start_num = AV_RL64(buf+12);
  54. start_den = AV_RL64(buf+20);
  55. if (start_den) {
  56. int base_den;
  57. av_reduce(&start_time, &base_den, start_num, start_den, INT_MAX);
  58. avpriv_set_pts_info(st, 64, 1, base_den);
  59. os->lastpts =
  60. st->start_time = start_time;
  61. }
  62. } else if (!strncmp(buf, "fisbone", 8)) {
  63. if (os->psize < 52)
  64. return -1;
  65. target_idx = ogg_find_stream(ogg, AV_RL32(buf+12));
  66. start_granule = AV_RL64(buf+36);
  67. if (os->start_granule != OGG_NOGRANULE_VALUE) {
  68. av_log_missing_feature(s,
  69. "Multiple fisbone for the same stream", 0);
  70. return 1;
  71. }
  72. if (target_idx >= 0 && start_granule != OGG_NOGRANULE_VALUE) {
  73. os->start_granule = start_granule;
  74. }
  75. }
  76. return 1;
  77. }
  78. const struct ogg_codec ff_skeleton_codec = {
  79. .magic = "fishead",
  80. .magicsize = 8,
  81. .header = skeleton_header,
  82. .nb_header = 0,
  83. };