diff options
| author | Nicky Dasmijn <nicky.dasmijn@posteo.nl> | 2024-04-05 19:25:02 +0200 | 
|---|---|---|
| committer | GitHub <noreply@github.com> | 2024-04-05 20:25:02 +0300 | 
| commit | 57d423745fd1d3d0ea6a0c69b869a20c27e27fc5 (patch) | |
| tree | f590e086983c7277e4b05fcd54c3eebf6b12f43a /indra/media_plugins | |
| parent | cc8d71c18c124138e76e85e663498d2ee9776b3c (diff) | |
Linux viewer (ReleaseOS) resurrection (#1099)
Co-authored-by: AiraYumi <aira.youme@airanyumi.net>
Diffstat (limited to 'indra/media_plugins')
7 files changed, 1415 insertions, 1 deletions
| diff --git a/indra/media_plugins/CMakeLists.txt b/indra/media_plugins/CMakeLists.txt index 972bb7dd2d..fe8fee5e7e 100644 --- a/indra/media_plugins/CMakeLists.txt +++ b/indra/media_plugins/CMakeLists.txt @@ -3,7 +3,7 @@  add_subdirectory(base)  if (LINUX) -    #add_subdirectory(gstreamer010) +    add_subdirectory(gstreamer10)      add_subdirectory(example)  endif (LINUX) diff --git a/indra/media_plugins/gstreamer10/CMakeLists.txt b/indra/media_plugins/gstreamer10/CMakeLists.txt new file mode 100644 index 0000000000..ffa3c30519 --- /dev/null +++ b/indra/media_plugins/gstreamer10/CMakeLists.txt @@ -0,0 +1,46 @@ +# -*- cmake -*- + +project(media_plugin_gstreamer10) + +include(00-Common) +include(LLCommon) +include(LLImage) +include(LLMath) +include(LLWindow) +include(Linking) +include(PluginAPI) +include(OpenGL) +include(GLIB) + +include(GStreamer10Plugin) + +### media_plugin_gstreamer10 + +if(NOT WINDOWS) # not windows therefore gcc LINUX and DARWIN +add_definitions(-fPIC) +endif() + +set(media_plugin_gstreamer10_SOURCE_FILES +    media_plugin_gstreamer10.cpp +    llmediaimplgstreamer_syms.cpp +    ) + +set(media_plugin_gstreamer10_HEADER_FILES +    llmediaimplgstreamer_syms.h +    llmediaimplgstreamertriviallogging.h +    ) + +add_library(media_plugin_gstreamer10 +    SHARED +    ${media_plugin_gstreamer10_SOURCE_FILES} +) + +target_link_libraries(media_plugin_gstreamer10 media_plugin_base ll::gstreamer10 ) + +if (WINDOWS) +  set_target_properties( +    media_plugin_gstreamer10 +    PROPERTIES +    LINK_FLAGS "/MANIFEST:NO /SAFESEH:NO /NODEFAULTLIB:LIBCMT" +    ) +endif (WINDOWS) diff --git a/indra/media_plugins/gstreamer10/llmediaimplgstreamer.h b/indra/media_plugins/gstreamer10/llmediaimplgstreamer.h new file mode 100644 index 0000000000..6bc272c009 --- /dev/null +++ b/indra/media_plugins/gstreamer10/llmediaimplgstreamer.h @@ -0,0 +1,53 @@ +/**  + * @file llmediaimplgstreamer.h + * @author Tofu Linden + * @brief implementation that supports media playback via GStreamer. + * + * @cond + * $LicenseInfo:firstyear=2007&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library 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 + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + * @endcond + */ + +// header guard +#ifndef llmediaimplgstreamer_h +#define llmediaimplgstreamer_h + +#if LL_GSTREAMER010_ENABLED + +extern "C" { +#include <stdio.h> +#include <gst/gst.h> + +#include "apr_pools.h" +#include "apr_dso.h" +} + + +extern "C" { +gboolean llmediaimplgstreamer_bus_callback (GstBus     *bus, +					    GstMessage *message, +					    gpointer    data); +} + +#endif // LL_GSTREAMER010_ENABLED + +#endif // llmediaimplgstreamer_h diff --git a/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms.cpp b/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms.cpp new file mode 100644 index 0000000000..e5e5c1c9a3 --- /dev/null +++ b/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms.cpp @@ -0,0 +1,197 @@ +/**  + * @file llmediaimplgstreamer_syms.cpp + * @brief dynamic GStreamer symbol-grabbing code + * + * @cond + * $LicenseInfo:firstyear=2007&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library 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 + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + * @endcond + */ + +#include <string> +#include <iostream> +#include <vector> + +#ifdef LL_WINDOWS +#undef _WIN32_WINNT +#define _WIN32_WINNT 0x0502 +#include <Windows.h> +#endif + +#include "linden_common.h" + +extern "C" { +#include <gst/gst.h> +#include <gst/app/gstappsink.h> +} + +#include "apr_pools.h" +#include "apr_dso.h" + +#ifdef LL_WINDOWS + +#ifndef _M_AMD64 +#define GSTREAMER_REG_KEY "Software\\GStreamer1.0\\x86" +#define GSTREAMER_DIR_SUFFIX "1.0\\x86\\bin\\" +#else +#define GSTREAMER_REG_KEY "Software\\GStreamer1.0\\x86_64" +#define GSTREAMER_DIR_SUFFIX "1.0\\x86_64\\bin\\" +#endif + +bool openRegKey( HKEY &aKey ) +{ +	// Try native (32 bit view/64 bit view) of registry first. +	if( ERROR_SUCCESS == ::RegOpenKeyExA( HKEY_LOCAL_MACHINE, GSTREAMER_REG_KEY, 0, KEY_QUERY_VALUE, &aKey ) ) +		return true; + +	// If native view fails, use 32 bit view or registry. +	if( ERROR_SUCCESS == ::RegOpenKeyExA( HKEY_LOCAL_MACHINE, GSTREAMER_REG_KEY, 0, KEY_QUERY_VALUE | KEY_WOW64_32KEY, &aKey ) ) +		return true; + +	return false; +} + +std::string getGStreamerDir() +{ +	std::string ret; +	HKEY hKey; + +	if( openRegKey( hKey ) ) +	{ +		DWORD dwLen(0); +		::RegQueryValueExA( hKey, "InstallDir", nullptr, nullptr, nullptr, &dwLen ); + +		if( dwLen > 0 ) +		{ +			std::vector< char > vctBuffer; +			vctBuffer.resize( dwLen ); +			::RegQueryValueExA( hKey, "InstallDir", nullptr, nullptr, reinterpret_cast< LPBYTE>(&vctBuffer[ 0 ]), &dwLen ); +			ret = &vctBuffer[0]; + +			if( ret[ dwLen-1 ] != '\\' ) +				ret += "\\"; +			ret += GSTREAMER_DIR_SUFFIX; + +			SetDllDirectoryA( ret.c_str() ); +		} +		::RegCloseKey( hKey ); +	} +	return ret; +} +#else +std::string getGStreamerDir() { return ""; } +#endif + +#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) RTN (*ll##GSTSYM)(__VA_ARGS__) = NULL; +#include "llmediaimplgstreamer_syms_raw.inc" +#undef LL_GST_SYM + +struct Symloader +{ +	bool mRequired; +	char const *mName; +	apr_dso_handle_sym_t *mPPFunc; +}; + +#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) { REQ, #GSTSYM , (apr_dso_handle_sym_t*)&ll##GSTSYM},  +Symloader sSyms[] = { +#include "llmediaimplgstreamer_syms_raw.inc" +{ false, 0, 0 } }; +#undef LL_GST_SYM + +// a couple of stubs for disgusting reasons +GstDebugCategory* +ll_gst_debug_category_new(gchar *name, guint color, gchar *description) +{ +	static GstDebugCategory dummy; +	return &dummy; +} +void ll_gst_debug_register_funcptr(GstDebugFuncPtr func, gchar* ptrname) +{ +} + +static bool sSymsGrabbed = false; +static apr_pool_t *sSymGSTDSOMemoryPool = NULL; + +std::vector< apr_dso_handle_t* > sLoadedLibraries; + +bool grab_gst_syms( std::vector< std::string > const &aDSONames ) +{ +	if (sSymsGrabbed) +		return true; + +	//attempt to load the shared libraries +	apr_pool_create(&sSymGSTDSOMemoryPool, NULL); +   +	for( std::vector< std::string >::const_iterator itr = aDSONames.begin(); itr != aDSONames.end(); ++itr ) +	{ +		apr_dso_handle_t *pDSO(NULL); +		std::string strDSO = getGStreamerDir() + *itr; +		if( APR_SUCCESS == apr_dso_load( &pDSO, strDSO.c_str(), sSymGSTDSOMemoryPool )) +			sLoadedLibraries.push_back( pDSO ); +		 +		for( int i = 0; sSyms[ i ].mName; ++i ) +		{ +			if( !*sSyms[ i ].mPPFunc ) +			{ +				apr_dso_sym( sSyms[ i ].mPPFunc, pDSO, sSyms[ i ].mName ); +			} +		} +	} + +	std::stringstream strm; +	bool sym_error = false; +	for( int i = 0; sSyms[ i ].mName; ++i ) +	{ +		if( sSyms[ i ].mRequired && ! *sSyms[ i ].mPPFunc ) +		{ +			sym_error = true; +			strm << sSyms[ i ].mName << std::endl; +		} +	} + +	sSymsGrabbed = !sym_error; +	return sSymsGrabbed; +} + + +void ungrab_gst_syms() +{  +	// should be safe to call regardless of whether we've +	// actually grabbed syms. + +	for( std::vector< apr_dso_handle_t* >::iterator itr = sLoadedLibraries.begin(); itr != sLoadedLibraries.end(); ++itr ) +		apr_dso_unload( *itr ); + +	sLoadedLibraries.clear(); + +	if ( sSymGSTDSOMemoryPool ) +	{ +		apr_pool_destroy(sSymGSTDSOMemoryPool); +		sSymGSTDSOMemoryPool = NULL; +	} + +	for( int i = 0; sSyms[ i ].mName; ++i ) +		*sSyms[ i ].mPPFunc = NULL; + +	sSymsGrabbed = false; +} + diff --git a/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms.h b/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms.h new file mode 100644 index 0000000000..0874644ee6 --- /dev/null +++ b/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms.h @@ -0,0 +1,68 @@ +/**  + * @file llmediaimplgstreamer_syms.h + * @brief dynamic GStreamer symbol-grabbing code + * + * @cond + * $LicenseInfo:firstyear=2007&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library 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 + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + * @endcond + */ + +#include "linden_common.h" +#include <vector> +extern "C" { +#include <gst/gst.h> +} + +bool grab_gst_syms( std::vector< std::string > const&); +void ungrab_gst_syms(); + +#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) extern RTN (*ll##GSTSYM)(__VA_ARGS__); +#include "llmediaimplgstreamer_syms_raw.inc" +#undef LL_GST_SYM + +// regrettable hacks to give us better runtime compatibility with older systems +#define llg_return_if_fail(COND) do{if (!(COND)) return;}while(0) +#define llg_return_val_if_fail(COND,V) do{if (!(COND)) return V;}while(0) + +// regrettable hacks because GStreamer was not designed for runtime loading +#undef GST_TYPE_MESSAGE +#define GST_TYPE_MESSAGE (llgst_message_get_type()) +#undef GST_TYPE_OBJECT +#define GST_TYPE_OBJECT (llgst_object_get_type()) +#undef GST_TYPE_PIPELINE +#define GST_TYPE_PIPELINE (llgst_pipeline_get_type()) +#undef GST_TYPE_ELEMENT +#define GST_TYPE_ELEMENT (llgst_element_get_type()) +#undef GST_TYPE_VIDEO_SINK +#define GST_TYPE_VIDEO_SINK (llgst_video_sink_get_type()) +// more regrettable hacks to stub-out these .h-exposed GStreamer internals +void ll_gst_debug_register_funcptr(GstDebugFuncPtr func, gchar* ptrname); +#undef _gst_debug_register_funcptr +#define _gst_debug_register_funcptr ll_gst_debug_register_funcptr +GstDebugCategory* ll_gst_debug_category_new(gchar *name, guint color, gchar *description); +#undef _gst_debug_category_new +#define _gst_debug_category_new ll_gst_debug_category_new +#undef __gst_debug_enabled +#define __gst_debug_enabled (0) + +// more hacks +#define LLGST_MESSAGE_TYPE_NAME(M) (llgst_message_type_get_name(GST_MESSAGE_TYPE(M))) diff --git a/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms_raw.inc b/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms_raw.inc new file mode 100644 index 0000000000..155eeb6809 --- /dev/null +++ b/indra/media_plugins/gstreamer10/llmediaimplgstreamer_syms_raw.inc @@ -0,0 +1,68 @@ +LL_GST_SYM(true, gst_buffer_new, GstBuffer*, void) +LL_GST_SYM(true, gst_structure_set_value, void, GstStructure *, const gchar *, const GValue*) +LL_GST_SYM(true, gst_init_check, gboolean, int *argc, char **argv[], GError ** err) +LL_GST_SYM(true, gst_message_get_type, GType, void) +LL_GST_SYM(true, gst_message_type_get_name, const gchar*, GstMessageType type) +LL_GST_SYM(true, gst_message_parse_error, void, GstMessage *message, GError **gerror, gchar **debug) +LL_GST_SYM(true, gst_message_parse_warning, void, GstMessage *message, GError **gerror, gchar **debug) +LL_GST_SYM(true, gst_message_parse_state_changed, void, GstMessage *message, GstState *oldstate, GstState *newstate, GstState *pending) +LL_GST_SYM(true, gst_element_set_state, GstStateChangeReturn, GstElement *element, GstState state) +LL_GST_SYM(true, gst_object_unref, void, gpointer object) +LL_GST_SYM(true, gst_object_get_type, GType, void) +LL_GST_SYM(true, gst_pipeline_get_type, GType, void) +LL_GST_SYM(true, gst_pipeline_get_bus, GstBus*, GstPipeline *pipeline) +LL_GST_SYM(true, gst_bus_add_watch, guint, GstBus * bus, GstBusFunc func, gpointer user_data) +LL_GST_SYM(true, gst_element_factory_make, GstElement*, const gchar *factoryname, const gchar *name) +LL_GST_SYM(true, gst_element_get_type, GType, void) +LL_GST_SYM(true, gst_static_pad_template_get, GstPadTemplate*, GstStaticPadTemplate *pad_template) +LL_GST_SYM(true, gst_element_class_add_pad_template, void, GstElementClass *klass, GstPadTemplate *temp) +LL_GST_SYM(true, gst_caps_from_string, GstCaps *, const gchar *string) +LL_GST_SYM(true, gst_caps_get_structure, GstStructure *, const GstCaps *caps, guint index) +LL_GST_SYM(true, gst_element_register, gboolean, GstPlugin *plugin, const gchar *name, guint rank, GType type) +LL_GST_SYM(true, gst_structure_get_int, gboolean, const GstStructure *structure, const gchar *fieldname, gint *value) +LL_GST_SYM(true, gst_structure_get_value, const GValue *, const GstStructure *structure, const gchar *fieldname) +LL_GST_SYM(true, gst_value_get_fraction_numerator, gint, const GValue *value) +LL_GST_SYM(true, gst_value_get_fraction_denominator, gint, const GValue *value) +LL_GST_SYM(true, gst_structure_get_name, const gchar *, const GstStructure *structure) +LL_GST_SYM(true, gst_element_seek, bool, GstElement *, gdouble, GstFormat, GstSeekFlags, GstSeekType, gint64, GstSeekType, gint64) + +LL_GST_SYM(false, gst_registry_fork_set_enabled, void, gboolean enabled) +LL_GST_SYM(false, gst_segtrap_set_enabled, void, gboolean enabled) +LL_GST_SYM(false, gst_message_parse_buffering, void, GstMessage *message, gint *percent) +LL_GST_SYM(false, gst_message_parse_info, void, GstMessage *message, GError **gerror, gchar **debug) +LL_GST_SYM(false, gst_element_query_position, gboolean, GstElement *element, GstFormat *format, gint64 *cur) +LL_GST_SYM(false, gst_version, void, guint *major, guint *minor, guint *micro, guint *nano) + +LL_GST_SYM( true, gst_message_parse_tag, void, GstMessage *, GstTagList **) +LL_GST_SYM( true, gst_tag_list_foreach, void, const GstTagList *, GstTagForeachFunc, gpointer) +LL_GST_SYM( true, gst_tag_list_get_tag_size, guint, const GstTagList *, const gchar *) +LL_GST_SYM( true, gst_tag_list_get_value_index, const GValue *, const GstTagList *, const gchar *, guint) + +LL_GST_SYM( true, gst_caps_new_simple, GstCaps*, const char *, const char*, ... ) + +LL_GST_SYM( true, gst_sample_get_caps, GstCaps*, GstSample* ) +LL_GST_SYM( true, gst_sample_get_buffer, GstBuffer*, GstSample* ) +LL_GST_SYM( true, gst_buffer_map, gboolean, GstBuffer*, GstMapInfo*, GstMapFlags ) +LL_GST_SYM( true, gst_buffer_unmap, void, GstBuffer*, GstMapInfo* ) + +LL_GST_SYM( true, gst_app_sink_set_caps, void, GstAppSink*, GstCaps const* ) +LL_GST_SYM( true, gst_app_sink_pull_sample, GstSample*, GstAppSink* ) + +LL_GST_SYM( true, g_free, void, gpointer ) +LL_GST_SYM( true, g_error_free, void, GError* ) + +LL_GST_SYM( true, g_main_context_pending, gboolean, GMainContext* ) +LL_GST_SYM( true, g_main_loop_get_context, GMainContext*, GMainLoop* ) +LL_GST_SYM( true, g_main_context_iteration, gboolean, GMainContext*, gboolean ) +LL_GST_SYM( true, g_main_loop_new, GMainLoop*, GMainContext*, gboolean ) +LL_GST_SYM( true, g_main_loop_quit, void, GMainLoop* ) +LL_GST_SYM( true, gst_mini_object_unref, void, GstMiniObject* ) +LL_GST_SYM( true, g_object_set, void, gpointer, gchar const*, ... ) +LL_GST_SYM( true, g_source_remove, gboolean, guint ) +LL_GST_SYM( true, g_value_get_string, gchar const*, GValue const* ) + + +LL_GST_SYM( true, gst_debug_set_active, void, gboolean ) +LL_GST_SYM( true, gst_debug_add_log_function, void, GstLogFunction, gpointer, GDestroyNotify ) +LL_GST_SYM( true, gst_debug_set_default_threshold, void, GstDebugLevel ) +LL_GST_SYM( true, gst_debug_message_get , gchar const*, GstDebugMessage * )
\ No newline at end of file diff --git a/indra/media_plugins/gstreamer10/media_plugin_gstreamer10.cpp b/indra/media_plugins/gstreamer10/media_plugin_gstreamer10.cpp new file mode 100644 index 0000000000..07bfb67283 --- /dev/null +++ b/indra/media_plugins/gstreamer10/media_plugin_gstreamer10.cpp @@ -0,0 +1,982 @@ +/**  + * @file media_plugin_gstreamer10.cpp + * @brief GStreamer-1.0 plugin for LLMedia API plugin system + * + * @cond + * $LicenseInfo:firstyear=2016&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2016, Linden Research, Inc. / Nicky Dasmijn + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library 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 + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + * @endcond + */ + +#define FLIP_Y + +#include "linden_common.h" + +#include "llgl.h" + +#include "llplugininstance.h" +#include "llpluginmessage.h" +#include "llpluginmessageclasses.h" +#include "media_plugin_base.h" + +#define G_DISABLE_CAST_CHECKS +extern "C" { +#include <gst/gst.h> +#include <gst/app/gstappsink.h> + +} + +#include "llmediaimplgstreamer.h" +#include "llmediaimplgstreamer_syms.h" + +static inline void llgst_caps_unref( GstCaps * caps ) +{ +    llgst_mini_object_unref( GST_MINI_OBJECT_CAST( caps ) ); +} + +static inline void llgst_sample_unref( GstSample *aSample ) +{ +    llgst_mini_object_unref( GST_MINI_OBJECT_CAST( aSample ) ); +} + +////////////////////////////////////////////////////////////////////////////// +// +class MediaPluginGStreamer10 : public MediaPluginBase +{ +public: +    MediaPluginGStreamer10(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data); +    ~MediaPluginGStreamer10(); + +    /* virtual */ void receiveMessage(const char *message_string); + +    static bool startup(); +    static bool closedown(); + +    gboolean processGSTEvents(GstBus *bus, GstMessage *message); + +private: +    std::string getVersion(); +    bool navigateTo( const std::string urlIn ); +    bool seek( double time_sec ); +    bool setVolume( float volume ); +     +    // misc +    bool pause(); +    bool stop(); +    bool play(double rate); +    bool getTimePos(double &sec_out); + +    double MIN_LOOP_SEC = 1.0F; +    U32 INTERNAL_TEXTURE_SIZE = 1024; +     +    bool mIsLooping; + +    enum ECommand { +        COMMAND_NONE, +        COMMAND_STOP, +        COMMAND_PLAY, +        COMMAND_FAST_FORWARD, +        COMMAND_FAST_REWIND, +        COMMAND_PAUSE, +        COMMAND_SEEK, +    }; +    ECommand mCommand; + +private: +    bool unload(); +    bool load(); + +    bool update(int milliseconds); +    void mouseDown( int x, int y ); +    void mouseUp( int x, int y ); +    void mouseMove( int x, int y ); + +    static bool mDoneInit; +     +    guint mBusWatchID; +     +    float mVolume; + +    int mDepth; + +    // padded texture size we need to write into +    int mTextureWidth; +    int mTextureHeight; +     +    bool mSeekWanted; +    double mSeekDestination; +     +    // Very GStreamer-specific +    GMainLoop *mPump; // event pump for this media +    GstElement *mPlaybin; +    GstAppSink *mAppSink; +}; + +//static +bool MediaPluginGStreamer10::mDoneInit = false; + +MediaPluginGStreamer10::MediaPluginGStreamer10( LLPluginInstance::sendMessageFunction host_send_func, +                                                void *host_user_data ) +    : MediaPluginBase(host_send_func, host_user_data) +    , mBusWatchID ( 0 ) +    , mSeekWanted(false) +    , mSeekDestination(0.0) +    , mPump ( NULL ) +    , mPlaybin ( NULL ) +    , mAppSink ( NULL ) +    , mCommand ( COMMAND_NONE ) +{ +} + +gboolean MediaPluginGStreamer10::processGSTEvents(GstBus *bus, GstMessage *message) +{ +    if (!message)  +        return TRUE; // shield against GStreamer bug + +    switch (GST_MESSAGE_TYPE (message)) +    { +        case GST_MESSAGE_BUFFERING: +        { +            // NEEDS GST 0.10.11+ +            if (llgst_message_parse_buffering) +            { +                gint percent = 0; +                llgst_message_parse_buffering(message, &percent); +            } +            break; +        } +        case GST_MESSAGE_STATE_CHANGED: +        { +            GstState old_state; +            GstState new_state; +            GstState pending_state; +            llgst_message_parse_state_changed(message, +                                              &old_state, +                                              &new_state, +                                              &pending_state); + +            switch (new_state) +            { +                case GST_STATE_VOID_PENDING: +                    break; +                case GST_STATE_NULL: +                    break; +                case GST_STATE_READY: +                    setStatus(STATUS_LOADED); +                    break; +                case GST_STATE_PAUSED: +                    setStatus(STATUS_PAUSED); +                    break; +                case GST_STATE_PLAYING: +                    setStatus(STATUS_PLAYING); +                    break; +            } +            break; +        } +        case GST_MESSAGE_ERROR: +        { +            GError *err = NULL; +            gchar *debug = NULL; + +            llgst_message_parse_error (message, &err, &debug); +            if (err) +                llg_error_free (err); +            llg_free (debug); + +            mCommand = COMMAND_STOP; + +            setStatus(STATUS_ERROR); + +            break; +        } +        case GST_MESSAGE_INFO: +        { +            if (llgst_message_parse_info) +            { +                GError *err = NULL; +                gchar *debug = NULL; +             +                llgst_message_parse_info (message, &err, &debug); +                if (err) +                    llg_error_free (err); +                llg_free (debug); +            } +            break; +        } +        case GST_MESSAGE_WARNING: +        { +            GError *err = NULL; +            gchar *debug = NULL; +             +            llgst_message_parse_warning (message, &err, &debug); +            if (err) +                llg_error_free (err); +            llg_free (debug); + +            break; +        } +        case GST_MESSAGE_EOS: +            /* end-of-stream */ +            if (mIsLooping) +            { +                double eos_pos_sec = 0.0F; +                bool got_eos_position = getTimePos(eos_pos_sec); +                 +                if (got_eos_position && eos_pos_sec < MIN_LOOP_SEC) +                { +                    // if we know that the movie is really short, don't +                    // loop it else it can easily become a time-hog +                    // because of GStreamer spin-up overhead +                    // inject a COMMAND_PAUSE +                    mCommand = COMMAND_PAUSE; +                } +                else +                { +                    stop(); +                    play(1.0); +                } +            } +            else // not a looping media +            { +                // inject a COMMAND_STOP +                mCommand = COMMAND_STOP; +            } +            break; +        default: +            /* unhandled message */ +            break; +    } + +    /* we want to be notified again the next time there is a message +     * on the bus, so return true (false means we want to stop watching +     * for messages on the bus and our callback should not be called again) +     */ +    return TRUE; +} + +extern "C" { +    gboolean llmediaimplgstreamer_bus_callback (GstBus     *bus, +                                                GstMessage *message, +                                                gpointer    data) +    { +        MediaPluginGStreamer10 *impl = (MediaPluginGStreamer10*)data; +        return impl->processGSTEvents(bus, message); +    } +} // extern "C" + + + +bool MediaPluginGStreamer10::navigateTo ( const std::string urlIn ) +{ +    if (!mDoneInit) +        return false; // error + +    setStatus(STATUS_LOADING); + +    mSeekWanted = false; + +    if (NULL == mPump ||  NULL == mPlaybin) +    { +        setStatus(STATUS_ERROR); +        return false; // error +    } + +    llg_object_set (G_OBJECT (mPlaybin), "uri", urlIn.c_str(), NULL); + +    // navigateTo implicitly plays, too. +    play(1.0); + +    return true; +} + + +class GstSampleUnref +{ +    GstSample *mT; +public: +    GstSampleUnref( GstSample *aT ) +        : mT( aT ) +    { llassert_always( mT ); } + +    ~GstSampleUnref( ) +    { llgst_sample_unref( mT ); } +}; + +bool MediaPluginGStreamer10::update(int milliseconds) +{ +    if (!mDoneInit) +        return false; // error + +    //  DEBUGMSG("updating media..."); +     +    // sanity check +    if (NULL == mPump || NULL == mPlaybin) +    { +        return false; +    } + +    // see if there's an outstanding seek wanted +    if (mSeekWanted && +        // bleh, GST has to be happy that the movie is really truly playing +        // or it may quietly ignore the seek (with rtsp:// at least). +        (GST_STATE(mPlaybin) == GST_STATE_PLAYING)) +    { +        seek(mSeekDestination); +        mSeekWanted = false; +    } + +    // *TODO: time-limit - but there isn't a lot we can do here, most +    // time is spent in gstreamer's own opaque worker-threads.  maybe +    // we can do something sneaky like only unlock the video object +    // for 'milliseconds' and otherwise hold the lock. +    while (llg_main_context_pending(llg_main_loop_get_context(mPump))) +    { +           llg_main_context_iteration(llg_main_loop_get_context(mPump), FALSE); +    } + +    // check for availability of a new frame +     +    if( !mAppSink ) +        return true; + +    if( GST_STATE(mPlaybin) != GST_STATE_PLAYING) // Do not try to pull a sample if not in playing state +        return true; +     +    GstSample *pSample = llgst_app_sink_pull_sample( mAppSink ); +    if(!pSample) +        return false; // Done playing + +    GstSampleUnref oSampleUnref( pSample ); +    GstCaps *pCaps = llgst_sample_get_caps ( pSample ); +    if (!pCaps) +        return false; + +    gint width = 0, height = 0; +    GstStructure *pStruct = llgst_caps_get_structure ( pCaps, 0); + +    if(!llgst_structure_get_int ( pStruct, "width", &width) ) +        width = 0; +    if(!llgst_structure_get_int ( pStruct, "height", &height) ) +        height = 0; + +    if( !mPixels || width == 0 || height == 0) +        return true; +     +    GstBuffer *pBuffer = llgst_sample_get_buffer ( pSample ); +    GstMapInfo map; +    llgst_buffer_map ( pBuffer, &map, GST_MAP_READ); + +    // Our render buffer is always 1kx1k + +    U32 rowSkip = INTERNAL_TEXTURE_SIZE / mTextureHeight; +    U32 colSkip = INTERNAL_TEXTURE_SIZE / mTextureWidth; + +    for (int row = 0; row < mTextureHeight; ++row) +    { +        U8 const *pTexelIn = map.data + (row*rowSkip * width *3); +#ifndef FLIP_Y +        U8 *pTexelOut = mPixels + (row * mTextureWidth * mDepth ); +#else +        U8 *pTexelOut = mPixels + ((mTextureHeight-row-1) * mTextureWidth * mDepth ); +#endif       +        for( int col = 0; col < mTextureWidth; ++col ) +        { +            pTexelOut[ 0 ] = pTexelIn[0]; +            pTexelOut[ 1 ] = pTexelIn[1]; +            pTexelOut[ 2 ] = pTexelIn[2]; +            pTexelOut += mDepth; +            pTexelIn += colSkip*3; +        } +    } + +    llgst_buffer_unmap( pBuffer, &map ); +    setDirty(0,0,mTextureWidth,mTextureHeight); + +    return true; +} + +void MediaPluginGStreamer10::mouseDown( int x, int y ) +{ +  // do nothing +} + +void MediaPluginGStreamer10::mouseUp( int x, int y ) +{ +  // do nothing +} + +void MediaPluginGStreamer10::mouseMove( int x, int y ) +{ +  // do nothing +} + + +bool MediaPluginGStreamer10::pause() +{ +    // todo: error-check this? +    if (mDoneInit && mPlaybin) +    { +        llgst_element_set_state(mPlaybin, GST_STATE_PAUSED); +        return true; +    } +    return false; +} + +bool MediaPluginGStreamer10::stop() +{ +    // todo: error-check this? +    if (mDoneInit && mPlaybin) +    { +        llgst_element_set_state(mPlaybin, GST_STATE_READY); +        return true; +    } +    return false; +} + +bool MediaPluginGStreamer10::play(double rate) +{ +    // NOTE: we don't actually support non-natural rate. + +    // todo: error-check this? +    if (mDoneInit && mPlaybin) +    { +        llgst_element_set_state(mPlaybin, GST_STATE_PLAYING); +        return true; +    } +    return false; +} + +bool MediaPluginGStreamer10::setVolume( float volume ) +{ +    // we try to only update volume as conservatively as +    // possible, as many gst-plugins-base versions up to at least +    // November 2008 have critical race-conditions in setting volume - sigh +    if (mVolume == volume) +        return true; // nothing to do, everything's fine + +    mVolume = volume; +    if (mDoneInit && mPlaybin) +    { +        llg_object_set(mPlaybin, "volume", mVolume, NULL); +        return true; +    } + +    return false; +} + +bool MediaPluginGStreamer10::seek(double time_sec) +{ +    bool success = false; +    if (mDoneInit && mPlaybin) +    { +        success = llgst_element_seek(mPlaybin, 1.0F, GST_FORMAT_TIME, +                GstSeekFlags(GST_SEEK_FLAG_FLUSH | +                         GST_SEEK_FLAG_KEY_UNIT), +                GST_SEEK_TYPE_SET, gint64(time_sec*GST_SECOND), +                GST_SEEK_TYPE_NONE, GST_CLOCK_TIME_NONE); +    } +    return success; +} + +bool MediaPluginGStreamer10::getTimePos(double &sec_out) +{ +    bool got_position = false; +    if (mDoneInit && mPlaybin) +    { +        gint64 pos(0); +        GstFormat timefmt = GST_FORMAT_TIME; +        got_position = +            llgst_element_query_position && +            llgst_element_query_position(mPlaybin, +                             &timefmt, +                             &pos); +        got_position = got_position +            && (timefmt == GST_FORMAT_TIME); +        // GStreamer may have other ideas, but we consider the current position +        // undefined if not PLAYING or PAUSED +        got_position = got_position && +            (GST_STATE(mPlaybin) == GST_STATE_PLAYING || +             GST_STATE(mPlaybin) == GST_STATE_PAUSED); +        if (got_position && !GST_CLOCK_TIME_IS_VALID(pos)) +        { +            if (GST_STATE(mPlaybin) == GST_STATE_PLAYING) +            { +                // if we're playing then we treat an invalid clock time +                // as 0, for complicated reasons (insert reason here) +                pos = 0; +            } +            else +            { +                got_position = false; +            } +             +        } +        // If all the preconditions succeeded... we can trust the result. +        if (got_position) +        { +            sec_out = double(pos) / double(GST_SECOND); // gst to sec +        } +    } +    return got_position; +} + +bool MediaPluginGStreamer10::load() +{ +    if (!mDoneInit) +        return false; // error + +    setStatus(STATUS_LOADING); + +    mIsLooping = false; +    mVolume = 0.1234567f; // minor hack to force an initial volume update + +    // Create a pumpable main-loop for this media +    mPump = llg_main_loop_new (NULL, FALSE); +    if (!mPump) +    { +        setStatus(STATUS_ERROR); +        return false; // error +    } + +    // instantiate a playbin element to do the hard work +    mPlaybin = llgst_element_factory_make ("playbin", ""); +    if (!mPlaybin) +    { +        setStatus(STATUS_ERROR); +        return false; // error +    } + +    // get playbin's bus +    GstBus *bus = llgst_pipeline_get_bus (GST_PIPELINE (mPlaybin)); +    if (!bus) +    { +        setStatus(STATUS_ERROR); +        return false; // error +    } +    mBusWatchID = llgst_bus_add_watch (bus, +                       llmediaimplgstreamer_bus_callback, +                       this); +    llgst_object_unref (bus); + +    mAppSink = (GstAppSink*)(llgst_element_factory_make ("appsink", "")); + +    GstCaps* pCaps = llgst_caps_new_simple( "video/x-raw", +                                            "format", G_TYPE_STRING, "RGB", +                                            "width", G_TYPE_INT, INTERNAL_TEXTURE_SIZE, +                                            "height", G_TYPE_INT, INTERNAL_TEXTURE_SIZE, +                                            NULL ); + +    llgst_app_sink_set_caps( mAppSink, pCaps ); +    llgst_caps_unref( pCaps ); + +    if (!mAppSink) +    { +        setStatus(STATUS_ERROR); +        return false; +    } +     +    llg_object_set(mPlaybin, "video-sink", mAppSink, NULL); + +    return true; +} + +bool MediaPluginGStreamer10::unload () +{ +    if (!mDoneInit) +        return false; // error + +    // stop getting callbacks for this bus +    llg_source_remove(mBusWatchID); +    mBusWatchID = 0; + +    if (mPlaybin) +    { +        llgst_element_set_state (mPlaybin, GST_STATE_NULL); +        llgst_object_unref (GST_OBJECT (mPlaybin)); +        mPlaybin = NULL; +    } + +    if (mPump) +    { +        llg_main_loop_quit(mPump); +        mPump = NULL; +    } + +    mAppSink = NULL; + +    setStatus(STATUS_NONE); + +    return true; +} + +void LogFunction(GstDebugCategory *category, GstDebugLevel level, const gchar *file, const gchar *function, gint line, GObject *object, GstDebugMessage *message, gpointer user_data ) +#ifndef LL_LINUX // Docu says we need G_GNUC_NO_INSTRUMENT, but GCC says 'error' +    G_GNUC_NO_INSTRUMENT +#endif +{ +#ifdef LL_LINUX +    std::cerr << file << ":" << line << "(" << function << "): " << llgst_debug_message_get( message ) << std::endl; +#endif +} + +//static +bool MediaPluginGStreamer10::startup() +{ +    // first - check if GStreamer is explicitly disabled +    if (NULL != getenv("LL_DISABLE_GSTREAMER")) +        return false; + +    // only do global GStreamer initialization once. +    if (!mDoneInit) +    { +        ll_init_apr(); + +        // Get symbols! +        std::vector< std::string > vctDSONames; +#if LL_DARWIN +#elif LL_WINDOWS +        vctDSONames.push_back( "libgstreamer-1.0-0.dll"  ); +        vctDSONames.push_back( "libgstapp-1.0-0.dll"  ); +        vctDSONames.push_back( "libglib-2.0-0.dll" ); +        vctDSONames.push_back( "libgobject-2.0-0.dll" ); +#else // linux or other ELFy unixoid +        vctDSONames.push_back( "libgstreamer-1.0.so.0"  ); +        vctDSONames.push_back( "libgstapp-1.0.so.0"  ); +        vctDSONames.push_back( "libglib-2.0.so.0" ); +        vctDSONames.push_back( "libgobject-2.0.so" ); +#endif +        if( !grab_gst_syms( vctDSONames ) ) +        { +            return false; +        } + +        if (llgst_segtrap_set_enabled) +        { +            llgst_segtrap_set_enabled(FALSE); +        } +#if LL_LINUX +        // Gstreamer tries a fork during init, waitpid-ing on it, +        // which conflicts with any installed SIGCHLD handler... +        struct sigaction tmpact, oldact; +        if (llgst_registry_fork_set_enabled ) { +            // if we can disable SIGCHLD-using forking behaviour, +            // do it. +            llgst_registry_fork_set_enabled(false); +        } +        else { +            // else temporarily install default SIGCHLD handler +            // while GStreamer initialises +            tmpact.sa_handler = SIG_DFL; +            sigemptyset( &tmpact.sa_mask ); +            tmpact.sa_flags = SA_SIGINFO; +            sigaction(SIGCHLD, &tmpact, &oldact); +        } +#endif // LL_LINUX +        // Protect against GStreamer resetting the locale, yuck. +        static std::string saved_locale; +        saved_locale = setlocale(LC_ALL, NULL); +         +//      _putenv_s( "GST_PLUGIN_PATH", "E:\\gstreamer\\1.0\\x86\\lib\\gstreamer-1.0" ); + +        llgst_debug_set_default_threshold( GST_LEVEL_WARNING ); +        llgst_debug_add_log_function( LogFunction, NULL, NULL ); +        llgst_debug_set_active( false ); + +        // finally, try to initialize GStreamer! +        GError *err = NULL; +        gboolean init_gst_success = llgst_init_check(NULL, NULL, &err); + +        // restore old locale +        setlocale(LC_ALL, saved_locale.c_str() ); + +#if LL_LINUX +        // restore old SIGCHLD handler +        if (!llgst_registry_fork_set_enabled) +            sigaction(SIGCHLD, &oldact, NULL); +#endif // LL_LINUX + +        if (!init_gst_success) // fail +        { +            if (err) +            { +                llg_error_free(err); +            } +            return false; +        } +         +        mDoneInit = true; +    } + +    return true; +} + +//static +bool MediaPluginGStreamer10::closedown() +{ +    if (!mDoneInit) +        return false; // error + +    ungrab_gst_syms(); + +    mDoneInit = false; + +    return true; +} + +MediaPluginGStreamer10::~MediaPluginGStreamer10() +{ +    closedown(); +} + + +std::string MediaPluginGStreamer10::getVersion() +{ +    std::string plugin_version = "GStreamer10 media plugin, GStreamer version "; +    if (mDoneInit && +        llgst_version) +    { +        guint major, minor, micro, nano; +        llgst_version(&major, &minor, µ, &nano); +        plugin_version += llformat("%u.%u.%u.%u (runtime), %u.%u.%u.%u (headers)", (unsigned int)major, (unsigned int)minor, +                                   (unsigned int)micro, (unsigned int)nano, (unsigned int)GST_VERSION_MAJOR, (unsigned int)GST_VERSION_MINOR, +                                   (unsigned int)GST_VERSION_MICRO, (unsigned int)GST_VERSION_NANO); +    } +    else +    { +        plugin_version += "(unknown)"; +    } +    return plugin_version; +} + +void MediaPluginGStreamer10::receiveMessage(const char *message_string) +{ +    LLPluginMessage message_in; + +    if(message_in.parse(message_string) >= 0) +    { +        std::string message_class = message_in.getClass(); +        std::string message_name = message_in.getName(); + +        if(message_class == LLPLUGIN_MESSAGE_CLASS_BASE) +        { +            if(message_name == "init") +            { +                LLPluginMessage message("base", "init_response"); +                LLSD versions = LLSD::emptyMap(); +                versions[LLPLUGIN_MESSAGE_CLASS_BASE] = LLPLUGIN_MESSAGE_CLASS_BASE_VERSION; +                versions[LLPLUGIN_MESSAGE_CLASS_MEDIA] = LLPLUGIN_MESSAGE_CLASS_MEDIA_VERSION; +                versions[LLPLUGIN_MESSAGE_CLASS_MEDIA_TIME] = LLPLUGIN_MESSAGE_CLASS_MEDIA_TIME_VERSION; +                message.setValueLLSD("versions", versions); + +                load(); + +                message.setValue("plugin_version", getVersion()); +                sendMessage(message); +            } +            else if(message_name == "idle") +            { +                // no response is necessary here. +                double time = message_in.getValueReal("time"); +                 +                // Convert time to milliseconds for update() +                update((int)(time * 1000.0f)); +            } +            else if(message_name == "cleanup") +            { +                unload(); +                closedown(); +            } +            else if(message_name == "shm_added") +            { +                SharedSegmentInfo info; +                info.mAddress = message_in.getValuePointer("address"); +                info.mSize = (size_t)message_in.getValueS32("size"); +                std::string name = message_in.getValue("name"); + +                mSharedSegments.insert(SharedSegmentMap::value_type(name, info)); +            } +            else if(message_name == "shm_remove") +            { +                std::string name = message_in.getValue("name"); + +                SharedSegmentMap::iterator iter = mSharedSegments.find(name); +                if(iter != mSharedSegments.end()) +                { +                    if(mPixels == iter->second.mAddress) +                    { +                        // This is the currently active pixel buffer.  Make sure we stop drawing to it. +                        mPixels = NULL; +                        mTextureSegmentName.clear(); +                    } +                    mSharedSegments.erase(iter); +                } + +                // Send the response so it can be cleaned up. +                LLPluginMessage message("base", "shm_remove_response"); +                message.setValue("name", name); +                sendMessage(message); +            } +        } +        else if(message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA) +        { +            if(message_name == "init") +            { +                // Plugin gets to decide the texture parameters to use. +                LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "texture_params"); +                // lame to have to decide this now, it depends on the movie.  Oh well. +                mDepth = 4; + +                mTextureWidth = 1; +                mTextureHeight = 1; + +                message.setValueU32("format", GL_RGBA); +                message.setValueU32("type", GL_UNSIGNED_INT_8_8_8_8_REV); + +                message.setValueS32("depth", mDepth); +                message.setValueS32("default_width", INTERNAL_TEXTURE_SIZE ); +                message.setValueS32("default_height", INTERNAL_TEXTURE_SIZE ); +                message.setValueU32("internalformat", GL_RGBA8); +                message.setValueBoolean("coords_opengl", true); // true == use OpenGL-style coordinates, false == (0,0) is upper left. +                message.setValueBoolean("allow_downsample", true); // we respond with grace and performance if asked to downscale +                sendMessage(message); +            } +            else if(message_name == "size_change") +            { +                std::string name = message_in.getValue("name"); +                S32 width = message_in.getValueS32("width"); +                S32 height = message_in.getValueS32("height"); +                S32 texture_width = message_in.getValueS32("texture_width"); +                S32 texture_height = message_in.getValueS32("texture_height"); + +                LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "size_change_response"); +                message.setValue("name", name); +                message.setValueS32("width", width); +                message.setValueS32("height", height); +                message.setValueS32("texture_width", texture_width); +                message.setValueS32("texture_height", texture_height); +                sendMessage(message); + +                if(!name.empty()) +                { +                    // Find the shared memory region with this name +                    SharedSegmentMap::iterator iter = mSharedSegments.find(name); +                    if(iter != mSharedSegments.end()) +                    { +                        mPixels = (unsigned char*)iter->second.mAddress; +                        mTextureSegmentName = name; + +                        mTextureWidth = texture_width; +                        mTextureHeight = texture_height; +                        memset( mPixels, 0, mTextureWidth*mTextureHeight*mDepth ); +                    } +                     +                    LLPluginMessage message(LLPLUGIN_MESSAGE_CLASS_MEDIA, "size_change_request"); +                    message.setValue("name", mTextureSegmentName); +                    message.setValueS32("width", INTERNAL_TEXTURE_SIZE ); +                    message.setValueS32("height", INTERNAL_TEXTURE_SIZE ); +                    sendMessage(message); + +                } +            } +            else if(message_name == "load_uri") +            { +                std::string uri = message_in.getValue("uri"); +                navigateTo( uri ); +                sendStatus();        +            } +            else if(message_name == "mouse_event") +            { +                std::string event = message_in.getValue("event"); +                S32 x = message_in.getValueS32("x"); +                S32 y = message_in.getValueS32("y"); +                 +                if(event == "down") +                { +                    mouseDown(x, y); +                } +                else if(event == "up") +                { +                    mouseUp(x, y); +                } +                else if(event == "move") +                { +                    mouseMove(x, y); +                }; +            }; +        } +        else if(message_class == LLPLUGIN_MESSAGE_CLASS_MEDIA_TIME) +        { +            if(message_name == "stop") +            { +                stop(); +            } +            else if(message_name == "start") +            { +                double rate = 0.0; +                if(message_in.hasValue("rate")) +                { +                    rate = message_in.getValueReal("rate"); +                } +                // NOTE: we don't actually support rate. +                play(rate); +            } +            else if(message_name == "pause") +            { +                pause(); +            } +            else if(message_name == "seek") +            { +                double time = message_in.getValueReal("time"); +                // defer the actual seek in case we haven't +                // really truly started yet in which case there +                // is nothing to seek upon +                mSeekWanted = true; +                mSeekDestination = time; +            } +            else if(message_name == "set_loop") +            { +                bool loop = message_in.getValueBoolean("loop"); +                mIsLooping = loop; +            } +            else if(message_name == "set_volume") +            { +                double volume = message_in.getValueReal("volume"); +                setVolume(volume); +            } +        } +    } +} + +int init_media_plugin(LLPluginInstance::sendMessageFunction host_send_func, void *host_user_data, LLPluginInstance::sendMessageFunction *plugin_send_func, void **plugin_user_data) +{ +    if( MediaPluginGStreamer10::startup() ) +    { +        MediaPluginGStreamer10 *self = new MediaPluginGStreamer10(host_send_func, host_user_data); +        *plugin_send_func = MediaPluginGStreamer10::staticReceiveMessage; +        *plugin_user_data = (void*)self; +         +        return 0; // okay +    } +    else  +    { +        return -1; // failed to init +    } +} | 
