annotate vqavideo.c @ 2497:69adfbbdcdeb libavcodec

- samples from mplayer ftp in the "adv" profile seem to have profile=2, which isn't the advanced one; and indeed, using adv. profile parser fails. Using normal parser works, and that's what is done - attempt at taking care of stride for NORM2 bitplane decoding - duplication of much code from msmpeg4.c; this code isn't yet used, but goes down as far as the block layer (mainly Transform Type stuff, the remains are wild editing without checking). Unusable yet, and lacks the AC decoding (but a step further in bitstream parsing) patch by anonymous
author michael
date Fri, 04 Feb 2005 02:20:38 +0000
parents 18b8b2dcc037
children ef2149182f1c
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
1 /*
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
2 * Westwood Studios VQA Video Decoder
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
3 * Copyright (C) 2003 the ffmpeg project
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
4 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
5 * This library is free software; you can redistribute it and/or
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
6 * modify it under the terms of the GNU Lesser General Public
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
7 * License as published by the Free Software Foundation; either
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
8 * version 2 of the License, or (at your option) any later version.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
9 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
10 * This library is distributed in the hope that it will be useful,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
13 * Lesser General Public License for more details.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
14 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
15 * You should have received a copy of the GNU Lesser General Public
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
16 * License along with this library; if not, write to the Free Software
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
18 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
19 */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
20
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
21 /**
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
22 * @file vqavideo.c
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
23 * VQA Video Decoder by Mike Melanson (melanson@pcisys.net)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
24 * For more information about the RPZA format, visit:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
25 * http://www.pcisys.net/~melanson/codecs/
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
26 *
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
27 * The VQA video decoder outputs PAL8 or RGB555 colorspace data, depending
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
28 * on the type of data in the file.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
29 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
30 * This decoder needs the 42-byte VQHD header from the beginning
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
31 * of the VQA file passed through the extradata field. The VQHD header
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
32 * is laid out as:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
33 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
34 * bytes 0-3 chunk fourcc: 'VQHD'
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
35 * bytes 4-7 chunk size in big-endian format, should be 0x0000002A
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
36 * bytes 8-49 VQHD chunk data
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
37 *
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
38 * Bytes 8-49 are what this decoder expects to see.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
39 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
40 * Briefly, VQA is a vector quantized animation format that operates in a
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
41 * VGA palettized colorspace. It operates on pixel vectors (blocks)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
42 * of either 4x2 or 4x4 in size. Compressed VQA chunks can contain vector
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
43 * codebooks, palette information, and code maps for rendering vectors onto
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
44 * frames. Any of these components can also be compressed with a run-length
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
45 * encoding (RLE) algorithm commonly referred to as "format80".
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
46 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
47 * VQA takes a novel approach to rate control. Each group of n frames
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
48 * (usually, n = 8) relies on a different vector codebook. Rather than
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
49 * transporting an entire codebook every 8th frame, the new codebook is
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
50 * broken up into 8 pieces and sent along with the compressed video chunks
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
51 * for each of the 8 frames preceding the 8 frames which require the
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
52 * codebook. A full codebook is also sent on the very first frame of a
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
53 * file. This is an interesting technique, although it makes random file
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
54 * seeking difficult despite the fact that the frames are all intracoded.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
55 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
56 * V1,2 VQA uses 12-bit codebook indices. If the 12-bit indices were
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
57 * packed into bytes and then RLE compressed, bytewise, the results would
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
58 * be poor. That is why the coding method divides each index into 2 parts,
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
59 * the top 4 bits and the bottom 8 bits, then RL encodes the 4-bit pieces
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
60 * together and the 8-bit pieces together. If most of the vectors are
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
61 * clustered into one group of 256 vectors, most of the 4-bit index pieces
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
62 * should be the same.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
63 */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
64
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
65 #include <stdio.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
66 #include <stdlib.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
67 #include <string.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
68 #include <unistd.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
69
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
70 #include "common.h"
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
71 #include "avcodec.h"
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
72 #include "dsputil.h"
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
73
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
74 #define PALETTE_COUNT 256
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
75 #define VQA_HEADER_SIZE 0x2A
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
76 #define CHUNK_PREAMBLE_SIZE 8
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
77
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
78 /* allocate the maximum vector space, regardless of the file version:
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
79 * (0xFF00 codebook vectors + 0x100 solid pixel vectors) * (4x4 pixels/block) */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
80 #define MAX_CODEBOOK_VECTORS 0xFF00
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
81 #define SOLID_PIXEL_VECTORS 0x100
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
82 #define MAX_VECTORS (MAX_CODEBOOK_VECTORS + SOLID_PIXEL_VECTORS)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
83 #define MAX_CODEBOOK_SIZE (MAX_VECTORS * 4 * 4)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
84
1881
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
85 #define CBF0_TAG MKBETAG('C', 'B', 'F', '0')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
86 #define CBFZ_TAG MKBETAG('C', 'B', 'F', 'Z')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
87 #define CBP0_TAG MKBETAG('C', 'B', 'P', '0')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
88 #define CBPZ_TAG MKBETAG('C', 'B', 'P', 'Z')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
89 #define CPL0_TAG MKBETAG('C', 'P', 'L', '0')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
90 #define CPLZ_TAG MKBETAG('C', 'P', 'L', 'Z')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
91 #define VPTZ_TAG MKBETAG('V', 'P', 'T', 'Z')
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
92
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
93 #define VQA_DEBUG 0
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
94
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
95 #if VQA_DEBUG
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
96 #define vqa_debug printf
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
97 #else
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
98 static inline void vqa_debug(const char *format, ...) { }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
99 #endif
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
100
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
101 typedef struct VqaContext {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
102
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
103 AVCodecContext *avctx;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
104 DSPContext dsp;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
105 AVFrame frame;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
106
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
107 unsigned char *buf;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
108 int size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
109
1585
6b224ca24033 revised palette API, courtesy of Roberto Togni (rtogni at freemail.it)
melanson
parents: 1506
diff changeset
110 unsigned int palette[PALETTE_COUNT];
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
111
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
112 int width; /* width of a frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
113 int height; /* height of a frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
114 int vector_width; /* width of individual vector */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
115 int vector_height; /* height of individual vector */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
116 int vqa_version; /* this should be either 1, 2 or 3 */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
117
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
118 unsigned char *codebook; /* the current codebook */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
119 int codebook_size;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
120 unsigned char *next_codebook_buffer; /* accumulator for next codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
121 int next_codebook_buffer_index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
122
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
123 unsigned char *decode_buffer;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
124 int decode_buffer_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
125
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
126 /* number of frames to go before replacing codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
127 int partial_countdown;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
128 int partial_count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
129
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
130 } VqaContext;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
131
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
132 static int vqa_decode_init(AVCodecContext *avctx)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
133 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
134 VqaContext *s = (VqaContext *)avctx->priv_data;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
135 unsigned char *vqa_header;
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
136 int i, j, codebook_index;;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
137
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
138 s->avctx = avctx;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
139 avctx->pix_fmt = PIX_FMT_PAL8;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
140 avctx->has_b_frames = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
141 dsputil_init(&s->dsp, avctx);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
142
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
143 /* make sure the extradata made it */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
144 if (s->avctx->extradata_size != VQA_HEADER_SIZE) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
145 av_log(s->avctx, AV_LOG_ERROR, " VQA video: expected extradata size of %d\n", VQA_HEADER_SIZE);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
146 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
147 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
148
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
149 /* load up the VQA parameters from the header */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
150 vqa_header = (unsigned char *)s->avctx->extradata;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
151 s->vqa_version = vqa_header[0];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
152 s->width = LE_16(&vqa_header[6]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
153 s->height = LE_16(&vqa_header[8]);
2422
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
154 if(avcodec_check_dimensions(avctx, s->width, s->height)){
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
155 s->width= s->height= 0;
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
156 return -1;
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
157 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
158 s->vector_width = vqa_header[10];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
159 s->vector_height = vqa_header[11];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
160 s->partial_count = s->partial_countdown = vqa_header[13];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
161
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
162 /* the vector dimensions have to meet very stringent requirements */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
163 if ((s->vector_width != 4) ||
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
164 ((s->vector_height != 2) && (s->vector_height != 4))) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
165 /* return without further initialization */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
166 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
167 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
168
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
169 /* allocate codebooks */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
170 s->codebook_size = MAX_CODEBOOK_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
171 s->codebook = av_malloc(s->codebook_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
172 s->next_codebook_buffer = av_malloc(s->codebook_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
173
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
174 /* initialize the solid-color vectors */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
175 if (s->vector_height == 4) {
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
176 codebook_index = 0xFF00 * 16;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
177 for (i = 0; i < 256; i++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
178 for (j = 0; j < 16; j++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
179 s->codebook[codebook_index++] = i;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
180 } else {
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
181 codebook_index = 0xF00 * 8;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
182 for (i = 0; i < 256; i++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
183 for (j = 0; j < 8; j++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
184 s->codebook[codebook_index++] = i;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
185 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
186 s->next_codebook_buffer_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
187
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
188 /* allocate decode buffer */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
189 s->decode_buffer_size = (s->width / s->vector_width) *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
190 (s->height / s->vector_height) * 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
191 s->decode_buffer = av_malloc(s->decode_buffer_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
192
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
193 s->frame.data[0] = NULL;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
194
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
195 return 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
196 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
197
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
198 #define CHECK_COUNT() \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
199 if (dest_index + count > dest_size) { \
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
200 av_log(NULL, AV_LOG_ERROR, " VQA video: decode_format80 problem: next op would overflow dest_index\n"); \
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
201 av_log(NULL, AV_LOG_ERROR, " VQA video: current dest_index = %d, count = %d, dest_size = %d\n", \
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
202 dest_index, count, dest_size); \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
203 return; \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
204 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
205
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
206 static void decode_format80(unsigned char *src, int src_size,
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
207 unsigned char *dest, int dest_size, int check_size) {
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
208
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
209 int src_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
210 int dest_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
211 int count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
212 int src_pos;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
213 unsigned char color;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
214 int i;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
215
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
216 while (src_index < src_size) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
217
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
218 vqa_debug(" opcode %02X: ", src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
219
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
220 /* 0x80 means that frame is finished */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
221 if (src[src_index] == 0x80)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
222 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
223
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
224 if (dest_index >= dest_size) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
225 av_log(NULL, AV_LOG_ERROR, " VQA video: decode_format80 problem: dest_index (%d) exceeded dest_size (%d)\n",
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
226 dest_index, dest_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
227 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
228 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
229
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
230 if (src[src_index] == 0xFF) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
231
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
232 src_index++;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
233 count = LE_16(&src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
234 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
235 src_pos = LE_16(&src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
236 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
237 vqa_debug("(1) copy %X bytes from absolute pos %X\n", count, src_pos);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
238 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
239 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
240 dest[dest_index + i] = dest[src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
241 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
242
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
243 } else if (src[src_index] == 0xFE) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
244
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
245 src_index++;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
246 count = LE_16(&src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
247 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
248 color = src[src_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
249 vqa_debug("(2) set %X bytes to %02X\n", count, color);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
250 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
251 memset(&dest[dest_index], color, count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
252 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
253
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
254 } else if ((src[src_index] & 0xC0) == 0xC0) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
255
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
256 count = (src[src_index++] & 0x3F) + 3;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
257 src_pos = LE_16(&src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
258 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
259 vqa_debug("(3) copy %X bytes from absolute pos %X\n", count, src_pos);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
260 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
261 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
262 dest[dest_index + i] = dest[src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
263 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
264
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
265 } else if (src[src_index] > 0x80) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
266
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
267 count = src[src_index++] & 0x3F;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
268 vqa_debug("(4) copy %X bytes from source to dest\n", count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
269 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
270 memcpy(&dest[dest_index], &src[src_index], count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
271 src_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
272 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
273
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
274 } else {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
275
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
276 count = ((src[src_index] & 0x70) >> 4) + 3;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
277 src_pos = BE_16(&src[src_index]) & 0x0FFF;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
278 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
279 vqa_debug("(5) copy %X bytes from relpos %X\n", count, src_pos);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
280 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
281 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
282 dest[dest_index + i] = dest[dest_index - src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
283 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
284 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
285 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
286
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
287 /* validate that the entire destination buffer was filled; this is
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
288 * important for decoding frame maps since each vector needs to have a
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
289 * codebook entry; it is not important for compressed codebooks because
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
290 * not every entry needs to be filled */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
291 if (check_size)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
292 if (dest_index < dest_size)
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
293 av_log(NULL, AV_LOG_ERROR, " VQA video: decode_format80 problem: decode finished with dest_index (%d) < dest_size (%d)\n",
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
294 dest_index, dest_size);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
295 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
296
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
297 static void vqa_decode_chunk(VqaContext *s)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
298 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
299 unsigned int chunk_type;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
300 unsigned int chunk_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
301 int byte_skip;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
302 unsigned int index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
303 int i;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
304 unsigned char r, g, b;
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
305 int index_shift;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
306
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
307 int cbf0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
308 int cbfz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
309 int cbp0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
310 int cbpz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
311 int cpl0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
312 int cplz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
313 int vptz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
314
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
315 int x, y;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
316 int lines = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
317 int pixel_ptr;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
318 int vector_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
319 int lobyte = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
320 int hibyte = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
321 int lobytes = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
322 int hibytes = s->decode_buffer_size / 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
323
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
324 /* first, traverse through the frame and find the subchunks */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
325 while (index < s->size) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
326
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
327 chunk_type = BE_32(&s->buf[index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
328 chunk_size = BE_32(&s->buf[index + 4]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
329
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
330 switch (chunk_type) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
331
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
332 case CBF0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
333 cbf0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
334 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
335
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
336 case CBFZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
337 cbfz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
338 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
339
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
340 case CBP0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
341 cbp0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
342 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
343
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
344 case CBPZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
345 cbpz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
346 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
347
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
348 case CPL0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
349 cpl0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
350 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
351
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
352 case CPLZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
353 cplz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
354 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
355
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
356 case VPTZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
357 vptz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
358 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
359
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
360 default:
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
361 av_log(s->avctx, AV_LOG_ERROR, " VQA video: Found unknown chunk type: %c%c%c%c (%08X)\n",
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
362 (chunk_type >> 24) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
363 (chunk_type >> 16) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
364 (chunk_type >> 8) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
365 (chunk_type >> 0) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
366 chunk_type);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
367 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
368 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
369
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
370 byte_skip = chunk_size & 0x01;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
371 index += (CHUNK_PREAMBLE_SIZE + chunk_size + byte_skip);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
372 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
373
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
374 /* next, deal with the palette */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
375 if ((cpl0_chunk != -1) && (cplz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
376
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
377 /* a chunk should not have both chunk types */
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
378 av_log(s->avctx, AV_LOG_ERROR, " VQA video: problem: found both CPL0 and CPLZ chunks\n");
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
379 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
380 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
381
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
382 /* decompress the palette chunk */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
383 if (cplz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
384
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
385 /* yet to be handled */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
386
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
387 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
388
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
389 /* convert the RGB palette into the machine's endian format */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
390 if (cpl0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
391
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
392 chunk_size = BE_32(&s->buf[cpl0_chunk + 4]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
393 /* sanity check the palette size */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
394 if (chunk_size / 3 > 256) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
395 av_log(s->avctx, AV_LOG_ERROR, " VQA video: problem: found a palette chunk with %d colors\n",
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
396 chunk_size / 3);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
397 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
398 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
399 cpl0_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
400 for (i = 0; i < chunk_size / 3; i++) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
401 /* scale by 4 to transform 6-bit palette -> 8-bit */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
402 r = s->buf[cpl0_chunk++] * 4;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
403 g = s->buf[cpl0_chunk++] * 4;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
404 b = s->buf[cpl0_chunk++] * 4;
1585
6b224ca24033 revised palette API, courtesy of Roberto Togni (rtogni at freemail.it)
melanson
parents: 1506
diff changeset
405 s->palette[i] = (r << 16) | (g << 8) | (b);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
406 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
407 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
408
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
409 /* next, look for a full codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
410 if ((cbf0_chunk != -1) && (cbfz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
411
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
412 /* a chunk should not have both chunk types */
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
413 av_log(s->avctx, AV_LOG_ERROR, " VQA video: problem: found both CBF0 and CBFZ chunks\n");
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
414 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
415 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
416
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
417 /* decompress the full codebook chunk */
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
418 if (cbfz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
419
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
420 chunk_size = BE_32(&s->buf[cbfz_chunk + 4]);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
421 cbfz_chunk += CHUNK_PREAMBLE_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
422 decode_format80(&s->buf[cbfz_chunk], chunk_size,
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
423 s->codebook, s->codebook_size, 0);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
424 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
425
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
426 /* copy a full codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
427 if (cbf0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
428
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
429 chunk_size = BE_32(&s->buf[cbf0_chunk + 4]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
430 /* sanity check the full codebook size */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
431 if (chunk_size > MAX_CODEBOOK_SIZE) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
432 av_log(s->avctx, AV_LOG_ERROR, " VQA video: problem: CBF0 chunk too large (0x%X bytes)\n",
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
433 chunk_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
434 return;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
435 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
436 cbf0_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
437
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
438 memcpy(s->codebook, &s->buf[cbf0_chunk], chunk_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
439 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
440
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
441 /* decode the frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
442 if (vptz_chunk == -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
443
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
444 /* something is wrong if there is no VPTZ chunk */
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
445 av_log(s->avctx, AV_LOG_ERROR, " VQA video: problem: no VPTZ chunk found\n");
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
446 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
447 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
448
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
449 chunk_size = BE_32(&s->buf[vptz_chunk + 4]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
450 vptz_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
451 decode_format80(&s->buf[vptz_chunk], chunk_size,
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
452 s->decode_buffer, s->decode_buffer_size, 1);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
453
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
454 /* render the final PAL8 frame */
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
455 if (s->vector_height == 4)
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
456 index_shift = 4;
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
457 else
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
458 index_shift = 3;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
459 for (y = 0; y < s->frame.linesize[0] * s->height;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
460 y += s->frame.linesize[0] * s->vector_height) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
461
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
462 for (x = y; x < y + s->width; x += 4, lobytes++, hibytes++) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
463 pixel_ptr = x;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
464
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
465 /* get the vector index, the method for which varies according to
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
466 * VQA file version */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
467 switch (s->vqa_version) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
468
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
469 case 1:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
470 /* still need sample media for this case (only one game, "Legend of
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
471 * Kyrandia III : Malcolm's Revenge", is known to use this version) */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
472 lines = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
473 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
474
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
475 case 2:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
476 lobyte = s->decode_buffer[lobytes];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
477 hibyte = s->decode_buffer[hibytes];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
478 vector_index = (hibyte << 8) | lobyte;
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
479 vector_index <<= index_shift;
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
480 lines = s->vector_height;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
481 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
482
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
483 case 3:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
484 /* not implemented yet */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
485 lines = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
486 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
487 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
488
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
489 while (lines--) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
490 s->frame.data[0][pixel_ptr + 0] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
491 s->frame.data[0][pixel_ptr + 1] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
492 s->frame.data[0][pixel_ptr + 2] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
493 s->frame.data[0][pixel_ptr + 3] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
494 pixel_ptr += s->frame.linesize[0];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
495 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
496 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
497 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
498
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
499 /* handle partial codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
500 if ((cbp0_chunk != -1) && (cbpz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
501 /* a chunk should not have both chunk types */
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
502 av_log(s->avctx, AV_LOG_ERROR, " VQA video: problem: found both CBP0 and CBPZ chunks\n");
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
503 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
504 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
505
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
506 if (cbp0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
507
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
508 chunk_size = BE_32(&s->buf[cbp0_chunk + 4]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
509 cbp0_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
510
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
511 /* accumulate partial codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
512 memcpy(&s->next_codebook_buffer[s->next_codebook_buffer_index],
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
513 &s->buf[cbp0_chunk], chunk_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
514 s->next_codebook_buffer_index += chunk_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
515
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
516 s->partial_countdown--;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
517 if (s->partial_countdown == 0) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
518
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
519 /* time to replace codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
520 memcpy(s->codebook, s->next_codebook_buffer,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
521 s->next_codebook_buffer_index);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
522
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
523 /* reset accounting */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
524 s->next_codebook_buffer_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
525 s->partial_countdown = s->partial_count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
526 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
527 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
528
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
529 if (cbpz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
530
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
531 chunk_size = BE_32(&s->buf[cbpz_chunk + 4]);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
532 cbpz_chunk += CHUNK_PREAMBLE_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
533
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
534 /* accumulate partial codebook */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
535 memcpy(&s->next_codebook_buffer[s->next_codebook_buffer_index],
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
536 &s->buf[cbpz_chunk], chunk_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
537 s->next_codebook_buffer_index += chunk_size;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
538
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
539 s->partial_countdown--;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
540 if (s->partial_countdown == 0) {
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
541
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
542 /* decompress codebook */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
543 decode_format80(s->next_codebook_buffer,
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
544 s->next_codebook_buffer_index,
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
545 s->codebook, s->codebook_size, 0);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
546
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
547 /* reset accounting */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
548 s->next_codebook_buffer_index = 0;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
549 s->partial_countdown = s->partial_count;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
550 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
551 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
552 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
553
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
554 static int vqa_decode_frame(AVCodecContext *avctx,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
555 void *data, int *data_size,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
556 uint8_t *buf, int buf_size)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
557 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
558 VqaContext *s = (VqaContext *)avctx->priv_data;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
559
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
560 s->buf = buf;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
561 s->size = buf_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
562
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
563 if (s->frame.data[0])
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
564 avctx->release_buffer(avctx, &s->frame);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
565
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
566 if (avctx->get_buffer(avctx, &s->frame)) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
567 av_log(s->avctx, AV_LOG_ERROR, " VQA Video: get_buffer() failed\n");
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
568 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
569 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
570
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
571 vqa_decode_chunk(s);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
572
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
573 /* make the palette available on the way out */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
574 memcpy(s->frame.data[1], s->palette, PALETTE_COUNT * 4);
1585
6b224ca24033 revised palette API, courtesy of Roberto Togni (rtogni at freemail.it)
melanson
parents: 1506
diff changeset
575 s->frame.palette_has_changed = 1;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
576
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
577 *data_size = sizeof(AVFrame);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
578 *(AVFrame*)data = s->frame;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
579
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
580 /* report that the buffer was completely consumed */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
581 return buf_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
582 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
583
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
584 static int vqa_decode_end(AVCodecContext *avctx)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
585 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
586 VqaContext *s = (VqaContext *)avctx->priv_data;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
587
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
588 av_free(s->codebook);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
589 av_free(s->next_codebook_buffer);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
590 av_free(s->decode_buffer);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
591
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
592 if (s->frame.data[0])
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
593 avctx->release_buffer(avctx, &s->frame);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
594
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
595 return 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
596 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
597
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
598 AVCodec vqa_decoder = {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
599 "vqavideo",
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
600 CODEC_TYPE_VIDEO,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
601 CODEC_ID_WS_VQA,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
602 sizeof(VqaContext),
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
603 vqa_decode_init,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
604 NULL,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
605 vqa_decode_end,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
606 vqa_decode_frame,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
607 CODEC_CAP_DR1,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
608 };