tiertexseq.c 9.2 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
 * Tiertex Limited SEQ File Demuxer
 * Copyright (c) 2006 Gregory Montoir (cyx@users.sourceforge.net)
 *
 * 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
 */

/**
23
 * @file
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
 * Tiertex Limited SEQ file demuxer
 */

#include "avformat.h"

#define SEQ_FRAME_SIZE         6144
#define SEQ_FRAME_W            256
#define SEQ_FRAME_H            128
#define SEQ_NUM_FRAME_BUFFERS  30
#define SEQ_AUDIO_BUFFER_SIZE  882
#define SEQ_SAMPLE_RATE        22050
#define SEQ_FRAME_RATE         25


typedef struct TiertexSeqFrameBuffer {
    int fill_size;
    int data_size;
    unsigned char *data;
} TiertexSeqFrameBuffer;

typedef struct SeqDemuxContext {
    int audio_stream_index;
    int video_stream_index;
    int current_frame_pts;
    int current_frame_offs;
    TiertexSeqFrameBuffer frame_buffers[SEQ_NUM_FRAME_BUFFERS];
    int frame_buffers_count;
    unsigned int current_audio_data_size;
    unsigned int current_audio_data_offs;
    unsigned int current_pal_data_size;
    unsigned int current_pal_data_offs;
    unsigned int current_video_data_size;
    unsigned char *current_video_data_ptr;
    int audio_buffer_full;
} SeqDemuxContext;


static int seq_probe(AVProbeData *p)
{
    int i;

65
    if (p->buf_size < 258)
66 67 68 69 70
        return 0;

    /* there's no real header in a .seq file, the only thing they have in common */
    /* is the first 256 bytes of the file which are always filled with 0 */
    for (i = 0; i < 256; i++)
Michael Niedermayer's avatar
Michael Niedermayer committed
71
        if (p->buf[i])
72 73
            return 0;

74 75 76
    if(p->buf[256]==0 && p->buf[257]==0)
        return 0;

77 78 79 80
    /* only one fourth of the score since the previous check is too naive */
    return AVPROBE_SCORE_MAX / 4;
}

81
static int seq_init_frame_buffers(SeqDemuxContext *seq, AVIOContext *pb)
82 83 84 85
{
    int i, sz;
    TiertexSeqFrameBuffer *seq_buffer;

86
    avio_seek(pb, 256, SEEK_SET);
87 88

    for (i = 0; i < SEQ_NUM_FRAME_BUFFERS; i++) {
89
        sz = avio_rl16(pb);
90 91 92 93 94 95 96 97
        if (sz == 0)
            break;
        else {
            seq_buffer = &seq->frame_buffers[i];
            seq_buffer->fill_size = 0;
            seq_buffer->data_size = sz;
            seq_buffer->data = av_malloc(sz);
            if (!seq_buffer->data)
98
                return AVERROR(ENOMEM);
99 100 101 102 103 104
        }
    }
    seq->frame_buffers_count = i;
    return 0;
}

105
static int seq_fill_buffer(SeqDemuxContext *seq, AVIOContext *pb, int buffer_num, unsigned int data_offs, int data_size)
106 107 108 109 110 111 112
{
    TiertexSeqFrameBuffer *seq_buffer;

    if (buffer_num >= SEQ_NUM_FRAME_BUFFERS)
        return AVERROR_INVALIDDATA;

    seq_buffer = &seq->frame_buffers[buffer_num];
113
    if (seq_buffer->fill_size + data_size > seq_buffer->data_size || data_size <= 0)
114 115
        return AVERROR_INVALIDDATA;

116
    avio_seek(pb, seq->current_frame_offs + data_offs, SEEK_SET);
117
    if (avio_read(pb, seq_buffer->data + seq_buffer->fill_size, data_size) != data_size)
118
        return AVERROR(EIO);
119 120 121 122 123

    seq_buffer->fill_size += data_size;
    return 0;
}

