/* 2004.02.01 first released source code for IOMP */ /* * Copyright (C) 2002 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 * * based on overview of Cinepak algorithm and example decoder * by Tim Ferguson: http://www.csse.monash.edu.au/~timf/ * * $Id: cinepak.c,v 1.2 2003/11/25 04:26:06 georgedon Exp $ */ #include #include #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_HEADER) { xine_bmiheader *bih; int chroma_size; bih = (xine_bmiheader *) buf->content; this->video_step = buf->decoder_info[1]; 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 */ this->stream->meta_info[XINE_META_INFO_VIDEOCODEC] = strdup("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_FRAMERATE) this->video_step = buf->decoder_info[0]; 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 *) malloc (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, 15, "cinepak", XINE_VERSION_CODE, &dec_info_video, init_plugin }, { PLUGIN_NONE, 0, "", 0, NULL, NULL } };