1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Wing Commander/Xan Video Decoder 3cabdff1aSopenharmony_ci * Copyright (C) 2003 The FFmpeg project 4cabdff1aSopenharmony_ci * 5cabdff1aSopenharmony_ci * This file is part of FFmpeg. 6cabdff1aSopenharmony_ci * 7cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 8cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 9cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 10cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 11cabdff1aSopenharmony_ci * 12cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 13cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 14cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15cabdff1aSopenharmony_ci * Lesser General Public License for more details. 16cabdff1aSopenharmony_ci * 17cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 18cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 19cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 20cabdff1aSopenharmony_ci */ 21cabdff1aSopenharmony_ci 22cabdff1aSopenharmony_ci/** 23cabdff1aSopenharmony_ci * @file 24cabdff1aSopenharmony_ci * Xan video decoder for Wing Commander III computer game 25cabdff1aSopenharmony_ci * by Mario Brito (mbrito@student.dei.uc.pt) 26cabdff1aSopenharmony_ci * and Mike Melanson (melanson@pcisys.net) 27cabdff1aSopenharmony_ci * 28cabdff1aSopenharmony_ci * The xan_wc3 decoder outputs PAL8 data. 29cabdff1aSopenharmony_ci */ 30cabdff1aSopenharmony_ci 31cabdff1aSopenharmony_ci#include <stdio.h> 32cabdff1aSopenharmony_ci#include <stdlib.h> 33cabdff1aSopenharmony_ci#include <string.h> 34cabdff1aSopenharmony_ci 35cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 36cabdff1aSopenharmony_ci#include "libavutil/mem.h" 37cabdff1aSopenharmony_ci 38cabdff1aSopenharmony_ci#define BITSTREAM_READER_LE 39cabdff1aSopenharmony_ci#include "avcodec.h" 40cabdff1aSopenharmony_ci#include "bytestream.h" 41cabdff1aSopenharmony_ci#include "codec_internal.h" 42cabdff1aSopenharmony_ci#include "get_bits.h" 43cabdff1aSopenharmony_ci#include "internal.h" 44cabdff1aSopenharmony_ci 45cabdff1aSopenharmony_ci#define RUNTIME_GAMMA 0 46cabdff1aSopenharmony_ci 47cabdff1aSopenharmony_ci#define VGA__TAG MKTAG('V', 'G', 'A', ' ') 48cabdff1aSopenharmony_ci#define PALT_TAG MKTAG('P', 'A', 'L', 'T') 49cabdff1aSopenharmony_ci#define SHOT_TAG MKTAG('S', 'H', 'O', 'T') 50cabdff1aSopenharmony_ci#define PALETTE_COUNT 256 51cabdff1aSopenharmony_ci#define PALETTE_SIZE (PALETTE_COUNT * 3) 52cabdff1aSopenharmony_ci#define PALETTES_MAX 256 53cabdff1aSopenharmony_ci 54cabdff1aSopenharmony_citypedef struct XanContext { 55cabdff1aSopenharmony_ci 56cabdff1aSopenharmony_ci AVCodecContext *avctx; 57cabdff1aSopenharmony_ci AVFrame *last_frame; 58cabdff1aSopenharmony_ci 59cabdff1aSopenharmony_ci const uint8_t *buf; 60cabdff1aSopenharmony_ci int size; 61cabdff1aSopenharmony_ci 62cabdff1aSopenharmony_ci /* scratch space */ 63cabdff1aSopenharmony_ci uint8_t *buffer1; 64cabdff1aSopenharmony_ci int buffer1_size; 65cabdff1aSopenharmony_ci uint8_t *buffer2; 66cabdff1aSopenharmony_ci int buffer2_size; 67cabdff1aSopenharmony_ci 68cabdff1aSopenharmony_ci unsigned *palettes; 69cabdff1aSopenharmony_ci int palettes_count; 70cabdff1aSopenharmony_ci int cur_palette; 71cabdff1aSopenharmony_ci 72cabdff1aSopenharmony_ci int frame_size; 73cabdff1aSopenharmony_ci 74cabdff1aSopenharmony_ci} XanContext; 75cabdff1aSopenharmony_ci 76cabdff1aSopenharmony_cistatic av_cold int xan_decode_end(AVCodecContext *avctx) 77cabdff1aSopenharmony_ci{ 78cabdff1aSopenharmony_ci XanContext *s = avctx->priv_data; 79cabdff1aSopenharmony_ci 80cabdff1aSopenharmony_ci av_frame_free(&s->last_frame); 81cabdff1aSopenharmony_ci 82cabdff1aSopenharmony_ci av_freep(&s->buffer1); 83cabdff1aSopenharmony_ci av_freep(&s->buffer2); 84cabdff1aSopenharmony_ci av_freep(&s->palettes); 85cabdff1aSopenharmony_ci 86cabdff1aSopenharmony_ci return 0; 87cabdff1aSopenharmony_ci} 88cabdff1aSopenharmony_ci 89cabdff1aSopenharmony_cistatic av_cold int xan_decode_init(AVCodecContext *avctx) 90cabdff1aSopenharmony_ci{ 91cabdff1aSopenharmony_ci XanContext *s = avctx->priv_data; 92cabdff1aSopenharmony_ci 93cabdff1aSopenharmony_ci s->avctx = avctx; 94cabdff1aSopenharmony_ci s->frame_size = 0; 95cabdff1aSopenharmony_ci 96cabdff1aSopenharmony_ci avctx->pix_fmt = AV_PIX_FMT_PAL8; 97cabdff1aSopenharmony_ci 98cabdff1aSopenharmony_ci s->buffer1_size = avctx->width * avctx->height; 99cabdff1aSopenharmony_ci s->buffer1 = av_malloc(s->buffer1_size); 100cabdff1aSopenharmony_ci if (!s->buffer1) 101cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 102cabdff1aSopenharmony_ci s->buffer2_size = avctx->width * avctx->height; 103cabdff1aSopenharmony_ci s->buffer2 = av_malloc(s->buffer2_size + 130); 104cabdff1aSopenharmony_ci if (!s->buffer2) 105cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 106cabdff1aSopenharmony_ci 107cabdff1aSopenharmony_ci s->last_frame = av_frame_alloc(); 108cabdff1aSopenharmony_ci if (!s->last_frame) 109cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 110cabdff1aSopenharmony_ci 111cabdff1aSopenharmony_ci return 0; 112cabdff1aSopenharmony_ci} 113cabdff1aSopenharmony_ci 114cabdff1aSopenharmony_cistatic int xan_huffman_decode(uint8_t *dest, int dest_len, 115cabdff1aSopenharmony_ci const uint8_t *src, int src_len) 116cabdff1aSopenharmony_ci{ 117cabdff1aSopenharmony_ci uint8_t byte = *src++; 118cabdff1aSopenharmony_ci uint8_t ival = byte + 0x16; 119cabdff1aSopenharmony_ci const uint8_t * ptr = src + byte*2; 120cabdff1aSopenharmony_ci int ptr_len = src_len - 1 - byte*2; 121cabdff1aSopenharmony_ci uint8_t val = ival; 122cabdff1aSopenharmony_ci uint8_t *dest_end = dest + dest_len; 123cabdff1aSopenharmony_ci uint8_t *dest_start = dest; 124cabdff1aSopenharmony_ci int ret; 125cabdff1aSopenharmony_ci GetBitContext gb; 126cabdff1aSopenharmony_ci 127cabdff1aSopenharmony_ci if ((ret = init_get_bits8(&gb, ptr, ptr_len)) < 0) 128cabdff1aSopenharmony_ci return ret; 129cabdff1aSopenharmony_ci 130cabdff1aSopenharmony_ci while (val != 0x16) { 131cabdff1aSopenharmony_ci unsigned idx; 132cabdff1aSopenharmony_ci if (get_bits_left(&gb) < 1) 133cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 134cabdff1aSopenharmony_ci idx = val - 0x17 + get_bits1(&gb) * byte; 135cabdff1aSopenharmony_ci if (idx >= 2 * byte) 136cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 137cabdff1aSopenharmony_ci val = src[idx]; 138cabdff1aSopenharmony_ci 139cabdff1aSopenharmony_ci if (val < 0x16) { 140cabdff1aSopenharmony_ci if (dest >= dest_end) 141cabdff1aSopenharmony_ci return dest_len; 142cabdff1aSopenharmony_ci *dest++ = val; 143cabdff1aSopenharmony_ci val = ival; 144cabdff1aSopenharmony_ci } 145cabdff1aSopenharmony_ci } 146cabdff1aSopenharmony_ci 147cabdff1aSopenharmony_ci return dest - dest_start; 148cabdff1aSopenharmony_ci} 149cabdff1aSopenharmony_ci 150cabdff1aSopenharmony_ci/** 151cabdff1aSopenharmony_ci * unpack simple compression 152cabdff1aSopenharmony_ci * 153cabdff1aSopenharmony_ci * @param dest destination buffer of dest_len, must be padded with at least 130 bytes 154cabdff1aSopenharmony_ci */ 155cabdff1aSopenharmony_cistatic void xan_unpack(uint8_t *dest, int dest_len, 156cabdff1aSopenharmony_ci const uint8_t *src, int src_len) 157cabdff1aSopenharmony_ci{ 158cabdff1aSopenharmony_ci uint8_t opcode; 159cabdff1aSopenharmony_ci int size; 160cabdff1aSopenharmony_ci uint8_t *dest_org = dest; 161cabdff1aSopenharmony_ci uint8_t *dest_end = dest + dest_len; 162cabdff1aSopenharmony_ci GetByteContext ctx; 163cabdff1aSopenharmony_ci 164cabdff1aSopenharmony_ci bytestream2_init(&ctx, src, src_len); 165cabdff1aSopenharmony_ci while (dest < dest_end && bytestream2_get_bytes_left(&ctx)) { 166cabdff1aSopenharmony_ci opcode = bytestream2_get_byte(&ctx); 167cabdff1aSopenharmony_ci 168cabdff1aSopenharmony_ci if (opcode < 0xe0) { 169cabdff1aSopenharmony_ci int size2, back; 170cabdff1aSopenharmony_ci if ((opcode & 0x80) == 0) { 171cabdff1aSopenharmony_ci size = opcode & 3; 172cabdff1aSopenharmony_ci 173cabdff1aSopenharmony_ci back = ((opcode & 0x60) << 3) + bytestream2_get_byte(&ctx) + 1; 174cabdff1aSopenharmony_ci size2 = ((opcode & 0x1c) >> 2) + 3; 175cabdff1aSopenharmony_ci } else if ((opcode & 0x40) == 0) { 176cabdff1aSopenharmony_ci size = bytestream2_peek_byte(&ctx) >> 6; 177cabdff1aSopenharmony_ci 178cabdff1aSopenharmony_ci back = (bytestream2_get_be16(&ctx) & 0x3fff) + 1; 179cabdff1aSopenharmony_ci size2 = (opcode & 0x3f) + 4; 180cabdff1aSopenharmony_ci } else { 181cabdff1aSopenharmony_ci size = opcode & 3; 182cabdff1aSopenharmony_ci 183cabdff1aSopenharmony_ci back = ((opcode & 0x10) << 12) + bytestream2_get_be16(&ctx) + 1; 184cabdff1aSopenharmony_ci size2 = ((opcode & 0x0c) << 6) + bytestream2_get_byte(&ctx) + 5; 185cabdff1aSopenharmony_ci } 186cabdff1aSopenharmony_ci 187cabdff1aSopenharmony_ci if (dest_end - dest < size + size2 || 188cabdff1aSopenharmony_ci dest + size - dest_org < back || 189cabdff1aSopenharmony_ci bytestream2_get_bytes_left(&ctx) < size) 190cabdff1aSopenharmony_ci return; 191cabdff1aSopenharmony_ci bytestream2_get_buffer(&ctx, dest, size); 192cabdff1aSopenharmony_ci dest += size; 193cabdff1aSopenharmony_ci av_memcpy_backptr(dest, back, size2); 194cabdff1aSopenharmony_ci dest += size2; 195cabdff1aSopenharmony_ci } else { 196cabdff1aSopenharmony_ci int finish = opcode >= 0xfc; 197cabdff1aSopenharmony_ci size = finish ? opcode & 3 : ((opcode & 0x1f) << 2) + 4; 198cabdff1aSopenharmony_ci 199cabdff1aSopenharmony_ci if (dest_end - dest < size || bytestream2_get_bytes_left(&ctx) < size) 200cabdff1aSopenharmony_ci return; 201cabdff1aSopenharmony_ci bytestream2_get_buffer(&ctx, dest, size); 202cabdff1aSopenharmony_ci dest += size; 203cabdff1aSopenharmony_ci if (finish) 204cabdff1aSopenharmony_ci return; 205cabdff1aSopenharmony_ci } 206cabdff1aSopenharmony_ci } 207cabdff1aSopenharmony_ci} 208cabdff1aSopenharmony_ci 209cabdff1aSopenharmony_cistatic inline void xan_wc3_output_pixel_run(XanContext *s, AVFrame *frame, 210cabdff1aSopenharmony_ci const uint8_t *pixel_buffer, int x, int y, int pixel_count) 211cabdff1aSopenharmony_ci{ 212cabdff1aSopenharmony_ci int stride; 213cabdff1aSopenharmony_ci int line_inc; 214cabdff1aSopenharmony_ci int index; 215cabdff1aSopenharmony_ci int current_x; 216cabdff1aSopenharmony_ci int width = s->avctx->width; 217cabdff1aSopenharmony_ci uint8_t *palette_plane; 218cabdff1aSopenharmony_ci 219cabdff1aSopenharmony_ci palette_plane = frame->data[0]; 220cabdff1aSopenharmony_ci stride = frame->linesize[0]; 221cabdff1aSopenharmony_ci line_inc = stride - width; 222cabdff1aSopenharmony_ci index = y * stride + x; 223cabdff1aSopenharmony_ci current_x = x; 224cabdff1aSopenharmony_ci while (pixel_count && index < s->frame_size) { 225cabdff1aSopenharmony_ci int count = FFMIN(pixel_count, width - current_x); 226cabdff1aSopenharmony_ci memcpy(palette_plane + index, pixel_buffer, count); 227cabdff1aSopenharmony_ci pixel_count -= count; 228cabdff1aSopenharmony_ci index += count; 229cabdff1aSopenharmony_ci pixel_buffer += count; 230cabdff1aSopenharmony_ci current_x += count; 231cabdff1aSopenharmony_ci 232cabdff1aSopenharmony_ci if (current_x >= width) { 233cabdff1aSopenharmony_ci index += line_inc; 234cabdff1aSopenharmony_ci current_x = 0; 235cabdff1aSopenharmony_ci } 236cabdff1aSopenharmony_ci } 237cabdff1aSopenharmony_ci} 238cabdff1aSopenharmony_ci 239cabdff1aSopenharmony_cistatic inline void xan_wc3_copy_pixel_run(XanContext *s, AVFrame *frame, 240cabdff1aSopenharmony_ci int x, int y, 241cabdff1aSopenharmony_ci int pixel_count, int motion_x, 242cabdff1aSopenharmony_ci int motion_y) 243cabdff1aSopenharmony_ci{ 244cabdff1aSopenharmony_ci int stride; 245cabdff1aSopenharmony_ci int line_inc; 246cabdff1aSopenharmony_ci int curframe_index, prevframe_index; 247cabdff1aSopenharmony_ci int curframe_x, prevframe_x; 248cabdff1aSopenharmony_ci int width = s->avctx->width; 249cabdff1aSopenharmony_ci uint8_t *palette_plane, *prev_palette_plane; 250cabdff1aSopenharmony_ci 251cabdff1aSopenharmony_ci if (y + motion_y < 0 || y + motion_y >= s->avctx->height || 252cabdff1aSopenharmony_ci x + motion_x < 0 || x + motion_x >= s->avctx->width) 253cabdff1aSopenharmony_ci return; 254cabdff1aSopenharmony_ci 255cabdff1aSopenharmony_ci palette_plane = frame->data[0]; 256cabdff1aSopenharmony_ci prev_palette_plane = s->last_frame->data[0]; 257cabdff1aSopenharmony_ci if (!prev_palette_plane) 258cabdff1aSopenharmony_ci prev_palette_plane = palette_plane; 259cabdff1aSopenharmony_ci stride = frame->linesize[0]; 260cabdff1aSopenharmony_ci line_inc = stride - width; 261cabdff1aSopenharmony_ci curframe_index = y * stride + x; 262cabdff1aSopenharmony_ci curframe_x = x; 263cabdff1aSopenharmony_ci prevframe_index = (y + motion_y) * stride + x + motion_x; 264cabdff1aSopenharmony_ci prevframe_x = x + motion_x; 265cabdff1aSopenharmony_ci 266cabdff1aSopenharmony_ci if (prev_palette_plane == palette_plane && FFABS(motion_x + width*motion_y) < pixel_count) { 267cabdff1aSopenharmony_ci avpriv_request_sample(s->avctx, "Overlapping copy"); 268cabdff1aSopenharmony_ci return ; 269cabdff1aSopenharmony_ci } 270cabdff1aSopenharmony_ci 271cabdff1aSopenharmony_ci while (pixel_count && 272cabdff1aSopenharmony_ci curframe_index < s->frame_size && 273cabdff1aSopenharmony_ci prevframe_index < s->frame_size) { 274cabdff1aSopenharmony_ci int count = FFMIN3(pixel_count, width - curframe_x, 275cabdff1aSopenharmony_ci width - prevframe_x); 276cabdff1aSopenharmony_ci 277cabdff1aSopenharmony_ci memcpy(palette_plane + curframe_index, 278cabdff1aSopenharmony_ci prev_palette_plane + prevframe_index, count); 279cabdff1aSopenharmony_ci pixel_count -= count; 280cabdff1aSopenharmony_ci curframe_index += count; 281cabdff1aSopenharmony_ci prevframe_index += count; 282cabdff1aSopenharmony_ci curframe_x += count; 283cabdff1aSopenharmony_ci prevframe_x += count; 284cabdff1aSopenharmony_ci 285cabdff1aSopenharmony_ci if (curframe_x >= width) { 286cabdff1aSopenharmony_ci curframe_index += line_inc; 287cabdff1aSopenharmony_ci curframe_x = 0; 288cabdff1aSopenharmony_ci } 289cabdff1aSopenharmony_ci 290cabdff1aSopenharmony_ci if (prevframe_x >= width) { 291cabdff1aSopenharmony_ci prevframe_index += line_inc; 292cabdff1aSopenharmony_ci prevframe_x = 0; 293cabdff1aSopenharmony_ci } 294cabdff1aSopenharmony_ci } 295cabdff1aSopenharmony_ci} 296cabdff1aSopenharmony_ci 297cabdff1aSopenharmony_cistatic int xan_wc3_decode_frame(XanContext *s, AVFrame *frame) 298cabdff1aSopenharmony_ci{ 299cabdff1aSopenharmony_ci 300cabdff1aSopenharmony_ci int width = s->avctx->width; 301cabdff1aSopenharmony_ci int height = s->avctx->height; 302cabdff1aSopenharmony_ci int total_pixels = width * height; 303cabdff1aSopenharmony_ci uint8_t opcode; 304cabdff1aSopenharmony_ci uint8_t flag = 0; 305cabdff1aSopenharmony_ci int size = 0; 306cabdff1aSopenharmony_ci int motion_x, motion_y; 307cabdff1aSopenharmony_ci int x, y, ret; 308cabdff1aSopenharmony_ci 309cabdff1aSopenharmony_ci uint8_t *opcode_buffer = s->buffer1; 310cabdff1aSopenharmony_ci uint8_t *opcode_buffer_end = s->buffer1 + s->buffer1_size; 311cabdff1aSopenharmony_ci int opcode_buffer_size = s->buffer1_size; 312cabdff1aSopenharmony_ci const uint8_t *imagedata_buffer = s->buffer2; 313cabdff1aSopenharmony_ci 314cabdff1aSopenharmony_ci /* pointers to segments inside the compressed chunk */ 315cabdff1aSopenharmony_ci const uint8_t *huffman_segment; 316cabdff1aSopenharmony_ci GetByteContext size_segment; 317cabdff1aSopenharmony_ci GetByteContext vector_segment; 318cabdff1aSopenharmony_ci const uint8_t *imagedata_segment; 319cabdff1aSopenharmony_ci int huffman_offset, size_offset, vector_offset, imagedata_offset, 320cabdff1aSopenharmony_ci imagedata_size; 321cabdff1aSopenharmony_ci 322cabdff1aSopenharmony_ci if (s->size < 8) 323cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 324cabdff1aSopenharmony_ci 325cabdff1aSopenharmony_ci huffman_offset = AV_RL16(&s->buf[0]); 326cabdff1aSopenharmony_ci size_offset = AV_RL16(&s->buf[2]); 327cabdff1aSopenharmony_ci vector_offset = AV_RL16(&s->buf[4]); 328cabdff1aSopenharmony_ci imagedata_offset = AV_RL16(&s->buf[6]); 329cabdff1aSopenharmony_ci 330cabdff1aSopenharmony_ci if (huffman_offset >= s->size || 331cabdff1aSopenharmony_ci size_offset >= s->size || 332cabdff1aSopenharmony_ci vector_offset >= s->size || 333cabdff1aSopenharmony_ci imagedata_offset >= s->size) 334cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 335cabdff1aSopenharmony_ci 336cabdff1aSopenharmony_ci huffman_segment = s->buf + huffman_offset; 337cabdff1aSopenharmony_ci bytestream2_init(&size_segment, s->buf + size_offset, s->size - size_offset); 338cabdff1aSopenharmony_ci bytestream2_init(&vector_segment, s->buf + vector_offset, s->size - vector_offset); 339cabdff1aSopenharmony_ci imagedata_segment = s->buf + imagedata_offset; 340cabdff1aSopenharmony_ci 341cabdff1aSopenharmony_ci if ((ret = xan_huffman_decode(opcode_buffer, opcode_buffer_size, 342cabdff1aSopenharmony_ci huffman_segment, s->size - huffman_offset)) < 0) 343cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 344cabdff1aSopenharmony_ci opcode_buffer_end = opcode_buffer + ret; 345cabdff1aSopenharmony_ci 346cabdff1aSopenharmony_ci if (imagedata_segment[0] == 2) { 347cabdff1aSopenharmony_ci xan_unpack(s->buffer2, s->buffer2_size, 348cabdff1aSopenharmony_ci &imagedata_segment[1], s->size - imagedata_offset - 1); 349cabdff1aSopenharmony_ci imagedata_size = s->buffer2_size; 350cabdff1aSopenharmony_ci } else { 351cabdff1aSopenharmony_ci imagedata_size = s->size - imagedata_offset - 1; 352cabdff1aSopenharmony_ci imagedata_buffer = &imagedata_segment[1]; 353cabdff1aSopenharmony_ci } 354cabdff1aSopenharmony_ci 355cabdff1aSopenharmony_ci /* use the decoded data segments to build the frame */ 356cabdff1aSopenharmony_ci x = y = 0; 357cabdff1aSopenharmony_ci while (total_pixels && opcode_buffer < opcode_buffer_end) { 358cabdff1aSopenharmony_ci 359cabdff1aSopenharmony_ci opcode = *opcode_buffer++; 360cabdff1aSopenharmony_ci size = 0; 361cabdff1aSopenharmony_ci 362cabdff1aSopenharmony_ci switch (opcode) { 363cabdff1aSopenharmony_ci 364cabdff1aSopenharmony_ci case 0: 365cabdff1aSopenharmony_ci flag ^= 1; 366cabdff1aSopenharmony_ci continue; 367cabdff1aSopenharmony_ci 368cabdff1aSopenharmony_ci case 1: 369cabdff1aSopenharmony_ci case 2: 370cabdff1aSopenharmony_ci case 3: 371cabdff1aSopenharmony_ci case 4: 372cabdff1aSopenharmony_ci case 5: 373cabdff1aSopenharmony_ci case 6: 374cabdff1aSopenharmony_ci case 7: 375cabdff1aSopenharmony_ci case 8: 376cabdff1aSopenharmony_ci size = opcode; 377cabdff1aSopenharmony_ci break; 378cabdff1aSopenharmony_ci 379cabdff1aSopenharmony_ci case 12: 380cabdff1aSopenharmony_ci case 13: 381cabdff1aSopenharmony_ci case 14: 382cabdff1aSopenharmony_ci case 15: 383cabdff1aSopenharmony_ci case 16: 384cabdff1aSopenharmony_ci case 17: 385cabdff1aSopenharmony_ci case 18: 386cabdff1aSopenharmony_ci size += (opcode - 10); 387cabdff1aSopenharmony_ci break; 388cabdff1aSopenharmony_ci 389cabdff1aSopenharmony_ci case 9: 390cabdff1aSopenharmony_ci case 19: 391cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&size_segment) < 1) { 392cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "size_segment overread\n"); 393cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 394cabdff1aSopenharmony_ci } 395cabdff1aSopenharmony_ci size = bytestream2_get_byte(&size_segment); 396cabdff1aSopenharmony_ci break; 397cabdff1aSopenharmony_ci 398cabdff1aSopenharmony_ci case 10: 399cabdff1aSopenharmony_ci case 20: 400cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&size_segment) < 2) { 401cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "size_segment overread\n"); 402cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 403cabdff1aSopenharmony_ci } 404cabdff1aSopenharmony_ci size = bytestream2_get_be16(&size_segment); 405cabdff1aSopenharmony_ci break; 406cabdff1aSopenharmony_ci 407cabdff1aSopenharmony_ci case 11: 408cabdff1aSopenharmony_ci case 21: 409cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&size_segment) < 3) { 410cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "size_segment overread\n"); 411cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 412cabdff1aSopenharmony_ci } 413cabdff1aSopenharmony_ci size = bytestream2_get_be24(&size_segment); 414cabdff1aSopenharmony_ci break; 415cabdff1aSopenharmony_ci } 416cabdff1aSopenharmony_ci 417cabdff1aSopenharmony_ci if (size > total_pixels) 418cabdff1aSopenharmony_ci break; 419cabdff1aSopenharmony_ci 420cabdff1aSopenharmony_ci if (opcode < 12) { 421cabdff1aSopenharmony_ci flag ^= 1; 422cabdff1aSopenharmony_ci if (flag) { 423cabdff1aSopenharmony_ci /* run of (size) pixels is unchanged from last frame */ 424cabdff1aSopenharmony_ci xan_wc3_copy_pixel_run(s, frame, x, y, size, 0, 0); 425cabdff1aSopenharmony_ci } else { 426cabdff1aSopenharmony_ci /* output a run of pixels from imagedata_buffer */ 427cabdff1aSopenharmony_ci if (imagedata_size < size) 428cabdff1aSopenharmony_ci break; 429cabdff1aSopenharmony_ci xan_wc3_output_pixel_run(s, frame, imagedata_buffer, x, y, size); 430cabdff1aSopenharmony_ci imagedata_buffer += size; 431cabdff1aSopenharmony_ci imagedata_size -= size; 432cabdff1aSopenharmony_ci } 433cabdff1aSopenharmony_ci } else { 434cabdff1aSopenharmony_ci uint8_t vector; 435cabdff1aSopenharmony_ci if (bytestream2_get_bytes_left(&vector_segment) <= 0) { 436cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "vector_segment overread\n"); 437cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 438cabdff1aSopenharmony_ci } 439cabdff1aSopenharmony_ci /* run-based motion compensation from last frame */ 440cabdff1aSopenharmony_ci vector = bytestream2_get_byte(&vector_segment); 441cabdff1aSopenharmony_ci motion_x = sign_extend(vector >> 4, 4); 442cabdff1aSopenharmony_ci motion_y = sign_extend(vector & 0xF, 4); 443cabdff1aSopenharmony_ci 444cabdff1aSopenharmony_ci /* copy a run of pixels from the previous frame */ 445cabdff1aSopenharmony_ci xan_wc3_copy_pixel_run(s, frame, x, y, size, motion_x, motion_y); 446cabdff1aSopenharmony_ci 447cabdff1aSopenharmony_ci flag = 0; 448cabdff1aSopenharmony_ci } 449cabdff1aSopenharmony_ci 450cabdff1aSopenharmony_ci /* coordinate accounting */ 451cabdff1aSopenharmony_ci total_pixels -= size; 452cabdff1aSopenharmony_ci y += (x + size) / width; 453cabdff1aSopenharmony_ci x = (x + size) % width; 454cabdff1aSopenharmony_ci } 455cabdff1aSopenharmony_ci return 0; 456cabdff1aSopenharmony_ci} 457cabdff1aSopenharmony_ci 458cabdff1aSopenharmony_ci#if RUNTIME_GAMMA 459cabdff1aSopenharmony_cistatic inline unsigned mul(unsigned a, unsigned b) 460cabdff1aSopenharmony_ci{ 461cabdff1aSopenharmony_ci return (a * b) >> 16; 462cabdff1aSopenharmony_ci} 463cabdff1aSopenharmony_ci 464cabdff1aSopenharmony_cistatic inline unsigned pow4(unsigned a) 465cabdff1aSopenharmony_ci{ 466cabdff1aSopenharmony_ci unsigned square = mul(a, a); 467cabdff1aSopenharmony_ci return mul(square, square); 468cabdff1aSopenharmony_ci} 469cabdff1aSopenharmony_ci 470cabdff1aSopenharmony_cistatic inline unsigned pow5(unsigned a) 471cabdff1aSopenharmony_ci{ 472cabdff1aSopenharmony_ci return mul(pow4(a), a); 473cabdff1aSopenharmony_ci} 474cabdff1aSopenharmony_ci 475cabdff1aSopenharmony_cistatic uint8_t gamma_corr(uint8_t in) { 476cabdff1aSopenharmony_ci unsigned lo, hi = 0xff40, target; 477cabdff1aSopenharmony_ci int i = 15; 478cabdff1aSopenharmony_ci in = (in << 2) | (in >> 6); 479cabdff1aSopenharmony_ci /* equivalent float code: 480cabdff1aSopenharmony_ci if (in >= 252) 481cabdff1aSopenharmony_ci return 253; 482cabdff1aSopenharmony_ci return round(pow(in / 256.0, 0.8) * 256); 483cabdff1aSopenharmony_ci */ 484cabdff1aSopenharmony_ci lo = target = in << 8; 485cabdff1aSopenharmony_ci do { 486cabdff1aSopenharmony_ci unsigned mid = (lo + hi) >> 1; 487cabdff1aSopenharmony_ci unsigned pow = pow5(mid); 488cabdff1aSopenharmony_ci if (pow > target) hi = mid; 489cabdff1aSopenharmony_ci else lo = mid; 490cabdff1aSopenharmony_ci } while (--i); 491cabdff1aSopenharmony_ci return (pow4((lo + hi) >> 1) + 0x80) >> 8; 492cabdff1aSopenharmony_ci} 493cabdff1aSopenharmony_ci#else 494cabdff1aSopenharmony_ci/** 495cabdff1aSopenharmony_ci * This is a gamma correction that xan3 applies to all palette entries. 496cabdff1aSopenharmony_ci * 497cabdff1aSopenharmony_ci * There is a peculiarity, namely that the values are clamped to 253 - 498cabdff1aSopenharmony_ci * it seems likely that this table was calculated by a buggy fixed-point 499cabdff1aSopenharmony_ci * implementation, the one above under RUNTIME_GAMMA behaves like this for 500cabdff1aSopenharmony_ci * example. 501cabdff1aSopenharmony_ci * The exponent value of 0.8 can be explained by this as well, since 0.8 = 4/5 502cabdff1aSopenharmony_ci * and thus pow(x, 0.8) is still easy to calculate. 503cabdff1aSopenharmony_ci * Also, the input values are first rotated to the left by 2. 504cabdff1aSopenharmony_ci */ 505cabdff1aSopenharmony_cistatic const uint8_t gamma_lookup[256] = { 506cabdff1aSopenharmony_ci 0x00, 0x09, 0x10, 0x16, 0x1C, 0x21, 0x27, 0x2C, 507cabdff1aSopenharmony_ci 0x31, 0x35, 0x3A, 0x3F, 0x43, 0x48, 0x4C, 0x50, 508cabdff1aSopenharmony_ci 0x54, 0x59, 0x5D, 0x61, 0x65, 0x69, 0x6D, 0x71, 509cabdff1aSopenharmony_ci 0x75, 0x79, 0x7D, 0x80, 0x84, 0x88, 0x8C, 0x8F, 510cabdff1aSopenharmony_ci 0x93, 0x97, 0x9A, 0x9E, 0xA2, 0xA5, 0xA9, 0xAC, 511cabdff1aSopenharmony_ci 0xB0, 0xB3, 0xB7, 0xBA, 0xBE, 0xC1, 0xC5, 0xC8, 512cabdff1aSopenharmony_ci 0xCB, 0xCF, 0xD2, 0xD5, 0xD9, 0xDC, 0xDF, 0xE3, 513cabdff1aSopenharmony_ci 0xE6, 0xE9, 0xED, 0xF0, 0xF3, 0xF6, 0xFA, 0xFD, 514cabdff1aSopenharmony_ci 0x03, 0x0B, 0x12, 0x18, 0x1D, 0x23, 0x28, 0x2D, 515cabdff1aSopenharmony_ci 0x32, 0x36, 0x3B, 0x40, 0x44, 0x49, 0x4D, 0x51, 516cabdff1aSopenharmony_ci 0x56, 0x5A, 0x5E, 0x62, 0x66, 0x6A, 0x6E, 0x72, 517cabdff1aSopenharmony_ci 0x76, 0x7A, 0x7D, 0x81, 0x85, 0x89, 0x8D, 0x90, 518cabdff1aSopenharmony_ci 0x94, 0x98, 0x9B, 0x9F, 0xA2, 0xA6, 0xAA, 0xAD, 519cabdff1aSopenharmony_ci 0xB1, 0xB4, 0xB8, 0xBB, 0xBF, 0xC2, 0xC5, 0xC9, 520cabdff1aSopenharmony_ci 0xCC, 0xD0, 0xD3, 0xD6, 0xDA, 0xDD, 0xE0, 0xE4, 521cabdff1aSopenharmony_ci 0xE7, 0xEA, 0xED, 0xF1, 0xF4, 0xF7, 0xFA, 0xFD, 522cabdff1aSopenharmony_ci 0x05, 0x0D, 0x13, 0x19, 0x1F, 0x24, 0x29, 0x2E, 523cabdff1aSopenharmony_ci 0x33, 0x38, 0x3C, 0x41, 0x45, 0x4A, 0x4E, 0x52, 524cabdff1aSopenharmony_ci 0x57, 0x5B, 0x5F, 0x63, 0x67, 0x6B, 0x6F, 0x73, 525cabdff1aSopenharmony_ci 0x77, 0x7B, 0x7E, 0x82, 0x86, 0x8A, 0x8D, 0x91, 526cabdff1aSopenharmony_ci 0x95, 0x99, 0x9C, 0xA0, 0xA3, 0xA7, 0xAA, 0xAE, 527cabdff1aSopenharmony_ci 0xB2, 0xB5, 0xB9, 0xBC, 0xBF, 0xC3, 0xC6, 0xCA, 528cabdff1aSopenharmony_ci 0xCD, 0xD0, 0xD4, 0xD7, 0xDA, 0xDE, 0xE1, 0xE4, 529cabdff1aSopenharmony_ci 0xE8, 0xEB, 0xEE, 0xF1, 0xF5, 0xF8, 0xFB, 0xFD, 530cabdff1aSopenharmony_ci 0x07, 0x0E, 0x15, 0x1A, 0x20, 0x25, 0x2A, 0x2F, 531cabdff1aSopenharmony_ci 0x34, 0x39, 0x3D, 0x42, 0x46, 0x4B, 0x4F, 0x53, 532cabdff1aSopenharmony_ci 0x58, 0x5C, 0x60, 0x64, 0x68, 0x6C, 0x70, 0x74, 533cabdff1aSopenharmony_ci 0x78, 0x7C, 0x7F, 0x83, 0x87, 0x8B, 0x8E, 0x92, 534cabdff1aSopenharmony_ci 0x96, 0x99, 0x9D, 0xA1, 0xA4, 0xA8, 0xAB, 0xAF, 535cabdff1aSopenharmony_ci 0xB2, 0xB6, 0xB9, 0xBD, 0xC0, 0xC4, 0xC7, 0xCB, 536cabdff1aSopenharmony_ci 0xCE, 0xD1, 0xD5, 0xD8, 0xDB, 0xDF, 0xE2, 0xE5, 537cabdff1aSopenharmony_ci 0xE9, 0xEC, 0xEF, 0xF2, 0xF6, 0xF9, 0xFC, 0xFD 538cabdff1aSopenharmony_ci}; 539cabdff1aSopenharmony_ci#endif 540cabdff1aSopenharmony_ci 541cabdff1aSopenharmony_cistatic int xan_decode_frame(AVCodecContext *avctx, AVFrame *frame, 542cabdff1aSopenharmony_ci int *got_frame, AVPacket *avpkt) 543cabdff1aSopenharmony_ci{ 544cabdff1aSopenharmony_ci const uint8_t *buf = avpkt->data; 545cabdff1aSopenharmony_ci int ret, buf_size = avpkt->size; 546cabdff1aSopenharmony_ci XanContext *s = avctx->priv_data; 547cabdff1aSopenharmony_ci GetByteContext ctx; 548cabdff1aSopenharmony_ci int tag = 0; 549cabdff1aSopenharmony_ci 550cabdff1aSopenharmony_ci bytestream2_init(&ctx, buf, buf_size); 551cabdff1aSopenharmony_ci while (bytestream2_get_bytes_left(&ctx) > 8 && tag != VGA__TAG) { 552cabdff1aSopenharmony_ci unsigned *tmpptr; 553cabdff1aSopenharmony_ci uint32_t new_pal; 554cabdff1aSopenharmony_ci int size; 555cabdff1aSopenharmony_ci int i; 556cabdff1aSopenharmony_ci tag = bytestream2_get_le32(&ctx); 557cabdff1aSopenharmony_ci size = bytestream2_get_be32(&ctx); 558cabdff1aSopenharmony_ci if (size < 0) { 559cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Invalid tag size %d\n", size); 560cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 561cabdff1aSopenharmony_ci } 562cabdff1aSopenharmony_ci size = FFMIN(size, bytestream2_get_bytes_left(&ctx)); 563cabdff1aSopenharmony_ci switch (tag) { 564cabdff1aSopenharmony_ci case PALT_TAG: 565cabdff1aSopenharmony_ci if (size < PALETTE_SIZE) 566cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 567cabdff1aSopenharmony_ci if (s->palettes_count >= PALETTES_MAX) 568cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 569cabdff1aSopenharmony_ci tmpptr = av_realloc_array(s->palettes, 570cabdff1aSopenharmony_ci s->palettes_count + 1, AVPALETTE_SIZE); 571cabdff1aSopenharmony_ci if (!tmpptr) 572cabdff1aSopenharmony_ci return AVERROR(ENOMEM); 573cabdff1aSopenharmony_ci s->palettes = tmpptr; 574cabdff1aSopenharmony_ci tmpptr += s->palettes_count * AVPALETTE_COUNT; 575cabdff1aSopenharmony_ci for (i = 0; i < PALETTE_COUNT; i++) { 576cabdff1aSopenharmony_ci#if RUNTIME_GAMMA 577cabdff1aSopenharmony_ci int r = gamma_corr(bytestream2_get_byteu(&ctx)); 578cabdff1aSopenharmony_ci int g = gamma_corr(bytestream2_get_byteu(&ctx)); 579cabdff1aSopenharmony_ci int b = gamma_corr(bytestream2_get_byteu(&ctx)); 580cabdff1aSopenharmony_ci#else 581cabdff1aSopenharmony_ci int r = gamma_lookup[bytestream2_get_byteu(&ctx)]; 582cabdff1aSopenharmony_ci int g = gamma_lookup[bytestream2_get_byteu(&ctx)]; 583cabdff1aSopenharmony_ci int b = gamma_lookup[bytestream2_get_byteu(&ctx)]; 584cabdff1aSopenharmony_ci#endif 585cabdff1aSopenharmony_ci *tmpptr++ = (0xFFU << 24) | (r << 16) | (g << 8) | b; 586cabdff1aSopenharmony_ci } 587cabdff1aSopenharmony_ci s->palettes_count++; 588cabdff1aSopenharmony_ci break; 589cabdff1aSopenharmony_ci case SHOT_TAG: 590cabdff1aSopenharmony_ci if (size < 4) 591cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 592cabdff1aSopenharmony_ci new_pal = bytestream2_get_le32(&ctx); 593cabdff1aSopenharmony_ci if (new_pal < s->palettes_count) { 594cabdff1aSopenharmony_ci s->cur_palette = new_pal; 595cabdff1aSopenharmony_ci } else 596cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Invalid palette selected\n"); 597cabdff1aSopenharmony_ci break; 598cabdff1aSopenharmony_ci case VGA__TAG: 599cabdff1aSopenharmony_ci break; 600cabdff1aSopenharmony_ci default: 601cabdff1aSopenharmony_ci bytestream2_skip(&ctx, size); 602cabdff1aSopenharmony_ci break; 603cabdff1aSopenharmony_ci } 604cabdff1aSopenharmony_ci } 605cabdff1aSopenharmony_ci buf_size = bytestream2_get_bytes_left(&ctx); 606cabdff1aSopenharmony_ci 607cabdff1aSopenharmony_ci if (s->palettes_count <= 0) { 608cabdff1aSopenharmony_ci av_log(s->avctx, AV_LOG_ERROR, "No palette found\n"); 609cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 610cabdff1aSopenharmony_ci } 611cabdff1aSopenharmony_ci 612cabdff1aSopenharmony_ci if ((ret = ff_get_buffer(avctx, frame, AV_GET_BUFFER_FLAG_REF)) < 0) 613cabdff1aSopenharmony_ci return ret; 614cabdff1aSopenharmony_ci 615cabdff1aSopenharmony_ci if (!s->frame_size) 616cabdff1aSopenharmony_ci s->frame_size = frame->linesize[0] * s->avctx->height; 617cabdff1aSopenharmony_ci 618cabdff1aSopenharmony_ci memcpy(frame->data[1], 619cabdff1aSopenharmony_ci s->palettes + s->cur_palette * AVPALETTE_COUNT, AVPALETTE_SIZE); 620cabdff1aSopenharmony_ci 621cabdff1aSopenharmony_ci s->buf = ctx.buffer; 622cabdff1aSopenharmony_ci s->size = buf_size; 623cabdff1aSopenharmony_ci 624cabdff1aSopenharmony_ci if (xan_wc3_decode_frame(s, frame) < 0) 625cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 626cabdff1aSopenharmony_ci 627cabdff1aSopenharmony_ci av_frame_unref(s->last_frame); 628cabdff1aSopenharmony_ci if ((ret = av_frame_ref(s->last_frame, frame)) < 0) 629cabdff1aSopenharmony_ci return ret; 630cabdff1aSopenharmony_ci 631cabdff1aSopenharmony_ci *got_frame = 1; 632cabdff1aSopenharmony_ci 633cabdff1aSopenharmony_ci /* always report that the buffer was completely consumed */ 634cabdff1aSopenharmony_ci return buf_size; 635cabdff1aSopenharmony_ci} 636cabdff1aSopenharmony_ci 637cabdff1aSopenharmony_ciconst FFCodec ff_xan_wc3_decoder = { 638cabdff1aSopenharmony_ci .p.name = "xan_wc3", 639cabdff1aSopenharmony_ci .p.long_name = NULL_IF_CONFIG_SMALL("Wing Commander III / Xan"), 640cabdff1aSopenharmony_ci .p.type = AVMEDIA_TYPE_VIDEO, 641cabdff1aSopenharmony_ci .p.id = AV_CODEC_ID_XAN_WC3, 642cabdff1aSopenharmony_ci .priv_data_size = sizeof(XanContext), 643cabdff1aSopenharmony_ci .init = xan_decode_init, 644cabdff1aSopenharmony_ci .close = xan_decode_end, 645cabdff1aSopenharmony_ci FF_CODEC_DECODE_CB(xan_decode_frame), 646cabdff1aSopenharmony_ci .p.capabilities = AV_CODEC_CAP_DR1, 647cabdff1aSopenharmony_ci .caps_internal = FF_CODEC_CAP_INIT_CLEANUP | FF_CODEC_CAP_INIT_THREADSAFE, 648cabdff1aSopenharmony_ci}; 649