/* * Copyright (C) 2000-2006 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.c,v 1.342 2007/02/20 00:37:02 dgp85 Exp $ */ /* * top-level xine functions */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include #include #include #if defined (__linux__) #include #elif defined (__FreeBSD__) #include #endif #ifdef HAVE_SETLOCALE #include #endif #define LOG_MODULE "xine" #define LOG_VERBOSE /* #define LOG */ #define XINE_ENABLE_EXPERIMENTAL_FEATURES #define XINE_ENGINE_INTERNAL #define METRONOM_CLOCK_INTERNAL #include "xine_internal.h" #include "plugin_catalog.h" #include "audio_out.h" #include "video_out.h" #include "demuxers/demux.h" #include "buffer.h" #include "spu_decoder.h" #include "input/input_plugin.h" #include "metronom.h" #include "configfile.h" #include "osd.h" #include "xineutils.h" #include "compat.h" #ifdef WIN32 # include # include #endif /* WIN32 */ static void mutex_cleanup (void *mutex) { pthread_mutex_unlock ((pthread_mutex_t *) mutex); } void _x_handle_stream_end (xine_stream_t *stream, int non_user) { if (stream->status == XINE_STATUS_QUIT) return; stream->status = XINE_STATUS_STOP; if (non_user) { /* frontends will not be interested in receiving this event * if they have called xine_stop explicitly, so only send * it if stream playback finished because of stream end reached */ xine_event_t event; event.data_length = 0; event.type = XINE_EVENT_UI_PLAYBACK_FINISHED; xine_event_send (stream, &event); } } void _x_extra_info_reset( extra_info_t *extra_info ) { memset( extra_info, 0, sizeof(extra_info_t) ); } void _x_extra_info_merge( extra_info_t *dst, extra_info_t *src ) { if (!src->invalid) { if( src->input_normpos ) dst->input_normpos = src->input_normpos; if( src->input_time ) dst->input_time = src->input_time; if( src->frame_number ) dst->frame_number = src->frame_number; if( src->seek_count ) dst->seek_count = src->seek_count; if( src->vpts ) dst->vpts = src->vpts; } } static void ticket_acquire(xine_ticket_t *this, int irrevocable) { pthread_mutex_lock(&this->lock); if (this->ticket_revoked && !this->irrevocable_tickets) pthread_cond_wait(&this->issued, &this->lock); else if (this->atomic_revoke && !pthread_equal(this->atomic_revoker_thread, pthread_self())) pthread_cond_wait(&this->issued, &this->lock); this->tickets_granted++; if (irrevocable) this->irrevocable_tickets++; pthread_mutex_unlock(&this->lock); } static void ticket_release(xine_ticket_t *this, int irrevocable) { pthread_mutex_lock(&this->lock); this->tickets_granted--; if (irrevocable) this->irrevocable_tickets--; if (this->ticket_revoked && !this->tickets_granted) pthread_cond_broadcast(&this->revoked); if (this->ticket_revoked && !this->irrevocable_tickets) pthread_cond_wait(&this->issued, &this->lock); pthread_mutex_unlock(&this->lock); } static void ticket_renew(xine_ticket_t *this, int irrevocable) { pthread_mutex_lock(&this->lock); this->tickets_granted--; _x_assert(this->ticket_revoked); if (!this->tickets_granted) pthread_cond_broadcast(&this->revoked); if (!this->irrevocable_tickets || !irrevocable) pthread_cond_wait(&this->issued, &this->lock); this->tickets_granted++; pthread_mutex_unlock(&this->lock); } static void ticket_issue(xine_ticket_t *this, int atomic) { if (!atomic) pthread_mutex_lock(&this->revoke_lock); pthread_mutex_lock(&this->lock); this->pending_revocations--; if (!this->pending_revocations) pthread_cond_broadcast(&this->issued); this->atomic_revoke = 0; pthread_mutex_unlock(&this->lock); pthread_mutex_unlock(&this->revoke_lock); } static void ticket_revoke(xine_ticket_t *this, int atomic) { pthread_mutex_lock(&this->revoke_lock); pthread_mutex_lock(&this->lock); this->pending_revocations++; this->ticket_revoked = 1; if (this->tickets_granted) pthread_cond_wait(&this->revoked, &this->lock); _x_assert(!this->tickets_granted); this->ticket_revoked = 0; if (atomic) { this->atomic_revoke = 1; this->atomic_revoker_thread = pthread_self(); } pthread_mutex_unlock(&this->lock); if (!atomic) pthread_mutex_unlock(&this->revoke_lock); } static void ticket_dispose(xine_ticket_t *this) { pthread_mutex_destroy(&this->lock); pthread_mutex_destroy(&this->revoke_lock); pthread_cond_destroy(&this->issued); pthread_cond_destroy(&this->revoked); free(this); } static xine_ticket_t *ticket_init(void) { xine_ticket_t *port_ticket; port_ticket = (xine_ticket_t *) xine_xmalloc(sizeof(xine_ticket_t)); port_ticket->acquire = ticket_acquire; port_ticket->release = ticket_release; port_ticket->renew = ticket_renew; port_ticket->issue = ticket_issue; port_ticket->revoke = ticket_revoke; port_ticket->dispose = ticket_dispose; pthread_mutex_init(&port_ticket->lock, NULL); pthread_mutex_init(&port_ticket->revoke_lock, NULL); pthread_cond_init(&port_ticket->issued, NULL); pthread_cond_init(&port_ticket->revoked, NULL); return port_ticket; } static void set_speed_internal (xine_stream_t *stream, int speed) { xine_t *xine = stream->xine; if (xine->clock->speed != XINE_SPEED_PAUSE && speed == XINE_SPEED_PAUSE) /* get all decoder and post threads in a state where they agree to be blocked */ xine->port_ticket->revoke(xine->port_ticket, 0); if (xine->clock->speed == XINE_SPEED_PAUSE && speed != XINE_SPEED_PAUSE) /* all decoder and post threads may continue now */ xine->port_ticket->issue(xine->port_ticket, 0); stream->xine->clock->set_fine_speed (stream->xine->clock, speed); /* see coment on audio_out loop about audio_paused */ if( stream->audio_out ) { xine->port_ticket->acquire(xine->port_ticket, 1); /* inform audio_out that speed has changed - he knows what to do */ stream->audio_out->set_property (stream->audio_out, AO_PROP_CLOCK_SPEED, speed); xine->port_ticket->release(xine->port_ticket, 1); } } /* stream->ignore_speed_change must be set, when entering this function */ static void stop_internal (xine_stream_t *stream) { lprintf ("status before = %d\n", stream->status); if ( stream->status == XINE_STATUS_IDLE || stream->status == XINE_STATUS_STOP ) { _x_demux_control_end(stream, 0); lprintf("ignored"); } else { /* make sure we're not in "paused" state */ set_speed_internal (stream, XINE_FINE_SPEED_NORMAL); /* Don't change status if we're quitting */ if (stream->status != XINE_STATUS_QUIT) stream->status = XINE_STATUS_STOP; } /* * stop demux */ if (stream->demux_plugin && stream->demux_thread_created) { lprintf ("stopping demux\n"); _x_demux_stop_thread( stream ); lprintf ("demux stopped\n"); } lprintf ("done\n"); } void xine_stop (xine_stream_t *stream) { pthread_mutex_lock (&stream->frontend_lock); pthread_cleanup_push (mutex_cleanup, (void *) &stream->frontend_lock); /* make sure that other threads cannot change the speed, especially pauseing the stream */ pthread_mutex_lock(&stream->speed_change_lock); stream->ignore_speed_change = 1; pthread_mutex_unlock(&stream->speed_change_lock); stream->xine->port_ticket->acquire(stream->xine->port_ticket, 1); if (stream->audio_out) stream->audio_out->set_property(stream->audio_out, AO_PROP_DISCARD_BUFFERS, 1); if (stream->video_out) stream->video_out->set_property(stream->video_out, VO_PROP_DISCARD_FRAMES, 1); stop_internal (stream); if (stream->slave && (stream->slave_affection & XINE_MASTER_SLAVE_STOP)) xine_stop(stream->slave); if (stream->video_out) stream->video_out->set_property(stream->video_out, VO_PROP_DISCARD_FRAMES, 0); if (stream->audio_out) stream->audio_out->set_property(stream->audio_out, AO_PROP_DISCARD_BUFFERS, 0); stream->xine->port_ticket->release(stream->xine->port_ticket, 1); stream->ignore_speed_change = 0; pthread_cleanup_pop (0); pthread_mutex_unlock (&stream->frontend_lock); } static void close_internal (xine_stream_t *stream) { int i ; if( stream->slave ) { xine_close( stream->slave ); if( stream->slave_is_subtitle ) { xine_dispose(stream->slave); stream->slave = NULL; stream->slave_is_subtitle = 0; } } if( !stream->gapless_switch ) { /* make sure that other threads cannot change the speed, especially pauseing the stream */ pthread_mutex_lock(&stream->speed_change_lock); stream->ignore_speed_change = 1; pthread_mutex_unlock(&stream->speed_change_lock); stream->xine->port_ticket->acquire(stream->xine->port_ticket, 1); if (stream->audio_out) stream->audio_out->set_property(stream->audio_out, AO_PROP_DISCARD_BUFFERS, 1); if (stream->video_out) stream->video_out->set_property(stream->video_out, VO_PROP_DISCARD_FRAMES, 1); } stop_internal( stream ); if( !stream->gapless_switch ) { if (stream->video_out) stream->video_out->set_property(stream->video_out, VO_PROP_DISCARD_FRAMES, 0); if (stream->audio_out) stream->audio_out->set_property(stream->audio_out, AO_PROP_DISCARD_BUFFERS, 0); stream->xine->port_ticket->release(stream->xine->port_ticket, 1); stream->ignore_speed_change = 0; } if (stream->demux_plugin) { _x_free_demux_plugin(stream, stream->demux_plugin); stream->demux_plugin = NULL; } /* * close input plugin */ if (stream->input_plugin) { _x_free_input_plugin(stream, stream->input_plugin); stream->input_plugin = NULL; } /* * reset / free meta info */ for (i=0; iaudio_track_map_entries = 0; stream->spu_track_map_entries = 0; } void xine_close (xine_stream_t *stream) { pthread_mutex_lock (&stream->frontend_lock); pthread_cleanup_push (mutex_cleanup, (void *) &stream->frontend_lock); close_internal (stream); /* * set status to idle. * not putting this into close_internal because it is also called * by open_internal. */ /* Don't change status if we're quitting */ if (stream->status != XINE_STATUS_QUIT) stream->status = XINE_STATUS_IDLE; pthread_cleanup_pop (0); pthread_mutex_unlock (&stream->frontend_lock); } static int stream_rewire_audio(xine_post_out_t *output, void *data) { xine_stream_t *stream = (xine_stream_t *)output->data; xine_audio_port_t *new_port = (xine_audio_port_t *)data; uint32_t bits, rate; int mode; if (!data) return 0; stream->xine->port_ticket->revoke(stream->xine->port_ticket, 1); if (stream->audio_out->status(stream->audio_out, stream, &bits, &rate, &mode)) { /* register our stream at the new output port */ new_port->open(new_port, stream, bits, rate, mode); stream->audio_out->close(stream->audio_out, stream); } stream->audio_out = new_port; stream->xine->port_ticket->issue(stream->xine->port_ticket, 1); return 1; } static int stream_rewire_video(xine_post_out_t *output, void *data) { xine_stream_t *stream = (xine_stream_t *)output->data; xine_video_port_t *new_port = (xine_video_port_t *)data; int64_t img_duration; int width, height; if (!data) return 0; stream->xine->port_ticket->revoke(stream->xine->port_ticket, 1); if (stream->video_out->status(stream->video_out, stream, &width, &height, &img_duration)) { /* register our stream at the new output port */ new_port->open(new_port, stream); stream->video_out->close(stream->video_out, stream); } stream->video_out = new_port; stream->xine->port_ticket->issue(stream->xine->port_ticket, 1); return 1; } void xine_dispose_internal (xine_stream_t *stream); xine_stream_t *xine_stream_new (xine_t *this, xine_audio_port_t *ao, xine_video_port_t *vo) { xine_stream_t *stream; int i; pthread_mutexattr_t attr; xprintf (this, XINE_VERBOSITY_DEBUG, "xine_stream_new\n"); /* * create a new stream object */ pthread_mutex_lock (&this->streams_lock); stream = (xine_stream_t *) xine_xmalloc (sizeof (xine_stream_t)) ; stream->current_extra_info = malloc( sizeof( extra_info_t ) ); stream->audio_decoder_extra_info = malloc( sizeof( extra_info_t ) ); stream->video_decoder_extra_info = malloc( sizeof( extra_info_t ) ); _x_extra_info_reset( stream->current_extra_info ); _x_extra_info_reset( stream->video_decoder_extra_info ); _x_extra_info_reset( stream->audio_decoder_extra_info ); stream->xine = this; stream->status = XINE_STATUS_IDLE; stream->spu_decoder_plugin = NULL; stream->spu_decoder_streamtype = -1; stream->audio_out = ao; stream->audio_channel_user = -1; stream->audio_channel_auto = -1; stream->audio_decoder_plugin = NULL; stream->audio_decoder_streamtype = -1; stream->spu_channel_auto = -1; stream->spu_channel_letterbox = -1; stream->spu_channel_pan_scan = -1; stream->spu_channel_user = -1; stream->spu_channel = -1; stream->early_finish_event = 0; stream->delay_finish_event = 0; stream->gapless_switch = 0; stream->video_out = vo; if (vo) stream->video_driver = vo->driver; else stream->video_driver = NULL; stream->video_channel = 0; stream->video_decoder_plugin = NULL; stream->video_decoder_streamtype = -1; stream->header_count_audio = 0; stream->header_count_video = 0; stream->finished_count_audio = 0; stream->finished_count_video = 0; stream->err = 0; stream->broadcaster = NULL; /* * initial master/slave */ stream->master = stream; stream->slave = NULL; stream->slave_is_subtitle = 0; /* * init mutexes and conditions */ pthread_mutex_init (&stream->info_mutex, NULL); pthread_mutex_init (&stream->meta_mutex, NULL); pthread_mutex_init (&stream->demux_lock, NULL); pthread_mutex_init (&stream->demux_mutex, NULL); pthread_mutex_init (&stream->event_queues_lock, NULL); pthread_mutex_init (&stream->counter_lock, NULL); pthread_cond_init (&stream->counter_changed, NULL); pthread_mutex_init (&stream->first_frame_lock, NULL); pthread_cond_init (&stream->first_frame_reached, NULL); pthread_mutex_init (&stream->current_extra_info_lock, NULL); pthread_mutex_init (&stream->speed_change_lock, NULL); /* warning: frontend_lock is a recursive mutex. it must NOT be * used with neither pthread_cond_wait() or pthread_cond_timedwait() */ pthread_mutexattr_init(&attr); pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE); pthread_mutex_init (&stream->frontend_lock, &attr); /* * Clear meta/stream info */ for (i = 0; i < XINE_STREAM_INFO_MAX; i++) { _x_stream_info_reset(stream, i); _x_stream_info_public_reset(stream, i); _x_meta_info_reset(stream, i); _x_meta_info_public_reset(stream, i); } /* * event queues */ stream->event_queues = xine_list_new (); /* * create a metronom */ stream->metronom = _x_metronom_init ( (vo != NULL), (ao != NULL), this); /* * alloc fifos, init and start decoder threads */ if (!_x_video_decoder_init (stream)) { free(stream->audio_decoder_extra_info); free(stream->current_extra_info); free(stream); pthread_mutex_unlock(&this->streams_lock); return NULL; } if (!_x_audio_decoder_init (stream)) { _x_video_decoder_shutdown(stream); free(stream->audio_decoder_extra_info); free(stream->current_extra_info); free(stream); pthread_mutex_unlock(&this->streams_lock); return NULL; } /* * osd */ if (vo) stream->osd_renderer = _x_osd_renderer_init(stream); else stream->osd_renderer = NULL; /* * create a reference counter */ stream->refcounter = _x_new_refcounter(stream, (refcounter_destructor)xine_dispose_internal); if (!stream->refcounter) { _x_video_decoder_shutdown(stream); _x_audio_decoder_shutdown(stream); free(stream->audio_decoder_extra_info); free(stream->current_extra_info); free(stream); pthread_mutex_unlock(&this->streams_lock); return NULL; } /* * register stream */ xine_list_push_back (this->streams, stream); pthread_mutex_unlock (&this->streams_lock); stream->video_source.name = "video source"; stream->video_source.type = XINE_POST_DATA_VIDEO; stream->video_source.data = stream; stream->video_source.rewire = stream_rewire_video; stream->audio_source.name = "audio source"; stream->audio_source.type = XINE_POST_DATA_AUDIO; stream->audio_source.data = stream; stream->audio_source.rewire = stream_rewire_audio; return stream; } static void mrl_unescape(char *mrl) { int i, len = strlen(mrl); for (i = 0; i < len; i++) { if ((mrl[i]=='%') && (i<(len-2))) { unsigned int c; if (sscanf(&mrl[i + 1], "%02x", &c) == 1) { mrl[i]= (char)c; memmove(mrl + i + 1, mrl + i + 3, len - i - 3); len -= 2; } } } mrl[len] = 0; } void _x_flush_events_queues (xine_stream_t *stream) { xine_list_iterator_t ite; pthread_mutex_lock (&stream->event_queues_lock); /* No events queue? */ for (ite = xine_list_front (stream->event_queues); ite; ite = xine_list_next (stream->event_queues, ite)) { xine_event_queue_t *queue = xine_list_get_value(stream->event_queues, ite); pthread_mutex_lock (&queue->lock); pthread_mutex_unlock (&stream->event_queues_lock); /* we might have been called from the very same function that * processes events, therefore waiting here would cause deadlock. * check only queues with listener threads which are not * currently executing their callback functions. */ if (queue->listener_thread != NULL && !queue->callback_running) { while (!xine_list_empty (queue->events)) { pthread_cond_wait (&queue->events_processed, &queue->lock); } } pthread_mutex_unlock (&queue->lock); pthread_mutex_lock (&stream->event_queues_lock); } pthread_mutex_unlock (&stream->event_queues_lock); } static int open_internal (xine_stream_t *stream, const char *mrl) { const char *stream_setup; int no_cache = 0; if (!mrl) { xprintf (stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; if (stream->status != XINE_STATUS_IDLE) stream->status = XINE_STATUS_STOP; return 0; } lprintf ("opening MRL '%s'...\n", mrl); /* * stop engine if necessary */ close_internal (stream); lprintf ("engine should be stopped now\n"); /* * look for a stream_setup in MRL and try finding an input plugin */ stream_setup = mrl; /* look for the next '#' or try the whole MRL, if none is found */ while (*stream_setup && (stream_setup = (strchr(stream_setup, '#') ? strchr(stream_setup, '#') : strlen(mrl) + mrl))) { char *input_source = (char *)malloc(stream_setup - mrl + 1); memcpy(input_source, mrl, stream_setup - mrl); input_source[stream_setup - mrl] = '\0'; /* * find an input plugin */ if ((stream->input_plugin = _x_find_input_plugin (stream, input_source))) { int res; xine_log (stream->xine, XINE_LOG_MSG, _("xine: found input plugin : %s\n"), stream->input_plugin->input_class->get_description(stream->input_plugin->input_class)); if (stream->input_plugin->input_class->eject_media) stream->eject_class = stream->input_plugin->input_class; _x_meta_info_set_utf8(stream, XINE_META_INFO_INPUT_PLUGIN, (stream->input_plugin->input_class->get_identifier (stream->input_plugin->input_class))); res = stream->input_plugin->open(stream->input_plugin); switch(res) { case 1: /* Open successfull */ free(input_source); break; case -1: /* Open unsuccessfull, but correct plugin */ free(input_source); stream->err = XINE_ERROR_INPUT_FAILED; _x_flush_events_queues (stream); return 0; default: xine_log (stream->xine, XINE_LOG_MSG, _("xine: input plugin cannot open MRL [%s]\n"),mrl); stream->input_plugin->dispose(stream->input_plugin); stream->input_plugin = NULL; stream->err = XINE_ERROR_INPUT_FAILED; } if ( res ) break; } free(input_source); /* if we fail when passing up to the first '#' to the input plugins, * maybe the user stated a (invalid) MRL, with a '#' belonging to the * input source -> look for the next '#' and try again */ if (*stream_setup) stream_setup++; } if (!stream->input_plugin) { xine_log (stream->xine, XINE_LOG_MSG, _("xine: cannot find input plugin for MRL [%s]\n"),mrl); stream->err = XINE_ERROR_NO_INPUT_PLUGIN; _x_flush_events_queues (stream); return 0; } if (*stream_setup) { while (stream_setup && *stream_setup && *(++stream_setup)) { if (strncasecmp(stream_setup, "demux", 5) == 0) { if (*(stream_setup += 5) == ':') { /* demuxer specified by name */ const char *tmp = ++stream_setup; char *demux_name; stream_setup = strchr(stream_setup, ';'); if (stream_setup) { demux_name = (char *)malloc(stream_setup - tmp + 1); memcpy(demux_name, tmp, stream_setup - tmp); demux_name[stream_setup - tmp] = '\0'; } else { demux_name = (char *)malloc(strlen(tmp) + 1); memcpy(demux_name, tmp, strlen(tmp)); demux_name[strlen(tmp)] = '\0'; } mrl_unescape(demux_name); if (!(stream->demux_plugin = _x_find_demux_plugin_by_name(stream, demux_name, stream->input_plugin))) { xine_log(stream->xine, XINE_LOG_MSG, _("xine: specified demuxer %s failed to start\n"), demux_name); stream->err = XINE_ERROR_NO_DEMUX_PLUGIN; stream->status = XINE_STATUS_IDLE; free(demux_name); return 0; } _x_meta_info_set_utf8(stream, XINE_META_INFO_SYSTEMLAYER, (stream->demux_plugin->demux_class->get_identifier(stream->demux_plugin->demux_class))); free(demux_name); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } continue; } if (strncasecmp(stream_setup, "save", 4) == 0) { if (*(stream_setup += 4) == ':') { /* filename to save */ const char *tmp = ++stream_setup; char *filename; input_plugin_t *input_saver; stream_setup = strchr(stream_setup, ';'); if (stream_setup) { filename = (char *)malloc(stream_setup - tmp + 1); memcpy(filename, tmp, stream_setup - tmp); filename[stream_setup - tmp] = '\0'; } else { filename = (char *)malloc(strlen(tmp) + 1); memcpy(filename, tmp, strlen(tmp)); filename[strlen(tmp)] = '\0'; } xine_log(stream->xine, XINE_LOG_MSG, _("xine: join rip input plugin\n")); input_saver = _x_rip_plugin_get_instance (stream, filename); free(filename); if( input_saver ) { stream->input_plugin = input_saver; } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error opening rip input plugin instance\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } continue; } if (strncasecmp(stream_setup, "lastdemuxprobe", 14) == 0) { if (*(stream_setup += 14) == ':') { /* all demuxers will be probed before the specified one */ const char *tmp = ++stream_setup; char *demux_name; stream_setup = strchr(stream_setup, ';'); if (stream_setup) { demux_name = (char *)malloc(stream_setup - tmp + 1); memcpy(demux_name, tmp, stream_setup - tmp); demux_name[stream_setup - tmp] = '\0'; } else { demux_name = (char *)malloc(strlen(tmp) + 1); memcpy(demux_name, tmp, strlen(tmp)); demux_name[strlen(tmp)] = '\0'; } mrl_unescape(demux_name); if (!(stream->demux_plugin = _x_find_demux_plugin_last_probe(stream, demux_name, stream->input_plugin))) { xine_log(stream->xine, XINE_LOG_MSG, _("xine: last_probed demuxer %s failed to start\n"), demux_name); stream->err = XINE_ERROR_NO_DEMUX_PLUGIN; stream->status = XINE_STATUS_IDLE; free(demux_name); return 0; } lprintf ("demux and input plugin found\n"); _x_meta_info_set_utf8(stream, XINE_META_INFO_SYSTEMLAYER, (stream->demux_plugin->demux_class->get_identifier(stream->demux_plugin->demux_class))); free(demux_name); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } continue; } if (strncasecmp(stream_setup, "novideo", 7) == 0) { stream_setup += 7; if (*stream_setup == ';' || *stream_setup == '\0') { _x_stream_info_set(stream, XINE_STREAM_INFO_IGNORE_VIDEO, 1); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } xprintf (stream->xine, XINE_VERBOSITY_LOG, _("ignoring video\n")); continue; } if (strncasecmp(stream_setup, "noaudio", 7) == 0) { stream_setup += 7; if (*stream_setup == ';' || *stream_setup == '\0') { _x_stream_info_set(stream, XINE_STREAM_INFO_IGNORE_AUDIO, 1); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } xprintf (stream->xine, XINE_VERBOSITY_LOG, _("ignoring audio\n")); continue; } if (strncasecmp(stream_setup, "nospu", 5) == 0) { stream_setup += 5; if (*stream_setup == ';' || *stream_setup == '\0') { _x_stream_info_set(stream, XINE_STREAM_INFO_IGNORE_SPU, 1); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } xprintf (stream->xine, XINE_VERBOSITY_LOG, _("ignoring subpicture\n")); continue; } if (strncasecmp(stream_setup, "nocache", 7) == 0) { stream_setup += 7; if (*stream_setup == ';' || *stream_setup == '\0') { no_cache = 1; } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } xprintf (stream->xine, XINE_VERBOSITY_LOG, _("input cache plugin disabled\n")); continue; } if (strncasecmp(stream_setup, "volume", 6) == 0) { if (*(stream_setup += 6) == ':') { const char *tmp = ++stream_setup; char *volume; stream_setup = strchr(stream_setup, ';'); if (stream_setup) { volume = (char *)malloc(stream_setup - tmp + 1); memcpy(volume, tmp, stream_setup - tmp); volume[stream_setup - tmp] = '\0'; } else { volume = (char *)malloc(strlen(tmp) + 1); memcpy(volume, tmp, strlen(tmp)); volume[strlen(tmp)] = '\0'; } mrl_unescape(volume); xine_set_param(stream, XINE_PARAM_AUDIO_VOLUME, atoi(volume)); free(volume); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } continue; } if (strncasecmp(stream_setup, "compression", 11) == 0) { if (*(stream_setup += 11) == ':') { const char *tmp = ++stream_setup; char *compression; stream_setup = strchr(stream_setup, ';'); if (stream_setup) { compression = (char *)malloc(stream_setup - tmp + 1); memcpy(compression, tmp, stream_setup - tmp); compression[stream_setup - tmp] = '\0'; } else { compression = (char *)malloc(strlen(tmp) + 1); memcpy(compression, tmp, strlen(tmp)); compression[strlen(tmp)] = '\0'; } mrl_unescape(compression); xine_set_param(stream, XINE_PARAM_AUDIO_COMPR_LEVEL, atoi(compression)); free(compression); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } continue; } if (strncasecmp(stream_setup, "subtitle", 8) == 0) { if (*(stream_setup += 8) == ':') { const char *tmp = ++stream_setup; char *subtitle_mrl; stream_setup = strchr(stream_setup, ';'); if (stream_setup) { subtitle_mrl = (char *)malloc(stream_setup - tmp + 1); memcpy(subtitle_mrl, tmp, stream_setup - tmp); subtitle_mrl[stream_setup - tmp] = '\0'; } else { subtitle_mrl = (char *)malloc(strlen(tmp) + 1); memcpy(subtitle_mrl, tmp, strlen(tmp)); subtitle_mrl[strlen(tmp)] = '\0'; } mrl_unescape(subtitle_mrl); stream->slave = xine_stream_new (stream->xine, NULL, stream->video_out ); stream->slave_affection = XINE_MASTER_SLAVE_PLAY | XINE_MASTER_SLAVE_STOP; if( xine_open( stream->slave, subtitle_mrl ) ) { xprintf (stream->xine, XINE_VERBOSITY_LOG, _("subtitle mrl opened '%s'\n"), subtitle_mrl); stream->slave->master = stream; stream->slave_is_subtitle = 1; } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error opening subtitle mrl\n")); xine_dispose( stream->slave ); stream->slave = NULL; } free(subtitle_mrl); } else { xprintf(stream->xine, XINE_VERBOSITY_LOG, _("xine: error while parsing mrl\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; return 0; } continue; } { /* when we got here, the stream setup parameter must be a config entry */ const char *tmp = stream_setup; char *config_entry; int retval; if ((stream_setup = strchr(stream_setup, ';'))) { config_entry = (char *)malloc(stream_setup - tmp + 1); memcpy(config_entry, tmp, stream_setup - tmp); config_entry[stream_setup - tmp] = '\0'; } else { config_entry = (char *)malloc(strlen(tmp) + 1); memcpy(config_entry, tmp, strlen(tmp)); config_entry[strlen(tmp)] = '\0'; } mrl_unescape(config_entry); retval = _x_config_change_opt(stream->xine->config, config_entry); if (retval <= 0) { if (retval == 0) { /* the option not found */ xine_log(stream->xine, XINE_LOG_MSG, _("xine: error while parsing MRL\n")); stream->err = XINE_ERROR_MALFORMED_MRL; stream->status = XINE_STATUS_IDLE; free(config_entry); return 0; } else { /* not permitted to change from MRL */ xine_log(stream->xine, XINE_LOG_MSG, _("xine: changing option '%s' from MRL isn't permitted\n"), config_entry); } } free(config_entry); } } } if( !no_cache ) /* enable buffered input plugin (request optimizer) */ stream->input_plugin = _x_cache_plugin_get_instance(stream, 0); if (!stream->demux_plugin) { /* * find a demux plugin */ if (!(stream->demux_plugin = _x_find_demux_plugin (stream, stream->input_plugin))) { xine_log (stream->xine, XINE_LOG_MSG, _("xine: couldn't find demux for >%s<\n"), mrl); stream->err = XINE_ERROR_NO_DEMUX_PLUGIN; stream->status = XINE_STATUS_IDLE; /* force the engine to unregister fifo callbacks */ _x_demux_control_nop(stream, BUF_FLAG_END_STREAM); return 0; } lprintf ("demux and input plugin found\n"); _x_meta_info_set_utf8(stream, XINE_META_INFO_SYSTEMLAYER, (stream->demux_plugin->demux_class->get_identifier(stream->demux_plugin->demux_class))); } xine_log (stream->xine, XINE_LOG_MSG, _("xine: found demuxer plugin: %s\n"), stream->demux_plugin->demux_class->get_description(stream->demux_plugin->demux_class)); _x_extra_info_reset( stream->current_extra_info ); _x_extra_info_reset( stream->video_decoder_extra_info ); _x_extra_info_reset( stream->audio_decoder_extra_info ); /* assume handled for now. we will only know for sure after trying * to init decoders (which should happen when headers are sent) */ _x_stream_info_set(stream, XINE_STREAM_INFO_VIDEO_HANDLED, 1); _x_stream_info_set(stream, XINE_STREAM_INFO_AUDIO_HANDLED, 1); /* * send and decode headers */ stream->demux_plugin->send_headers (stream->demux_plugin); if (stream->demux_plugin->get_status(stream->demux_plugin) != DEMUX_OK) { xine_log (stream->xine, XINE_LOG_MSG, _("xine: demuxer failed to start\n")); _x_free_demux_plugin(stream, stream->demux_plugin); stream->demux_plugin = NULL; xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "demux disposed\n"); _x_free_input_plugin(stream, stream->input_plugin); stream->input_plugin = NULL; stream->err = XINE_ERROR_NO_DEMUX_PLUGIN; stream->status = XINE_STATUS_IDLE; xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "return from\n"); return 0; } _x_demux_control_headers_done (stream); stream->status = XINE_STATUS_STOP; lprintf ("done\n"); return 1; } int xine_open (xine_stream_t *stream, const char *mrl) { int ret; pthread_mutex_lock (&stream->frontend_lock); pthread_cleanup_push (mutex_cleanup, (void *) &stream->frontend_lock); lprintf ("open MRL:%s\n", mrl); ret = open_internal (stream, mrl); pthread_cleanup_pop (0); pthread_mutex_unlock (&stream->frontend_lock); return ret; } static void wait_first_frame (xine_stream_t *stream) { if (stream->video_decoder_plugin) { pthread_mutex_lock (&stream->first_frame_lock); if (stream->first_frame_flag > 0) { struct timeval tv; struct timespec ts; gettimeofday(&tv, NULL); ts.tv_sec = tv.tv_sec + 10; ts.tv_nsec = tv.tv_usec * 1000; pthread_cond_timedwait(&stream->first_frame_reached, &stream->first_frame_lock, &ts); } pthread_mutex_unlock (&stream->first_frame_lock); } } static int play_internal (xine_stream_t *stream, int start_pos, int start_time) { int demux_status; int demux_thread_running; xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "xine_play\n"); if (!stream->demux_plugin) { xine_log (stream->xine, XINE_LOG_MSG, _("xine_play: no demux available\n")); stream->err = XINE_ERROR_NO_DEMUX_PLUGIN; return 0; } /* hint demuxer thread we want to interrupt it */ stream->demux_action_pending = 1; /* set normal speed */ if (_x_get_speed(stream) != XINE_SPEED_NORMAL) set_speed_internal (stream, XINE_FINE_SPEED_NORMAL); stream->xine->port_ticket->acquire(stream->xine->port_ticket, 1); /* only flush/discard output ports on master streams */ if( stream->master == stream && !stream->gapless_switch) { /* discard audio/video buffers to get engine going and take the lock faster */ if (stream->audio_out) stream->audio_out->set_property(stream->audio_out, AO_PROP_DISCARD_BUFFERS, 1); if (stream->video_out) stream->video_out->set_property(stream->video_out, VO_PROP_DISCARD_FRAMES, 1); } pthread_mutex_lock( &stream->demux_lock ); /* demux_lock taken. now demuxer is suspended */ stream->demux_action_pending = 0; /* set normal speed again (now that demuxer/input pair is suspended) * some input plugin may have changed speed by itself, we must ensure * the engine is not paused. */ if (_x_get_speed(stream) != XINE_SPEED_NORMAL) set_speed_internal (stream, XINE_FINE_SPEED_NORMAL); /* * start/seek demux */ /* seek to new position (no data is sent to decoders yet) */ demux_status = stream->demux_plugin->seek (stream->demux_plugin, start_pos, start_time, stream->demux_thread_running); /* only flush/discard output ports on master streams */ if( stream->master == stream && !stream->gapless_switch) { if (stream->audio_out) stream->audio_out->set_property(stream->audio_out, AO_PROP_DISCARD_BUFFERS, 0); if (stream->video_out) stream->video_out->set_property(stream->video_out, VO_PROP_DISCARD_FRAMES, 0); } stream->xine->port_ticket->release(stream->xine->port_ticket, 1); /* before resuming the demuxer, set first_frame_flag */ pthread_mutex_lock (&stream->first_frame_lock); stream->first_frame_flag = 2; pthread_mutex_unlock (&stream->first_frame_lock); /* before resuming the demuxer, reset current position information */ pthread_mutex_lock( &stream->current_extra_info_lock ); _x_extra_info_reset( stream->current_extra_info ); pthread_mutex_unlock( &stream->current_extra_info_lock ); demux_thread_running = stream->demux_thread_running; /* now resume demuxer thread if it is running already */ pthread_mutex_unlock( &stream->demux_lock ); if (demux_status != DEMUX_OK) { xine_log (stream->xine, XINE_LOG_MSG, _("xine_play: demux failed to start\n")); stream->err = XINE_ERROR_DEMUX_FAILED; stream->first_frame_flag = 0; return 0; } else { if (!demux_thread_running) { _x_demux_start_thread( stream ); stream->status = XINE_STATUS_PLAY; } } /* Wait until the first frame produced is displayed * see video_out.c */ wait_first_frame (stream); xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "play_internal ...done\n"); return 1; } int xine_play (xine_stream_t *stream, int start_pos, int start_time) { int ret; pthread_mutex_lock (&stream->frontend_lock); pthread_cleanup_push (mutex_cleanup, (void *) &stream->frontend_lock); stream->delay_finish_event = 0; ret = play_internal (stream, start_pos, start_time); if( stream->slave && (stream->slave_affection & XINE_MASTER_SLAVE_PLAY) ) xine_play (stream->slave, start_pos, start_time); stream->gapless_switch = 0; pthread_cleanup_pop (0); pthread_mutex_unlock (&stream->frontend_lock); return ret; } int xine_eject (xine_stream_t *stream) { int status; if (!stream->eject_class) return 0; pthread_mutex_lock (&stream->frontend_lock); pthread_cleanup_push (mutex_cleanup, (void *) &stream->frontend_lock); status = 0; /* only eject, if we are stopped OR a different input plugin is playing */ if (stream->eject_class && stream->eject_class->eject_media && ((stream->status == XINE_STATUS_STOP) || stream->eject_class != stream->input_plugin->input_class)) { status = stream->eject_class->eject_media (stream->eject_class); } pthread_cleanup_pop (0); pthread_mutex_unlock (&stream->frontend_lock); return status; } void xine_dispose_internal (xine_stream_t *stream) { xine_list_iterator_t *ite; lprintf("stream: %p\n", stream); pthread_mutex_destroy (&stream->info_mutex); pthread_mutex_destroy (&stream->meta_mutex); pthread_mutex_destroy (&stream->frontend_lock); pthread_mutex_destroy (&stream->counter_lock); pthread_mutex_destroy (&stream->event_queues_lock); pthread_mutex_destroy (&stream->current_extra_info_lock); pthread_cond_destroy (&stream->counter_changed); pthread_mutex_destroy (&stream->demux_mutex); pthread_mutex_destroy (&stream->demux_lock); pthread_mutex_destroy (&stream->first_frame_lock); pthread_cond_destroy (&stream->first_frame_reached); stream->metronom->exit (stream->metronom); pthread_mutex_lock(&stream->xine->streams_lock); ite = xine_list_find(stream->xine->streams, stream); if (ite) { xine_list_remove(stream->xine->streams, ite); } pthread_mutex_unlock(&stream->xine->streams_lock); _x_refcounter_dispose(stream->refcounter); free (stream->current_extra_info); free (stream->video_decoder_extra_info); free (stream->audio_decoder_extra_info); free (stream); } void xine_dispose (xine_stream_t *stream) { /* decrease the reference counter * if there is no more reference on this stream, the xine_dispose_internal * function is called */ xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "xine_dispose\n"); stream->status = XINE_STATUS_QUIT; xine_close(stream); if( stream->master != stream ) { stream->master->slave = NULL; } if( stream->slave && stream->slave->master == stream ) { stream->slave->master = NULL; } if(stream->broadcaster) _x_close_broadcaster(stream->broadcaster); xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "shutdown audio\n"); _x_audio_decoder_shutdown (stream); xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "shutdown video\n"); _x_video_decoder_shutdown (stream); if (stream->osd_renderer) stream->osd_renderer->close( stream->osd_renderer ); _x_refcounter_dec(stream->refcounter); } void xine_exit (xine_t *this) { int i; xprintf (this, XINE_VERBOSITY_DEBUG, "xine_exit: bye!\n"); for (i = 0; i < XINE_LOG_NUM; i++) if ( this->log_buffers[i] ) this->log_buffers[i]->dispose (this->log_buffers[i]); _x_dispose_plugins (this); if(this->streams) { xine_list_delete(this->streams); pthread_mutex_destroy(&this->streams_lock); } if(this->clock) this->clock->exit (this->clock); if(this->config) this->config->dispose(this->config); if(this->port_ticket) this->port_ticket->dispose(this->port_ticket); #if defined(WIN32) WSACleanup(); #endif free (this); } xine_t *xine_new (void) { xine_t *this; #ifdef WIN32 WSADATA Data; int i_err; #endif this = xine_xmalloc (sizeof (xine_t)); if (!this) _x_abort(); this->plugin_catalog = NULL; this->save_path = NULL; this->streams = NULL; this->clock = NULL; this->port_ticket = NULL; #ifdef ENABLE_NLS /* * i18n */ bindtextdomain(XINE_TEXTDOMAIN, XINE_LOCALEDIR); #endif /* * config */ this->config = _x_config_init (); /* * log buffers */ memset(this->log_buffers, 0, sizeof(this->log_buffers)); #ifdef WIN32 /* WinSock Library Init. */ i_err = WSAStartup( MAKEWORD( 1, 1 ), &Data ); if( i_err ) { fprintf( stderr, "error: can't initiate WinSocks, error %i\n", i_err ); } #endif /* WIN32 */ this->verbosity = XINE_VERBOSITY_NONE; return this; } void xine_engine_set_param(xine_t *this, int param, int value) { if(this) { switch(param) { case XINE_ENGINE_PARAM_VERBOSITY: this->verbosity = value; break; default: lprintf("Unknown parameter %d\n", param); break; } } } int xine_engine_get_param(xine_t *this, int param) { if(this) { switch(param) { case XINE_ENGINE_PARAM_VERBOSITY: return this->verbosity; break; default: lprintf("Unknown parameter %d\n", param); break; } } return -1; } static void config_demux_strategy_cb (void *this_gen, xine_cfg_entry_t *entry) { xine_t *this = (xine_t *)this_gen; this->demux_strategy = entry->num_value; } static void config_save_cb (void *this_gen, xine_cfg_entry_t *entry) { xine_t *this = (xine_t *)this_gen; char *homedir_trail_slash; homedir_trail_slash = (char *)malloc(strlen(xine_get_homedir()) + 2); sprintf(homedir_trail_slash, "%s/", xine_get_homedir()); if (entry->str_value[0] && (entry->str_value[0] != '/' || strstr(entry->str_value, "/.") || strcmp(entry->str_value, xine_get_homedir()) == 0 || strcmp(entry->str_value, homedir_trail_slash) == 0)) { xine_stream_t *stream; xine_list_iterator_t ite; xine_log(this, XINE_LOG_MSG, _("xine: The specified save_dir \"%s\" might be a security risk.\n"), entry->str_value); pthread_mutex_lock(&this->streams_lock); if ( (ite = xine_list_front(this->streams)) ) { stream = xine_list_get_value(this->streams, ite); _x_message(stream, XINE_MSG_SECURITY, _("The specified save_dir might be a security risk."), NULL); } pthread_mutex_unlock(&this->streams_lock); } free(homedir_trail_slash); this->save_path = entry->str_value; } void xine_init (xine_t *this) { static const char *demux_strategies[] = {"default", "reverse", "content", "extension", NULL}; /* initialize color conversion tables and functions */ init_yuv_conversion(); /* probe for optimized memcpy or config setting */ xine_probe_fast_memcpy (this); /* * plugins */ _x_scan_plugins(this); #ifdef HAVE_SETLOCALE if (!setlocale(LC_CTYPE, "")) xprintf(this, XINE_VERBOSITY_LOG, _("xine: locale not supported by C library\n")); #endif /* * content detection strategy */ this->demux_strategy = this->config->register_enum ( this->config, "engine.demux.strategy", 0, demux_strategies, _("media format detection strategy"), _("xine offers various methods to detect the media format of input to play. " "The individual values are:\n\n" "default\n" "First try to detect by content, then by file name extension.\n\n" "reverse\n" "First try to detect by file name extension, then by content.\n\n" "content\n" "Detect by content only.\n\n" "extension\n" "Detect by file name extension only.\n"), 20, config_demux_strategy_cb, this); /* * save directory */ this->save_path = this->config->register_filename ( this->config, "media.capture.save_dir", "", XINE_CONFIG_STRING_IS_DIRECTORY_NAME, _("directory for saving streams"), _("When using the stream save feature, files will be written only into this directory.\n" "This setting is security critical, because when changed to a different directory, xine " "can be used to fill files in it with arbitrary content. So you should be careful that " "the directory you specify is robust against any content in any file."), XINE_CONFIG_SECURITY, config_save_cb, this); /* * implicit configuration changes */ this->config->register_bool(this->config, "misc.implicit_config", 0, _("allow implicit changes to the configuration (e.g. by MRL)"), _("If enabled, you allow xine to change your configuration without " "explicit actions from your side. For example configuration changes " "demanded by MRLs or embedded into playlist will be executed.\n" "This setting is security critcal, because xine can receive MRLs or " "playlists from untrusted remote sources. If you allow them to " "arbitrarily change your configuration, you might end with a totally " "messed up xine."), XINE_CONFIG_SECURITY, NULL, this); /* * timeout for network I/O to avoid freezes */ this->config->register_num(this->config, "media.network.timeout", 30, _("Timeout for network stream reading (in seconds)"), _("Specifies the timeout when reading from network streams, in seconds. " "Too low values might stop streaming when the source is slow or the " "bandwidth is occupied, too high values will freeze the player if the " "connection is lost."), 0, NULL, this); /* * keep track of all opened streams */ this->streams = xine_list_new(); /* * locks */ pthread_mutex_init (&this->streams_lock, NULL); pthread_mutex_init (&this->log_lock, NULL); /* * start metronom clock */ this->clock = _x_metronom_clock_init(this); this->clock->start_clock (this->clock, 0); /* * tickets */ this->port_ticket = ticket_init(); } void _x_select_spu_channel (xine_stream_t *stream, int channel) { pthread_mutex_lock (&stream->frontend_lock); stream->spu_channel_user = (channel >= -2 ? channel : -2); stream->xine->port_ticket->acquire(stream->xine->port_ticket, 0); switch (stream->spu_channel_user) { case -2: stream->spu_channel = -1; if(stream->video_out) stream->video_out->enable_ovl (stream->video_out, 0); break; case -1: stream->spu_channel = stream->spu_channel_auto; if(stream->video_out) stream->video_out->enable_ovl (stream->video_out, 1); break; default: stream->spu_channel = stream->spu_channel_user; if(stream->video_out) stream->video_out->enable_ovl (stream->video_out, 1); } lprintf("set to %d\n",stream->spu_channel); stream->xine->port_ticket->release(stream->xine->port_ticket, 0); pthread_mutex_unlock (&stream->frontend_lock); } static int get_current_position (xine_stream_t *stream) { int pos; pthread_mutex_lock (&stream->frontend_lock); if (!stream->input_plugin) { lprintf ("no input source\n"); pthread_mutex_unlock (&stream->frontend_lock); return -1; } if ( (!stream->video_decoder_plugin && !stream->audio_decoder_plugin) ) { if( _x_stream_info_get(stream, XINE_STREAM_INFO_HAS_VIDEO) ) _x_extra_info_merge( stream->current_extra_info, stream->video_decoder_extra_info ); else _x_extra_info_merge( stream->current_extra_info, stream->audio_decoder_extra_info ); } if ( stream->current_extra_info->seek_count != stream->video_seek_count ) { pthread_mutex_unlock (&stream->frontend_lock); return -1; /* position not yet known */ } pthread_mutex_lock( &stream->current_extra_info_lock ); pos = stream->current_extra_info->input_normpos; pthread_mutex_unlock( &stream->current_extra_info_lock ); pthread_mutex_unlock (&stream->frontend_lock); return pos; } void _x_get_current_info (xine_stream_t *stream, extra_info_t *extra_info, int size) { pthread_mutex_lock( &stream->current_extra_info_lock ); memcpy( extra_info, stream->current_extra_info, size ); pthread_mutex_unlock( &stream->current_extra_info_lock ); } int xine_get_status (xine_stream_t *stream) { return stream->status; } /* * trick play */ void _x_set_fine_speed (xine_stream_t *stream, int speed) { pthread_mutex_lock(&stream->speed_change_lock); if (!stream->ignore_speed_change) { if (speed <= XINE_SPEED_PAUSE) speed = XINE_SPEED_PAUSE; xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "set_speed %d\n", speed); set_speed_internal (stream, speed); if (stream->slave && (stream->slave_affection & XINE_MASTER_SLAVE_SPEED)) set_speed_internal (stream->slave, speed); } pthread_mutex_unlock(&stream->speed_change_lock); } int _x_get_fine_speed (xine_stream_t *stream) { return stream->xine->clock->speed; } void _x_set_speed (xine_stream_t *stream, int speed) { if (speed > XINE_SPEED_FAST_4) speed = XINE_SPEED_FAST_4; _x_set_fine_speed (stream, speed * XINE_FINE_SPEED_NORMAL / XINE_SPEED_NORMAL); } int _x_get_speed (xine_stream_t *stream) { int speed = _x_get_fine_speed (stream); /* * ensure compatibility with old API, only valid XINE_SPEED_xxx * constants are allowed. XINE_SPEED_NORMAL may only be returned * if speed is exactly XINE_FINE_SPEED_NORMAL. */ if( speed <= XINE_SPEED_PAUSE ) return XINE_SPEED_PAUSE; if( speed <= XINE_SPEED_SLOW_4 * XINE_FINE_SPEED_NORMAL / XINE_SPEED_NORMAL ) return XINE_SPEED_SLOW_4; if( speed < XINE_FINE_SPEED_NORMAL ) return XINE_SPEED_SLOW_2; if( speed == XINE_FINE_SPEED_NORMAL ) return XINE_SPEED_NORMAL; if( speed <= XINE_SPEED_FAST_2 * XINE_FINE_SPEED_NORMAL / XINE_SPEED_NORMAL ) return XINE_SPEED_FAST_2; return XINE_SPEED_FAST_4; } /* * time measurement / seek */ static int get_stream_length (xine_stream_t *stream) { /* pthread_mutex_lock( &stream->demux_lock ); */ if (stream->demux_plugin) { int len = stream->demux_plugin->get_stream_length (stream->demux_plugin); /* pthread_mutex_unlock( &stream->demux_lock ); */ return len; } /* pthread_mutex_unlock( &stream->demux_lock ); */ return 0; } int xine_get_pos_length (xine_stream_t *stream, int *pos_stream, int *pos_time, int *length_time) { int pos = get_current_position (stream); /* force updating extra_info */ if (pos == -1) return 0; if (pos_stream) *pos_stream = pos; if (pos_time) { pthread_mutex_lock( &stream->current_extra_info_lock ); *pos_time = stream->current_extra_info->input_time; pthread_mutex_unlock( &stream->current_extra_info_lock ); } if (length_time) *length_time = get_stream_length (stream); return 1; } int xine_get_current_frame (xine_stream_t *stream, int *width, int *height, int *ratio_code, int *format, uint8_t *img) { vo_frame_t *frame; stream->xine->port_ticket->acquire(stream->xine->port_ticket, 0); frame = stream->video_out->get_last_frame (stream->video_out); stream->xine->port_ticket->release(stream->xine->port_ticket, 0); if (!frame) return 0; *width = frame->width; *height = frame->height; *ratio_code = 10000.0 * frame->ratio; /* make ratio_code backward compatible */ #define RATIO_LIKE(a, b) ((b) - 1 <= (a) && (a) <= 1 + (b)) if (RATIO_LIKE(*ratio_code, 10000)) *ratio_code = XINE_VO_ASPECT_SQUARE; else if (RATIO_LIKE(*ratio_code, 13333)) *ratio_code = XINE_VO_ASPECT_4_3; else if (RATIO_LIKE(*ratio_code, 17778)) *ratio_code = XINE_VO_ASPECT_ANAMORPHIC; else if (RATIO_LIKE(*ratio_code, 21100)) *ratio_code = XINE_VO_ASPECT_DVB; *format = frame->format; if (img){ switch (frame->format) { case XINE_IMGFMT_YV12: yv12_to_yv12( /* Y */ frame->base[0], frame->pitches[0], img, frame->width, /* U */ frame->base[1], frame->pitches[1], img+frame->width*frame->height, frame->width/2, /* V */ frame->base[2], frame->pitches[2], img+frame->width*frame->height+frame->width*frame->height/4, frame->width/2, /* width x height */ frame->width, frame->height); break; case XINE_IMGFMT_YUY2: yuy2_to_yuy2( /* src */ frame->base[0], frame->pitches[0], /* dst */ img, frame->width*2, /* width x height */ frame->width, frame->height); break; default: xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "xine: error, snapshot function not implemented for format 0x%x\n", frame->format); _x_abort (); } } return 1; } int xine_get_video_frame (xine_stream_t *stream, int timestamp, /* msec */ int *width, int *height, int *ratio_code, int *duration, /* msec */ int *format, uint8_t *img) { xprintf (stream->xine, XINE_VERBOSITY_DEBUG, "xine: xine_get_video_frame not implemented yet.\n"); _x_abort (); return 0; } int xine_get_spu_lang (xine_stream_t *stream, int channel, char *lang) { /* Ask the demuxer first (e.g. TS extracts this information from * the stream) **/ if (stream->demux_plugin) { if (stream->demux_plugin->get_capabilities (stream->demux_plugin) & DEMUX_CAP_SPULANG) { /* pass the channel number to the plugin in the data field */ *((int *)lang) = channel; if (stream->demux_plugin->get_optional_data (stream->demux_plugin, lang, DEMUX_OPTIONAL_DATA_SPULANG) == DEMUX_OPTIONAL_SUCCESS) return 1; } } /* No match, check with input plugin instead (e.g. DVD gets this * info from the IFO). **/ if (stream->input_plugin) { if (stream->input_plugin->get_capabilities (stream->input_plugin) & INPUT_CAP_SPULANG) { /* pass the channel number to the plugin in the data field */ *((int *)lang) = channel; if (stream->input_plugin->get_optional_data (stream->input_plugin, lang, INPUT_OPTIONAL_DATA_SPULANG) == INPUT_OPTIONAL_SUCCESS) return 1; } } return 0; } int xine_get_audio_lang (xine_stream_t *stream, int channel, char *lang) { if (stream->demux_plugin) { if (stream->demux_plugin->get_capabilities (stream->demux_plugin) & DEMUX_CAP_AUDIOLANG) { /* pass the channel number to the plugin in the data field */ *((int *)lang) = channel; if (stream->demux_plugin->get_optional_data (stream->demux_plugin, lang, DEMUX_OPTIONAL_DATA_AUDIOLANG) == DEMUX_OPTIONAL_SUCCESS) return 1; } } if (stream->input_plugin) { if (stream->input_plugin->get_capabilities (stream->input_plugin) & INPUT_CAP_AUDIOLANG) { /* pass the channel number to the plugin in the data field */ *((int *)lang) = channel; if (stream->input_plugin->get_optional_data (stream->input_plugin, lang, INPUT_OPTIONAL_DATA_AUDIOLANG) == INPUT_OPTIONAL_SUCCESS) return 1; } } return 0; } int _x_get_spu_channel (xine_stream_t *stream) { return stream->spu_channel_user; } /* * log functions */ int xine_get_log_section_count (xine_t *this) { return XINE_LOG_NUM; } const char *const *xine_get_log_names (xine_t *this) { static const char *log_sections[XINE_LOG_NUM + 1]; log_sections[XINE_LOG_MSG] = _("messages"); log_sections[XINE_LOG_PLUGIN] = _("plugin"); log_sections[XINE_LOG_TRACE] = _("trace"); log_sections[XINE_LOG_NUM] = NULL; return log_sections; } static inline void check_log_alloc (xine_t *this, int buf) { pthread_mutex_lock (&this->log_lock); if ( ! this->log_buffers[buf] ) this->log_buffers[buf] = _x_new_scratch_buffer(150); pthread_mutex_unlock (&this->log_lock); } void xine_log (xine_t *this, int buf, const char *format, ...) { va_list argp; char buffer[SCRATCH_LINE_LEN_MAX]; check_log_alloc (this, buf); va_start (argp, format); this->log_buffers[buf]->scratch_printf (this->log_buffers[buf], format, argp); va_end(argp); if(this->verbosity) { va_start(argp, format); vsnprintf(buffer, SCRATCH_LINE_LEN_MAX, format, argp); printf("%s", buffer); va_end (argp); } } void xine_vlog(xine_t *this, int buf, const char *format, va_list args) { check_log_alloc (this, buf); this->log_buffers[buf]->scratch_printf(this->log_buffers[buf], format, args); } char *const *xine_get_log (xine_t *this, int buf) { if(buf >= XINE_LOG_NUM) return NULL; if ( this->log_buffers[buf] ) return this->log_buffers[buf]->get_content (this->log_buffers[buf]); else return NULL; } void xine_register_log_cb (xine_t *this, xine_log_cb_t cb, void *user_data) { printf ("xine: xine_register_log_cb: not implemented yet.\n"); _x_abort(); } int xine_get_error (xine_stream_t *stream) { return stream->err; } int xine_trick_mode (xine_stream_t *stream, int mode, int value) { printf ("xine: xine_trick_mode not implemented yet.\n"); _x_abort (); } int xine_stream_master_slave(xine_stream_t *master, xine_stream_t *slave, int affection) { master->slave = slave; master->slave_affection = affection; /* respect transitivity: if our designated master already has a master * of its own, we point to this master's master; if our master is a * standalone stream, its master pointer will point to itself */ slave->master = master->master; return 1; }