| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Interplay C93 demuxer | 
					
						
							|  |  |  |  * Copyright (c) 2007 Anssi Hannula <anssi.hannula@gmail.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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 | 
					
						
							| 
									
										
										
										
											2007-07-05 10:40:25 +00:00
										 |  |  |  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "avformat.h"
 | 
					
						
							|  |  |  | #include "voc.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-14 20:01:32 +00:00
										 |  |  | #include "libavutil/intreadwrite.h"
 | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |     uint16_t index; | 
					
						
							|  |  |  |     uint8_t length; | 
					
						
							|  |  |  |     uint8_t frames; | 
					
						
							|  |  |  | } C93BlockRecord; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2008-12-11 22:34:14 +00:00
										 |  |  |     VocDecContext voc; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     C93BlockRecord block_records[512]; | 
					
						
							|  |  |  |     int current_block; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     uint32_t frame_offsets[32]; | 
					
						
							|  |  |  |     int current_frame; | 
					
						
							|  |  |  |     int next_pkt_is_audio; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AVStream *audio; | 
					
						
							|  |  |  | } C93DemuxContext; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-04-08 09:50:08 +00:00
										 |  |  | static int probe(AVProbeData *p) | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-09-14 20:01:32 +00:00
										 |  |  |     int i; | 
					
						
							|  |  |  |     int index = 1; | 
					
						
							|  |  |  |     if (p->buf_size < 16) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     for (i = 0; i < 16; i += 4) { | 
					
						
							|  |  |  |         if (AV_RL16(p->buf + i) != index || !p->buf[i + 2] || !p->buf[i + 3]) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         index += p->buf[i + 2]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return AVPROBE_SCORE_MAX; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-04-08 09:50:08 +00:00
										 |  |  | static int read_header(AVFormatContext *s, | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |                            AVFormatParameters *ap) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AVStream *video; | 
					
						
							| 
									
										
										
										
											2011-02-20 11:04:12 +01:00
										 |  |  |     AVIOContext *pb = s->pb; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     C93DemuxContext *c93 = s->priv_data; | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     int framecount = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < 512; i++) { | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |         c93->block_records[i].index = avio_rl16(pb); | 
					
						
							|  |  |  |         c93->block_records[i].length = avio_r8(pb); | 
					
						
							|  |  |  |         c93->block_records[i].frames = avio_r8(pb); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         if (c93->block_records[i].frames > 32) { | 
					
						
							|  |  |  |             av_log(s, AV_LOG_ERROR, "too many frames in block\n"); | 
					
						
							|  |  |  |             return AVERROR_INVALIDDATA; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         framecount += c93->block_records[i].frames; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Audio streams are added if audio packets are found */ | 
					
						
							|  |  |  |     s->ctx_flags |= AVFMTCTX_NOHEADER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     video = av_new_stream(s, 0); | 
					
						
							|  |  |  |     if (!video) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:21:30 +00:00
										 |  |  |         return AVERROR(ENOMEM); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-30 23:30:55 +00:00
										 |  |  |     video->codec->codec_type = AVMEDIA_TYPE_VIDEO; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     video->codec->codec_id = CODEC_ID_C93; | 
					
						
							|  |  |  |     video->codec->width = 320; | 
					
						
							|  |  |  |     video->codec->height = 192; | 
					
						
							|  |  |  |     /* 4:3 320x200 with 8 empty lines */ | 
					
						
							| 
									
										
										
										
											2008-08-23 23:43:20 +00:00
										 |  |  |     video->sample_aspect_ratio = (AVRational) { 5, 6 }; | 
					
						
							| 
									
										
										
										
											2011-02-05 10:08:29 +01:00
										 |  |  |     av_set_pts_info(video, 64, 2, 25); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     video->nb_frames = framecount; | 
					
						
							|  |  |  |     video->duration = framecount; | 
					
						
							|  |  |  |     video->start_time = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     c93->current_block = 0; | 
					
						
							|  |  |  |     c93->current_frame = 0; | 
					
						
							|  |  |  |     c93->next_pkt_is_audio = 0; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define C93_HAS_PALETTE 0x01
 | 
					
						
							|  |  |  | #define C93_FIRST_FRAME 0x02
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-04-08 09:50:08 +00:00
										 |  |  | static int read_packet(AVFormatContext *s, AVPacket *pkt) | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-20 11:04:12 +01:00
										 |  |  |     AVIOContext *pb = s->pb; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     C93DemuxContext *c93 = s->priv_data; | 
					
						
							|  |  |  |     C93BlockRecord *br = &c93->block_records[c93->current_block]; | 
					
						
							|  |  |  |     int datasize; | 
					
						
							|  |  |  |     int ret, i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (c93->next_pkt_is_audio) { | 
					
						
							|  |  |  |         c93->current_frame++; | 
					
						
							|  |  |  |         c93->next_pkt_is_audio = 0; | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |         datasize = avio_rl16(pb); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         if (datasize > 42) { | 
					
						
							|  |  |  |             if (!c93->audio) { | 
					
						
							|  |  |  |                 c93->audio = av_new_stream(s, 1); | 
					
						
							|  |  |  |                 if (!c93->audio) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:21:30 +00:00
										 |  |  |                     return AVERROR(ENOMEM); | 
					
						
							| 
									
										
										
										
											2010-03-30 23:30:55 +00:00
										 |  |  |                 c93->audio->codec->codec_type = AVMEDIA_TYPE_AUDIO; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |             url_fskip(pb, 26); /* VOC header */ | 
					
						
							|  |  |  |             ret = voc_get_packet(s, pkt, c93->audio, datasize - 26); | 
					
						
							|  |  |  |             if (ret > 0) { | 
					
						
							|  |  |  |                 pkt->stream_index = 1; | 
					
						
							| 
									
										
										
										
											2010-03-31 12:29:58 +00:00
										 |  |  |                 pkt->flags |= AV_PKT_FLAG_KEY; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |                 return ret; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (c93->current_frame >= br->frames) { | 
					
						
							|  |  |  |         if (c93->current_block >= 511 || !br[1].length) | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |             return AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         br++; | 
					
						
							|  |  |  |         c93->current_block++; | 
					
						
							|  |  |  |         c93->current_frame = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (c93->current_frame == 0) { | 
					
						
							|  |  |  |         url_fseek(pb, br->index * 2048, SEEK_SET); | 
					
						
							|  |  |  |         for (i = 0; i < 32; i++) { | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |             c93->frame_offsets[i] = avio_rl32(pb); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     url_fseek(pb,br->index * 2048 + | 
					
						
							|  |  |  |             c93->frame_offsets[c93->current_frame], SEEK_SET); | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |     datasize = avio_rl16(pb); /* video frame size */ | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ret = av_new_packet(pkt, datasize + 768 + 1); | 
					
						
							|  |  |  |     if (ret < 0) | 
					
						
							|  |  |  |         return ret; | 
					
						
							|  |  |  |     pkt->data[0] = 0; | 
					
						
							|  |  |  |     pkt->size = datasize + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |     ret = avio_read(pb, pkt->data + 1, datasize); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     if (ret < datasize) { | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |         ret = AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |     datasize = avio_rl16(pb); /* palette size */ | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     if (datasize) { | 
					
						
							|  |  |  |         if (datasize != 768) { | 
					
						
							|  |  |  |             av_log(s, AV_LOG_ERROR, "invalid palette size %u\n", datasize); | 
					
						
							|  |  |  |             ret = AVERROR_INVALIDDATA; | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pkt->data[0] |= C93_HAS_PALETTE; | 
					
						
							| 
									
										
										
										
											2011-02-21 16:43:01 +01:00
										 |  |  |         ret = avio_read(pb, pkt->data + pkt->size, datasize); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         if (ret < datasize) { | 
					
						
							| 
									
										
										
										
											2007-07-19 15:23:32 +00:00
										 |  |  |             ret = AVERROR(EIO); | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pkt->size += 768; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pkt->stream_index = 0; | 
					
						
							|  |  |  |     c93->next_pkt_is_audio = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* only the first frame is guaranteed to not reference previous frames */ | 
					
						
							|  |  |  |     if (c93->current_block == 0 && c93->current_frame == 0) { | 
					
						
							| 
									
										
										
										
											2010-03-31 12:29:58 +00:00
										 |  |  |         pkt->flags |= AV_PKT_FLAG_KEY; | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |         pkt->data[0] |= C93_FIRST_FRAME; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fail: | 
					
						
							|  |  |  |     av_free_packet(pkt); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-25 22:03:28 +00:00
										 |  |  | AVInputFormat ff_c93_demuxer = { | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     "c93", | 
					
						
							| 
									
										
										
										
											2008-06-03 16:20:54 +00:00
										 |  |  |     NULL_IF_CONFIG_SMALL("Interplay C93"), | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  |     sizeof(C93DemuxContext), | 
					
						
							| 
									
										
										
										
											2007-04-08 09:50:08 +00:00
										 |  |  |     probe, | 
					
						
							|  |  |  |     read_header, | 
					
						
							|  |  |  |     read_packet, | 
					
						
							| 
									
										
										
										
											2007-04-07 14:25:25 +00:00
										 |  |  | }; |