/* 2004.02.01 first released source code for IOMP */ /* * Copyright (C) 2001-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 * * $Id: xine_decoder.c,v 1.2 2003/11/25 04:26:02 georgedon Exp $ * * xine decoder plugin using libtheora * */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include #include #include "xine_internal.h" #include "video_out.h" #include "buffer.h" #include "metronom.h" #include "xineutils.h" /* #define LOG */ typedef struct theora_class_s { video_decoder_class_t decoder_class; } theora_class_t; typedef struct theora_decoder_s { video_decoder_t theora_decoder; theora_class_t *class; theora_info t_info; theora_comment t_comment; theora_state t_state; ogg_packet op; yuv_buffer yuv; xine_stream_t* stream; int reject; int op_max_size; char* packet; int done; int width, height; double ratio; int offset_x, offset_y; int frame_duration; int skipframes; int hp_read; int initialized; } theora_decoder_t; static void readin_op (theora_decoder_t *this, char* src, int size) { if ( this->done+size > this->op_max_size) { while (this->op_max_size < this->done+size) this->op_max_size=this->op_max_size*2; this->packet=realloc(this->packet, this->op_max_size); this->op.packet=this->packet; } xine_fast_memcpy ( this->packet+this->done, src, size); this->done=this->done+size; } static void yuv2frame(yuv_buffer *yuv, vo_frame_t *frame, int offset_x, int offset_y) { int i; int crop_offset; /* fixme - direct rendering (exchaning pointers) may be possible. * frame->base[0] = yuv->y could work if one could change the * pitches[0,1,2] values, and rely on the drawing routine using * the new pitches. With cropping and offsets, it's a bit trickier, * but it would still be possible. * Attempts at doing this have yielded nothing but SIGSEVs so far. */ /* Copy yuv data onto the frame. Cropping and offset as specified * by the frame_width, frame_height, offset_x and offset_y fields * in the theora header is carried out. */ crop_offset=offset_x+yuv->y_stride*offset_y; for(i=0;iheight;i++) xine_fast_memcpy(frame->base[0]+frame->pitches[0]*i, yuv->y+crop_offset+yuv->y_stride*i, frame->width); crop_offset=(offset_x/2)+(yuv->uv_stride)*(offset_y/2); for(i=0;iheight/2;i++){ xine_fast_memcpy(frame->base[1]+frame->pitches[1]*i, yuv->u+crop_offset+yuv->uv_stride*i, frame->width/2); xine_fast_memcpy(frame->base[2]+frame->pitches[2]*i, yuv->v+crop_offset+yuv->uv_stride*i, frame->width/2); } } static int collect_data (theora_decoder_t *this, buf_element_t *buf ) { /* Assembles an ogg_packet which was sent with send_ogg_packet over xinebuffers */ /* this->done, this->rejected, this->op and this->decoder->flags are needed*/ int op_size = sizeof (ogg_packet); if (buf->decoder_flags & BUF_FLAG_FRAME_START) { this->done=0; /*start from the beginnig*/ this->reject=0;/*new packet - new try*/ /*copy the ogg_packet struct and the sum, correct the adress of the packet*/ xine_fast_memcpy (&this->op, buf->content, op_size); this->op.packet=this->packet; readin_op (this, buf->content + op_size, buf->size - op_size ); /*read the rest of the data*/ } else { if (this->done==0 || this->reject) { /*we are starting to collect an packet without the beginnig reject the rest*/ printf ("libtheora: rejecting packet\n"); this->reject=1; return 0; } readin_op (this, buf->content, buf->size ); } if ((buf->decoder_flags & BUF_FLAG_FRAME_END) && !this->reject) { if ( this->done != this->op.bytes ) { printf ("libtheora: A packet changed its size during transfer - rejected\n"); printf (" size %d should be %ld\n", this->done , this->op.bytes); this->op.bytes=this->done; } return 1; } return 0; } static void theora_decode_data (video_decoder_t *this_gen, buf_element_t *buf) { /* * decode data from buf and feed decoded frames to * video output */ theora_decoder_t *this = (theora_decoder_t *) this_gen; vo_frame_t *frame; yuv_buffer yuv; int ret; if (!collect_data(this, buf)) return; /*return, until a entire packets is collected*/ if ( (buf->decoder_flags & BUF_FLAG_PREVIEW) ) { /*get the first 3 packets and decode the header during preview*/ if (this->hp_read<=2) { /*decode three header packets*/ if (theora_decode_header(&this->t_info, &this->t_comment,&this->op)<0) { printf ("libtheora: Was unable to decode header #%d, corrput stream?\n",this->hp_read); } else { this->hp_read++; } } if (this->hp_read==3) { /*headers are now decoded. initialize the decoder*/ theora_decode_init (&this->t_state, &this->t_info); #ifdef LOG printf("libtheora: theora stream is Theora %dx%d %.02f fps video.\n" " frame content is %dx%d with offset (%d,%d).\n" " pixel aspect is %d:%d.\n", this->t_info.width,this->t_info.height, (double)this->t_info.fps_numerator/this->t_info.fps_denominator, this->t_info.frame_width, this->t_info.frame_height, this->t_info.offset_x, this->t_info.offset_y, this->t_info.aspect_numerator, this->t_info.aspect_denominator); #endif this->frame_duration=((int64_t)90000*this->t_info.fps_denominator)/this->t_info.fps_numerator; this->width=this->t_info.frame_width; this->height=this->t_info.frame_height; if (this->t_info.aspect_numerator==0 || this->t_info.aspect_denominator==0) /* 0-values are undefined, so don't do any scaling. */ this->ratio=(double)this->width/(double)this->height; else /* Yes, this video needs to be scaled. */ this->ratio=(double)(this->width*this->t_info.aspect_numerator) / (double)(this->height*this->t_info.aspect_denominator); this->offset_x=this->t_info.offset_x; this->offset_y=this->t_info.offset_y; this->initialized=1; this->hp_read++; } } else if (buf->decoder_flags & BUF_FLAG_HEADER) { /*ignore headerpackets*/ return; } else { /*decode videodata*/ if (!this->initialized) { printf ("libtheora: cannot decode stream without header\n"); return; } ret=theora_decode_packetin( &this->t_state, &this->op); if ( ret!=0) { if (this->stream->xine->verbosity >= XINE_VERBOSITY_LOG) printf ("libtheora:Received an bad packet\n"); } else if (!this->skipframes) { theora_decode_YUVout(&this->t_state,&yuv); /*fixme - aspectratio from theora is not considered*/ frame = this->stream->video_out->get_frame( this->stream->video_out, this->width, this->height, this->ratio, XINE_IMGFMT_YV12, VO_BOTH_FIELDS); yuv2frame(&yuv, frame, this->offset_x, this->offset_y); frame->pts = buf->pts; frame->duration=this->frame_duration; this->skipframes=frame->draw(frame, this->stream); frame->free(frame); } else { this->skipframes=this->skipframes-1; } } } static void theora_flush (video_decoder_t *this_gen) { /* * flush out any frames that are still stored in the decoder */ theora_decoder_t *this = (theora_decoder_t *) this_gen; this->skipframes=0; } static void theora_reset (video_decoder_t *this_gen) { /* * reset decoder after engine flush (prepare for new * video data not related to recently decoded data) */ theora_decoder_t *this = (theora_decoder_t *) this_gen; this->skipframes=0; } static void theora_discontinuity (video_decoder_t *this_gen) { /* * inform decoder that a time reference discontinuity has happened. * that is, it must forget any currently held pts value */ theora_decoder_t *this = (theora_decoder_t *) this_gen; this->skipframes=0; } static void theora_dispose (video_decoder_t *this_gen) { /* * close down, free all resources */ theora_decoder_t *this = (theora_decoder_t *) this_gen; #ifdef LOG printf ("libtheora: dispose \n"); #endif theora_clear (&this->t_state); theora_comment_clear (&this->t_comment); theora_info_clear (&this->t_info); this->stream->video_out->close(this->stream->video_out, this->stream); free (this->packet); free (this); } static video_decoder_t *theora_open_plugin (video_decoder_class_t *class_gen, xine_stream_t *stream) { /* * open a new instance of this plugin class */ theora_decoder_t *this ; printf ("You are trying to decode an theorastream. At the moment, theora is in\n"); printf ("development. If the stream could not be played back go to\n"); printf ("http://xine.sourceforge.net and get the latest release of xine.\n"); printf ("This release will play back streams which have been encoded with\n"); printf ("libtheora-alpha2.\n"); this = (theora_decoder_t *) malloc (sizeof (theora_decoder_t)); memset(this, 0, sizeof (theora_decoder_t)); this->theora_decoder.decode_data = theora_decode_data; this->theora_decoder.flush = theora_flush; this->theora_decoder.reset = theora_reset; this->theora_decoder.discontinuity = theora_discontinuity; this->theora_decoder.dispose = theora_dispose; this->stream = stream; this->class = (theora_class_t *) class_gen; this->op_max_size = 4096; this->packet = malloc(this->op_max_size); this->done = 0; this->stream = stream; this->initialized = 0; theora_comment_init (&this->t_comment); theora_info_init (&this->t_info); stream->video_out->open (stream->video_out, stream); return &this->theora_decoder; } /* * theora plugin class */ static char *theora_get_identifier (video_decoder_class_t *this) { /* * return short, human readable identifier for this plugin class */ return "theora video"; } static char *theora_get_description (video_decoder_class_t *this) { /* * return human readable (verbose = 1 line) description for * this plugin class */ return "experimental theora video decoder plugin"; } static void theora_dispose_class (video_decoder_class_t *this) { /* * free all class-related resources */ free (this); } static void *init_plugin (xine_t *xine, void *data) { /*initialize our plugin*/ theora_class_t *this; this = (theora_class_t *) malloc (sizeof (theora_class_t)); this->decoder_class.open_plugin = theora_open_plugin; this->decoder_class.get_identifier = theora_get_identifier; this->decoder_class.get_description = theora_get_description; this->decoder_class.dispose = theora_dispose_class; return this; } /* * exported plugin catalog entry */ static uint32_t supported_types[] = { BUF_VIDEO_THEORA, 0 }; static decoder_info_t dec_info_video = { supported_types, /* supported types */ 5 /* priority */ }; plugin_info_t xine_plugin_info[] = { /* type, API, "name", version, special_info, init_function */ { PLUGIN_VIDEO_DECODER, 15, "theora", XINE_VERSION_CODE, &dec_info_video, init_plugin }, { PLUGIN_NONE, 0, "", 0, NULL, NULL } };