annotate vqavideo.c @ 4765:85298e8c55c4 libavcodec

bfin dsputils, basic pixel operations sads, diffs, motion compensation and standard IEEE 8x8 block transforms patch by Marc Hoffman, mmh pleasantst com
author diego
date Sun, 01 Apr 2007 22:28:45 +0000
parents 05e932ddaaa9
children 66ef3690d108
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 *
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3799
diff changeset
5 * This file is part of FFmpeg.
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3799
diff changeset
6 *
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3799
diff changeset
7 * FFmpeg is free software; you can redistribute it and/or
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
8 * modify it under the terms of the GNU Lesser General Public
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
9 * License as published by the Free Software Foundation; either
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3799
diff changeset
10 * version 2.1 of the License, or (at your option) any later version.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
11 *
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3799
diff changeset
12 * FFmpeg is distributed in the hope that it will be useful,
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
15 * Lesser General Public License for more details.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
16 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
17 * You should have received a copy of the GNU Lesser General Public
3947
c8c591fe26f8 Change license headers to say 'FFmpeg' instead of 'this program/this library'
diego
parents: 3799
diff changeset
18 * License along with FFmpeg; if not, write to the Free Software
3036
0b546eab515d Update licensing information: The FSF changed postal address.
diego
parents: 2967
diff changeset
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
1496
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
23 /**
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
24 * @file vqavideo.c
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
25 * VQA Video Decoder by Mike Melanson (melanson@pcisys.net)
4243
8d1744586070 Correct information in header
kostya
parents: 3947
diff changeset
26 * For more information about the VQA format, visit:
8d1744586070 Correct information in header
kostya
parents: 3947
diff changeset
27 * http://wiki.multimedia.cx/index.php?title=VQA
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
28 *
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
29 * 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
30 * on the type of data in the file.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
31 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
32 * This decoder needs the 42-byte VQHD header from the beginning
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
33 * of the VQA file passed through the extradata field. The VQHD header
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
34 * is laid out as:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
35 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
36 * bytes 0-3 chunk fourcc: 'VQHD'
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
37 * 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
38 * bytes 8-49 VQHD chunk data
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
39 *
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
40 * Bytes 8-49 are what this decoder expects to see.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
41 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
42 * 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
43 * VGA palettized colorspace. It operates on pixel vectors (blocks)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
44 * 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
45 * codebooks, palette information, and code maps for rendering vectors onto
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
46 * 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
47 * encoding (RLE) algorithm commonly referred to as "format80".
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
48 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
49 * 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
50 * (usually, n = 8) relies on a different vector codebook. Rather than
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
51 * transporting an entire codebook every 8th frame, the new codebook is
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
52 * 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
53 * 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
54 * 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
55 * file. This is an interesting technique, although it makes random file
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
56 * seeking difficult despite the fact that the frames are all intracoded.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
57 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
58 * 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
59 * packed into bytes and then RLE compressed, bytewise, the results would
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
60 * 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
61 * 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
62 * 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
63 * 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
64 * should be the same.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
65 */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
66
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
67 #include <stdio.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
68 #include <stdlib.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
69 #include <string.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
70 #include <unistd.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
71
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
72 #include "common.h"
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
73 #include "avcodec.h"
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
74 #include "dsputil.h"
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
75
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
76 #define PALETTE_COUNT 256
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
77 #define VQA_HEADER_SIZE 0x2A
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
78 #define CHUNK_PREAMBLE_SIZE 8
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
79
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
80 /* 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
81 * (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
82 #define MAX_CODEBOOK_VECTORS 0xFF00
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
83 #define SOLID_PIXEL_VECTORS 0x100
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
84 #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
85 #define MAX_CODEBOOK_SIZE (MAX_VECTORS * 4 * 4)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
86
1881
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
87 #define CBF0_TAG MKBETAG('C', 'B', 'F', '0')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
88 #define CBFZ_TAG MKBETAG('C', 'B', 'F', 'Z')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
89 #define CBP0_TAG MKBETAG('C', 'B', 'P', '0')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
90 #define CBPZ_TAG MKBETAG('C', 'B', 'P', 'Z')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
91 #define CPL0_TAG MKBETAG('C', 'P', 'L', '0')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
92 #define CPLZ_TAG MKBETAG('C', 'P', 'L', 'Z')
39ad6cd5d4a6 remove numerous definitions of BE_*/LE_* macros; convert FOURCC_TAG ->
melanson
parents: 1598
diff changeset
93 #define VPTZ_TAG MKBETAG('V', 'P', 'T', 'Z')
1496
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 #define VQA_DEBUG 0
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
96
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
97 #if VQA_DEBUG
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
98 #define vqa_debug printf
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
99 #else
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
100 static inline void vqa_debug(const char *format, ...) { }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
101 #endif
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 typedef struct VqaContext {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
104
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
105 AVCodecContext *avctx;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
106 DSPContext dsp;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
107 AVFrame frame;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
108
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
109 unsigned char *buf;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
110 int size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
111
3799
81638b2fbeba palette (if we memcpy it into AVFrame) must be uint32_t
michael
parents: 3036
diff changeset
112 uint32_t palette[PALETTE_COUNT];
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
113
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
114 int width; /* width of a frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
115 int height; /* height of a frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
116 int vector_width; /* width of individual vector */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
117 int vector_height; /* height of individual vector */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
118 int vqa_version; /* this should be either 1, 2 or 3 */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
119
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
120 unsigned char *codebook; /* the current codebook */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
121 int codebook_size;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
122 unsigned char *next_codebook_buffer; /* accumulator for next codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
123 int next_codebook_buffer_index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
124
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
125 unsigned char *decode_buffer;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
126 int decode_buffer_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
127
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
128 /* number of frames to go before replacing codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
129 int partial_countdown;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
130 int partial_count;
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 } VqaContext;
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 static int vqa_decode_init(AVCodecContext *avctx)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
135 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
136 VqaContext *s = (VqaContext *)avctx->priv_data;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
137 unsigned char *vqa_header;
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
138 int i, j, codebook_index;;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
139
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
140 s->avctx = avctx;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
141 avctx->pix_fmt = PIX_FMT_PAL8;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
142 avctx->has_b_frames = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
143 dsputil_init(&s->dsp, avctx);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
144
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
145 /* make sure the extradata made it */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
146 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
147 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
148 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
149 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
150
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
151 /* load up the VQA parameters from the header */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
152 vqa_header = (unsigned char *)s->avctx->extradata;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
153 s->vqa_version = vqa_header[0];
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
154 s->width = AV_RL16(&vqa_header[6]);
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
155 s->height = AV_RL16(&vqa_header[8]);
2422
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
156 if(avcodec_check_dimensions(avctx, s->width, s->height)){
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
157 s->width= s->height= 0;
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
158 return -1;
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
159 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
160 s->vector_width = vqa_header[10];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
161 s->vector_height = vqa_header[11];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
162 s->partial_count = s->partial_countdown = vqa_header[13];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
163
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
164 /* the vector dimensions have to meet very stringent requirements */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
165 if ((s->vector_width != 4) ||
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
166 ((s->vector_height != 2) && (s->vector_height != 4))) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
167 /* return without further initialization */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
168 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
169 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
170
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
171 /* allocate codebooks */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
172 s->codebook_size = MAX_CODEBOOK_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
173 s->codebook = av_malloc(s->codebook_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
174 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
175
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
176 /* initialize the solid-color vectors */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
177 if (s->vector_height == 4) {
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
178 codebook_index = 0xFF00 * 16;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
179 for (i = 0; i < 256; i++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
180 for (j = 0; j < 16; j++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
181 s->codebook[codebook_index++] = i;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
182 } else {
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
183 codebook_index = 0xF00 * 8;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
184 for (i = 0; i < 256; i++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
185 for (j = 0; j < 8; j++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
186 s->codebook[codebook_index++] = i;
1496
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 s->next_codebook_buffer_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
189
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
190 /* allocate decode buffer */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
191 s->decode_buffer_size = (s->width / s->vector_width) *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
192 (s->height / s->vector_height) * 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
193 s->decode_buffer = av_malloc(s->decode_buffer_size);
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 s->frame.data[0] = NULL;
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 return 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
198 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
199
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
200 #define CHECK_COUNT() \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
201 if (dest_index + count > dest_size) { \
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
202 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
203 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
204 dest_index, count, dest_size); \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
205 return; \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
206 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
207
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
208 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
209 unsigned char *dest, int dest_size, int check_size) {
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
210
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
211 int src_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
212 int dest_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
213 int count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
214 int src_pos;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
215 unsigned char color;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
216 int i;
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 while (src_index < src_size) {
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 vqa_debug(" opcode %02X: ", src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
221
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
222 /* 0x80 means that frame is finished */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
223 if (src[src_index] == 0x80)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
224 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
225
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
226 if (dest_index >= dest_size) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
227 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
228 dest_index, dest_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
229 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
230 }
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 if (src[src_index] == 0xFF) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
233
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
234 src_index++;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
235 count = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
236 src_index += 2;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
237 src_pos = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
238 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
239 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
240 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
241 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
242 dest[dest_index + i] = dest[src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
243 dest_index += count;
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 } else if (src[src_index] == 0xFE) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
246
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
247 src_index++;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
248 count = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
249 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
250 color = src[src_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
251 vqa_debug("(2) set %X bytes to %02X\n", count, color);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
252 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
253 memset(&dest[dest_index], color, count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
254 dest_index += count;
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 } else if ((src[src_index] & 0xC0) == 0xC0) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
257
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
258 count = (src[src_index++] & 0x3F) + 3;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
259 src_pos = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
260 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
261 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
262 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
263 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
264 dest[dest_index + i] = dest[src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
265 dest_index += count;
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 } else if (src[src_index] > 0x80) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
268
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
269 count = src[src_index++] & 0x3F;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
270 vqa_debug("(4) copy %X bytes from source to dest\n", count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
271 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
272 memcpy(&dest[dest_index], &src[src_index], count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
273 src_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
274 dest_index += count;
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 } else {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
277
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
278 count = ((src[src_index] & 0x70) >> 4) + 3;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
279 src_pos = AV_RB16(&src[src_index]) & 0x0FFF;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
280 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
281 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
282 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
283 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
284 dest[dest_index + i] = dest[dest_index - src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
285 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
286 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
287 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
288
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
289 /* 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
290 * 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
291 * 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
292 * not every entry needs to be filled */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
293 if (check_size)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
294 if (dest_index < dest_size)
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
295 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
296 dest_index, dest_size);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
297 }
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 static void vqa_decode_chunk(VqaContext *s)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
300 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
301 unsigned int chunk_type;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
302 unsigned int chunk_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
303 int byte_skip;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
304 unsigned int index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
305 int i;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
306 unsigned char r, g, b;
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
307 int index_shift;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
308
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
309 int cbf0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
310 int cbfz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
311 int cbp0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
312 int cbpz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
313 int cpl0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
314 int cplz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
315 int vptz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
316
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
317 int x, y;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
318 int lines = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
319 int pixel_ptr;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
320 int vector_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
321 int lobyte = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
322 int hibyte = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
323 int lobytes = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
324 int hibytes = s->decode_buffer_size / 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
325
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
326 /* first, traverse through the frame and find the subchunks */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
327 while (index < s->size) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
328
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
329 chunk_type = AV_RB32(&s->buf[index]);
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
330 chunk_size = AV_RB32(&s->buf[index + 4]);
1496
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 switch (chunk_type) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
333
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
334 case CBF0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
335 cbf0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
336 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
337
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
338 case CBFZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
339 cbfz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
340 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
341
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
342 case CBP0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
343 cbp0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
344 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
345
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
346 case CBPZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
347 cbpz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
348 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
349
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
350 case CPL0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
351 cpl0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
352 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
353
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
354 case CPLZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
355 cplz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
356 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
357
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
358 case VPTZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
359 vptz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
360 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
361
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
362 default:
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
363 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
364 (chunk_type >> 24) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
365 (chunk_type >> 16) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
366 (chunk_type >> 8) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
367 (chunk_type >> 0) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
368 chunk_type);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
369 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
370 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
371
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
372 byte_skip = chunk_size & 0x01;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
373 index += (CHUNK_PREAMBLE_SIZE + chunk_size + byte_skip);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
374 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
375
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
376 /* next, deal with the palette */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
377 if ((cpl0_chunk != -1) && (cplz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
378
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
379 /* 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
380 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
381 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
382 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
383
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
384 /* decompress the palette chunk */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
385 if (cplz_chunk != -1) {
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 /* yet to be handled */
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 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
390
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
391 /* convert the RGB palette into the machine's endian format */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
392 if (cpl0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
393
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
394 chunk_size = AV_RB32(&s->buf[cpl0_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
395 /* sanity check the palette size */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
396 if (chunk_size / 3 > 256) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
397 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
398 chunk_size / 3);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
399 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
400 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
401 cpl0_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
402 for (i = 0; i < chunk_size / 3; i++) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
403 /* scale by 4 to transform 6-bit palette -> 8-bit */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
404 r = s->buf[cpl0_chunk++] * 4;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
405 g = s->buf[cpl0_chunk++] * 4;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
406 b = s->buf[cpl0_chunk++] * 4;
1585
6b224ca24033 revised palette API, courtesy of Roberto Togni (rtogni at freemail.it)
melanson
parents: 1506
diff changeset
407 s->palette[i] = (r << 16) | (g << 8) | (b);
1496
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 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
410
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
411 /* next, look for a full codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
412 if ((cbf0_chunk != -1) && (cbfz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
413
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
414 /* 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
415 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
416 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
417 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
418
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
419 /* decompress the full codebook chunk */
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
420 if (cbfz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
421
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
422 chunk_size = AV_RB32(&s->buf[cbfz_chunk + 4]);
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
423 cbfz_chunk += CHUNK_PREAMBLE_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
424 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
425 s->codebook, s->codebook_size, 0);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
426 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
427
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
428 /* copy a full codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
429 if (cbf0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
430
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
431 chunk_size = AV_RB32(&s->buf[cbf0_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
432 /* sanity check the full codebook size */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
433 if (chunk_size > MAX_CODEBOOK_SIZE) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
434 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
435 chunk_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
436 return;
1496
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 cbf0_chunk += CHUNK_PREAMBLE_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 memcpy(s->codebook, &s->buf[cbf0_chunk], chunk_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
441 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
442
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
443 /* decode the frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
444 if (vptz_chunk == -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
445
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
446 /* 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
447 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
448 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
449 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
450
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
451 chunk_size = AV_RB32(&s->buf[vptz_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
452 vptz_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
453 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
454 s->decode_buffer, s->decode_buffer_size, 1);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
455
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
456 /* render the final PAL8 frame */
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
457 if (s->vector_height == 4)
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
458 index_shift = 4;
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
459 else
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
460 index_shift = 3;
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
461 for (y = 0; y < s->frame.linesize[0] * s->height;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
462 y += s->frame.linesize[0] * s->vector_height) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
463
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
464 for (x = y; x < y + s->width; x += 4, lobytes++, hibytes++) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
465 pixel_ptr = x;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
466
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
467 /* get the vector index, the method for which varies according to
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
468 * VQA file version */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
469 switch (s->vqa_version) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
470
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
471 case 1:
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
472 /* still need sample media for this case (only one game, "Legend of
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
473 * Kyrandia III : Malcolm's Revenge", is known to use this version) */
4245
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
474 lobyte = s->decode_buffer[lobytes * 2];
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
475 hibyte = s->decode_buffer[(lobytes * 2) + 1];
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
476 vector_index = ((hibyte << 8) | lobyte) >> 3;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
477 vector_index <<= index_shift;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
478 lines = s->vector_height;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
479 /* uniform color fill - a quick hack */
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
480 if (hibyte == 0xFF) {
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
481 while (lines--) {
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
482 s->frame.data[0][pixel_ptr + 0] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
483 s->frame.data[0][pixel_ptr + 1] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
484 s->frame.data[0][pixel_ptr + 2] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
485 s->frame.data[0][pixel_ptr + 3] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
486 pixel_ptr += s->frame.linesize[0];
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
487 }
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
488 lines=0;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
489 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
490 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
491
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
492 case 2:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
493 lobyte = s->decode_buffer[lobytes];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
494 hibyte = s->decode_buffer[hibytes];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
495 vector_index = (hibyte << 8) | lobyte;
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
496 vector_index <<= index_shift;
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
497 lines = s->vector_height;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
498 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
499
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
500 case 3:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
501 /* not implemented yet */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
502 lines = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
503 break;
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 while (lines--) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
507 s->frame.data[0][pixel_ptr + 0] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
508 s->frame.data[0][pixel_ptr + 1] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
509 s->frame.data[0][pixel_ptr + 2] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
510 s->frame.data[0][pixel_ptr + 3] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
511 pixel_ptr += s->frame.linesize[0];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
512 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
513 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
514 }
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 /* handle partial codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
517 if ((cbp0_chunk != -1) && (cbpz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
518 /* 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
519 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
520 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
521 }
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 if (cbp0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
524
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
525 chunk_size = AV_RB32(&s->buf[cbp0_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
526 cbp0_chunk += CHUNK_PREAMBLE_SIZE;
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 /* accumulate partial codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
529 memcpy(&s->next_codebook_buffer[s->next_codebook_buffer_index],
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
530 &s->buf[cbp0_chunk], chunk_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
531 s->next_codebook_buffer_index += chunk_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
532
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
533 s->partial_countdown--;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
534 if (s->partial_countdown == 0) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
535
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
536 /* time to replace codebook */
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
537 memcpy(s->codebook, s->next_codebook_buffer,
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
538 s->next_codebook_buffer_index);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
539
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
540 /* reset accounting */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
541 s->next_codebook_buffer_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
542 s->partial_countdown = s->partial_count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
543 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
544 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
545
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
546 if (cbpz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
547
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
548 chunk_size = AV_RB32(&s->buf[cbpz_chunk + 4]);
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
549 cbpz_chunk += CHUNK_PREAMBLE_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
550
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
551 /* accumulate partial codebook */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
552 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
553 &s->buf[cbpz_chunk], chunk_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
554 s->next_codebook_buffer_index += chunk_size;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
555
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
556 s->partial_countdown--;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
557 if (s->partial_countdown == 0) {
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
558
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
559 /* decompress codebook */
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
560 decode_format80(s->next_codebook_buffer,
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
561 s->next_codebook_buffer_index,
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
562 s->codebook, s->codebook_size, 0);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
563
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
564 /* reset accounting */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
565 s->next_codebook_buffer_index = 0;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
566 s->partial_countdown = s->partial_count;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
567 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
568 }
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 static int vqa_decode_frame(AVCodecContext *avctx,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
572 void *data, int *data_size,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
573 uint8_t *buf, int buf_size)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
574 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
575 VqaContext *s = (VqaContext *)avctx->priv_data;
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 s->buf = buf;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
578 s->size = buf_size;
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 if (s->frame.data[0])
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
581 avctx->release_buffer(avctx, &s->frame);
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 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
584 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
585 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
586 }
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 vqa_decode_chunk(s);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
589
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
590 /* make the palette available on the way out */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
591 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
592 s->frame.palette_has_changed = 1;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
593
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
594 *data_size = sizeof(AVFrame);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
595 *(AVFrame*)data = s->frame;
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 /* report that the buffer was completely consumed */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
598 return buf_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
599 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
600
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
601 static int vqa_decode_end(AVCodecContext *avctx)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
602 {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
603 VqaContext *s = (VqaContext *)avctx->priv_data;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
604
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
605 av_free(s->codebook);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
606 av_free(s->next_codebook_buffer);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
607 av_free(s->decode_buffer);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
608
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
609 if (s->frame.data[0])
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
610 avctx->release_buffer(avctx, &s->frame);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
611
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
612 return 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
613 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
614
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
615 AVCodec vqa_decoder = {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
616 "vqavideo",
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
617 CODEC_TYPE_VIDEO,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
618 CODEC_ID_WS_VQA,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
619 sizeof(VqaContext),
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
620 vqa_decode_init,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
621 NULL,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
622 vqa_decode_end,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
623 vqa_decode_frame,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
624 CODEC_CAP_DR1,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
625 };