| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * 4X Technologies .4xm File Demuxer (no muxer) | 
					
						
							|  |  |  |  * Copyright (c) 2003  The ffmpeg Project | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2006-10-07 15:30:46 +00:00
										 |  |  |  * This file is part of FFmpeg. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * FFmpeg is free software; you can redistribute it and/or | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							| 
									
										
										
										
											2006-10-07 15:30:46 +00:00
										 |  |  |  * version 2.1 of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2006-10-07 15:30:46 +00:00
										 |  |  |  * FFmpeg is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |  * 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 | 
					
						
							| 
									
										
										
										
											2006-10-07 15:30:46 +00:00
										 |  |  |  * License along with FFmpeg; if not, write to the Free Software | 
					
						
							| 
									
										
										
										
											2006-01-12 22:43:26 +00:00
										 |  |  |  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * @file 4xm.c | 
					
						
							|  |  |  |  * 4X Technologies file demuxer | 
					
						
							|  |  |  |  * by Mike Melanson (melanson@pcisys.net) | 
					
						
							|  |  |  |  * for more information on the .4xm file format, visit: | 
					
						
							|  |  |  |  *   http://www.pcisys.net/~melanson/codecs/
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-11 22:19:48 +00:00
										 |  |  | #include "libavutil/intreadwrite.h"
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | #include "avformat.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-11 21:08:52 +00:00
										 |  |  | #define     RIFF_TAG MKTAG('R', 'I', 'F', 'F')
 | 
					
						
							| 
									
										
										
										
											2009-01-11 21:07:12 +00:00
										 |  |  | #define  FOURXMV_TAG MKTAG('4', 'X', 'M', 'V')
 | 
					
						
							| 
									
										
										
										
											2009-01-11 21:08:52 +00:00
										 |  |  | #define     LIST_TAG MKTAG('L', 'I', 'S', 'T')
 | 
					
						
							|  |  |  | #define     HEAD_TAG MKTAG('H', 'E', 'A', 'D')
 | 
					
						
							|  |  |  | #define     TRK__TAG MKTAG('T', 'R', 'K', '_')
 | 
					
						
							|  |  |  | #define     MOVI_TAG MKTAG('M', 'O', 'V', 'I')
 | 
					
						
							|  |  |  | #define     VTRK_TAG MKTAG('V', 'T', 'R', 'K')
 | 
					
						
							|  |  |  | #define     STRK_TAG MKTAG('S', 'T', 'R', 'K')
 | 
					
						
							|  |  |  | #define     std__TAG MKTAG('s', 't', 'd', '_')
 | 
					
						
							|  |  |  | #define     name_TAG MKTAG('n', 'a', 'm', 'e')
 | 
					
						
							|  |  |  | #define     vtrk_TAG MKTAG('v', 't', 'r', 'k')
 | 
					
						
							|  |  |  | #define     strk_TAG MKTAG('s', 't', 'r', 'k')
 | 
					
						
							|  |  |  | #define     ifrm_TAG MKTAG('i', 'f', 'r', 'm')
 | 
					
						
							|  |  |  | #define     pfrm_TAG MKTAG('p', 'f', 'r', 'm')
 | 
					
						
							|  |  |  | #define     cfrm_TAG MKTAG('c', 'f', 'r', 'm')
 | 
					
						
							|  |  |  | #define     ifr2_TAG MKTAG('i', 'f', 'r', '2')
 | 
					
						
							|  |  |  | #define     pfr2_TAG MKTAG('p', 'f', 'r', '2')
 | 
					
						
							|  |  |  | #define     cfr2_TAG MKTAG('c', 'f', 'r', '2')
 | 
					
						
							|  |  |  | #define     snd__TAG MKTAG('s', 'n', 'd', '_')
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define vtrk_SIZE 0x44
 | 
					
						
							|  |  |  | #define strk_SIZE 0x28
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GET_LIST_HEADER() \
 | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |     fourcc_tag = get_le32(pb); \ | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     size = get_le32(pb); \ | 
					
						
							|  |  |  |     if (fourcc_tag != LIST_TAG) \ | 
					
						
							|  |  |  |         return AVERROR_INVALIDDATA; \ | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |     fourcc_tag = get_le32(pb); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct AudioTrack { | 
					
						
							|  |  |  |     int sample_rate; | 
					
						
							|  |  |  |     int bits; | 
					
						
							|  |  |  |     int channels; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |     int stream_index; | 
					
						
							| 
									
										
										
										
											2003-06-03 13:58:44 +00:00
										 |  |  |     int adpcm; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | } AudioTrack; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct FourxmDemuxContext { | 
					
						
							|  |  |  |     int width; | 
					
						
							|  |  |  |     int height; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |     int video_stream_index; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     int track_count; | 
					
						
							|  |  |  |     AudioTrack *tracks; | 
					
						
							|  |  |  |     int selected_track; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |     int64_t audio_pts; | 
					
						
							|  |  |  |     int64_t video_pts; | 
					
						
							| 
									
										
										
										
											2003-11-16 14:26:50 +00:00
										 |  |  |     float fps; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | } FourxmDemuxContext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fourxm_probe(AVProbeData *p) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |     if ((AV_RL32(&p->buf[0]) != RIFF_TAG) || | 
					
						
							| 
									
										
										
										
											2009-01-11 21:07:12 +00:00
										 |  |  |         (AV_RL32(&p->buf[8]) != FOURXMV_TAG)) | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return AVPROBE_SCORE_MAX; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fourxm_read_header(AVFormatContext *s, | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |                               AVFormatParameters *ap) | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-11-21 07:41:00 +00:00
										 |  |  |     ByteIOContext *pb = s->pb; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     unsigned int fourcc_tag; | 
					
						
							|  |  |  |     unsigned int size; | 
					
						
							|  |  |  |     int header_size; | 
					
						
							| 
									
										
										
										
											2007-04-08 20:24:16 +00:00
										 |  |  |     FourxmDemuxContext *fourxm = s->priv_data; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     unsigned char *header; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |     int i, ret; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     int current_track = -1; | 
					
						
							|  |  |  |     AVStream *st; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fourxm->track_count = 0; | 
					
						
							|  |  |  |     fourxm->tracks = NULL; | 
					
						
							|  |  |  |     fourxm->selected_track = 0; | 
					
						
							| 
									
										
										
										
											2003-11-16 14:26:50 +00:00
										 |  |  |     fourxm->fps = 1.0; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* skip the first 3 32-bit numbers */ | 
					
						
							|  |  |  |     url_fseek(pb, 12, SEEK_CUR); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* check for LIST-HEAD */ | 
					
						
							|  |  |  |     GET_LIST_HEADER(); | 
					
						
							|  |  |  |     header_size = size - 4; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:01:47 +00:00
										 |  |  |     if (fourcc_tag != HEAD_TAG || header_size < 0) | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         return AVERROR_INVALIDDATA; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* allocate space for the header and load the whole thing */ | 
					
						
							|  |  |  |     header = av_malloc(header_size); | 
					
						
							|  |  |  |     if (!header) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:21:30 +00:00
										 |  |  |         return AVERROR(ENOMEM); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |     if (get_buffer(pb, header, header_size) != header_size){ | 
					
						
							|  |  |  |         av_free(header); | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |         return AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* take the lazy approach and search for any and all vtrk and strk chunks */ | 
					
						
							|  |  |  |     for (i = 0; i < header_size - 8; i++) { | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |         fourcc_tag = AV_RL32(&header[i]); | 
					
						
							|  |  |  |         size = AV_RL32(&header[i + 4]); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |         if (fourcc_tag == std__TAG) { | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |             fourxm->fps = av_int2flt(AV_RL32(&header[i + 12])); | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |         } else if (fourcc_tag == vtrk_TAG) { | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             /* check that there is enough data */ | 
					
						
							|  |  |  |             if (size != vtrk_SIZE) { | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |                 ret= AVERROR_INVALIDDATA; | 
					
						
							|  |  |  |                 goto fail; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-01-27 21:09:22 +00:00
										 |  |  |             fourxm->width  = AV_RL32(&header[i + 36]); | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |             fourxm->height = AV_RL32(&header[i + 40]); | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* allocate a new AVStream */ | 
					
						
							|  |  |  |             st = av_new_stream(s, 0); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |             if (!st){ | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |                 ret= AVERROR(ENOMEM); | 
					
						
							|  |  |  |                 goto fail; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2005-04-26 10:01:17 +00:00
										 |  |  |             av_set_pts_info(st, 60, 1, fourxm->fps); | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             fourxm->video_stream_index = st->index; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-17 22:24:36 +00:00
										 |  |  |             st->codec->codec_type = CODEC_TYPE_VIDEO; | 
					
						
							|  |  |  |             st->codec->codec_id = CODEC_ID_4XM; | 
					
						
							| 
									
										
										
										
											2007-08-27 21:58:39 +00:00
										 |  |  |             st->codec->extradata_size = 4; | 
					
						
							|  |  |  |             st->codec->extradata = av_malloc(4); | 
					
						
							|  |  |  |             AV_WL32(st->codec->extradata, AV_RL32(&header[i + 16])); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:09:22 +00:00
										 |  |  |             st->codec->width  = fourxm->width; | 
					
						
							| 
									
										
										
										
											2005-07-17 22:24:36 +00:00
										 |  |  |             st->codec->height = fourxm->height; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-13 00:44:35 +00:00
										 |  |  |             i += 8 + size; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         } else if (fourcc_tag == strk_TAG) { | 
					
						
							|  |  |  |             /* check that there is enough data */ | 
					
						
							|  |  |  |             if (size != strk_SIZE) { | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |                 ret= AVERROR_INVALIDDATA; | 
					
						
							|  |  |  |                 goto fail; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |             current_track = AV_RL32(&header[i + 8]); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             if (current_track + 1 > fourxm->track_count) { | 
					
						
							| 
									
										
										
										
											2003-05-26 20:53:09 +00:00
										 |  |  |                 fourxm->track_count = current_track + 1; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |                 if((unsigned)fourxm->track_count >= UINT_MAX / sizeof(AudioTrack)){ | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |                     ret= -1; | 
					
						
							|  |  |  |                     goto fail; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2005-12-17 18:14:38 +00:00
										 |  |  |                 fourxm->tracks = av_realloc(fourxm->tracks, | 
					
						
							| 
									
										
										
										
											2003-05-26 20:53:09 +00:00
										 |  |  |                     fourxm->track_count * sizeof(AudioTrack)); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |                 if (!fourxm->tracks) { | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |                     ret=  AVERROR(ENOMEM); | 
					
						
							|  |  |  |                     goto fail; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-01-27 21:09:22 +00:00
										 |  |  |             fourxm->tracks[current_track].adpcm       = AV_RL32(&header[i + 12]); | 
					
						
							|  |  |  |             fourxm->tracks[current_track].channels    = AV_RL32(&header[i + 36]); | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |             fourxm->tracks[current_track].sample_rate = AV_RL32(&header[i + 40]); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:09:22 +00:00
										 |  |  |             fourxm->tracks[current_track].bits        = AV_RL32(&header[i + 44]); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             i += 8 + size; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* allocate a new AVStream */ | 
					
						
							|  |  |  |             st = av_new_stream(s, current_track); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |             if (!st){ | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |                 ret= AVERROR(ENOMEM); | 
					
						
							|  |  |  |                 goto fail; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:06:19 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-26 10:01:17 +00:00
										 |  |  |             av_set_pts_info(st, 60, 1, fourxm->tracks[current_track].sample_rate); | 
					
						
							| 
									
										
										
										
											2004-05-21 20:43:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |             fourxm->tracks[current_track].stream_index = st->index; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-17 22:24:36 +00:00
										 |  |  |             st->codec->codec_type = CODEC_TYPE_AUDIO; | 
					
						
							| 
									
										
										
										
											2006-11-05 16:31:52 +00:00
										 |  |  |             st->codec->codec_tag = 0; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:09:22 +00:00
										 |  |  |             st->codec->channels              = fourxm->tracks[current_track].channels; | 
					
						
							|  |  |  |             st->codec->sample_rate           = fourxm->tracks[current_track].sample_rate; | 
					
						
							| 
									
										
										
										
											2008-09-08 14:24:59 +00:00
										 |  |  |             st->codec->bits_per_coded_sample = fourxm->tracks[current_track].bits; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:09:22 +00:00
										 |  |  |             st->codec->bit_rate              = st->codec->channels * st->codec->sample_rate * | 
					
						
							| 
									
										
										
										
											2008-09-08 14:24:59 +00:00
										 |  |  |                 st->codec->bits_per_coded_sample; | 
					
						
							|  |  |  |             st->codec->block_align = st->codec->channels * st->codec->bits_per_coded_sample; | 
					
						
							| 
									
										
										
										
											2003-06-03 13:58:44 +00:00
										 |  |  |             if (fourxm->tracks[current_track].adpcm) | 
					
						
							| 
									
										
										
										
											2005-07-17 22:24:36 +00:00
										 |  |  |                 st->codec->codec_id = CODEC_ID_ADPCM_4XM; | 
					
						
							| 
									
										
										
										
											2008-09-08 14:24:59 +00:00
										 |  |  |             else if (st->codec->bits_per_coded_sample == 8) | 
					
						
							| 
									
										
										
										
											2005-07-17 22:24:36 +00:00
										 |  |  |                 st->codec->codec_id = CODEC_ID_PCM_U8; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |             else | 
					
						
							| 
									
										
										
										
											2005-07-17 22:24:36 +00:00
										 |  |  |                 st->codec->codec_id = CODEC_ID_PCM_S16LE; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* skip over the LIST-MOVI chunk (which is where the stream should be */ | 
					
						
							|  |  |  |     GET_LIST_HEADER(); | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |     if (fourcc_tag != MOVI_TAG){ | 
					
						
							|  |  |  |         ret= AVERROR_INVALIDDATA; | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  |     av_free(header); | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |     /* initialize context members */ | 
					
						
							| 
									
										
										
										
											2005-04-26 10:01:17 +00:00
										 |  |  |     fourxm->video_pts = -1;  /* first frame will push to 0 */ | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |     fourxm->audio_pts = 0; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2009-01-27 21:56:28 +00:00
										 |  |  | fail: | 
					
						
							|  |  |  |     av_freep(&fourxm->tracks); | 
					
						
							|  |  |  |     av_free(header); | 
					
						
							|  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fourxm_read_packet(AVFormatContext *s, | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |                               AVPacket *pkt) | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     FourxmDemuxContext *fourxm = s->priv_data; | 
					
						
							| 
									
										
										
										
											2007-11-21 07:41:00 +00:00
										 |  |  |     ByteIOContext *pb = s->pb; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     unsigned int fourcc_tag; | 
					
						
							| 
									
										
										
										
											2003-06-03 13:58:44 +00:00
										 |  |  |     unsigned int size, out_size; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     int ret = 0; | 
					
						
							|  |  |  |     int track_number; | 
					
						
							|  |  |  |     int packet_read = 0; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |     unsigned char header[8]; | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |     int audio_frame_count; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     while (!packet_read) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-21 07:41:00 +00:00
										 |  |  |         if ((ret = get_buffer(s->pb, header, 8)) < 0) | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |             return ret; | 
					
						
							| 
									
										
										
										
											2007-01-19 22:12:59 +00:00
										 |  |  |         fourcc_tag = AV_RL32(&header[0]); | 
					
						
							|  |  |  |         size = AV_RL32(&header[4]); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         if (url_feof(pb)) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |             return AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         switch (fourcc_tag) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-28 23:50:00 +00:00
										 |  |  |         case LIST_TAG: | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |             /* this is a good time to bump the video pts */ | 
					
						
							| 
									
										
										
										
											2005-04-26 10:01:17 +00:00
										 |  |  |             fourxm->video_pts ++; | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-28 23:50:00 +00:00
										 |  |  |             /* skip the LIST-* tag and move on to the next fourcc */ | 
					
						
							|  |  |  |             get_le32(pb); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         case ifrm_TAG: | 
					
						
							|  |  |  |         case pfrm_TAG: | 
					
						
							| 
									
										
										
										
											2007-08-12 00:23:47 +00:00
										 |  |  |         case cfrm_TAG: | 
					
						
							|  |  |  |         case ifr2_TAG: | 
					
						
							|  |  |  |         case pfr2_TAG: | 
					
						
							|  |  |  |         case cfr2_TAG: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* allocate 8 more bytes than 'size' to account for fourcc
 | 
					
						
							|  |  |  |              * and size */ | 
					
						
							| 
									
										
										
										
											2005-01-12 00:16:25 +00:00
										 |  |  |             if (size + 8 < size || av_new_packet(pkt, size + 8)) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |                 return AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |             pkt->stream_index = fourxm->video_stream_index; | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |             pkt->pts = fourxm->video_pts; | 
					
						
							| 
									
										
										
										
											2007-11-21 07:41:00 +00:00
										 |  |  |             pkt->pos = url_ftell(s->pb); | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |             memcpy(pkt->data, header, 8); | 
					
						
							| 
									
										
										
										
											2007-11-21 07:41:00 +00:00
										 |  |  |             ret = get_buffer(s->pb, &pkt->data[8], size); | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (ret < 0) | 
					
						
							|  |  |  |                 av_free_packet(pkt); | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 packet_read = 1; | 
					
						
							| 
									
										
										
										
											2003-05-25 23:41:25 +00:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |         case snd__TAG: | 
					
						
							|  |  |  |             track_number = get_le32(pb); | 
					
						
							| 
									
										
										
										
											2003-06-03 13:58:44 +00:00
										 |  |  |             out_size= get_le32(pb); | 
					
						
							|  |  |  |             size-=8; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             if (track_number == fourxm->selected_track) { | 
					
						
							| 
									
										
										
										
											2007-11-21 07:41:00 +00:00
										 |  |  |                 ret= av_get_packet(s->pb, pkt, size); | 
					
						
							| 
									
										
										
										
											2005-05-26 20:17:12 +00:00
										 |  |  |                 if(ret<0) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |                     return AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2005-12-17 18:14:38 +00:00
										 |  |  |                 pkt->stream_index = | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  |                     fourxm->tracks[fourxm->selected_track].stream_index; | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |                 pkt->pts = fourxm->audio_pts; | 
					
						
							| 
									
										
										
										
											2005-05-26 20:17:12 +00:00
										 |  |  |                 packet_read = 1; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |                 /* pts accounting */ | 
					
						
							|  |  |  |                 audio_frame_count = size; | 
					
						
							|  |  |  |                 if (fourxm->tracks[fourxm->selected_track].adpcm) | 
					
						
							| 
									
										
										
										
											2005-12-17 18:14:38 +00:00
										 |  |  |                     audio_frame_count -= | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |                         2 * (fourxm->tracks[fourxm->selected_track].channels); | 
					
						
							|  |  |  |                 audio_frame_count /= | 
					
						
							|  |  |  |                       fourxm->tracks[fourxm->selected_track].channels; | 
					
						
							|  |  |  |                 if (fourxm->tracks[fourxm->selected_track].adpcm) | 
					
						
							|  |  |  |                     audio_frame_count *= 2; | 
					
						
							| 
									
										
										
										
											2005-12-17 18:14:38 +00:00
										 |  |  |                 else | 
					
						
							| 
									
										
										
										
											2003-11-15 18:18:42 +00:00
										 |  |  |                     audio_frame_count /= | 
					
						
							|  |  |  |                     (fourxm->tracks[fourxm->selected_track].bits / 8); | 
					
						
							| 
									
										
										
										
											2005-04-26 10:01:17 +00:00
										 |  |  |                 fourxm->audio_pts += audio_frame_count; | 
					
						
							| 
									
										
										
										
											2003-05-28 02:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 url_fseek(pb, size, SEEK_CUR); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             url_fseek(pb, size, SEEK_CUR); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fourxm_read_close(AVFormatContext *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2007-04-08 20:24:16 +00:00
										 |  |  |     FourxmDemuxContext *fourxm = s->priv_data; | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     av_free(fourxm->tracks); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-07-10 21:14:37 +00:00
										 |  |  | AVInputFormat fourxm_demuxer = { | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     "4xm", | 
					
						
							| 
									
										
										
										
											2008-06-03 16:20:54 +00:00
										 |  |  |     NULL_IF_CONFIG_SMALL("4X Technologies format"), | 
					
						
							| 
									
										
										
										
											2003-05-25 05:07:36 +00:00
										 |  |  |     sizeof(FourxmDemuxContext), | 
					
						
							|  |  |  |     fourxm_probe, | 
					
						
							|  |  |  |     fourxm_read_header, | 
					
						
							|  |  |  |     fourxm_read_packet, | 
					
						
							|  |  |  |     fourxm_read_close, | 
					
						
							|  |  |  | }; |