diff options
Diffstat (limited to 'src/libxinevdec/cinepak.c')
-rw-r--r-- | src/libxinevdec/cinepak.c | 566 |
1 files changed, 0 insertions, 566 deletions
diff --git a/src/libxinevdec/cinepak.c b/src/libxinevdec/cinepak.c deleted file mode 100644 index a2fa78cce..000000000 --- a/src/libxinevdec/cinepak.c +++ /dev/null @@ -1,566 +0,0 @@ -/* - * Copyright (C) 2002-2003 the xine project - * - * This file is part of xine, a free video player. - * - * xine is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * xine is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA - * - * by Ewald Snel <ewald@rambo.its.tudelft.nl> - * - * based on overview of Cinepak algorithm and example decoder - * by Tim Ferguson: http://www.csse.monash.edu.au/~timf/ - * - * $Id: cinepak.c,v 1.38 2004/02/09 22:04:11 jstembridge Exp $ - */ - -#include <stdlib.h> -#include <unistd.h> - -#include "xine_internal.h" -#include "video_out.h" -#include "buffer.h" -#include "bswap.h" -#include "xineutils.h" - -#define MAX_STRIPS 32 -#define VIDEOBUFSIZE 128 * 1024 - - -typedef struct { - uint8_t y0, y1, y2, y3; - uint8_t u, v; -} cvid_codebook_t; - -typedef struct { - uint16_t id; - uint16_t x1, y1; - uint16_t x2, y2; - cvid_codebook_t v4_codebook[256]; - cvid_codebook_t v1_codebook[256]; -} cvid_strip_t; - -typedef struct { - video_decoder_class_t decoder_class; -} cvid_class_t; - -typedef struct cvid_decoder_s { - video_decoder_t video_decoder; - - cvid_class_t *class; - - xine_stream_t *stream; - int64_t video_step; - int decoder_ok; - - unsigned char *buf; - int bufsize; - int size; - - cvid_strip_t strips[MAX_STRIPS]; - - unsigned int coded_width; - unsigned int coded_height; - int luma_pitch; - int chroma_pitch; - uint8_t *current; - int offsets[3]; - - unsigned int width; - unsigned int height; - double ratio; -} cvid_decoder_t; - -static unsigned char yuv_palette[256 * 4]; -static int color_depth; - -static void cinepak_decode_codebook (cvid_codebook_t *codebook, - int chunk_id, int size, uint8_t *data) -{ - uint8_t *eod = (data + size); - uint32_t flag, mask; - int i, n; - - n = (chunk_id & 0x0400) ? 4 : 6; - flag = 0; - mask = 0; - - for (i=0; i < 256; i++) { - if ((chunk_id & 0x0100) && !(mask >>= 1)) { - if ((data + 4) > eod) - break; - - flag = BE_32 (data); - data += 4; - mask = 0x80000000; - } - - if (!(chunk_id & 0x0100) || (flag & mask)) { - if ((data + n) > eod) - break; - - if (n == 6) { - codebook[i].y0 = *data++; - codebook[i].y1 = *data++; - codebook[i].y2 = *data++; - codebook[i].y3 = *data++; - codebook[i].u = 128 + *data++; - codebook[i].v = 128 + *data++; - } else if (color_depth == 8) { - /* if color depth is 8, this codebook type indicates palette lookup */ - codebook[i].y0 = yuv_palette[*(data + 0) * 4]; - codebook[i].y1 = yuv_palette[*(data + 1) * 4]; - codebook[i].y2 = yuv_palette[*(data + 2) * 4]; - codebook[i].y3 = yuv_palette[*(data + 3) * 4]; - codebook[i].u = - (yuv_palette[*(data + 0) * 4 + 1] + - yuv_palette[*(data + 1) * 4 + 1] + - yuv_palette[*(data + 2) * 4 + 1] + - yuv_palette[*(data + 3) * 4 + 1]) / 4; - codebook[i].v = - (yuv_palette[*(data + 0) * 4 + 2] + - yuv_palette[*(data + 1) * 4 + 2] + - yuv_palette[*(data + 2) * 4 + 2] + - yuv_palette[*(data + 3) * 4 + 2]) / 4; - data += 4; - } else { - /* if color depth is 40, this codebook type indicates greyscale */ - codebook[i].y0 = *data++; - codebook[i].y1 = *data++; - codebook[i].y2 = *data++; - codebook[i].y3 = *data++; - codebook[i].u = 128; - codebook[i].v = 128; - } - } - } -} - -static int cinepak_decode_vectors (cvid_decoder_t *this, cvid_strip_t *strip, - int chunk_id, int size, uint8_t *data) -{ - uint8_t *eod = (data + size); - uint32_t flag, mask; - cvid_codebook_t *codebook; - unsigned int x, y; - uint8_t *iy[4]; - uint8_t *iu[2]; - uint8_t *iv[2]; - - flag = 0; - mask = 0; - - for (y=strip->y1; y < strip->y2; y+=4) { - - iy[0] = &this->current[strip->x1 + (y * this->luma_pitch) + this->offsets[0]]; - iy[1] = iy[0] + this->luma_pitch; - iy[2] = iy[1] + this->luma_pitch; - iy[3] = iy[2] + this->luma_pitch; - iu[0] = &this->current[(strip->x1/2) + ((y/2) * this->chroma_pitch) + this->offsets[1]]; - iu[1] = iu[0] + this->chroma_pitch; - iv[0] = &this->current[(strip->x1/2) + ((y/2) * this->chroma_pitch) + this->offsets[2]]; - iv[1] = iv[0] + this->chroma_pitch; - - for (x=strip->x1; x < strip->x2; x+=4) { - if ((chunk_id & 0x0100) && !(mask >>= 1)) { - if ((data + 4) > eod) - return -1; - - flag = BE_32 (data); - data += 4; - mask = 0x80000000; - } - - if (!(chunk_id & 0x0100) || (flag & mask)) { - if (!(chunk_id & 0x0200) && !(mask >>= 1)) { - if ((data + 4) > eod) - return -1; - - flag = BE_32 (data); - data += 4; - mask = 0x80000000; - } - - if ((chunk_id & 0x0200) || (~flag & mask)) { - if (data >= eod) - return -1; - - codebook = &strip->v1_codebook[*data++]; - iy[0][0] = codebook->y0; iy[0][1] = codebook->y0; - iy[1][0] = codebook->y0; iy[1][1] = codebook->y0; - iu[0][0] = codebook->u; iv[0][0] = codebook->v; - - iy[0][2] = codebook->y1; iy[0][3] = codebook->y1; - iy[1][2] = codebook->y1; iy[1][3] = codebook->y1; - iu[0][1] = codebook->u; iv[0][1] = codebook->v; - - iy[2][0] = codebook->y2; iy[2][1] = codebook->y2; - iy[3][0] = codebook->y2; iy[3][1] = codebook->y2; - iu[1][0] = codebook->u; iv[1][0] = codebook->v; - - iy[2][2] = codebook->y3; iy[2][3] = codebook->y3; - iy[3][2] = codebook->y3; iy[3][3] = codebook->y3; - iu[1][1] = codebook->u; iv[1][1] = codebook->v; - - } else if (flag & mask) { - if ((data + 4) > eod) - return -1; - - codebook = &strip->v4_codebook[*data++]; - iy[0][0] = codebook->y0; iy[0][1] = codebook->y1; - iy[1][0] = codebook->y2; iy[1][1] = codebook->y3; - iu[0][0] = codebook->u; iv[0][0] = codebook->v; - - codebook = &strip->v4_codebook[*data++]; - iy[0][2] = codebook->y0; iy[0][3] = codebook->y1; - iy[1][2] = codebook->y2; iy[1][3] = codebook->y3; - iu[0][1] = codebook->u; iv[0][1] = codebook->v; - - codebook = &strip->v4_codebook[*data++]; - iy[2][0] = codebook->y0; iy[2][1] = codebook->y1; - iy[3][0] = codebook->y2; iy[3][1] = codebook->y3; - iu[1][0] = codebook->u; iv[1][0] = codebook->v; - - codebook = &strip->v4_codebook[*data++]; - iy[2][2] = codebook->y0; iy[2][3] = codebook->y1; - iy[3][2] = codebook->y2; iy[3][3] = codebook->y3; - iu[1][1] = codebook->u; iv[1][1] = codebook->v; - } - } - - iy[0] += 4; iy[1] += 4; - iy[2] += 4; iy[3] += 4; - iu[0] += 2; iu[1] += 2; - iv[0] += 2; iv[1] += 2; - } - } - - return 0; -} - -static int cinepak_decode_strip (cvid_decoder_t *this, - cvid_strip_t *strip, uint8_t *data, int size) -{ - uint8_t *eod = (data + size); - int chunk_id, chunk_size; - - if (strip->x1 >= this->coded_width || strip->x2 > this->coded_width || - strip->y1 >= this->coded_height || strip->y2 > this->coded_height || - strip->x1 >= strip->x2 || strip->y1 >= strip->y2) - return -1; - - while ((data + 4) <= eod) { - chunk_id = BE_16 (&data[0]); - chunk_size = BE_16 (&data[2]) - 4; - data += 4; - chunk_size = ((data + chunk_size) > eod) ? (eod - data) : chunk_size; - - switch (chunk_id) { - case 0x2000: - case 0x2100: - case 0x2400: - case 0x2500: - cinepak_decode_codebook (strip->v4_codebook, chunk_id, chunk_size, data); - break; - case 0x2200: - case 0x2300: - case 0x2600: - case 0x2700: - cinepak_decode_codebook (strip->v1_codebook, chunk_id, chunk_size, data); - break; - case 0x3000: - case 0x3100: - case 0x3200: - return cinepak_decode_vectors (this, strip, chunk_id, chunk_size, data); - } - - data += chunk_size; - } - - return -1; -} - -static int cinepak_decode_frame (cvid_decoder_t *this, uint8_t *data, int size) { - uint8_t *eod = (data + size); - int i, result, strip_size, frame_flags, num_strips; - int y0 = 0; - - if (size < 10) - return -1; - - frame_flags = data[0]; - num_strips = BE_16 (&data[8]); - data += 10; - - if (num_strips > MAX_STRIPS) - num_strips = MAX_STRIPS; - - for (i=0; i < num_strips; i++) { - if ((data + 12) > eod) - return -1; - - this->strips[i].id = BE_16 (data); - this->strips[i].y1 = y0; - this->strips[i].x1 = 0; - this->strips[i].y2 = y0 + BE_16 (&data[8]); - this->strips[i].x2 = this->coded_width; - - strip_size = BE_16 (&data[2]) - 12; - data += 12; - strip_size = ((data + strip_size) > eod) ? (eod - data) : strip_size; - - if ((i > 0) && !(frame_flags & 0x01)) { - xine_fast_memcpy (this->strips[i].v4_codebook, this->strips[i-1].v4_codebook, - sizeof(this->strips[i].v4_codebook)); - - xine_fast_memcpy (this->strips[i].v1_codebook, this->strips[i-1].v1_codebook, - sizeof(this->strips[i].v1_codebook)); - } - - result = cinepak_decode_strip (this, &this->strips[i], data, strip_size); - - if (result != 0) - return result; - - data += strip_size; - y0 = this->strips[i].y2; - } - - return 0; -} - -static void cinepak_copy_frame (cvid_decoder_t *this, uint8_t *base[3], int pitches[3]) { - int i, j; - uint8_t *src, *dst; - - src = &this->current[this->offsets[0]]; - dst = base[0]; - - for (i=0; i < this->height; ++i) { - memcpy (dst, src, this->width); - src += this->luma_pitch; - dst += pitches[0]; - } - - for (i=1; i < 3; i++) { - src = &this->current[this->offsets[i]]; - dst = base[i]; - - for (j=0; j < (this->height / 2); ++j) { - memcpy (dst, src, (this->width / 2)); - src += this->chroma_pitch; - dst += pitches[i]; - } - } -} - -static void cvid_decode_data (video_decoder_t *this_gen, buf_element_t *buf) { - cvid_decoder_t *this = (cvid_decoder_t *) this_gen; - - palette_entry_t *palette; - int i; - - if (buf->decoder_flags & BUF_FLAG_PREVIEW) - return; - - /* convert the RGB palette to a YUV palette */ - if ((buf->decoder_flags & BUF_FLAG_SPECIAL) && - (buf->decoder_info[1] == BUF_SPECIAL_PALETTE)) { - palette = (palette_entry_t *)buf->decoder_info_ptr[2]; - for (i = 0; i < buf->decoder_info[2]; i++) { - yuv_palette[i * 4 + 0] = - COMPUTE_Y(palette[i].r, palette[i].g, palette[i].b); - yuv_palette[i * 4 + 1] = - COMPUTE_U(palette[i].r, palette[i].g, palette[i].b); - yuv_palette[i * 4 + 2] = - COMPUTE_V(palette[i].r, palette[i].g, palette[i].b); - } - } - - if (buf->decoder_flags & BUF_FLAG_FRAMERATE) { - this->video_step = buf->decoder_info[0]; - _x_stream_info_set(this->stream, XINE_STREAM_INFO_FRAME_DURATION, this->video_step); - } - - if (buf->decoder_flags & BUF_FLAG_STDHEADER) { - xine_bmiheader *bih; - int chroma_size; - - bih = (xine_bmiheader *) buf->content; - - this->width = (bih->biWidth + 1) & ~0x1; - this->height = (bih->biHeight + 1) & ~0x1; - this->ratio = (double)this->width/(double)this->height; - this->coded_width = (this->width + 3) & ~0x3; - this->coded_height = (this->height + 3) & ~0x3; - this->luma_pitch = this->coded_width; - this->chroma_pitch = this->coded_width / 2; - - chroma_size = (this->chroma_pitch * (this->coded_height / 2)); - this->current = (uint8_t *) realloc (this->current, 6*chroma_size); - this->offsets[0] = 0; - this->offsets[1] = 4*chroma_size; - this->offsets[2] = 5*chroma_size; - - if (this->buf) - free (this->buf); - this->bufsize = VIDEOBUFSIZE; - this->buf = malloc(this->bufsize); - this->size = 0; - - color_depth = bih->biBitCount; - - this->stream->video_out->open (this->stream->video_out, this->stream); - this->decoder_ok = 1; - - /* stream/meta info */ - _x_meta_info_set(this->stream, XINE_META_INFO_VIDEOCODEC, "Cinepak"); - - } else if (this->decoder_ok) { - - if (this->size + buf->size > this->bufsize) { - this->bufsize = this->size + 2 * buf->size; - this->buf = realloc (this->buf, this->bufsize); - } - - xine_fast_memcpy (&this->buf[this->size], buf->content, buf->size); - - this->size += buf->size; - - if (buf->decoder_flags & BUF_FLAG_FRAME_END) { - - vo_frame_t *img; - int result; - - result = cinepak_decode_frame (this, this->buf, this->size); - - img = this->stream->video_out->get_frame (this->stream->video_out, - this->width, this->height, - this->ratio, - XINE_IMGFMT_YV12, VO_BOTH_FIELDS); - - img->duration = this->video_step; - img->pts = buf->pts; - img->bad_frame = (result != 0); - - if (result == 0) { - cinepak_copy_frame (this, img->base, img->pitches); - } - - img->draw(img, this->stream); - img->free(img); - - this->size = 0; - } - } -} - -static void cvid_flush (video_decoder_t *this_gen) { -} - -static void cvid_reset (video_decoder_t *this_gen) { - cvid_decoder_t *this = (cvid_decoder_t *) this_gen; - - this->size = 0; -} - -static void cvid_discontinuity (video_decoder_t *this_gen) { -} - -static void cvid_dispose (video_decoder_t *this_gen) { - cvid_decoder_t *this = (cvid_decoder_t *) this_gen; - - if (this->current) { - free (this->current); - } - - if (this->buf) { - free (this->buf); - } - - if (this->decoder_ok) { - this->decoder_ok = 0; - this->stream->video_out->close(this->stream->video_out, this->stream); - } - - free (this_gen); -} - -static video_decoder_t *open_plugin (video_decoder_class_t *class_gen, xine_stream_t *stream) { - - cvid_decoder_t *this ; - - this = (cvid_decoder_t *) xine_xmalloc (sizeof (cvid_decoder_t)); - - this->video_decoder.decode_data = cvid_decode_data; - this->video_decoder.flush = cvid_flush; - this->video_decoder.reset = cvid_reset; - this->video_decoder.discontinuity = cvid_discontinuity; - this->video_decoder.dispose = cvid_dispose; - this->size = 0; - - this->stream = stream; - this->class = (cvid_class_t *) class_gen; - - this->decoder_ok = 0; - this->buf = NULL; - - return &this->video_decoder; -} - -static char *get_identifier (video_decoder_class_t *this) { - return "Cinepak"; -} - -static char *get_description (video_decoder_class_t *this) { - return "Cinepak (CVID) video decoder plugin"; -} - -static void dispose_class (video_decoder_class_t *this) { - free (this); -} - -static void *init_plugin (xine_t *xine, void *data) { - - cvid_class_t *this; - - this = (cvid_class_t *) xine_xmalloc (sizeof (cvid_class_t)); - - this->decoder_class.open_plugin = open_plugin; - this->decoder_class.get_identifier = get_identifier; - this->decoder_class.get_description = get_description; - this->decoder_class.dispose = dispose_class; - - return this; -} - -/* - * exported plugin catalog entry - */ - -static uint32_t video_types[] = { - BUF_VIDEO_CINEPAK, - 0 - }; - -static decoder_info_t dec_info_video = { - video_types, /* supported types */ - 5 /* priority */ -}; - -plugin_info_t xine_plugin_info[] = { - /* type, API, "name", version, special_info, init_function */ - { PLUGIN_VIDEO_DECODER, 18, "cinepak", XINE_VERSION_CODE, &dec_info_video, init_plugin }, - { PLUGIN_NONE, 0, "", 0, NULL, NULL } -}; |