libavformat/thp.c
efb0c399
 /*
  * THP Demuxer
406792e7
  * Copyright (c) 2007 Marco Gerards
efb0c399
  *
  * This file is part of FFmpeg.
  *
  * FFmpeg is free software; you can redistribute it and/or
  * modify it under the terms of the GNU Lesser General Public
  * License as published by the Free Software Foundation; either
  * version 2.1 of the License, or (at your option) any later version.
  *
  * FFmpeg is distributed in the hope that it will be useful,
  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  * Lesser General Public License for more details.
  *
  * You should have received a copy of the GNU Lesser General Public
  * License along with FFmpeg; if not, write to the Free Software
  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  */
 
6a5d31ac
 #include "libavutil/intreadwrite.h"
3383a53e
 #include "libavutil/intfloat.h"
efb0c399
 #include "avformat.h"
c3f9ebf7
 #include "internal.h"
efb0c399
 
 typedef struct ThpDemuxContext {
     int              version;
     int              first_frame;
     int              first_framesz;
     int              last_frame;
     int              compoff;
     int              framecnt;
     AVRational       fps;
     int              frame;
     int              next_frame;
     int              next_framesz;
     int              video_stream_index;
d1e0d21f
     int              audio_stream_index;
efb0c399
     int              compcount;
     unsigned char    components[16];
     AVStream*        vst;
     int              has_audio;
482ee636
     unsigned         audiosize;
efb0c399
 } ThpDemuxContext;
 
 
 static int thp_probe(AVProbeData *p)
 {
     /* check file header */
     if (AV_RL32(p->buf) == MKTAG('T', 'H', 'P', '\0'))
         return AVPROBE_SCORE_MAX;
     else
         return 0;
 }
 
6e9651d1
 static int thp_read_header(AVFormatContext *s)
