annotate vqavideo.c @ 12530:63edd10ad4bc libavcodec tip

Try to fix crashes introduced by r25218 r25218 made assumptions about the existence of past reference frames that weren't necessarily true.
author darkshikari
date Tue, 28 Sep 2010 09:06:22 +0000
parents ffb3668ff7af
children
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 /**
11644
7dd2a45249a9 Remove explicit filename from Doxygen @file commands.
diego
parents: 11560
diff changeset
23 * @file
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
24 * VQA Video Decoder by Mike Melanson (melanson@pcisys.net)
4243
8d1744586070 Correct information in header
kostya
parents: 3947
diff changeset
25 * For more information about the VQA format, visit:
8d1744586070 Correct information in header
kostya
parents: 3947
diff changeset
26 * http://wiki.multimedia.cx/index.php?title=VQA
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
27 *
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
28 * 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
29 * on the type of data in the file.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
30 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
31 * This decoder needs the 42-byte VQHD header from the beginning
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
32 * of the VQA file passed through the extradata field. The VQHD header
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
33 * is laid out as:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
34 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
35 * bytes 0-3 chunk fourcc: 'VQHD'
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
36 * 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
37 * bytes 8-49 VQHD chunk data
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
38 *
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
39 * Bytes 8-49 are what this decoder expects to see.
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
40 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
41 * 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
42 * VGA palettized colorspace. It operates on pixel vectors (blocks)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
43 * 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
44 * codebooks, palette information, and code maps for rendering vectors onto
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
45 * 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
46 * encoding (RLE) algorithm commonly referred to as "format80".
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
47 *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
48 * 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
49 * (usually, n = 8) relies on a different vector codebook. Rather than
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
50 * transporting an entire codebook every 8th frame, the new codebook is
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
51 * 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
52 * 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
53 * 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
54 * file. This is an interesting technique, although it makes random file
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
55 * seeking difficult despite the fact that the frames are all intracoded.
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
56 *
6903
0f63fc62ea8b consistency cosmetics: indices --> indexes
diego
parents: 6722
diff changeset
57 * V1,2 VQA uses 12-bit codebook indexes. If the 12-bit indexes were
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
58 * packed into bytes and then RLE compressed, bytewise, the results would
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
59 * 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
60 * 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
61 * 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
62 * 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
63 * should be the same.
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
66 #include <stdio.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
67 #include <stdlib.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
68 #include <string.h>
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
69
8573
2acf0ae7b041 Fix build: Add intreadwrite.h and bswap.h #includes where necessary.
diego
parents: 7040
diff changeset
70 #include "libavutil/intreadwrite.h"
12372
914f484bb476 Remove use of the deprecated function avcodec_check_dimensions(), use
stefano
parents: 11644
diff changeset
71 #include "libavcore/imgutils.h"
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
72 #include "avcodec.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 AVFrame frame;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
105
6295
michael
parents: 5215
diff changeset
106 const unsigned char *buf;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
107 int size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
108
3799
81638b2fbeba palette (if we memcpy it into AVFrame) must be uint32_t
michael
parents: 3036
diff changeset
109 uint32_t palette[PALETTE_COUNT];
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
110
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
111 int width; /* width of a frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
112 int height; /* height of a frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
113 int vector_width; /* width of individual vector */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
114 int vector_height; /* height of individual vector */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
115 int vqa_version; /* this should be either 1, 2 or 3 */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
116
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
117 unsigned char *codebook; /* the current codebook */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
118 int codebook_size;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
119 unsigned char *next_codebook_buffer; /* accumulator for next codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
120 int next_codebook_buffer_index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
121
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
122 unsigned char *decode_buffer;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
123 int decode_buffer_size;
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 /* number of frames to go before replacing codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
126 int partial_countdown;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
127 int partial_count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
128
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
129 } VqaContext;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
130
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6484
diff changeset
131 static av_cold int vqa_decode_init(AVCodecContext *avctx)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
132 {
4827
b3ee9a1526b0 Get rid of unnecessary pointer casts.
diego
parents: 4801
diff changeset
133 VqaContext *s = avctx->priv_data;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
134 unsigned char *vqa_header;
6377
michael
parents: 6295
diff changeset
135 int i, j, codebook_index;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
136
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
137 s->avctx = avctx;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
138 avctx->pix_fmt = PIX_FMT_PAL8;
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 /* make sure the extradata made it */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
141 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
142 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
143 return -1;
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
146 /* load up the VQA parameters from the header */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
147 vqa_header = (unsigned char *)s->avctx->extradata;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
148 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
149 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
150 s->height = AV_RL16(&vqa_header[8]);
12462
ffb3668ff7af Use new imgutils.h API names, fix deprecation warnings.
stefano
parents: 12372
diff changeset
151 if(av_image_check_size(s->width, s->height, 0, avctx)){
2422
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
152 s->width= s->height= 0;
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
153 return -1;
18b8b2dcc037 various security fixes and precautionary checks
michael
parents: 1881
diff changeset
154 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
155 s->vector_width = vqa_header[10];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
156 s->vector_height = vqa_header[11];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
157 s->partial_count = s->partial_countdown = vqa_header[13];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
158
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
159 /* the vector dimensions have to meet very stringent requirements */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
160 if ((s->vector_width != 4) ||
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
161 ((s->vector_height != 2) && (s->vector_height != 4))) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
162 /* return without further initialization */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
163 return -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
164 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
165
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
166 /* allocate codebooks */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
167 s->codebook_size = MAX_CODEBOOK_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
168 s->codebook = av_malloc(s->codebook_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
169 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
170
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
171 /* initialize the solid-color vectors */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
172 if (s->vector_height == 4) {
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
173 codebook_index = 0xFF00 * 16;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
174 for (i = 0; i < 256; i++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
175 for (j = 0; j < 16; j++)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
176 s->codebook[codebook_index++] = i;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
177 } else {
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
178 codebook_index = 0xF00 * 8;
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 < 8; 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 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
183 s->next_codebook_buffer_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
184
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
185 /* allocate decode buffer */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
186 s->decode_buffer_size = (s->width / s->vector_width) *
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
187 (s->height / s->vector_height) * 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
188 s->decode_buffer = av_malloc(s->decode_buffer_size);
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 s->frame.data[0] = NULL;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
191
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
192 return 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
193 }
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 #define CHECK_COUNT() \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
196 if (dest_index + count > dest_size) { \
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
197 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
198 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
199 dest_index, count, dest_size); \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
200 return; \
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
201 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
202
6295
michael
parents: 5215
diff changeset
203 static void decode_format80(const unsigned char *src, int src_size,
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
204 unsigned char *dest, int dest_size, int check_size) {
1496
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 int src_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
207 int dest_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
208 int count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
209 int src_pos;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
210 unsigned char color;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
211 int i;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
212
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
213 while (src_index < src_size) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
214
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
215 vqa_debug(" opcode %02X: ", src[src_index]);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
216
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
217 /* 0x80 means that frame is finished */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
218 if (src[src_index] == 0x80)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
219 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
220
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
221 if (dest_index >= dest_size) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
222 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
223 dest_index, dest_size);
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
227 if (src[src_index] == 0xFF) {
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 src_index++;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
230 count = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
231 src_index += 2;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
232 src_pos = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
233 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
234 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
235 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
236 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
237 dest[dest_index + i] = dest[src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
238 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
239
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
240 } else if (src[src_index] == 0xFE) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
241
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
242 src_index++;
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
243 count = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
244 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
245 color = src[src_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
246 vqa_debug("(2) set %X bytes to %02X\n", count, color);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
247 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
248 memset(&dest[dest_index], color, count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
249 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
250
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
251 } else if ((src[src_index] & 0xC0) == 0xC0) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
252
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
253 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
254 src_pos = AV_RL16(&src[src_index]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
255 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
256 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
257 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
258 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
259 dest[dest_index + i] = dest[src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
260 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
261
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
262 } else if (src[src_index] > 0x80) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
263
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
264 count = src[src_index++] & 0x3F;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
265 vqa_debug("(4) copy %X bytes from source to dest\n", count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
266 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
267 memcpy(&dest[dest_index], &src[src_index], count);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
268 src_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
269 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
270
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
271 } else {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
272
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
273 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
274 src_pos = AV_RB16(&src[src_index]) & 0x0FFF;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
275 src_index += 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
276 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
277 CHECK_COUNT();
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
278 for (i = 0; i < count; i++)
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
279 dest[dest_index + i] = dest[dest_index - src_pos + i];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
280 dest_index += count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
281 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
282 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
283
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
284 /* 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
285 * 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
286 * 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
287 * not every entry needs to be filled */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
288 if (check_size)
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
289 if (dest_index < dest_size)
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
290 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
291 dest_index, dest_size);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
292 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
293
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
294 static void vqa_decode_chunk(VqaContext *s)
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 unsigned int chunk_type;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
297 unsigned int chunk_size;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
298 int byte_skip;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
299 unsigned int index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
300 int i;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
301 unsigned char r, g, b;
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
302 int index_shift;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
303
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
304 int cbf0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
305 int cbfz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
306 int cbp0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
307 int cbpz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
308 int cpl0_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
309 int cplz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
310 int vptz_chunk = -1;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
311
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
312 int x, y;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
313 int lines = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
314 int pixel_ptr;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
315 int vector_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
316 int lobyte = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
317 int hibyte = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
318 int lobytes = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
319 int hibytes = s->decode_buffer_size / 2;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
320
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
321 /* first, traverse through the frame and find the subchunks */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
322 while (index < s->size) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
323
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
324 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
325 chunk_size = AV_RB32(&s->buf[index + 4]);
1496
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 switch (chunk_type) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
328
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
329 case CBF0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
330 cbf0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
331 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
332
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
333 case CBFZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
334 cbfz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
335 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
336
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
337 case CBP0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
338 cbp0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
339 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
340
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
341 case CBPZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
342 cbpz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
343 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
344
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
345 case CPL0_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
346 cpl0_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
347 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
348
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
349 case CPLZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
350 cplz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
351 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
352
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
353 case VPTZ_TAG:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
354 vptz_chunk = index;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
355 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
356
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
357 default:
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
358 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
359 (chunk_type >> 24) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
360 (chunk_type >> 16) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
361 (chunk_type >> 8) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
362 (chunk_type >> 0) & 0xFF,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
363 chunk_type);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
364 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
365 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
366
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
367 byte_skip = chunk_size & 0x01;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
368 index += (CHUNK_PREAMBLE_SIZE + chunk_size + byte_skip);
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
371 /* next, deal with the palette */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
372 if ((cpl0_chunk != -1) && (cplz_chunk != -1)) {
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 /* 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
375 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
376 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
377 }
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 /* decompress the palette chunk */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
380 if (cplz_chunk != -1) {
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 /* yet to be handled */
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 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
385
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
386 /* convert the RGB palette into the machine's endian format */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
387 if (cpl0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
388
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
389 chunk_size = AV_RB32(&s->buf[cpl0_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
390 /* sanity check the palette size */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
391 if (chunk_size / 3 > 256) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
392 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
393 chunk_size / 3);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
394 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
395 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
396 cpl0_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
397 for (i = 0; i < chunk_size / 3; i++) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
398 /* scale by 4 to transform 6-bit palette -> 8-bit */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
399 r = s->buf[cpl0_chunk++] * 4;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
400 g = s->buf[cpl0_chunk++] * 4;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
401 b = s->buf[cpl0_chunk++] * 4;
1585
6b224ca24033 revised palette API, courtesy of Roberto Togni (rtogni at freemail.it)
melanson
parents: 1506
diff changeset
402 s->palette[i] = (r << 16) | (g << 8) | (b);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
403 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
404 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
405
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
406 /* next, look for a full codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
407 if ((cbf0_chunk != -1) && (cbfz_chunk != -1)) {
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 /* 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
410 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
411 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
412 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
413
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
414 /* decompress the full codebook chunk */
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
415 if (cbfz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
416
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
417 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
418 cbfz_chunk += CHUNK_PREAMBLE_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
419 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
420 s->codebook, s->codebook_size, 0);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
421 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
422
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
423 /* copy a full codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
424 if (cbf0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
425
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
426 chunk_size = AV_RB32(&s->buf[cbf0_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
427 /* sanity check the full codebook size */
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
428 if (chunk_size > MAX_CODEBOOK_SIZE) {
1598
932d306bf1dc av_log() patch by (Michel Bardiaux <mbardiaux at peaktime dot be>)
michael
parents: 1585
diff changeset
429 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
430 chunk_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
431 return;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
432 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
433 cbf0_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
434
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
435 memcpy(s->codebook, &s->buf[cbf0_chunk], chunk_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
436 }
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 /* decode the frame */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
439 if (vptz_chunk == -1) {
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 /* 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
442 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
443 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
444 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
445
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
446 chunk_size = AV_RB32(&s->buf[vptz_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
447 vptz_chunk += CHUNK_PREAMBLE_SIZE;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
448 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
449 s->decode_buffer, s->decode_buffer_size, 1);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
450
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
451 /* render the final PAL8 frame */
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
452 if (s->vector_height == 4)
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
453 index_shift = 4;
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
454 else
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
455 index_shift = 3;
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
456 for (y = 0; y < s->frame.linesize[0] * s->height;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
457 y += s->frame.linesize[0] * s->vector_height) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
458
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
459 for (x = y; x < y + s->width; x += 4, lobytes++, hibytes++) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
460 pixel_ptr = x;
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 /* get the vector index, the method for which varies according to
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
463 * VQA file version */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
464 switch (s->vqa_version) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
465
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
466 case 1:
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
467 /* 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
468 * Kyrandia III : Malcolm's Revenge", is known to use this version) */
4245
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
469 lobyte = s->decode_buffer[lobytes * 2];
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
470 hibyte = s->decode_buffer[(lobytes * 2) + 1];
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
471 vector_index = ((hibyte << 8) | lobyte) >> 3;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
472 vector_index <<= index_shift;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
473 lines = s->vector_height;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
474 /* uniform color fill - a quick hack */
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
475 if (hibyte == 0xFF) {
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
476 while (lines--) {
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
477 s->frame.data[0][pixel_ptr + 0] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
478 s->frame.data[0][pixel_ptr + 1] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
479 s->frame.data[0][pixel_ptr + 2] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
480 s->frame.data[0][pixel_ptr + 3] = 255 - lobyte;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
481 pixel_ptr += s->frame.linesize[0];
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
482 }
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
483 lines=0;
4397b7436222 VQA v1 support
kostya
parents: 4243
diff changeset
484 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
485 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
486
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
487 case 2:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
488 lobyte = s->decode_buffer[lobytes];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
489 hibyte = s->decode_buffer[hibytes];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
490 vector_index = (hibyte << 8) | lobyte;
1506
e0c02032d0f2 support a few more types of VQA files
melanson
parents: 1501
diff changeset
491 vector_index <<= index_shift;
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
492 lines = s->vector_height;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
493 break;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
494
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
495 case 3:
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
496 /* not implemented yet */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
497 lines = 0;
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
501 while (lines--) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
502 s->frame.data[0][pixel_ptr + 0] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
503 s->frame.data[0][pixel_ptr + 1] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
504 s->frame.data[0][pixel_ptr + 2] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
505 s->frame.data[0][pixel_ptr + 3] = s->codebook[vector_index++];
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
506 pixel_ptr += s->frame.linesize[0];
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 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
509 }
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 /* handle partial codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
512 if ((cbp0_chunk != -1) && (cbpz_chunk != -1)) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
513 /* 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
514 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
515 return;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
516 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
517
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
518 if (cbp0_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
519
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
520 chunk_size = AV_RB32(&s->buf[cbp0_chunk + 4]);
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
521 cbp0_chunk += CHUNK_PREAMBLE_SIZE;
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 /* accumulate partial codebook */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
524 memcpy(&s->next_codebook_buffer[s->next_codebook_buffer_index],
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
525 &s->buf[cbp0_chunk], chunk_size);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
526 s->next_codebook_buffer_index += chunk_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 s->partial_countdown--;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
529 if (s->partial_countdown == 0) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
530
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
531 /* time to replace codebook */
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
532 memcpy(s->codebook, s->next_codebook_buffer,
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
533 s->next_codebook_buffer_index);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
534
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
535 /* reset accounting */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
536 s->next_codebook_buffer_index = 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
537 s->partial_countdown = s->partial_count;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
538 }
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
541 if (cbpz_chunk != -1) {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
542
4364
05e932ddaaa9 rename BE/LE_8/16/32 to AV_RL/B_8/16/32
alex
parents: 4245
diff changeset
543 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
544 cbpz_chunk += CHUNK_PREAMBLE_SIZE;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
545
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
546 /* accumulate partial codebook */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
547 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
548 &s->buf[cbpz_chunk], chunk_size);
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
549 s->next_codebook_buffer_index += chunk_size;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
550
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
551 s->partial_countdown--;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
552 if (s->partial_countdown == 0) {
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
553
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
554 /* decompress codebook */
2967
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
555 decode_format80(s->next_codebook_buffer,
ef2149182f1c COSMETICS: Remove all trailing whitespace.
diego
parents: 2422
diff changeset
556 s->next_codebook_buffer_index,
1501
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
557 s->codebook, s->codebook_size, 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 /* reset accounting */
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
560 s->next_codebook_buffer_index = 0;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
561 s->partial_countdown = s->partial_count;
dc6f46cd4a7b added solid color vectors; basic PAL8, 4x2-vector video (as in
tmmm
parents: 1496
diff changeset
562 }
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
563 }
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
564 }
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 static int vqa_decode_frame(AVCodecContext *avctx,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
567 void *data, int *data_size,
9355
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 8718
diff changeset
568 AVPacket *avpkt)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
569 {
9355
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 8718
diff changeset
570 const uint8_t *buf = avpkt->data;
54bc8a2727b0 Implement avcodec_decode_video2(), _audio3() and _subtitle2() which takes an
rbultje
parents: 8718
diff changeset
571 int buf_size = avpkt->size;
4827
b3ee9a1526b0 Get rid of unnecessary pointer casts.
diego
parents: 4801
diff changeset
572 VqaContext *s = avctx->priv_data;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
573
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
574 s->buf = buf;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
575 s->size = buf_size;
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 if (s->frame.data[0])
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
578 avctx->release_buffer(avctx, &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 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
581 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
582 return -1;
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
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
585 vqa_decode_chunk(s);
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 /* make the palette available on the way out */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
588 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
589 s->frame.palette_has_changed = 1;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
590
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
591 *data_size = sizeof(AVFrame);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
592 *(AVFrame*)data = s->frame;
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 /* report that the buffer was completely consumed */
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
595 return buf_size;
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
6517
48759bfbd073 Apply 'cold' attribute to init/uninit functions in libavcodec
zuxy
parents: 6484
diff changeset
598 static av_cold int vqa_decode_end(AVCodecContext *avctx)
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
599 {
4827
b3ee9a1526b0 Get rid of unnecessary pointer casts.
diego
parents: 4801
diff changeset
600 VqaContext *s = avctx->priv_data;
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
601
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
602 av_free(s->codebook);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
603 av_free(s->next_codebook_buffer);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
604 av_free(s->decode_buffer);
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
605
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
606 if (s->frame.data[0])
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
607 avctx->release_buffer(avctx, &s->frame);
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 return 0;
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
610 }
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 AVCodec vqa_decoder = {
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
613 "vqavideo",
11560
8a4984c5cacc Define AVMediaType enum, and use it instead of enum CodecType, which
stefano
parents: 9981
diff changeset
614 AVMEDIA_TYPE_VIDEO,
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
615 CODEC_ID_WS_VQA,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
616 sizeof(VqaContext),
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
617 vqa_decode_init,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
618 NULL,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
619 vqa_decode_end,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
620 vqa_decode_frame,
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
621 CODEC_CAP_DR1,
7040
e943e1409077 Make AVCodec long_names definition conditional depending on CONFIG_SMALL.
stefano
parents: 6903
diff changeset
622 .long_name = NULL_IF_CONFIG_SMALL("Westwood Studios VQA (Vector Quantized Animation) video"),
1496
b78a9ba6a568 first pass at a VQA video decoder
tmmm
parents:
diff changeset
623 };