124
static int seq_parse_frame_data(SeqDemuxContext *seq, AVIOContext *pb)
125 126 127
{
    unsigned int offset_table[4], buffer_num[4];
    TiertexSeqFrameBuffer *seq_buffer;
128
    int i, e, err;
129 130

    seq->current_frame_offs += SEQ_FRAME_SIZE;
131
    avio_seek(pb, seq->current_frame_offs, SEEK_SET);
132 133

    /* sound data */
134
    seq->current_audio_data_offs = avio_rl16(pb);
Michael Niedermayer's avatar
Michael Niedermayer committed
135
    if (seq->current_audio_data_offs) {
136 137 138 139 140 141
        seq->current_audio_data_size = SEQ_AUDIO_BUFFER_SIZE * 2;
    } else {
        seq->current_audio_data_size = 0;
    }

    /* palette data */
142
    seq->current_pal_data_offs = avio_rl16(pb);
Michael Niedermayer's avatar
Michael Niedermayer committed
143
    if (seq->current_pal_data_offs) {
144 145 146 147 148 149 150
        seq->current_pal_data_size = 768;
    } else {
        seq->current_pal_data_size = 0;
    }

    /* video data */
    for (i = 0; i < 4; i++)
151
        buffer_num[i] = avio_r8(pb);
152 153

    for (i = 0; i < 4; i++)
154
        offset_table[i] = avio_rl16(pb);
155 156

    for (i = 0; i < 3; i++) {
Michael Niedermayer's avatar
Michael Niedermayer committed
157
        if (offset_table[i]) {
158
            for (e = i + 1; e < 3 && offset_table[e] == 0; e++);
159
            err = seq_fill_buffer(seq, pb, buffer_num[1 + i],
160 161
              offset_table[i],
              offset_table[e] - offset_table[i]);
Michael Niedermayer's avatar
Michael Niedermayer committed
162
            if (err)
163
                return err;
164 165 166 167
        }
    }

    if (buffer_num[0] != 255) {
168 169 170
        if (buffer_num[0] >= SEQ_NUM_FRAME_BUFFERS)
            return AVERROR_INVALIDDATA;

171 172 173 174 175 176 177 178 179 180 181 182 183 184 185
        seq_buffer = &seq->frame_buffers[buffer_num[0]];
        seq->current_video_data_size = seq_buffer->fill_size;
        seq->current_video_data_ptr  = seq_buffer->data;
        seq_buffer->fill_size = 0;
    } else {
        seq->current_video_data_size = 0;
        seq->current_video_data_ptr  = 0;
    }

    return 0;
}

static int seq_read_header(AVFormatContext *s, AVFormatParameters *ap)
{
    int i, rc;
186
    SeqDemuxContext *seq = s->priv_data;
187
    AVIOContext *pb = s->pb;
188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
    AVStream *st;

    /* init internal buffers */
    rc = seq_init_frame_buffers(seq, pb);
    if (rc)
        return rc;

    seq->current_frame_offs = 0;

    /* preload (no audio data, just buffer operations related data) */
    for (i = 1; i <= 100; i++) {
        rc = seq_parse_frame_data(seq, pb);
        if (rc)
            return rc;
    }

    seq->current_frame_pts = 0;

    seq->audio_buffer_full = 0;

    /* initialize the video decoder stream */
209
    st = avformat_new_stream(s, NULL);
210
    if (!st)
211
        return AVERROR(ENOMEM);
212 213 214

    av_set_pts_info(st, 32, 1, SEQ_FRAME_RATE);
    seq->video_stream_index = st->index;
215
    st->codec->codec_type = AVMEDIA_TYPE_VIDEO;
216 217 218 219 220 221
    st->codec->codec_id = CODEC_ID_TIERTEXSEQVIDEO;
    st->codec->codec_tag = 0;  /* no fourcc */
    st->codec->width = SEQ_FRAME_W;
    st->codec->height = SEQ_FRAME_H;

    /* initialize the audio decoder stream */
222
    st = avformat_new_stream(s, NULL);
223
    if (!st)
224
        return AVERROR(ENOMEM);
225 226 227

    av_set_pts_info(st, 32, 1, SEQ_SAMPLE_RATE);
    seq->audio_stream_index = st->index;
228
    st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
229 230 231 232
    st->codec->codec_id = CODEC_ID_PCM_S16BE;
    st->codec->codec_tag = 0;  /* no tag */
    st->codec->channels = 1;
    st->codec->sample_rate = SEQ_SAMPLE_RATE;
233 234 235
    st->codec->bits_per_coded_sample = 16;
    st->codec->bit_rate = st->codec->sample_rate * st->codec->bits_per_coded_sample * st->codec->channels;
    st->codec->block_align = st->codec->channels * st->codec->bits_per_coded_sample;
236 237 238 239 240 241 242

    return 0;
}