efb0c399
 {
90f2a1a0
     ThpDemuxContext *thp = s->priv_data;
     AVStream *st;
471fe57e
     AVIOContext *pb = s->pb;
482ee636
     int64_t fsize= avio_size(pb);
90f2a1a0
     int i;
 
     /* Read the file header.  */
e63a3628
                            avio_rb32(pb); /* Skip Magic.  */
     thp->version         = avio_rb32(pb);
90f2a1a0
 
e63a3628
                            avio_rb32(pb); /* Max buf size.  */
                            avio_rb32(pb); /* Max samples.  */
90f2a1a0
 
3383a53e
     thp->fps             = av_d2q(av_int2float(avio_rb32(pb)), INT_MAX);
e63a3628
     thp->framecnt        = avio_rb32(pb);
     thp->first_framesz   = avio_rb32(pb);
a0009754
     pb->maxsize          = avio_rb32(pb);
     if(fsize>0 && (!pb->maxsize || fsize < pb->maxsize))
         pb->maxsize= fsize;
90f2a1a0
 
e63a3628
     thp->compoff         = avio_rb32(pb);
                            avio_rb32(pb); /* offsetDataOffset.  */
     thp->first_frame     = avio_rb32(pb);
     thp->last_frame      = avio_rb32(pb);
90f2a1a0
 
     thp->next_framesz    = thp->first_framesz;
     thp->next_frame      = thp->first_frame;
 
     /* Read the component structure.  */
f59d8ff8
     avio_seek (pb, thp->compoff, SEEK_SET);
e63a3628
     thp->compcount       = avio_rb32(pb);
90f2a1a0
 
     /* Read the list of component types.  */
e63a3628
     avio_read(pb, thp->components, 16);
90f2a1a0
 
     for (i = 0; i < thp->compcount; i++) {
         if (thp->components[i] == 0) {
             if (thp->vst != 0)
                 break;
 
             /* Video component.  */
3b3bbdd3
             st = avformat_new_stream(s, NULL);
90f2a1a0
             if (!st)
769e10f0
                 return AVERROR(ENOMEM);
90f2a1a0
 
             /* The denominator and numerator are switched because 1/fps
                is required.  */
c3f9ebf7
             avpriv_set_pts_info(st, 64, thp->fps.den, thp->fps.num);
72415b2a
             st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
36ef5369
             st->codec->codec_id = AV_CODEC_ID_THP;
90f2a1a0
             st->codec->codec_tag = 0;  /* no fourcc */
e63a3628
             st->codec->width = avio_rb32(pb);
             st->codec->height = avio_rb32(pb);
90f2a1a0
             st->codec->sample_rate = av_q2d(thp->fps);
d0bdcbcb
             st->nb_frames =
             st->duration = thp->framecnt;
90f2a1a0
             thp->vst = st;
             thp->video_stream_index = st->index;
 
             if (thp->version == 0x11000)
e63a3628
                 avio_rb32(pb); /* Unknown.  */
90f2a1a0
         } else if (thp->components[i] == 1) {
             if (thp->has_audio != 0)
                 break;
 
             /* Audio component.  */
3b3bbdd3
             st = avformat_new_stream(s, NULL);
90f2a1a0
             if (!st)
769e10f0
                 return AVERROR(ENOMEM);
90f2a1a0
 
72415b2a
             st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
36ef5369
             st->codec->codec_id = AV_CODEC_ID_ADPCM_THP;
90f2a1a0
             st->codec->codec_tag = 0;  /* no fourcc */
e63a3628
             st->codec->channels    = avio_rb32(pb); /* numChannels.  */
             st->codec->sample_rate = avio_rb32(pb); /* Frequency.  */
90f2a1a0
 
c3f9ebf7
             avpriv_set_pts_info(st, 64, 1, st->codec->sample_rate);
90f2a1a0
 
             thp->audio_stream_index = st->index;
             thp->has_audio = 1;
efb0c399
         }
     }
 
90f2a1a0
     return 0;
efb0c399
 }
 
 static int thp_read_packet(AVFormatContext *s,
                             AVPacket *pkt)
 {
     ThpDemuxContext *thp = s->priv_data;
471fe57e
     AVIOContext *pb = s->pb;
482ee636
     unsigned int size;
efb0c399
     int ret;
 
d1e0d21f
     if (thp->audiosize == 0) {
90f2a1a0
         /* Terminate when last frame is reached.  */
         if (thp->frame >= thp->framecnt)
25d8ebd4
             return AVERROR_EOF;
d1e0d21f
 
f59d8ff8
         avio_seek(pb, thp->next_frame, SEEK_SET);
efb0c399
 
90f2a1a0
         /* Locate the next frame and read out its size.  */
         thp->next_frame += thp->next_framesz;
e63a3628
         thp->next_framesz = avio_rb32(pb);
efb0c399
 
e63a3628
                         avio_rb32(pb); /* Previous total size.  */
         size          = avio_rb32(pb); /* Total size of this frame.  */
efb0c399
 
90f2a1a0
         /* Store the audiosize so the next time this function is called,
            the audio can be read.  */
         if (thp->has_audio)
e63a3628
             thp->audiosize = avio_rb32(pb); /* Audio size.  */
90f2a1a0
         else
             thp->frame++;
efb0c399
 
90f2a1a0
         ret = av_get_packet(pb, pkt, size);
         if (ret != size) {
             av_free_packet(pkt);
6f3e0b21
             return AVERROR(EIO);
90f2a1a0
         }
efb0c399
 
90f2a1a0
         pkt->stream_index = thp->video_stream_index;
     } else {
d1e0d21f
         ret = av_get_packet(pb, pkt, thp->audiosize);
         if (ret != thp->audiosize) {
             av_free_packet(pkt);
6f3e0b21
             return AVERROR(EIO);
d1e0d21f
         }
 
         pkt->stream_index = thp->audio_stream_index;
5a9b9522
         if (thp->audiosize >= 8)
             pkt->duration = AV_RB32(&pkt->data[4]);
 
d1e0d21f
         thp->audiosize = 0;
         thp->frame++;
     }
efb0c399
 
     return 0;
 }
 
66355be3
 AVInputFormat ff_thp_demuxer = {
dfc2c4d9
     .name           = "thp",
     .long_name      = NULL_IF_CONFIG_SMALL("THP"),
     .priv_data_size = sizeof(ThpDemuxContext),
     .read_probe     = thp_probe,
     .read_header    = thp_read_header,
     .read_packet    = thp_read_packet
efb0c399
 };