static int seq_read_packet(AVFormatContext *s, AVPacket *pkt)
{
    int rc;
243
    SeqDemuxContext *seq = s->priv_data;
244
    AVIOContext *pb = s->pb;
245 246 247 248 249 250 251 252 253

    if (!seq->audio_buffer_full) {
        rc = seq_parse_frame_data(seq, pb);
        if (rc)
            return rc;

        /* video packet */
        if (seq->current_pal_data_size + seq->current_video_data_size != 0) {
            if (av_new_packet(pkt, 1 + seq->current_pal_data_size + seq->current_video_data_size))
254
                return AVERROR(ENOMEM);
255 256

            pkt->data[0] = 0;
Michael Niedermayer's avatar
Michael Niedermayer committed
257
            if (seq->current_pal_data_size) {
258
                pkt->data[0] |= 1;
259
                avio_seek(pb, seq->current_frame_offs + seq->current_pal_data_offs, SEEK_SET);
260
                if (avio_read(pb, &pkt->data[1], seq->current_pal_data_size) != seq->current_pal_data_size)
261
                    return AVERROR(EIO);
262
            }
Michael Niedermayer's avatar
Michael Niedermayer committed
263
            if (seq->current_video_data_size) {
264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279
                pkt->data[0] |= 2;
                memcpy(&pkt->data[1 + seq->current_pal_data_size],
                  seq->current_video_data_ptr,
                  seq->current_video_data_size);
            }
            pkt->stream_index = seq->video_stream_index;
            pkt->pts = seq->current_frame_pts;

            /* sound buffer will be processed on next read_packet() call */
            seq->audio_buffer_full = 1;
            return 0;
       }
    }

    /* audio packet */
    if (seq->current_audio_data_offs == 0) /* end of data reached */
280
        return AVERROR(EIO);
281

282
    avio_seek(pb, seq->current_frame_offs + seq->current_audio_data_offs, SEEK_SET);
283 284 285 286 287
    rc = av_get_packet(pb, pkt, seq->current_audio_data_size);
    if (rc < 0)
        return rc;

    pkt->stream_index = seq->audio_stream_index;
288
    seq->current_frame_pts++;
289 290 291 292 293 294 295 296

    seq->audio_buffer_full = 0;
    return 0;
}

static int seq_read_close(AVFormatContext *s)
{
    int i;
297
    SeqDemuxContext *seq = s->priv_data;
298 299 300 301 302 303 304

    for (i = 0; i < SEQ_NUM_FRAME_BUFFERS; i++)
        av_free(seq->frame_buffers[i].data);

    return 0;
}

305
AVInputFormat ff_tiertexseq_demuxer = {
306 307 308 309 310 311 312
    .name           = "tiertexseq",
    .long_name      = NULL_IF_CONFIG_SMALL("Tiertex Limited SEQ format"),
    .priv_data_size = sizeof(SeqDemuxContext),
    .read_probe     = seq_probe,
    .read_header    = seq_read_header,
    .read_packet    = seq_read_packet,
    .read_close     = seq_read_close,
313
};