aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/linden
diff options
context:
space:
mode:
Diffstat (limited to 'linden')
-rw-r--r--linden/indra/cmake/00-Common.cmake15
-rw-r--r--linden/indra/cmake/GStreamer.cmake10
-rw-r--r--linden/indra/llaudio/audioengine.cpp44
-rw-r--r--linden/indra/llcharacter/llbvhloader.cpp45
-rw-r--r--linden/indra/llimagej2coj/llimagej2coj.cpp19
-rw-r--r--linden/indra/llmedia/CMakeLists.txt4
-rw-r--r--linden/indra/llmedia/llgstplaythread.cpp54
-rw-r--r--linden/indra/llmedia/llgstplaythread.h62
-rw-r--r--linden/indra/llmedia/llmediabase.h15
-rw-r--r--linden/indra/llmedia/llmediaimplcommon.cpp12
-rw-r--r--linden/indra/llmedia/llmediaimplcommon.h2
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer.cpp335
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer.h27
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer_syms.cpp188
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer_syms.h78
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer_syms_raw.inc50
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer_syms_rawa.inc5
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamer_syms_rawv.inc5
-rw-r--r--linden/indra/llmedia/llmediaimplgstreamervidplug.cpp83
-rw-r--r--linden/indra/llmedia/llmediamanager.cpp14
-rw-r--r--linden/indra/llmessage/llhttpclient.cpp14
-rw-r--r--linden/indra/llui/llfloater.cpp1
-rw-r--r--linden/indra/newview/app_settings/keywords.ini1
-rw-r--r--linden/indra/newview/app_settings/settings.xml11
-rw-r--r--linden/indra/newview/llappviewer.cpp29
-rw-r--r--linden/indra/newview/llappviewer.h2
-rw-r--r--linden/indra/newview/llcompilequeue.cpp16
-rw-r--r--linden/indra/newview/lldrawable.cpp12
-rw-r--r--linden/indra/newview/llfilepicker.cpp39
-rw-r--r--linden/indra/newview/llfloaterchat.cpp6
-rw-r--r--linden/indra/newview/llviewercontrol.cpp18
-rw-r--r--linden/indra/newview/llviewermedia.cpp7
-rw-r--r--linden/indra/newview/llviewermenu.cpp5
-rw-r--r--linden/indra/newview/llviewermessage.cpp9
-rw-r--r--linden/indra/newview/llvoavatar.cpp45
-rw-r--r--linden/indra/newview/pipeline.cpp12
-rw-r--r--linden/indra/newview/skins/default/xui/de/panel_media_controls.xml1
-rw-r--r--linden/indra/newview/skins/default/xui/en-us/panel_preferences_im.xml2
-rw-r--r--linden/indra/newview/skins/default/xui/fr/panel_media_controls.xml1
-rw-r--r--linden/indra/newview/skins/default/xui/ja/panel_media_controls.xml2
-rw-r--r--linden/indra/newview/skins/default/xui/ko/panel_media_controls.xml1
-rw-r--r--linden/indra/newview/skins/silver/xui/en-us/panel_media_controls.xml14
42 files changed, 716 insertions, 599 deletions
diff --git a/linden/indra/cmake/00-Common.cmake b/linden/indra/cmake/00-Common.cmake
index c41591f..27838d6 100644
--- a/linden/indra/cmake/00-Common.cmake
+++ b/linden/indra/cmake/00-Common.cmake
@@ -57,6 +57,21 @@ if (WINDOWS)
57 /Zc:wchar_t- 57 /Zc:wchar_t-
58 ) 58 )
59 endif (MSVC80 OR MSVC90) 59 endif (MSVC80 OR MSVC90)
60
61 # VS optimization flags
62 if(MSVC80)
63
64 set(NUMBER_OF_CORES $ENV{NUMBER_OF_PROCESSORS})
65
66 set(CMAKE_CXX_FLAGS_RELEASE
67 "${CMAKE_CXX_FLAGS_RELEASE} /Ob2 /Oi /Ot /GT"
68 CACHE STRING "C++ compiler release options" FORCE)
69
70 add_definitions(
71 /MP${NUMBER_OF_CORES}
72 )
73
74 endif (MSVC80)
60 75
61 # Are we using the crummy Visual Studio KDU build workaround? 76 # Are we using the crummy Visual Studio KDU build workaround?
62 if (NOT VS_DISABLE_FATAL_WARNINGS) 77 if (NOT VS_DISABLE_FATAL_WARNINGS)
diff --git a/linden/indra/cmake/GStreamer.cmake b/linden/indra/cmake/GStreamer.cmake
index b4b984d..7b380a8 100644
--- a/linden/indra/cmake/GStreamer.cmake
+++ b/linden/indra/cmake/GStreamer.cmake
@@ -54,10 +54,6 @@ if (WINDOWS)
54 54
55else (WINDOWS) 55else (WINDOWS)
56 56
57 include(FindPkgConfig)
58
59 pkg_check_modules(GSTREAMER REQUIRED gstreamer-0.10)
60 pkg_check_modules(GSTREAMER_PLUGINS_BASE REQUIRED gstreamer-plugins-base-0.10)
61 set(GSTREAMER_INCLUDE_DIRS 57 set(GSTREAMER_INCLUDE_DIRS
62 ${LIBS_PREBUILT_DIR}/${LL_ARCH_DIR}/include/gstreamer-0.10 58 ${LIBS_PREBUILT_DIR}/${LL_ARCH_DIR}/include/gstreamer-0.10
63 ${LIBS_PREBUILT_DIR}/${LL_ARCH_DIR}/include/glib-2.0 59 ${LIBS_PREBUILT_DIR}/${LL_ARCH_DIR}/include/glib-2.0
@@ -66,9 +62,11 @@ else (WINDOWS)
66 ${LIBS_PREBUILT_DIR}/${LL_ARCH_DIR}/include/libxml2 62 ${LIBS_PREBUILT_DIR}/${LL_ARCH_DIR}/include/libxml2
67 ) 63 )
68 64
69 # We don't need to explicitly link against gstreamer itself, because
70 # LLMediaImplGStreamer probes for the system's copy at runtime.
71 set(GSTREAMER_LIBRARIES 65 set(GSTREAMER_LIBRARIES
66 gstvideo-0.10
67 gstaudio-0.10
68 gstbase-0.10
69 gstreamer-0.10
72 gobject-2.0 70 gobject-2.0
73 gmodule-2.0 71 gmodule-2.0
74 dl 72 dl
diff --git a/linden/indra/llaudio/audioengine.cpp b/linden/indra/llaudio/audioengine.cpp
index 7bb65e9..438aefc 100644
--- a/linden/indra/llaudio/audioengine.cpp
+++ b/linden/indra/llaudio/audioengine.cpp
@@ -191,39 +191,43 @@ void LLAudioEngine::startInternetStream(const std::string& url)
191 if (mgr) 191 if (mgr)
192 { 192 {
193 mInternetStreamMedia = mgr->createSourceFromMimeType(LLURI(url).scheme(), "audio/mpeg"); // assumes that whatever media implementation supports mp3 also supports vorbis. 193 mInternetStreamMedia = mgr->createSourceFromMimeType(LLURI(url).scheme(), "audio/mpeg"); // assumes that whatever media implementation supports mp3 also supports vorbis.
194 //LL_INFOS("AudioEngine") << "mInternetStreamMedia is now " << mInternetStreamMedia << llendl; 194 LL_INFOS("AudioEngine") << "mInternetStreamMedia is now " << mInternetStreamMedia << llendl;
195 } 195 }
196 } 196 }
197 197
198 if(!mInternetStreamMedia) 198 if(!mInternetStreamMedia)
199 {
199 return; 200 return;
200 201 }
201 // Check for a dead stream, just in case 202 // Check for a dead stream from gstreamer, just in case
202 if(getStatus() == LLMediaBase::STATUS_DEAD) 203 else if(getStatus() == LLMediaBase::STATUS_DEAD)
203 { 204 {
204 LL_INFOS("AudioEngine") << "don't play dead stream urls"<< llendl; 205 LL_INFOS("AudioEngine") << "don't play dead stream urls"<< llendl;
205 mInternetStreamURL.clear(); 206 mInternetStreamURL.clear();
206 mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_STOP); 207 mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_STOP);
207 mInternetStreamMedia->updateMedia(); 208 mInternetStreamMedia->updateMedia();
208 stopInternetStream(); 209 stopInternetStream();
209 return;
210 } 210 }
211 211 else if (url.empty())
212 if (!url.empty())
213 { 212 {
214 LL_INFOS("AudioEngine") << "Starting internet stream: " << url << llendl; 213 LL_INFOS("AudioEngine") << "url is emptly. Setting stream to NULL"<< llendl;
215 mInternetStreamURL = url; 214 mInternetStreamURL.clear();
216 mInternetStreamMedia->navigateTo ( url ); 215 mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_STOP);
217 LL_INFOS("AudioEngine") << "Playing....." << llendl;
218 mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_START);
219 mInternetStreamMedia->updateMedia(); 216 mInternetStreamMedia->updateMedia();
220 } 217 }
218 // Stream appears to be good, attempting to play
221 else 219 else
222 { 220 {
223 LL_INFOS("AudioEngine") << "setting stream to NULL"<< llendl; 221 // stop any other stream first
224 mInternetStreamURL.clear(); 222 stopInternetStream();
225 mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_STOP); 223
224 LL_INFOS("AudioEngine") << "Starting internet stream: " << url << llendl;
225 mInternetStreamURL = url;
226 mInternetStreamMedia->navigateTo(url);
227 //LL_INFOS("AudioEngine") << "Playing....." << llendl;
228 mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_START);
226 mInternetStreamMedia->updateMedia(); 229 mInternetStreamMedia->updateMedia();
230 mStatus = LLMediaBase::STATUS_STARTED;
227 } 231 }
228} 232}
229 233
@@ -231,7 +235,8 @@ void LLAudioEngine::startInternetStream(const std::string& url)
231void LLAudioEngine::stopInternetStream() 235void LLAudioEngine::stopInternetStream()
232{ 236{
233 LL_INFOS("AudioEngine") << "entered stopInternetStream()" << llendl; 237 LL_INFOS("AudioEngine") << "entered stopInternetStream()" << llendl;
234 238 mInternetStreamURL.clear();
239
235 if(mInternetStreamMedia) 240 if(mInternetStreamMedia)
236 { 241 {
237 if(!mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_STOP)) 242 if(!mInternetStreamMedia->addCommand(LLMediaBase::COMMAND_STOP))
@@ -240,8 +245,6 @@ void LLAudioEngine::stopInternetStream()
240 } 245 }
241 mInternetStreamMedia->updateMedia(); 246 mInternetStreamMedia->updateMedia();
242 } 247 }
243
244 mInternetStreamURL.clear();
245} 248}
246 249
247// virtual 250// virtual
@@ -879,7 +882,8 @@ void LLAudioEngine::triggerSound(const LLUUID &audio_uuid, const LLUUID& owner_i
879 // Create a new source (since this can't be associated with an existing source. 882 // Create a new source (since this can't be associated with an existing source.
880 //LL_INFOS("AudioEngine") << "Localized: " << audio_uuid << llendl; 883 //LL_INFOS("AudioEngine") << "Localized: " << audio_uuid << llendl;
881 884
882 if (mMuted) 885 //If we cannot hear it, dont even try to load the sound.
886 if (mMuted || gain == 0.0)
883 { 887 {
884 return; 888 return;
885 } 889 }
diff --git a/linden/indra/llcharacter/llbvhloader.cpp b/linden/indra/llcharacter/llbvhloader.cpp
index cd1f492..938f47d 100644
--- a/linden/indra/llcharacter/llbvhloader.cpp
+++ b/linden/indra/llcharacter/llbvhloader.cpp
@@ -1133,6 +1133,8 @@ void LLBVHLoader::optimize()
1133 1133
1134 F32 rot_threshold = ROTATION_KEYFRAME_THRESHOLD / llmax((F32)joint->mChildTreeMaxDepth * 0.33f, 1.f); 1134 F32 rot_threshold = ROTATION_KEYFRAME_THRESHOLD / llmax((F32)joint->mChildTreeMaxDepth * 0.33f, 1.f);
1135 1135
1136 double diff_max = 0;
1137 KeyVector::iterator ki_max = ki;
1136 for (; ki != joint->mKeys.end(); ++ki) 1138 for (; ki != joint->mKeys.end(); ++ki)
1137 { 1139 {
1138 if (ki_prev == ki_last_good_pos) 1140 if (ki_prev == ki_last_good_pos)
@@ -1193,30 +1195,55 @@ void LLBVHLoader::optimize()
1193 F32 x_delta; 1195 F32 x_delta;
1194 F32 y_delta; 1196 F32 y_delta;
1195 F32 rot_test; 1197 F32 rot_test;
1196 1198
1199 // Test if the rotation has changed significantly since the very first frame. If false
1200 // for all frames, then we'll just throw out this joint's rotation entirely.
1197 x_delta = dist_vec(LLVector3::x_axis * first_frame_rot, LLVector3::x_axis * test_rot); 1201 x_delta = dist_vec(LLVector3::x_axis * first_frame_rot, LLVector3::x_axis * test_rot);
1198 y_delta = dist_vec(LLVector3::y_axis * first_frame_rot, LLVector3::y_axis * test_rot); 1202 y_delta = dist_vec(LLVector3::y_axis * first_frame_rot, LLVector3::y_axis * test_rot);
1199 rot_test = x_delta + y_delta; 1203 rot_test = x_delta + y_delta;
1200
1201 if (rot_test > ROTATION_MOTION_THRESHOLD) 1204 if (rot_test > ROTATION_MOTION_THRESHOLD)
1202 { 1205 {
1203 rot_changed = TRUE; 1206 rot_changed = TRUE;
1204 } 1207 }
1205
1206 x_delta = dist_vec(LLVector3::x_axis * interp_rot, LLVector3::x_axis * test_rot); 1208 x_delta = dist_vec(LLVector3::x_axis * interp_rot, LLVector3::x_axis * test_rot);
1207 y_delta = dist_vec(LLVector3::y_axis * interp_rot, LLVector3::y_axis * test_rot); 1209 y_delta = dist_vec(LLVector3::y_axis * interp_rot, LLVector3::y_axis * test_rot);
1208 rot_test = x_delta + y_delta; 1210 rot_test = x_delta + y_delta;
1209 1211
1210 if (rot_test < rot_threshold) 1212 // Draw a line between the last good keyframe and current. Test the distance between the last frame (current-1, i.e. ki_prev)
1211 { 1213 // and the line. If it's greater than some threshold, then it represents a significant frame and we want to include it.
1212 ki_prev->mIgnoreRot = TRUE; 1214 if (rot_test >= rot_threshold ||
1213 numRotFramesConsidered++; 1215 (ki+1 == joint->mKeys.end() && numRotFramesConsidered > 2))
1214 }
1215 else
1216 { 1216 {
1217 // Add the current test keyframe (which is technically the previous key, i.e. ki_prev).
1217 numRotFramesConsidered = 2; 1218 numRotFramesConsidered = 2;
1218 ki_last_good_rot = ki_prev; 1219 ki_last_good_rot = ki_prev;
1219 joint->mNumRotKeys++; 1220 joint->mNumRotKeys++;
1221
1222 // Add another keyframe between the last good keyframe and current, at whatever point was the most "significant" (i.e.
1223 // had the largest deviation from the earlier tests). Note that a more robust approach would be test all intermediate
1224 // keyframes against the line between the last good keyframe and current, but we're settling for this other method
1225 // because it's significantly faster.
1226 if (diff_max > 0)
1227 {
1228 if (ki_max->mIgnoreRot == TRUE)
1229 {
1230 ki_max->mIgnoreRot = FALSE;
1231 joint->mNumRotKeys++;
1232 }
1233 diff_max = 0;
1234 }
1235 }
1236 else
1237 {
1238 // This keyframe isn't significant enough, throw it away.
1239 ki_prev->mIgnoreRot = TRUE;
1240 numRotFramesConsidered++;
1241 // Store away the keyframe that has the largest deviation from the interpolated line, for insertion later.
1242 if (rot_test > diff_max)
1243 {
1244 diff_max = rot_test;
1245 ki_max = ki;
1246 }
1220 } 1247 }
1221 } 1248 }
1222 1249
diff --git a/linden/indra/llimagej2coj/llimagej2coj.cpp b/linden/indra/llimagej2coj/llimagej2coj.cpp
index 983241c..a0c61d8 100644
--- a/linden/indra/llimagej2coj/llimagej2coj.cpp
+++ b/linden/indra/llimagej2coj/llimagej2coj.cpp
@@ -151,9 +151,11 @@ BOOL LLImageJ2COJ::decodeImpl(LLImageJ2C &base, LLImageRaw &raw_image, F32 decod
151 // dereference the array. 151 // dereference the array.
152 if(!image || !image->numcomps) 152 if(!image || !image->numcomps)
153 { 153 {
154 fprintf(stderr, "ERROR -> decodeImpl: failed to decode image!\n"); 154 llwarns << "ERROR -> decodeImpl: failed to decode image!" << llendl;
155 if (image) 155 if (image)
156 {
156 opj_image_destroy(image); 157 opj_image_destroy(image);
158 }
157 159
158 return TRUE; // done 160 return TRUE; // done
159 } 161 }
@@ -169,6 +171,17 @@ BOOL LLImageJ2COJ::decodeImpl(LLImageJ2C &base, LLImageRaw &raw_image, F32 decod
169 return TRUE; 171 return TRUE;
170 } 172 }
171 } 173 }
174
175 if(image->numcomps <= first_channel)
176 {
177 llwarns << "trying to decode more channels than are present in image: numcomps: " << image->numcomps << " first_channel: " << first_channel << llendl;
178 if (image)
179 {
180 opj_image_destroy(image);
181 }
182
183 return TRUE;
184 }
172 185
173 // Copy image data into our raw image format (instead of the separate channel format 186 // Copy image data into our raw image format (instead of the separate channel format
174 187
@@ -211,7 +224,7 @@ BOOL LLImageJ2COJ::decodeImpl(LLImageJ2C &base, LLImageRaw &raw_image, F32 decod
211 } 224 }
212 else // Some rare OpenJPEG versions have this bug. 225 else // Some rare OpenJPEG versions have this bug.
213 { 226 {
214 fprintf(stderr, "ERROR -> decodeImpl: failed to decode image! (NULL comp data - OpenJPEG bug)\n"); 227 llwarns << "ERROR -> decodeImpl: failed to decode image! (NULL comp data - OpenJPEG bug)" << llendl;
215 opj_image_destroy(image); 228 opj_image_destroy(image);
216 229
217 return TRUE; // done 230 return TRUE; // done
@@ -430,7 +443,7 @@ BOOL LLImageJ2COJ::getMetadata(LLImageJ2C &base)
430 443
431 if(!image) 444 if(!image)
432 { 445 {
433 fprintf(stderr, "ERROR -> getMetadata: failed to decode image!\n"); 446 llwarns << "ERROR -> getMetadata: failed to decode image!" << llendl;
434 return FALSE; 447 return FALSE;
435 } 448 }
436 449
diff --git a/linden/indra/llmedia/CMakeLists.txt b/linden/indra/llmedia/CMakeLists.txt
index 026afc1..0e5b261 100644
--- a/linden/indra/llmedia/CMakeLists.txt
+++ b/linden/indra/llmedia/CMakeLists.txt
@@ -30,8 +30,8 @@ set(llmedia_SOURCE_FILES
30 llmediaimplfactory.cpp 30 llmediaimplfactory.cpp
31 llmediamanager.cpp 31 llmediamanager.cpp
32 llmediaimplgstreamer.cpp 32 llmediaimplgstreamer.cpp
33 llmediaimplgstreamer_syms.cpp
34 llmediaimplgstreamervidplug.cpp 33 llmediaimplgstreamervidplug.cpp
34 llgstplaythread.cpp
35 ) 35 )
36 36
37set(llmedia_HEADER_FILES 37set(llmedia_HEADER_FILES
@@ -48,7 +48,7 @@ set(llmedia_HEADER_FILES
48 llmediaobserver.h 48 llmediaobserver.h
49 llmediaimplgstreamer.h 49 llmediaimplgstreamer.h
50 llmediaimplgstreamervidplug.h 50 llmediaimplgstreamervidplug.h
51 llmediaimplgstreamer_syms.h 51 llgstplaythread.h
52 ) 52 )
53 53
54 # Work around a bad interaction between broken gstreamer headers and 54 # Work around a bad interaction between broken gstreamer headers and
diff --git a/linden/indra/llmedia/llgstplaythread.cpp b/linden/indra/llmedia/llgstplaythread.cpp
new file mode 100644
index 0000000..152f9d9
--- /dev/null
+++ b/linden/indra/llmedia/llgstplaythread.cpp
@@ -0,0 +1,54 @@
1/**
2 * @file llgstplaythread.cpp
3 * @author Jacek Antonelli
4 * @brief GStreamer playback management thread class
5 *
6 * $LicenseInfo:firstyear=2009&license=viewergpl$
7 *
8 * Copyright (c) 2009, Jacek Antonelli
9 *
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of version 2 of the GNU General Public
12 * License as published by the Free Software Foundation. Terms of
13 * the GPL can be found in doc/GPL-license.txt in this distribution, or
14 * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
15 *
16 * There are special exceptions to the terms and conditions of the GPL as
17 * it is applied to this Source Code. View the full text of the exception
18 * in the file doc/FLOSS-exception.txt in this software distribution, or
19 * online at http://secondlifegrid.net/programs/open_source/licensing/flossexception
20 *
21 * This program is distributed in the hope that it will be useful,
22 * but WITHOUT ANY WARRANTY; without even the implied warranty of
23 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24 * GNU General Public License for more details.
25 *
26 * You should have received a copy of the GNU General Public License along
27 * with this program; if not, write to the Free Software Foundation, Inc.,
28 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
29 *
30 * $/LicenseInfo$
31 */
32
33#include "llgstplaythread.h"
34#include "llmediaimplgstreamer.h"
35
36LLGstPlayThread::
37LLGstPlayThread( LLMediaImplCommon *impl,
38 const std::string& name, apr_pool_t *poolp ):
39 LLThread( name, poolp ),
40 mMediaImpl( impl )
41{
42}
43
44
45LLGstPlayThread::~LLGstPlayThread()
46{
47}
48
49
50// virtual
51void LLGstPlayThread::run()
52{
53 ((LLMediaImplGStreamer *)mMediaImpl)->startPlay();
54}
diff --git a/linden/indra/llmedia/llgstplaythread.h b/linden/indra/llmedia/llgstplaythread.h
new file mode 100644
index 0000000..c3c36a7
--- /dev/null
+++ b/linden/indra/llmedia/llgstplaythread.h
@@ -0,0 +1,62 @@
1/**
2 * @file llgstplaythread.h
3 * @author Jacek Antonelli
4 * @brief GStreamer playback management thread class
5 *
6 * $LicenseInfo:firstyear=2009&license=viewergpl$
7 *
8 * Copyright (c) 2009, Jacek Antonelli
9 *
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of version 2 of the GNU General Public
12 * License as published by the Free Software Foundation. Terms of
13 * the GPL can be found in doc/GPL-license.txt in this distribution, or
14 * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
15 *
16 * There are special exceptions to the terms and conditions of the GPL as
17 * it is applied to this Source Code. View the full text of the exception
18 * in the file doc/FLOSS-exception.txt in this software distribution, or
19 * online at http://secondlifegrid.net/programs/open_source/licensing/flossexception
20 *
21 * This program is distributed in the hope that it will be useful,
22 * but WITHOUT ANY WARRANTY; without even the implied warranty of
23 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24 * GNU General Public License for more details.
25 *
26 * You should have received a copy of the GNU General Public License along
27 * with this program; if not, write to the Free Software Foundation, Inc.,
28 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
29 *
30 * $/LicenseInfo$
31 */
32
33
34#ifndef LL_LLGSTPLAYTHREAD_H
35#define LL_LLGSTPLAYTHREAD_H
36
37#include "linden_common.h"
38
39#include "llthread.h"
40#include "llmediaimplcommon.h"
41
42class LLGstPlayThread: public LLThread
43{
44 public:
45
46 LLGstPlayThread( LLMediaImplCommon *impl,
47 const std::string& name, apr_pool_t *poolp );
48
49 ~LLGstPlayThread();
50
51 virtual void run();
52
53 private:
54
55 // Actually, this will really only be an LLMediaImplGStreamer.
56 // But we have to jump through some hoops to mutual pointer-holding.
57 // There may be a better way, but I don't have the motivation to find it.
58 LLMediaImplCommon *mMediaImpl;
59};
60
61
62#endif // LL_LLGSTPLAYTHREAD_H
diff --git a/linden/indra/llmedia/llmediabase.h b/linden/indra/llmedia/llmediabase.h
index b5b9420..bf54a22 100644
--- a/linden/indra/llmedia/llmediabase.h
+++ b/linden/indra/llmedia/llmediabase.h
@@ -61,6 +61,21 @@ class LLMediaBase
61 // undoes everything init() didm called by the media manager when destroying a source 61 // undoes everything init() didm called by the media manager when destroying a source
62 virtual bool reset() = 0; 62 virtual bool reset() = 0;
63 63
64
65 /* Mirrors GStreamer debug levels. */
66 enum EDebugLevel {
67 DEBUG_LEVEL_NONE = 0,
68 DEBUG_LEVEL_ERROR,
69 DEBUG_LEVEL_WARNING,
70 DEBUG_LEVEL_INFO,
71 DEBUG_LEVEL_DEBUG,
72 DEBUG_LEVEL_LOG,
73 DEBUG_LEVEL_COUNT,
74 };
75
76 /* Set the debug verbosity level. Only implemented for GStreamer. */
77 virtual bool setDebugLevel( EDebugLevel level ) = 0;
78
64 // accessor for MIME type 79 // accessor for MIME type
65 virtual bool setMimeType( const std::string mime_type ) = 0; 80 virtual bool setMimeType( const std::string mime_type ) = 0;
66 virtual std::string getMimeType() const = 0; 81 virtual std::string getMimeType() const = 0;
diff --git a/linden/indra/llmedia/llmediaimplcommon.cpp b/linden/indra/llmedia/llmediaimplcommon.cpp
index abb61e1..48b3973 100644
--- a/linden/indra/llmedia/llmediaimplcommon.cpp
+++ b/linden/indra/llmedia/llmediaimplcommon.cpp
@@ -66,7 +66,8 @@ LLMediaImplCommon::LLMediaImplCommon() :
66 mCommand( LLMediaBase::COMMAND_NONE ), 66 mCommand( LLMediaBase::COMMAND_NONE ),
67 mStatus( LLMediaBase::STATUS_UNKNOWN ), 67 mStatus( LLMediaBase::STATUS_UNKNOWN ),
68 mVolume( 0 ), 68 mVolume( 0 ),
69 mLooping( false ) 69 mLooping( false ),
70 mDebugLevel( LLMediaBase::DEBUG_LEVEL_NONE )
70{ 71{
71} 72}
72 73
@@ -92,6 +93,15 @@ bool LLMediaImplCommon::reset()
92 93
93//////////////////////////////////////////////////////////////////////////////// 94////////////////////////////////////////////////////////////////////////////////
94// virtual (derives from LLMediaBase) 95// virtual (derives from LLMediaBase)
96bool LLMediaImplCommon::setDebugLevel( LLMediaBase::EDebugLevel level )
97{
98 mDebugLevel = level;
99
100 return true;
101}
102
103////////////////////////////////////////////////////////////////////////////////
104// virtual (derives from LLMediaBase)
95bool LLMediaImplCommon::setMimeType( const std::string mime_type ) 105bool LLMediaImplCommon::setMimeType( const std::string mime_type )
96{ 106{
97 mMimeType = mime_type; 107 mMimeType = mime_type;
diff --git a/linden/indra/llmedia/llmediaimplcommon.h b/linden/indra/llmedia/llmediaimplcommon.h
index aa6c4d5..8a726f1 100644
--- a/linden/indra/llmedia/llmediaimplcommon.h
+++ b/linden/indra/llmedia/llmediaimplcommon.h
@@ -54,6 +54,7 @@ class LLMediaImplCommon :
54 // housekeeping 54 // housekeeping
55 virtual bool init(); 55 virtual bool init();
56 virtual bool reset(); 56 virtual bool reset();
57 virtual bool setDebugLevel( LLMediaBase::EDebugLevel level );
57 virtual bool setMimeType( const std::string url ); 58 virtual bool setMimeType( const std::string url );
58 virtual std::string getMimeType() const; 59 virtual std::string getMimeType() const;
59 virtual std::string getMediaURL() const; 60 virtual std::string getMediaURL() const;
@@ -156,6 +157,7 @@ class LLMediaImplCommon :
156 LLMediaBase::ECommand mCommand; 157 LLMediaBase::ECommand mCommand;
157 LLMediaBase::EStatus mStatus; 158 LLMediaBase::EStatus mStatus;
158 bool mLooping; 159 bool mLooping;
160 LLMediaBase::EDebugLevel mDebugLevel;
159}; 161};
160 162
161#endif // LLMEDIAIMPLCOMMON_H 163#endif // LLMEDIAIMPLCOMMON_H
diff --git a/linden/indra/llmedia/llmediaimplgstreamer.cpp b/linden/indra/llmedia/llmediaimplgstreamer.cpp
index 30706f1..7ae6b02 100644
--- a/linden/indra/llmedia/llmediaimplgstreamer.cpp
+++ b/linden/indra/llmedia/llmediaimplgstreamer.cpp
@@ -30,23 +30,35 @@
30 * $/LicenseInfo$ 30 * $/LicenseInfo$
31 */ 31 */
32 32
33#include "llmediaimplgstreamer.h"
34
35///#if LL_GSTREAMER_ENABLED 33///#if LL_GSTREAMER_ENABLED
36 34
35#if LL_WINDOWS
36 // GStreamer 0.10.22 - gstutils.h - conversion from 'guint64' to 'guint8'.
37 // This was an intentional change to make GStreamer more threadsafe, and
38 // is okay. Delete this bit if GStreamer ever gets more VS-friendly -- McCabe
39 #pragma warning(disable : 4244)
40#endif
41
42#include "linden_common.h"
43#include "llmediaimplgstreamer.h"
44
37extern "C" { 45extern "C" {
38#include <gst/gst.h> 46#include <gst/gst.h>
39#include <gst/gstelement.h> 47#include <gst/gstelement.h>
40} 48}
41 49
50#if LL_WINDOWS
51 #pragma warning(default : 4244)
52#include <direct.h>
53#include <stdlib.h>
54#endif
55
42#include "llmediamanager.h" 56#include "llmediamanager.h"
43#include "llmediaimplregister.h" 57#include "llmediaimplregister.h"
44 58
45#include "llmediaimplgstreamervidplug.h" 59#include "llmediaimplgstreamervidplug.h"
60#include "llgstplaythread.h"
46 61
47#include "llmediaimplgstreamer_syms.h"
48
49#include "llerror.h"
50 62
51// register this impl with media manager factory 63// register this impl with media manager factory
52static LLMediaImplRegister sLLMediaImplGStreamerReg( "LLMediaImplGStreamer", new LLMediaImplGStreamerMaker() ); 64static LLMediaImplRegister sLLMediaImplGStreamerReg( "LLMediaImplGStreamer", new LLMediaImplGStreamerMaker() );
@@ -73,7 +85,8 @@ LLMediaImplGStreamer () :
73 mPump ( NULL ), 85 mPump ( NULL ),
74 mPlaybin ( NULL ), 86 mPlaybin ( NULL ),
75 mVideoSink ( NULL ), 87 mVideoSink ( NULL ),
76 mState( GST_STATE_NULL ) 88 mState( GST_STATE_NULL ),
89 mPlayThread ( NULL )
77{ 90{
78 startup( NULL ); // Startup gstreamer if it hasn't been already. 91 startup( NULL ); // Startup gstreamer if it hasn't been already.
79 92
@@ -90,7 +103,7 @@ LLMediaImplGStreamer () :
90 } 103 }
91 104
92 // instantiate a playbin element to do the hard work 105 // instantiate a playbin element to do the hard work
93 mPlaybin = llgst_element_factory_make ("playbin", "play"); 106 mPlaybin = gst_element_factory_make ("playbin", "play");
94 if (!mPlaybin) 107 if (!mPlaybin)
95 { 108 {
96 // todo: cleanup pump 109 // todo: cleanup pump
@@ -104,7 +117,7 @@ LLMediaImplGStreamer () :
104 117
105 // Plays inworld instead of in external player 118 // Plays inworld instead of in external player
106 mVideoSink = 119 mVideoSink =
107 GST_SLVIDEO(llgst_element_factory_make ("private-slvideo", "slvideo")); 120 GST_SLVIDEO(gst_element_factory_make ("private-slvideo", "slvideo"));
108 if (!mVideoSink) 121 if (!mVideoSink)
109 { 122 {
110 LL_WARNS("MediaImpl") << "Could not instantiate private-slvideo element." << LL_ENDL; 123 LL_WARNS("MediaImpl") << "Could not instantiate private-slvideo element." << LL_ENDL;
@@ -148,7 +161,7 @@ LLMediaImplGStreamer::
148std::string LLMediaImplGStreamer::getVersion() 161std::string LLMediaImplGStreamer::getVersion()
149{ 162{
150 guint major, minor, micro, nano; 163 guint major, minor, micro, nano;
151 llgst_version(&major, &minor, &micro, &nano); 164 gst_version(&major, &minor, &micro, &nano);
152 std::string version = llformat("%d.%d.%d.%d",major,minor,micro,nano); 165 std::string version = llformat("%d.%d.%d.%d",major,minor,micro,nano);
153 return version; 166 return version;
154} 167}
@@ -165,50 +178,163 @@ bool LLMediaImplGStreamer::startup (LLMediaManagerData* init_data)
165 // Init the glib type system - we need it. 178 // Init the glib type system - we need it.
166 g_type_init(); 179 g_type_init();
167 180
168 // Get symbols! 181 set_gst_plugin_path();
169#if LL_WINDOWS
170 if (! grab_gst_syms("libgstreamer-0.10.dll", "libgstvideo-0.10.dll", "libgstaudio-0.10.dll") )
171 {
172 LL_WARNS("MediaImpl") << "Couldn't find suitable GStreamer 0.10 support on this system - video playback disabled." << LL_ENDL;
173 return false;
174 }
175#else
176 if (! grab_gst_syms("libgstreamer-0.10.so.0", "libgstvideo-0.10.so.0", "libgstaudio-0.10.so.0") )
177 {
178 LL_WARNS("MediaImpl") << "Couldn't find suitable GStreamer 0.10 support on this system - video playback disabled." << LL_ENDL;
179 return false;
180 }
181#endif
182 if (llgst_segtrap_set_enabled)
183 llgst_segtrap_set_enabled(FALSE);
184 else
185 {
186 LL_WARNS("MediaImpl") << "gst_segtrap_set_enabled() is not available; Automated crash-reporter may cease to function until next restart." << LL_ENDL;
187 }
188 182
189 // Protect against GStreamer resetting the locale, yuck. 183 // Protect against GStreamer resetting the locale, yuck.
190 static std::string saved_locale; 184 static std::string saved_locale;
191 saved_locale = setlocale(LC_ALL, NULL); 185 saved_locale = setlocale(LC_ALL, NULL);
192 if (0 == llgst_init_check(NULL, NULL, NULL)) 186 if (0 == gst_init_check(NULL, NULL, NULL))
193 { 187 {
194 LL_WARNS("MediaImpl") << "GStreamer library failed to initialize and load standard plugins." << LL_ENDL; 188 LL_WARNS("MediaImpl") << "GStreamer library failed to initialize and load standard plugins." << LL_ENDL;
195 setlocale(LC_ALL, saved_locale.c_str() ); 189 setlocale(LC_ALL, saved_locale.c_str() );
196 return false; 190 return false;
197 } 191 }
198 setlocale(LC_ALL, saved_locale.c_str() ); 192 setlocale(LC_ALL, saved_locale.c_str() );
199 193
194 // Set up logging facilities
195 gst_debug_remove_log_function( gst_debug_log_default );
196 gst_debug_add_log_function( gstreamer_log, NULL );
197
200 // Init our custom plugins - only really need do this once. 198 // Init our custom plugins - only really need do this once.
201 gst_slvideo_init_class(); 199 gst_slvideo_init_class();
202 200
201
202 // List the plugins GStreamer can find
203 LL_DEBUGS("MediaImpl") << "Found GStreamer plugins:" << LL_ENDL;
204 GList *list;
205 GstRegistry *registry = gst_registry_get_default();
206 std::string loaded = "";
207 for (list = gst_registry_get_plugin_list(registry);
208 list != NULL;
209 list = g_list_next(list))
210 {
211 GstPlugin *list_plugin = (GstPlugin *)list->data;
212 (bool)gst_plugin_is_loaded(list_plugin) ? loaded = "Yes" : loaded = "No";
213 LL_DEBUGS("MediaImpl") << gst_plugin_get_name(list_plugin) << ", loaded? " << loaded << LL_ENDL;
214 }
215 gst_plugin_list_free(list);
216
217
203 done_init = true; 218 done_init = true;
204 } 219 }
205 return true; 220 return true;
206} 221}
207 222
208 223
224void LLMediaImplGStreamer::set_gst_plugin_path()
225{
226 // Only needed for Windows.
227 // Linux sets in wrapper.sh, Mac sets in Info-Imprudence.plist
228#ifdef LL_WINDOWS
229
230 char* imp_cwd;
231
232 // Get the current working directory:
233 imp_cwd = _getcwd(NULL,0);
234
235 if(imp_cwd == NULL)
236 {
237 LL_DEBUGS("MediaImpl") << "_getcwd failed, not setting GST_PLUGIN_PATH."
238 << LL_ENDL;
239 }
240 else
241 {
242 LL_DEBUGS("MediaImpl") << "Imprudence is installed at "
243 << imp_cwd << LL_ENDL;
244
245 // Grab the current path, if it's set.
246 std::string old_plugin_path = "";
247 char *old_path = getenv("GST_PLUGIN_PATH");
248 if(old_path == NULL)
249 {
250 LL_DEBUGS("MediaImpl") << "Did not find user-set GST_PLUGIN_PATH."
251 << LL_ENDL;
252 }
253 else
254 {
255 old_plugin_path = ";" + std::string( old_path );
256 }
257
258
259 // Search both Imprudence and Imprudence\lib\gstreamer-plugins.
260 // If those fail, search the path the user has set, if any.
261 std::string plugin_path =
262 "GST_PLUGIN_PATH=" +
263 std::string(imp_cwd) + "\\lib\\gstreamer-plugins;" +
264 std::string(imp_cwd) +
265 old_plugin_path;
266
267 // Place GST_PLUGIN_PATH in the environment settings for imprudence.exe
268 // Returns 0 on success
269 if(_putenv( (char*)plugin_path.c_str() ))
270 {
271 LL_WARNS("MediaImpl") << "Setting environment variable failed!" << LL_ENDL;
272 }
273 else
274 {
275 LL_DEBUGS("MediaImpl") << "GST_PLUGIN_PATH set to "
276 << getenv("GST_PLUGIN_PATH") << LL_ENDL;
277 }
278 }
279
280#endif //LL_WINDOWS
281}
282
283
284void LLMediaImplGStreamer::gstreamer_log(GstDebugCategory *category,
285 GstDebugLevel level,
286 const gchar *file,
287 const gchar *function,
288 gint line,
289 GObject *object,
290 GstDebugMessage *message,
291 gpointer data)
292{
293 std::stringstream log(std::stringstream::out);
294
295 // Log format example:
296 //
297 // GST_ELEMENT_PADS: removing pad 'sink' (in gstelement.c:757:gst_element_remove_pad)
298 //
299 log << gst_debug_category_get_name( category ) << ": "
300 << gst_debug_message_get(message) << " "
301 << "(in " << file << ":" << line << ":" << function << ")";
302
303 switch( level )
304 {
305 case GST_LEVEL_ERROR:
306 LL_ERRS("MediaImpl") << log.str() << LL_ENDL;
307 break;
308 case GST_LEVEL_WARNING:
309 LL_WARNS("MediaImpl") << log.str() << LL_ENDL;
310 break;
311 case GST_LEVEL_DEBUG:
312 LL_DEBUGS("MediaImpl") << log.str() << LL_ENDL;
313 break;
314 case GST_LEVEL_INFO:
315 LL_INFOS("MediaImpl") << log.str() << LL_ENDL;
316 break;
317 default:
318 // Do nothing.
319 break;
320 }
321}
322
323
209bool LLMediaImplGStreamer::closedown() 324bool LLMediaImplGStreamer::closedown()
210{ 325{
211 ungrab_gst_syms(); 326 return true;
327}
328
329
330bool LLMediaImplGStreamer::setDebugLevel( LLMediaBase::EDebugLevel level )
331{
332 // Do parent class stuff.
333 LLMediaImplCommon::setDebugLevel(level);
334
335 // Set GStreamer verbosity.
336 gst_debug_set_default_threshold( (GstDebugLevel)level );
337
212 return true; 338 return true;
213} 339}
214 340
@@ -236,7 +362,7 @@ static const char* get_gst_state_name(GstState state)
236gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gpointer data) 362gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gpointer data)
237{ 363{
238#ifdef LL_GST_REPORT_STATE_CHANGES 364#ifdef LL_GST_REPORT_STATE_CHANGES
239 LL_DEBUGS("MediaCallback") << "Got GST message type: " << LLGST_MESSAGE_TYPE_NAME (message) << LL_ENDL; 365 LL_DEBUGS("MediaCallback") << "Got GST message type: " << GST_MESSAGE_TYPE_NAME (message) << LL_ENDL;
240#endif 366#endif
241 367
242 LLMediaImplGStreamer *impl = (LLMediaImplGStreamer*)data; 368 LLMediaImplGStreamer *impl = (LLMediaImplGStreamer*)data;
@@ -245,17 +371,13 @@ gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gp
245 { 371 {
246 case GST_MESSAGE_BUFFERING: 372 case GST_MESSAGE_BUFFERING:
247 { 373 {
248 // NEEDS GST 0.10.11+ 374 gint percent = 0;
249 if (llgst_message_parse_buffering) 375 gst_message_parse_buffering(message, &percent);
250 {
251 gint percent = 0;
252 llgst_message_parse_buffering(message, &percent);
253#ifdef LL_GST_REPORT_STATE_CHANGES 376#ifdef LL_GST_REPORT_STATE_CHANGES
254 LL_DEBUGS("MediaBuffering") << "GST buffering: " << percent << "%%" << LL_ENDL; 377 LL_DEBUGS("MediaBuffering") << "GST buffering: " << percent << "%%" << LL_ENDL;
255#endif 378#endif
256 LLMediaEvent event( impl, percent ); 379 LLMediaEvent event( impl, percent );
257 impl->getEventEmitter().update( &LLMediaObserver::onUpdateProgress, event ); 380 impl->getEventEmitter().update( &LLMediaObserver::onUpdateProgress, event );
258 }
259 } 381 }
260 break; 382 break;
261 case GST_MESSAGE_STATE_CHANGED: 383 case GST_MESSAGE_STATE_CHANGED:
@@ -263,7 +385,7 @@ gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gp
263 GstState old_state; 385 GstState old_state;
264 GstState new_state; 386 GstState new_state;
265 GstState pending_state; 387 GstState pending_state;
266 llgst_message_parse_state_changed(message, 388 gst_message_parse_state_changed(message,
267 &old_state, 389 &old_state,
268 &new_state, 390 &new_state,
269 &pending_state); 391 &pending_state);
@@ -308,7 +430,7 @@ gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gp
308 GError *err = NULL; 430 GError *err = NULL;
309 gchar *debug = NULL; 431 gchar *debug = NULL;
310 432
311 llgst_message_parse_error (message, &err, &debug); 433 gst_message_parse_error (message, &err, &debug);
312 LL_WARNS("MediaImpl") << "GST Error: " << err->message << LL_ENDL; 434 LL_WARNS("MediaImpl") << "GST Error: " << err->message << LL_ENDL;
313 g_error_free (err); 435 g_error_free (err);
314 g_free (debug); 436 g_free (debug);
@@ -320,17 +442,14 @@ gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gp
320 } 442 }
321 case GST_MESSAGE_INFO: 443 case GST_MESSAGE_INFO:
322 { 444 {
323 if (llgst_message_parse_info) 445 GError *err = NULL;
324 { 446 gchar *debug = NULL;
325 GError *err = NULL;
326 gchar *debug = NULL;
327 447
328 llgst_message_parse_info (message, &err, &debug); 448 gst_message_parse_info (message, &err, &debug);
329 LL_INFOS("MediaImpl") << "GST info: " << err->message 449 LL_INFOS("MediaImpl") << "GST info: " << err->message
330 << LL_ENDL; 450 << LL_ENDL;
331 g_error_free (err); 451 g_error_free (err);
332 g_free (debug); 452 g_free (debug);
333 }
334 break; 453 break;
335 } 454 }
336 case GST_MESSAGE_WARNING: 455 case GST_MESSAGE_WARNING:
@@ -338,7 +457,7 @@ gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gp
338 GError *err = NULL; 457 GError *err = NULL;
339 gchar *debug = NULL; 458 gchar *debug = NULL;
340 459
341 llgst_message_parse_warning (message, &err, &debug); 460 gst_message_parse_warning (message, &err, &debug);
342 LL_WARNS("MediaImpl") << "GST warning: " << err->message 461 LL_WARNS("MediaImpl") << "GST warning: " << err->message
343 << LL_ENDL; 462 << LL_ENDL;
344 g_error_free (err); 463 g_error_free (err);
@@ -351,10 +470,10 @@ gboolean LLMediaImplGStreamer::bus_callback(GstBus *bus, GstMessage *message, gp
351 GstTagList *tag_list; 470 GstTagList *tag_list;
352 gchar *title; 471 gchar *title;
353 gchar *artist; 472 gchar *artist;
354 llgst_message_parse_tag(message, &tag_list); 473 gst_message_parse_tag(message, &tag_list);
355 gboolean hazTitle = llgst_tag_list_get_string(tag_list, 474 gboolean hazTitle = gst_tag_list_get_string(tag_list,
356 GST_TAG_TITLE, &title); 475 GST_TAG_TITLE, &title);
357 gboolean hazArtist = llgst_tag_list_get_string(tag_list, 476 gboolean hazArtist = gst_tag_list_get_string(tag_list,
358 GST_TAG_ARTIST, &artist); 477 GST_TAG_ARTIST, &artist);
359 if(hazTitle) 478 if(hazTitle)
360 LL_INFOS("MediaInfo") << "Title: " << title << LL_ENDL; 479 LL_INFOS("MediaInfo") << "Title: " << title << LL_ENDL;
@@ -408,13 +527,13 @@ bool LLMediaImplGStreamer::navigateTo (const std::string urlIn)
408 g_object_set (G_OBJECT (mPlaybin), "uri", urlIn.c_str(), NULL); 527 g_object_set (G_OBJECT (mPlaybin), "uri", urlIn.c_str(), NULL);
409 528
410 // get playbin's bus - perhaps this can/should be done in ctor 529 // get playbin's bus - perhaps this can/should be done in ctor
411 GstBus *bus = llgst_pipeline_get_bus (GST_PIPELINE (mPlaybin)); 530 GstBus *bus = gst_pipeline_get_bus (GST_PIPELINE (mPlaybin));
412 if (!bus) 531 if (!bus)
413 { 532 {
414 return false; 533 return false;
415 } 534 }
416 llgst_bus_add_watch (bus, bus_callback, this); 535 gst_bus_add_watch (bus, bus_callback, this);
417 llgst_object_unref (bus); 536 gst_object_unref (bus);
418 537
419 mState = GST_STATE_READY; 538 mState = GST_STATE_READY;
420 539
@@ -431,9 +550,9 @@ bool LLMediaImplGStreamer::unload()
431 LL_DEBUGS("MediaImpl") << "unloading media..." << LL_ENDL; 550 LL_DEBUGS("MediaImpl") << "unloading media..." << LL_ENDL;
432 if (mPlaybin) 551 if (mPlaybin)
433 { 552 {
434 llgst_element_set_state (mPlaybin, GST_STATE_NULL); 553 gst_element_set_state (mPlaybin, GST_STATE_NULL);
435 mState = GST_STATE_NULL; 554 mState = GST_STATE_NULL;
436 llgst_object_unref (GST_OBJECT (mPlaybin)); 555 gst_object_unref (GST_OBJECT (mPlaybin));
437 mPlaybin = NULL; 556 mPlaybin = NULL;
438 } 557 }
439 558
@@ -591,20 +710,23 @@ bool LLMediaImplGStreamer::stop()
591 if (!mPlaybin || mState == GST_STATE_NULL) 710 if (!mPlaybin || mState == GST_STATE_NULL)
592 return true; 711 return true;
593 712
594 GstElement *pipeline = (GstElement *)llgst_object_ref(GST_OBJECT(mPlaybin)); 713 GstStateChangeReturn state_change;
595 llgst_object_unref(pipeline);
596
597 llgst_element_set_state(pipeline, GST_STATE_READY);
598 714
599 if (mState == GST_STATE_PLAYING) 715 state_change = gst_element_set_state(mPlaybin, GST_STATE_READY);
600 mState = GST_STATE_VOID_PENDING;
601 else
602 mState = GST_STATE_READY;
603 716
604 GstStateChangeReturn state_change = llgst_element_get_state(mPlaybin, NULL, NULL, GST_CLOCK_TIME_NONE); 717 LL_DEBUGS("MediaImpl") << gst_element_state_change_return_get_name(state_change) << LL_ENDL;
605 LL_DEBUGS("MediaImpl") << "get_state: " << llgst_element_state_change_return_get_name(state_change) << LL_ENDL;
606 718
607 return true; 719 if (state_change == GST_STATE_CHANGE_FAILURE)
720 {
721 LL_WARNS("MediaImpl") << "could not stop stream!" << LL_ENDL;
722 return false;
723 }
724 else
725 {
726 // Going into pending after play keeps dead streams from looping
727 (mState == GST_STATE_PLAYING) ? (mState = GST_STATE_VOID_PENDING) : (mState = GST_STATE_READY);
728 return true;
729 }
608} 730}
609 731
610/////////////////////////////////////////////////////////////////////////////// 732///////////////////////////////////////////////////////////////////////////////
@@ -616,25 +738,47 @@ bool LLMediaImplGStreamer::play()
616 if (!mPlaybin || mState == GST_STATE_NULL) 738 if (!mPlaybin || mState == GST_STATE_NULL)
617 return true; 739 return true;
618 740
619 GstElement *pipeline = (GstElement *)llgst_object_ref(GST_OBJECT(mPlaybin)); 741 // Clean up the existing thread, if any.
620 llgst_object_unref(pipeline); 742 if( mPlayThread != NULL && mPlayThread->isStopped())
743 {
744 delete mPlayThread;
745 mPlayThread = NULL;
746 }
747
748 if( mPlayThread == NULL )
749 {
750 // Make a new thread to start playing. This keeps the viewer
751 // responsive while the stream is resolved and buffered.
752 mPlayThread = new LLGstPlayThread( (LLMediaImplCommon *)this, "GstPlayThread", NULL);
753 mPlayThread->start();
754 }
755
756 return true;
757}
758
759
760void LLMediaImplGStreamer::startPlay()
761{
762 GstElement *pipeline = (GstElement *)gst_object_ref(GST_OBJECT(mPlaybin));
763 gst_object_unref(pipeline);
621 764
622 llgst_element_set_state(pipeline, GST_STATE_PLAYING); 765 GstStateChangeReturn state_change;
766
767 state_change = gst_element_set_state(mPlaybin, GST_STATE_PLAYING);
623 mState = GST_STATE_PLAYING; 768 mState = GST_STATE_PLAYING;
624 /*llgst_element_set_state(mPlaybin, GST_STATE_PLAYING);
625 mState = GST_STATE_PLAYING;*/
626 769
627 GstStateChangeReturn state_change = llgst_element_get_state(mPlaybin, NULL, NULL, GST_CLOCK_TIME_NONE); 770 LL_DEBUGS("MediaImpl") << gst_element_state_change_return_get_name(state_change) << LL_ENDL;
628 LL_DEBUGS("MediaImpl") << "get_state: " << llgst_element_state_change_return_get_name(state_change) << LL_ENDL;
629 771
630 // Check to make sure playing was successful. If not, stop. 772 // Check to make sure playing was successful. If not, stop.
773 // NOTE: state_change is almost always GST_STATE_CHANGE_ASYNC
631 if (state_change == GST_STATE_CHANGE_FAILURE) 774 if (state_change == GST_STATE_CHANGE_FAILURE)
632 { 775 {
633 setStatus(LLMediaBase::STATUS_STOPPED); 776 // If failing from a bad stream, go into an unknown
777 // state to stop bus_callback from looping back.
778 // We also force a stop in case the operations don't sync
779 setStatus(LLMediaBase::STATUS_UNKNOWN);
634 stop(); 780 stop();
635 } 781 }
636
637 return true;
638} 782}
639 783
640/////////////////////////////////////////////////////////////////////////////// 784///////////////////////////////////////////////////////////////////////////////
@@ -646,13 +790,22 @@ bool LLMediaImplGStreamer::pause()
646 if (!mPlaybin || mState == GST_STATE_NULL) 790 if (!mPlaybin || mState == GST_STATE_NULL)
647 return true; 791 return true;
648 792
649 llgst_element_set_state(mPlaybin, GST_STATE_PAUSED); 793 GstStateChangeReturn state_change;
650 mState = GST_STATE_PAUSED;
651
652 GstStateChangeReturn state_change = llgst_element_get_state(mPlaybin, NULL, NULL, GST_CLOCK_TIME_NONE);
653 LL_DEBUGS("MediaImpl") << "get_state: " << llgst_element_state_change_return_get_name(state_change) << LL_ENDL;
654 794
655 return true; 795 state_change = gst_element_set_state(mPlaybin, GST_STATE_PAUSED);
796
797 LL_DEBUGS("MediaImpl") << gst_element_state_change_return_get_name(state_change) << LL_ENDL;
798
799 if (state_change == GST_STATE_CHANGE_FAILURE)
800 {
801 LL_WARNS("MediaImpl") << "could not pause stream!" << LL_ENDL;
802 return false;
803 }
804 else
805 {
806 mState = GST_STATE_PAUSED;
807 return true;
808 }
656}; 809};
657 810
658 811
@@ -671,7 +824,7 @@ bool LLMediaImplGStreamer::seek(double time)
671 bool success = false; 824 bool success = false;
672 if (mPlaybin) 825 if (mPlaybin)
673 { 826 {
674 success = llgst_element_seek(mPlaybin, 1.0F, GST_FORMAT_TIME, 827 success = gst_element_seek(mPlaybin, 1.0F, GST_FORMAT_TIME,
675 GstSeekFlags(GST_SEEK_FLAG_FLUSH | 828 GstSeekFlags(GST_SEEK_FLAG_FLUSH |
676 GST_SEEK_FLAG_KEY_UNIT), 829 GST_SEEK_FLAG_KEY_UNIT),
677 GST_SEEK_TYPE_SET, gint64(time*1000000000.0F), 830 GST_SEEK_TYPE_SET, gint64(time*1000000000.0F),
diff --git a/linden/indra/llmedia/llmediaimplgstreamer.h b/linden/indra/llmedia/llmediaimplgstreamer.h
index dec970a..32e6bd1 100644
--- a/linden/indra/llmedia/llmediaimplgstreamer.h
+++ b/linden/indra/llmedia/llmediaimplgstreamer.h
@@ -48,6 +48,7 @@ extern "C" {
48} 48}
49 49
50#include "llmediaimplgstreamervidplug.h" 50#include "llmediaimplgstreamervidplug.h"
51#include "llgstplaythread.h"
51 52
52class LLMediaManagerData; 53class LLMediaManagerData;
53class LLMediaImplMaker; 54class LLMediaImplMaker;
@@ -66,6 +67,26 @@ class LLMediaImplGStreamer:
66 static bool startup( LLMediaManagerData* init_data ); 67 static bool startup( LLMediaManagerData* init_data );
67 static bool closedown(); 68 static bool closedown();
68 69
70 // Sets GST_PLUGIN_PATH env var for GStreamer.
71 static void set_gst_plugin_path();
72
73 /* virtual */ bool setDebugLevel( LLMediaBase::EDebugLevel level );
74
75 // Function given to GStreamer for handling debug messages
76 static void gstreamer_log(GstDebugCategory *category,
77 GstDebugLevel level,
78 const gchar *file,
79 const gchar *function,
80 gint line,
81 GObject *object,
82 GstDebugMessage *message,
83 gpointer data)
84#if __GNUC__
85 // recommended by the gstreamer docs
86 G_GNUC_NO_INSTRUMENT
87#endif
88 ;
89
69 /* virtual */ std::string getVersion(); 90 /* virtual */ std::string getVersion();
70 /* virtual */ bool navigateTo( const std::string url ); 91 /* virtual */ bool navigateTo( const std::string url );
71 /* virtual */ bool updateMedia(); 92 /* virtual */ bool updateMedia();
@@ -76,9 +97,13 @@ class LLMediaImplGStreamer:
76 /* virtual */ bool seek( double time ); 97 /* virtual */ bool seek( double time );
77 /* virtual */ bool setVolume( float volume ); 98 /* virtual */ bool setVolume( float volume );
78 99
100 void startPlay();
101
102
79 LLMediaEmitter< LLMediaObserver > getEventEmitter() const {return mEventEmitter;}; 103 LLMediaEmitter< LLMediaObserver > getEventEmitter() const {return mEventEmitter;};
80 104
81 private: 105 private:
106
82 // misc 107 // misc
83 bool unload(); 108 bool unload();
84 bool pause(); 109 bool pause();
@@ -100,6 +125,8 @@ class LLMediaImplGStreamer:
100 GstSLVideo *mVideoSink; 125 GstSLVideo *mVideoSink;
101 GstState mState; 126 GstState mState;
102 GstState getState() const { return mState; } 127 GstState getState() const { return mState; }
128
129 LLGstPlayThread *mPlayThread;
103}; 130};
104 131
105class LLMediaImplGStreamerMaker : public LLMediaImplMaker 132class LLMediaImplGStreamerMaker : public LLMediaImplMaker
diff --git a/linden/indra/llmedia/llmediaimplgstreamer_syms.cpp b/linden/indra/llmedia/llmediaimplgstreamer_syms.cpp
deleted file mode 100644
index fb1949a..0000000
--- a/linden/indra/llmedia/llmediaimplgstreamer_syms.cpp
+++ /dev/null
@@ -1,188 +0,0 @@
1/**
2 * @file llmediaimplgstreamer_syms.cpp
3 * @brief dynamic GStreamer symbol-grabbing code
4 *
5 * $LicenseInfo:firstyear=2007&license=viewergpl$
6 *
7 * Copyright (c) 2007-2008, Linden Research, Inc.
8 *
9 * Second Life Viewer Source Code
10 * The source code in this file ("Source Code") is provided by Linden Lab
11 * to you under the terms of the GNU General Public License, version 2.0
12 * ("GPL"), unless you have obtained a separate licensing agreement
13 * ("Other License"), formally executed by you and Linden Lab. Terms of
14 * the GPL can be found in doc/GPL-license.txt in this distribution, or
15 * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
16 *
17 * There are special exceptions to the terms and conditions of the GPL as
18 * it is applied to this Source Code. View the full text of the exception
19 * in the file doc/FLOSS-exception.txt in this software distribution, or
20 * online at http://secondlifegrid.net/programs/open_source/licensing/flossexception
21 *
22 * By copying, modifying or distributing this software, you acknowledge
23 * that you have read and understood your obligations described above,
24 * and agree to abide by those obligations.
25 *
26 * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
27 * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
28 * COMPLETENESS OR PERFORMANCE.
29 * $/LicenseInfo$
30 */
31
32///#if LL_GSTREAMER_ENABLED
33
34extern "C" {
35#include <gst/gst.h>
36
37#include "apr_pools.h"
38#include "apr_dso.h"
39}
40
41#include "llmediaimplgstreamer.h"
42
43#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) RTN (*ll##GSTSYM)(__VA_ARGS__) = NULL
44#include "llmediaimplgstreamer_syms_raw.inc"
45#include "llmediaimplgstreamer_syms_rawa.inc"
46#include "llmediaimplgstreamer_syms_rawv.inc"
47#undef LL_GST_SYM
48
49
50static bool sSymsGrabbed = false;
51static apr_pool_t *sSymGSTDSOMemoryPool = NULL;
52static apr_dso_handle_t *sSymGSTDSOHandleG = NULL;
53static apr_dso_handle_t *sSymGSTDSOHandleV = NULL;
54static apr_dso_handle_t *sSymGSTDSOHandleA = NULL;
55
56
57bool grab_gst_syms(std::string gst_dso_name,
58 std::string gst_dso_name_vid,
59 std::string gst_dso_name_aud)
60{
61 if (sSymsGrabbed)
62 {
63 // already have grabbed good syms
64 return TRUE;
65 }
66
67 bool sym_error = false;
68 bool rtn = false;
69 apr_status_t rv;
70 apr_dso_handle_t *sSymGSTDSOHandle = NULL;
71
72#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) do{rv = apr_dso_sym((apr_dso_handle_sym_t*)&ll##GSTSYM, sSymGSTDSOHandle, #GSTSYM); if (rv != APR_SUCCESS) {INFOMSG("Failed to grab symbol: %s", #GSTSYM); if (REQ) sym_error = true;} else DEBUGMSG("grabbed symbol: %s from %p", #GSTSYM, (void*)ll##GSTSYM);}while(0)
73
74 //attempt to load the shared libraries
75 apr_pool_create(&sSymGSTDSOMemoryPool, NULL);
76
77 if ( APR_SUCCESS == (rv = apr_dso_load(&sSymGSTDSOHandle,
78 gst_dso_name.c_str(),
79 sSymGSTDSOMemoryPool) ))
80 {
81 INFOMSG("Found DSO: %s", gst_dso_name.c_str());
82#include "llmediaimplgstreamer_syms_raw.inc"
83
84 if ( sSymGSTDSOHandle )
85 {
86 sSymGSTDSOHandleG = sSymGSTDSOHandle;
87 sSymGSTDSOHandle = NULL;
88 }
89
90 if ( APR_SUCCESS == (rv = apr_dso_load(&sSymGSTDSOHandle,
91 gst_dso_name_aud.c_str(),
92 sSymGSTDSOMemoryPool) ))
93 {
94 INFOMSG("Found DSO: %s", gst_dso_name_aud.c_str());
95#include "llmediaimplgstreamer_syms_rawa.inc"
96
97 if ( sSymGSTDSOHandle )
98 {
99 sSymGSTDSOHandleA = sSymGSTDSOHandle;
100 sSymGSTDSOHandle = NULL;
101 }
102
103 if ( APR_SUCCESS ==
104 (rv = apr_dso_load(&sSymGSTDSOHandle,
105 gst_dso_name_vid.c_str(),
106 sSymGSTDSOMemoryPool) ))
107 {
108 INFOMSG("Found DSO: %s", gst_dso_name_vid.c_str());
109#include "llmediaimplgstreamer_syms_rawv.inc"
110 }
111 else
112 {
113 INFOMSG("Couldn't load DSO: %s", gst_dso_name_vid.c_str());
114 rtn = false; // failure
115 }
116 }
117 else
118 {
119 INFOMSG("Couldn't load DSO: %s", gst_dso_name_aud.c_str());
120 rtn = false; // failure
121 }
122
123 rtn = !sym_error;
124 }
125 else
126 {
127 INFOMSG("Couldn't load DSO: %s", gst_dso_name.c_str());
128 rtn = false; // failure
129 }
130
131 if (sym_error)
132 {
133 WARNMSG("Failed to find necessary symbols in GStreamer libraries.");
134 }
135
136 if ( sSymGSTDSOHandle )
137 {
138 sSymGSTDSOHandleV = sSymGSTDSOHandle;
139 sSymGSTDSOHandle = NULL;
140 }
141#undef LL_GST_SYM
142
143 sSymsGrabbed = !!rtn;
144 return rtn;
145}
146
147
148void ungrab_gst_syms()
149{
150 // should be safe to call regardless of whether we've
151 // actually grabbed syms.
152
153 if ( sSymGSTDSOHandleG )
154 {
155 apr_dso_unload(sSymGSTDSOHandleG);
156 sSymGSTDSOHandleG = NULL;
157 }
158
159 if ( sSymGSTDSOHandleA )
160 {
161 apr_dso_unload(sSymGSTDSOHandleA);
162 sSymGSTDSOHandleA = NULL;
163 }
164
165 if ( sSymGSTDSOHandleV )
166 {
167 apr_dso_unload(sSymGSTDSOHandleV);
168 sSymGSTDSOHandleV = NULL;
169 }
170
171 if ( sSymGSTDSOMemoryPool )
172 {
173 apr_pool_destroy(sSymGSTDSOMemoryPool);
174 sSymGSTDSOMemoryPool = NULL;
175 }
176
177 // NULL-out all of the symbols we'd grabbed
178#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) do{ll##GSTSYM = NULL;}while(0)
179#include "llmediaimplgstreamer_syms_raw.inc"
180#include "llmediaimplgstreamer_syms_rawa.inc"
181#include "llmediaimplgstreamer_syms_rawv.inc"
182#undef LL_GST_SYM
183
184 sSymsGrabbed = false;
185}
186
187
188///#endif // LL_GSTREAMER_ENABLED
diff --git a/linden/indra/llmedia/llmediaimplgstreamer_syms.h b/linden/indra/llmedia/llmediaimplgstreamer_syms.h
deleted file mode 100644
index ebebd80..0000000
--- a/linden/indra/llmedia/llmediaimplgstreamer_syms.h
+++ /dev/null
@@ -1,78 +0,0 @@
1/**
2 * @file llmediaimplgstreamer_syms.h
3 * @brief dynamic GStreamer symbol-grabbing code
4 *
5 * $LicenseInfo:firstyear=2007&license=viewergpl$
6 *
7 * Copyright (c) 2007-2008, Linden Research, Inc.
8 *
9 * Second Life Viewer Source Code
10 * The source code in this file ("Source Code") is provided by Linden Lab
11 * to you under the terms of the GNU General Public License, version 2.0
12 * ("GPL"), unless you have obtained a separate licensing agreement
13 * ("Other License"), formally executed by you and Linden Lab. Terms of
14 * the GPL can be found in doc/GPL-license.txt in this distribution, or
15 * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
16 *
17 * There are special exceptions to the terms and conditions of the GPL as
18 * it is applied to this Source Code. View the full text of the exception
19 * in the file doc/FLOSS-exception.txt in this software distribution, or
20 * online at http://secondlifegrid.net/programs/open_source/licensing/flossexception
21 *
22 * By copying, modifying or distributing this software, you acknowledge
23 * that you have read and understood your obligations described above,
24 * and agree to abide by those obligations.
25 *
26 * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
27 * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
28 * COMPLETENESS OR PERFORMANCE.
29 * $/LicenseInfo$
30 */
31
32#include "linden_common.h"
33
34///#if LL_GSTREAMER_ENABLED
35
36extern "C" {
37#include <gst/gst.h>
38}
39
40bool grab_gst_syms(std::string gst_dso_name,
41 std::string gst_dso_name_vid,
42 std::string gst_dso_name_aud);
43void ungrab_gst_syms();
44
45#define LL_GST_SYM(REQ, GSTSYM, RTN, ...) extern RTN (*ll##GSTSYM)(__VA_ARGS__)
46#include "llmediaimplgstreamer_syms_raw.inc"
47#include "llmediaimplgstreamer_syms_rawa.inc"
48#include "llmediaimplgstreamer_syms_rawv.inc"
49#undef LL_GST_SYM
50
51// regrettable hacks to give us better runtime compatibility with older systems
52#define llg_return_if_fail(COND) do{if (!(COND)) return;}while(0)
53#define llg_return_val_if_fail(COND,V) do{if (!(COND)) return V;}while(0)
54
55// regrettable hacks because GStreamer was not designed for runtime loading
56#undef GST_TYPE_MESSAGE
57#define GST_TYPE_MESSAGE (llgst_message_get_type())
58#undef GST_TYPE_OBJECT
59#define GST_TYPE_OBJECT (llgst_object_get_type())
60#undef GST_TYPE_PIPELINE
61#define GST_TYPE_PIPELINE (llgst_pipeline_get_type())
62#undef GST_TYPE_ELEMENT
63#define GST_TYPE_ELEMENT (llgst_element_get_type())
64#undef GST_TYPE_AUDIO_SINK
65#define GST_TYPE_AUDIO_SINK (llgst_audio_sink_get_type())
66#undef GST_TYPE_VIDEO_SINK
67#define GST_TYPE_VIDEO_SINK (llgst_video_sink_get_type())
68#undef _gst_debug_register_funcptr
69#define _gst_debug_register_funcptr ll_gst_debug_register_funcptr
70#undef _gst_debug_category_new
71#define _gst_debug_category_new ll_gst_debug_category_new
72#undef __gst_debug_enabled
73#define __gst_debug_enabled (0)
74
75// more hacks
76#define LLGST_MESSAGE_TYPE_NAME(M) (llgst_message_type_get_name(GST_MESSAGE_TYPE(M)))
77
78///#endif // LL_GSTREAMER_ENABLED
diff --git a/linden/indra/llmedia/llmediaimplgstreamer_syms_raw.inc b/linden/indra/llmedia/llmediaimplgstreamer_syms_raw.inc
deleted file mode 100644
index 0d0d764..0000000
--- a/linden/indra/llmedia/llmediaimplgstreamer_syms_raw.inc
+++ /dev/null
@@ -1,50 +0,0 @@
1
2// required symbols to grab
3LL_GST_SYM(true, gst_init_check, gboolean, int *argc, char **argv[], GError ** err);
4LL_GST_SYM(true, gst_message_get_type, GType, void);
5LL_GST_SYM(true, gst_message_type_get_name, const gchar*, GstMessageType type);
6LL_GST_SYM(true, gst_message_parse_error, void, GstMessage *message, GError **gerror, gchar **debug);
7LL_GST_SYM(true, gst_message_parse_warning, void, GstMessage *message, GError **gerror, gchar **debug);
8LL_GST_SYM(true, gst_message_parse_state_changed, void, GstMessage *message, GstState *oldstate, GstState *newstate, GstState *pending);
9LL_GST_SYM(true, gst_element_set_state, GstStateChangeReturn, GstElement *element, GstState state);
10LL_GST_SYM(true, gst_element_get_state, GstStateChangeReturn, GstElement *element, GstState *state, GstState *pending, GstClockTime timeout);
11LL_GST_SYM(true, gst_object_unref, void, gpointer object);
12LL_GST_SYM(true, gst_object_ref, gpointer, gpointer object);
13LL_GST_SYM(true, gst_object_get_type, GType, void);
14LL_GST_SYM(true, gst_pipeline_get_type, GType, void);
15LL_GST_SYM(true, gst_pipeline_get_bus, GstBus*, GstPipeline *pipeline);
16LL_GST_SYM(true, gst_bus_add_watch, guint, GstBus * bus, GstBusFunc func, gpointer user_data);
17LL_GST_SYM(true, gst_element_factory_make, GstElement*, const gchar *factoryname, const gchar *name);
18LL_GST_SYM(true, gst_element_get_type, GType, void);
19LL_GST_SYM(true, gst_static_pad_template_get, GstPadTemplate*, GstStaticPadTemplate *pad_template);
20LL_GST_SYM(true, gst_element_class_add_pad_template, void, GstElementClass *klass, GstPadTemplate *temp);
21LL_GST_SYM(true, gst_element_class_set_details, void, GstElementClass *klass, const GstElementDetails *details);
22LL_GST_SYM(true, gst_caps_unref, void, GstCaps* caps);
23LL_GST_SYM(true, gst_caps_ref, GstCaps *, GstCaps* caps);
24LL_GST_SYM(true, _gst_debug_register_funcptr, void, GstDebugFuncPtr func, gchar* ptrname);
25LL_GST_SYM(true, _gst_debug_category_new, GstDebugCategory *, gchar *name, guint color, gchar *description);
26LL_GST_SYM(true, gst_caps_is_empty, gboolean, const GstCaps *caps);
27LL_GST_SYM(true, gst_caps_from_string, GstCaps *, const gchar *string);
28LL_GST_SYM(true, gst_caps_replace, void, GstCaps **caps, GstCaps *newcaps);
29LL_GST_SYM(true, gst_caps_get_structure, GstStructure *, const GstCaps *caps, guint index);
30LL_GST_SYM(true, gst_caps_copy, GstCaps *, const GstCaps * caps);
31LL_GST_SYM(true, gst_caps_intersect, GstCaps *, const GstCaps *caps1, const GstCaps *caps2);
32LL_GST_SYM(true, gst_element_register, gboolean, GstPlugin *plugin, const gchar *name, guint rank, GType type);
33LL_GST_SYM(true, _gst_plugin_register_static, void, GstPluginDesc *desc);
34LL_GST_SYM(true, gst_structure_get_int, gboolean, const GstStructure *structure, const gchar *fieldname, gint *value);
35LL_GST_SYM(true, gst_structure_get_value, G_CONST_RETURN GValue *, const GstStructure *structure, const gchar *fieldname);
36LL_GST_SYM(true, gst_value_get_fraction_numerator, gint, const GValue *value);
37LL_GST_SYM(true, gst_value_get_fraction_denominator, gint, const GValue *value);
38LL_GST_SYM(true, gst_structure_get_name, G_CONST_RETURN gchar *, const GstStructure *structure);
39LL_GST_SYM(true, gst_element_seek, bool, GstElement *, gdouble, GstFormat, GstSeekFlags, GstSeekType, gint64, GstSeekType, gint64);
40LL_GST_SYM(true, gst_version, void, guint *major, guint *minor, guint *micro, guint *nano);
41LL_GST_SYM(true, gst_element_state_change_return_get_name, const gchar *, GstStateChangeReturn state_ret);
42
43// optional symbols to grab
44LL_GST_SYM(false, gst_segtrap_set_enabled, void, gboolean enabled);
45LL_GST_SYM(false, gst_message_parse_buffering, void, GstMessage *message, gint *percent);
46LL_GST_SYM(false, gst_message_parse_info, void, GstMessage *message, GError **gerror, gchar **debug);
47
48//aw tag infos (Artist, Title, ...tbc...)
49LL_GST_SYM(true, gst_message_parse_tag, void, GstMessage *message, GstTagList **tag_list);
50LL_GST_SYM(true, gst_tag_list_get_string, gboolean, const GstTagList *list, const gchar *tag, gchar **value);
diff --git a/linden/indra/llmedia/llmediaimplgstreamer_syms_rawa.inc b/linden/indra/llmedia/llmediaimplgstreamer_syms_rawa.inc
deleted file mode 100644
index 0be99b5..0000000
--- a/linden/indra/llmedia/llmediaimplgstreamer_syms_rawa.inc
+++ /dev/null
@@ -1,5 +0,0 @@
1
2// required symbols to grab
3LL_GST_SYM(true, gst_audio_sink_get_type, GType, void);
4
5// optional symbols to grab
diff --git a/linden/indra/llmedia/llmediaimplgstreamer_syms_rawv.inc b/linden/indra/llmedia/llmediaimplgstreamer_syms_rawv.inc
deleted file mode 100644
index 14fbcb4..0000000
--- a/linden/indra/llmedia/llmediaimplgstreamer_syms_rawv.inc
+++ /dev/null
@@ -1,5 +0,0 @@
1
2// required symbols to grab
3LL_GST_SYM(true, gst_video_sink_get_type, GType, void);
4
5// optional symbols to grab
diff --git a/linden/indra/llmedia/llmediaimplgstreamervidplug.cpp b/linden/indra/llmedia/llmediaimplgstreamervidplug.cpp
index c95ef36..eaf2c84 100644
--- a/linden/indra/llmedia/llmediaimplgstreamervidplug.cpp
+++ b/linden/indra/llmedia/llmediaimplgstreamervidplug.cpp
@@ -37,8 +37,6 @@
37#include <gst/video/video.h> 37#include <gst/video/video.h>
38#include <gst/video/gstvideosink.h> 38#include <gst/video/gstvideosink.h>
39 39
40#include "llmediaimplgstreamer_syms.h"
41
42#include "llthread.h" 40#include "llthread.h"
43 41
44#include "llmediaimplgstreamervidplug.h" 42#include "llmediaimplgstreamervidplug.h"
@@ -88,9 +86,9 @@ gst_slvideo_base_init (gpointer gclass)
88 }; 86 };
89 GstElementClass *element_class = GST_ELEMENT_CLASS (gclass); 87 GstElementClass *element_class = GST_ELEMENT_CLASS (gclass);
90 88
91 llgst_element_class_add_pad_template (element_class, 89 gst_element_class_add_pad_template (element_class,
92 llgst_static_pad_template_get (&sink_factory)); 90 gst_static_pad_template_get (&sink_factory));
93 llgst_element_class_set_details (element_class, &element_details); 91 gst_element_class_set_details (element_class, &element_details);
94} 92}
95 93
96 94
@@ -101,7 +99,7 @@ gst_slvideo_finalize (GObject * object)
101 slvideo = GST_SLVIDEO (object); 99 slvideo = GST_SLVIDEO (object);
102 if (slvideo->caps) 100 if (slvideo->caps)
103 { 101 {
104 llgst_caps_unref(slvideo->caps); 102 gst_caps_unref(slvideo->caps);
105 } 103 }
106 104
107 G_OBJECT_CLASS(parent_class)->finalize (object); 105 G_OBJECT_CLASS(parent_class)->finalize (object);
@@ -112,7 +110,7 @@ static GstFlowReturn
112gst_slvideo_show_frame (GstBaseSink * bsink, GstBuffer * buf) 110gst_slvideo_show_frame (GstBaseSink * bsink, GstBuffer * buf)
113{ 111{
114 GstSLVideo *slvideo; 112 GstSLVideo *slvideo;
115 llg_return_val_if_fail (buf != NULL, GST_FLOW_ERROR); 113 g_return_val_if_fail (buf != NULL, GST_FLOW_ERROR);
116 114
117 slvideo = GST_SLVIDEO(bsink); 115 slvideo = GST_SLVIDEO(bsink);
118 116
@@ -205,7 +203,7 @@ gst_slvideo_get_caps (GstBaseSink * bsink)
205 GstSLVideo *slvideo; 203 GstSLVideo *slvideo;
206 slvideo = GST_SLVIDEO(bsink); 204 slvideo = GST_SLVIDEO(bsink);
207 205
208 return llgst_caps_ref (slvideo->caps); 206 return gst_caps_ref (slvideo->caps);
209} 207}
210 208
211 209
@@ -221,36 +219,36 @@ gst_slvideo_set_caps (GstBaseSink * bsink, GstCaps * caps)
221 219
222 filter = GST_SLVIDEO(bsink); 220 filter = GST_SLVIDEO(bsink);
223 221
224 intersection = llgst_caps_intersect (filter->caps, caps); 222 intersection = gst_caps_intersect (filter->caps, caps);
225 if (llgst_caps_is_empty (intersection)) 223 if (gst_caps_is_empty (intersection))
226 { 224 {
227 // no overlap between our caps and requested caps 225 // no overlap between our caps and requested caps
228 return FALSE; 226 return FALSE;
229 } 227 }
230 llgst_caps_unref(intersection); 228 gst_caps_unref(intersection);
231 229
232 int width = 0; 230 int width = 0;
233 int height = 0; 231 int height = 0;
234 gboolean ret; 232 gboolean ret;
235 const GValue *fps; 233 const GValue *fps;
236 const GValue *par; 234 const GValue *par;
237 structure = llgst_caps_get_structure (caps, 0); 235 structure = gst_caps_get_structure (caps, 0);
238 ret = llgst_structure_get_int (structure, "width", &width); 236 ret = gst_structure_get_int (structure, "width", &width);
239 ret = ret && llgst_structure_get_int (structure, "height", &height); 237 ret = ret && gst_structure_get_int (structure, "height", &height);
240 fps = llgst_structure_get_value (structure, "framerate"); 238 fps = gst_structure_get_value (structure, "framerate");
241 ret = ret && (fps != NULL); 239 ret = ret && (fps != NULL);
242 par = llgst_structure_get_value (structure, "pixel-aspect-ratio"); 240 par = gst_structure_get_value (structure, "pixel-aspect-ratio");
243 if (!ret) 241 if (!ret)
244 return FALSE; 242 return FALSE;
245 243
246 filter->width = width; 244 filter->width = width;
247 filter->height = height; 245 filter->height = height;
248 filter->fps_n = llgst_value_get_fraction_numerator(fps); 246 filter->fps_n = gst_value_get_fraction_numerator(fps);
249 filter->fps_d = llgst_value_get_fraction_denominator(fps); 247 filter->fps_d = gst_value_get_fraction_denominator(fps);
250 if (par) 248 if (par)
251 { 249 {
252 filter->par_n = llgst_value_get_fraction_numerator(par); 250 filter->par_n = gst_value_get_fraction_numerator(par);
253 filter->par_d = llgst_value_get_fraction_denominator(par); 251 filter->par_d = gst_value_get_fraction_denominator(par);
254 } 252 }
255 else 253 else
256 { 254 {
@@ -261,15 +259,15 @@ gst_slvideo_set_caps (GstBaseSink * bsink, GstCaps * caps)
261 GST_VIDEO_SINK_HEIGHT(filter) = height; 259 GST_VIDEO_SINK_HEIGHT(filter) = height;
262 260
263 filter->format = SLV_PF_UNKNOWN; 261 filter->format = SLV_PF_UNKNOWN;
264 if (0 == strcmp(llgst_structure_get_name(structure), 262 if (0 == strcmp(gst_structure_get_name(structure),
265 "video/x-raw-rgb")) 263 "video/x-raw-rgb"))
266 { 264 {
267 int red_mask; 265 int red_mask;
268 int green_mask; 266 int green_mask;
269 int blue_mask; 267 int blue_mask;
270 llgst_structure_get_int(structure, "red_mask", &red_mask); 268 gst_structure_get_int(structure, "red_mask", &red_mask);
271 llgst_structure_get_int(structure, "green_mask", &green_mask); 269 gst_structure_get_int(structure, "green_mask", &green_mask);
272 llgst_structure_get_int(structure, "blue_mask", &blue_mask); 270 gst_structure_get_int(structure, "blue_mask", &blue_mask);
273 if ((unsigned int)red_mask == 0xFF000000 && 271 if ((unsigned int)red_mask == 0xFF000000 &&
274 (unsigned int)green_mask == 0x00FF0000 && 272 (unsigned int)green_mask == 0x00FF0000 &&
275 (unsigned int)blue_mask == 0x0000FF00) 273 (unsigned int)blue_mask == 0x0000FF00)
@@ -366,9 +364,9 @@ gst_slvideo_update_caps (GstSLVideo * slvideo)
366 // GStreamer will automatically convert colourspace if necessary. 364 // GStreamer will automatically convert colourspace if necessary.
367 // GStreamer will automatically resize media to one of these enumerated 365 // GStreamer will automatically resize media to one of these enumerated
368 // powers-of-two that we ask for (yay GStreamer!) 366 // powers-of-two that we ask for (yay GStreamer!)
369 caps = llgst_caps_from_string (SLV_ALLCAPS); 367 caps = gst_caps_from_string (SLV_ALLCAPS);
370 368
371 llgst_caps_replace (&slvideo->caps, caps); 369 gst_caps_replace (&slvideo->caps, caps);
372} 370}
373 371
374 372
@@ -401,7 +399,7 @@ static void
401gst_slvideo_set_property (GObject * object, guint prop_id, 399gst_slvideo_set_property (GObject * object, guint prop_id,
402 const GValue * value, GParamSpec * pspec) 400 const GValue * value, GParamSpec * pspec)
403{ 401{
404 llg_return_if_fail (GST_IS_SLVIDEO (object)); 402 g_return_if_fail (GST_IS_SLVIDEO (object));
405 403
406 if (prop_id) { 404 if (prop_id) {
407 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); 405 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
@@ -412,7 +410,7 @@ static void
412gst_slvideo_get_property (GObject * object, guint prop_id, 410gst_slvideo_get_property (GObject * object, guint prop_id,
413 GValue * value, GParamSpec * pspec) 411 GValue * value, GParamSpec * pspec)
414{ 412{
415 llg_return_if_fail (GST_IS_SLVIDEO (object)); 413 g_return_if_fail (GST_IS_SLVIDEO (object));
416 414
417 if (prop_id) { 415 if (prop_id) {
418 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); 416 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
@@ -433,30 +431,23 @@ plugin_init (GstPlugin * plugin)
433 GST_DEBUG_CATEGORY_INIT (gst_slvideo_debug, "private-slvideo-plugin", 431 GST_DEBUG_CATEGORY_INIT (gst_slvideo_debug, "private-slvideo-plugin",
434 0, "Second Life Video Sink"); 432 0, "Second Life Video Sink");
435 433
436 return llgst_element_register (plugin, "private-slvideo", 434 return gst_element_register (plugin, "private-slvideo",
437 GST_RANK_NONE, GST_TYPE_SLVIDEO); 435 GST_RANK_NONE, GST_TYPE_SLVIDEO);
438} 436}
439 437
440/* this is the structure that gstreamer looks for to register plugins
441 */
442/* NOTE: Can't rely upon GST_PLUGIN_DEFINE_STATIC to self-register, since
443 some g++ versions buggily avoid __attribute__((constructor)) functions -
444 so we provide an explicit plugin init function.
445 */
446#define PACKAGE "packagehack"
447 GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
448 GST_VERSION_MINOR,
449 "private-slvideoplugin",
450 "SL Video sink plugin",
451 plugin_init, "0.1", GST_LICENSE_UNKNOWN,
452 "Second Life",
453 "http://www.secondlife.com/");
454#undef PACKAGE
455 438
456void gst_slvideo_init_class (void) 439void gst_slvideo_init_class (void)
457{ 440{
458 ll_gst_plugin_register_static (&gst_plugin_desc); 441 gst_plugin_register_static( GST_VERSION_MAJOR,
459 //fprintf(stderr, "\n\n\nCLASS INIT\n\n\n"); 442 GST_VERSION_MINOR,
443 "private-slvideoplugin",
444 "SL Video sink plugin",
445 plugin_init,
446 "0.1",
447 GST_LICENSE_UNKNOWN,
448 "Second Life",
449 "Second Life",
450 "http://www.secondlife.com/" );
460} 451}
461 452
462///#endif // LL_GSTREAMER_ENABLED 453///#endif // LL_GSTREAMER_ENABLED
diff --git a/linden/indra/llmedia/llmediamanager.cpp b/linden/indra/llmedia/llmediamanager.cpp
index 5394b62..7fe0df3 100644
--- a/linden/indra/llmedia/llmediamanager.cpp
+++ b/linden/indra/llmedia/llmediamanager.cpp
@@ -30,12 +30,24 @@
30 */ 30 */
31 31
32#include "llmediamanager.h" 32#include "llmediamanager.h"
33
34#if LL_WINDOWS
35 // GStreamer 0.10.22 - gstutils.h - conversion from 'guint64' to 'guint8'.
36 // This was an intentional change to make GStreamer more threadsafe, and
37 // is okay. Delete this bit if GStreamer ever gets more VS-friendly -- McCabe
38 #pragma warning(disable : 4244)
39#endif
40#include "llmediaimplgstreamer.h"
41#if LL_WINDOWS
42 #pragma warning(default : 4244)
43#endif
44
33#include "llmediaimplfactory.h" 45#include "llmediaimplfactory.h"
34 46
35#include "llmediaimplexample1.h" 47#include "llmediaimplexample1.h"
36#include "llmediaimplexample2.h" 48#include "llmediaimplexample2.h"
37#include "llmediaimplquicktime.h" 49#include "llmediaimplquicktime.h"
38#include "llmediaimplgstreamer.h" 50
39#if LL_LLMOZLIB_ENABLED 51#if LL_LLMOZLIB_ENABLED
40# include "llmediaimplllmozlib.h" 52# include "llmediaimplllmozlib.h"
41#endif 53#endif
diff --git a/linden/indra/llmessage/llhttpclient.cpp b/linden/indra/llmessage/llhttpclient.cpp
index a07f379..8a4b063 100644
--- a/linden/indra/llmessage/llhttpclient.cpp
+++ b/linden/indra/llmessage/llhttpclient.cpp
@@ -160,10 +160,9 @@ namespace
160 fstream.seekg(0, std::ios::end); 160 fstream.seekg(0, std::ios::end);
161 U32 fileSize = fstream.tellg(); 161 U32 fileSize = fstream.tellg();
162 fstream.seekg(0, std::ios::beg); 162 fstream.seekg(0, std::ios::beg);
163 char* fileBuffer; 163 std::vector<char> fileBuffer(fileSize);
164 fileBuffer = new char [fileSize]; 164 fstream.read(&fileBuffer[0], fileSize);
165 fstream.read(fileBuffer, fileSize); 165 ostream.write(&fileBuffer[0], fileSize);
166 ostream.write(fileBuffer, fileSize);
167 fstream.close(); 166 fstream.close();
168 eos = true; 167 eos = true;
169 return STATUS_DONE; 168 return STATUS_DONE;
@@ -190,10 +189,9 @@ namespace
190 189
191 LLVFile vfile(gVFS, mUUID, mAssetType, LLVFile::READ); 190 LLVFile vfile(gVFS, mUUID, mAssetType, LLVFile::READ);
192 S32 fileSize = vfile.getSize(); 191 S32 fileSize = vfile.getSize();
193 U8* fileBuffer; 192 std::vector<U8> fileBuffer(fileSize);
194 fileBuffer = new U8 [fileSize]; 193 vfile.read(&fileBuffer[0], fileSize);
195 vfile.read(fileBuffer, fileSize); 194 ostream.write((char*)&fileBuffer[0], fileSize);
196 ostream.write((char*)fileBuffer, fileSize);
197 eos = true; 195 eos = true;
198 return STATUS_DONE; 196 return STATUS_DONE;
199 } 197 }
diff --git a/linden/indra/llui/llfloater.cpp b/linden/indra/llui/llfloater.cpp
index 9f8cd65..2f4e873 100644
--- a/linden/indra/llui/llfloater.cpp
+++ b/linden/indra/llui/llfloater.cpp
@@ -1580,6 +1580,7 @@ void LLFloater::updateButtons()
1580 S32 button_count = 0; 1580 S32 button_count = 0;
1581 for (S32 i = 0; i < BUTTON_COUNT; i++) 1581 for (S32 i = 0; i < BUTTON_COUNT; i++)
1582 { 1582 {
1583 if(!mButtons[i]) continue;
1583 mButtons[i]->setEnabled(mButtonsEnabled[i]); 1584 mButtons[i]->setEnabled(mButtonsEnabled[i]);
1584 1585
1585 if (mButtonsEnabled[i] 1586 if (mButtonsEnabled[i]
diff --git a/linden/indra/newview/app_settings/keywords.ini b/linden/indra/newview/app_settings/keywords.ini
index 2a911ba..56acf4c 100644
--- a/linden/indra/newview/app_settings/keywords.ini
+++ b/linden/indra/newview/app_settings/keywords.ini
@@ -455,6 +455,7 @@ PARCEL_FLAG_ALLOW_LANDMARK Used with llGetParcelFlags to find if a parcel allo
455PARCEL_FLAG_ALLOW_TERRAFORM Used with llGetParcelFlags to find if a parcel allows anyone to terraform the land 455PARCEL_FLAG_ALLOW_TERRAFORM Used with llGetParcelFlags to find if a parcel allows anyone to terraform the land
456PARCEL_FLAG_ALLOW_DAMAGE Used with llGetParcelFlags to find if a parcel allows damage 456PARCEL_FLAG_ALLOW_DAMAGE Used with llGetParcelFlags to find if a parcel allows damage
457PARCEL_FLAG_ALLOW_CREATE_OBJECTS Used with llGetParcelFlags to find if a parcel allows anyone to create objects 457PARCEL_FLAG_ALLOW_CREATE_OBJECTS Used with llGetParcelFlags to find if a parcel allows anyone to create objects
458PARCEL_FLAG_ALLOW_CREATE_GROUP_OBJECTS Used with llGetParcelFlags to find if a parcel allows group members or objects to create objects
458PARCEL_FLAG_USE_ACCESS_GROUP Used with llGetParcelFlags to find if a parcel limits access to a group 459PARCEL_FLAG_USE_ACCESS_GROUP Used with llGetParcelFlags to find if a parcel limits access to a group
459PARCEL_FLAG_USE_ACCESS_LIST Used with llGetParcelFlags to find if a parcel limits access to a list of residents 460PARCEL_FLAG_USE_ACCESS_LIST Used with llGetParcelFlags to find if a parcel limits access to a list of residents
460PARCEL_FLAG_USE_BAN_LIST Used with llGetParcelFlags to find if a parcel uses a ban list 461PARCEL_FLAG_USE_BAN_LIST Used with llGetParcelFlags to find if a parcel uses a ban list
diff --git a/linden/indra/newview/app_settings/settings.xml b/linden/indra/newview/app_settings/settings.xml
index 0f15236..55ff208 100644
--- a/linden/indra/newview/app_settings/settings.xml
+++ b/linden/indra/newview/app_settings/settings.xml
@@ -4659,6 +4659,17 @@
4659 <key>Value</key> 4659 <key>Value</key>
4660 <integer>0</integer> 4660 <integer>0</integer>
4661 </map> 4661 </map>
4662 <key>MediaDebugLevel</key>
4663 <map>
4664 <key>Comment</key>
4665 <string>How spammy media (GStreamer) debug output should be. (0-4)</string>
4666 <key>Persist</key>
4667 <integer>1</integer>
4668 <key>Type</key>
4669 <string>U32</string>
4670 <key>Value</key>
4671 <integer>0</integer>
4672 </map>
4662 <key>MemoryLogFrequency</key> 4673 <key>MemoryLogFrequency</key>
4663 <map> 4674 <map>
4664 <key>Comment</key> 4675 <key>Comment</key>
diff --git a/linden/indra/newview/llappviewer.cpp b/linden/indra/newview/llappviewer.cpp
index 15ce1a3..b2bfccf 100644
--- a/linden/indra/newview/llappviewer.cpp
+++ b/linden/indra/newview/llappviewer.cpp
@@ -75,9 +75,6 @@
75 75
76#if LL_WINDOWS 76#if LL_WINDOWS
77 #include "llwindebug.h" 77 #include "llwindebug.h"
78 #include <direct.h>
79 #include <errno.h>
80 #include <stdlib.h>
81#endif 78#endif
82 79
83#if LL_WINDOWS 80#if LL_WINDOWS
@@ -320,30 +317,6 @@ std::vector<std::string> gLoginURIs;
320static std::string gHelperURI; 317static std::string gHelperURI;
321 318
322 319
323void LLAppViewer::gst_plugin_path()
324{
325#ifdef LL_WINDOWS
326 char* buffer;
327
328 // Get the current working directory:
329 if((buffer = _getcwd(NULL,0)) == NULL)
330 {
331 LL_INFOS("InitInfo") << "_getcwd error" << LL_ENDL;
332 }
333 else
334 {
335 LL_INFOS("InitInfo") << "Imprudence is installed at " << buffer << LL_ENDL;
336
337 std::string plugin_path = "GST_PLUGIN_PATH=" + std::string(buffer) + "\\lib\\gstreamer-plugins";
338
339 // Place GST_PLUGIN_PATH in the environment settings for imprudence.exe
340 const char* gst_plugin_path = plugin_path.c_str();
341 putenv(gst_plugin_path);
342 LL_INFOS("InitInfo") << "GST_PLUGIN_PATH set to " << getenv("GST_PLUGIN_PATH") << LL_ENDL;
343 }
344#endif //LL_WINDOWS
345}
346
347void idle_afk_check() 320void idle_afk_check()
348{ 321{
349 // check idle timers 322 // check idle timers
@@ -643,8 +616,6 @@ bool LLAppViewer::init()
643 LL_VERSION_PATCH, 616 LL_VERSION_PATCH,
644 LL_VERSION_BUILD ); 617 LL_VERSION_BUILD );
645 618
646 gst_plugin_path();
647
648 ////////////////////////////////////////////////////////////////////////////// 619 //////////////////////////////////////////////////////////////////////////////
649 ////////////////////////////////////////////////////////////////////////////// 620 //////////////////////////////////////////////////////////////////////////////
650 ////////////////////////////////////////////////////////////////////////////// 621 //////////////////////////////////////////////////////////////////////////////
diff --git a/linden/indra/newview/llappviewer.h b/linden/indra/newview/llappviewer.h
index 2083a93..58dc835 100644
--- a/linden/indra/newview/llappviewer.h
+++ b/linden/indra/newview/llappviewer.h
@@ -138,8 +138,6 @@ public:
138 138
139protected: 139protected:
140 140
141 void gst_plugin_path(); // Sets GST_PLUGIN_PATH environment variable for GStreamer.
142
143 virtual bool initWindow(); // Initialize the viewer's window. 141 virtual bool initWindow(); // Initialize the viewer's window.
144 virtual bool initLogging(); // Initialize log files, logging system, return false on failure. 142 virtual bool initLogging(); // Initialize log files, logging system, return false on failure.
145 virtual void initConsole() {}; // Initialize OS level debugging console. 143 virtual void initConsole() {}; // Initialize OS level debugging console.
diff --git a/linden/indra/newview/llcompilequeue.cpp b/linden/indra/newview/llcompilequeue.cpp
index fc2742f..b4f1a22 100644
--- a/linden/indra/newview/llcompilequeue.cpp
+++ b/linden/indra/newview/llcompilequeue.cpp
@@ -59,6 +59,7 @@
59#include "llfloaterchat.h" 59#include "llfloaterchat.h"
60#include "llviewerstats.h" 60#include "llviewerstats.h"
61#include "lluictrlfactory.h" 61#include "lluictrlfactory.h"
62#include "llselectmgr.h"
62 63
63///---------------------------------------------------------------------------- 64///----------------------------------------------------------------------------
64/// Local function declarations, constants, enums, and typedefs 65/// Local function declarations, constants, enums, and typedefs
@@ -194,7 +195,20 @@ BOOL LLFloaterScriptQueue::start()
194{ 195{
195 //llinfos << "LLFloaterCompileQueue::start()" << llendl; 196 //llinfos << "LLFloaterCompileQueue::start()" << llendl;
196 std::string buffer; 197 std::string buffer;
197 buffer = llformat("Starting %s of %d items.", mStartString.c_str(), mObjectIDs.count()); // *TODO: Translate 198
199 LLSelectMgr *mgr = LLSelectMgr::getInstance();
200 LLObjectSelectionHandle selectHandle = mgr->getSelection();
201 U32 n_objects = 0;
202 if (gSavedSettings.getBOOL("EditLinkedParts"))
203 {
204 n_objects = selectHandle->getObjectCount();
205 }
206 else
207 {
208 n_objects = selectHandle->getRootObjectCount();
209 }
210
211 buffer = llformat("Starting %s of %d items.", mStartString.c_str(), n_objects); // *TODO: Translate
198 212
199 LLScrollListCtrl* list = getChild<LLScrollListCtrl>("queue output"); 213 LLScrollListCtrl* list = getChild<LLScrollListCtrl>("queue output");
200 list->addCommentText(buffer); 214 list->addCommentText(buffer);
diff --git a/linden/indra/newview/lldrawable.cpp b/linden/indra/newview/lldrawable.cpp
index 5e4aafc..e058681 100644
--- a/linden/indra/newview/lldrawable.cpp
+++ b/linden/indra/newview/lldrawable.cpp
@@ -497,7 +497,7 @@ F32 LLDrawable::updateXform(BOOL undamped)
497 F32 dist_squared = 0.f; 497 F32 dist_squared = 0.f;
498 F32 camdist2 = (mDistanceWRTCamera * mDistanceWRTCamera); 498 F32 camdist2 = (mDistanceWRTCamera * mDistanceWRTCamera);
499 499
500 if (damped && mDistanceWRTCamera > 0.0f) 500 if (damped && isVisible())
501 { 501 {
502 F32 lerp_amt = llclamp(LLCriticalDamp::getInterpolant(OBJECT_DAMPING_TIME_CONSTANT), 0.f, 1.f); 502 F32 lerp_amt = llclamp(LLCriticalDamp::getInterpolant(OBJECT_DAMPING_TIME_CONSTANT), 0.f, 1.f);
503 LLVector3 new_pos = lerp(old_pos, target_pos, lerp_amt); 503 LLVector3 new_pos = lerp(old_pos, target_pos, lerp_amt);
@@ -521,11 +521,19 @@ F32 LLDrawable::updateXform(BOOL undamped)
521 { 521 {
522 // snap to final position 522 // snap to final position
523 dist_squared = 0.0f; 523 dist_squared = 0.0f;
524 if (!isRoot())
525 { //child prim snapping to some position, needs a rebuild
526 gPipeline.markRebuild(this, LLDrawable::REBUILD_POSITION, TRUE);
527 }
524 } 528 }
525 } 529 }
526 530
527 if ((mCurrentScale != target_scale) || 531 if ((mCurrentScale != target_scale) ||
528 (!isRoot() && (dist_squared >= MIN_INTERPOLATE_DISTANCE_SQUARED) || !mVObjp->getAngularVelocity().isExactlyZero())) 532 (!isRoot() &&
533 (dist_squared >= MIN_INTERPOLATE_DISTANCE_SQUARED) ||
534 !mVObjp->getAngularVelocity().isExactlyZero() ||
535 target_pos != mXform.getPosition() ||
536 target_rot != mXform.getRotation()))
529 { //child prim moving or scale change requires immediate rebuild 537 { //child prim moving or scale change requires immediate rebuild
530 gPipeline.markRebuild(this, LLDrawable::REBUILD_POSITION, TRUE); 538 gPipeline.markRebuild(this, LLDrawable::REBUILD_POSITION, TRUE);
531 } 539 }
diff --git a/linden/indra/newview/llfilepicker.cpp b/linden/indra/newview/llfilepicker.cpp
index bbcfba6..1edfbe6 100644
--- a/linden/indra/newview/llfilepicker.cpp
+++ b/linden/indra/newview/llfilepicker.cpp
@@ -899,12 +899,41 @@ BOOL LLFilePicker::getSaveFile(ESaveFilter filter, const std::string& filename)
899// static 899// static
900void LLFilePicker::add_to_selectedfiles(gpointer data, gpointer user_data) 900void LLFilePicker::add_to_selectedfiles(gpointer data, gpointer user_data)
901{ 901{
902 LLFilePicker* picker = (LLFilePicker*) user_data; 902 // We need to run g_filename_to_utf8 in the user's locale
903 std::string old_locale(setlocale(LC_ALL, NULL));
904 setlocale(LC_ALL, "");
905
906 LLFilePicker* picker = (LLFilePicker*) user_data;
907 GError *error = NULL;
903 gchar* filename_utf8 = g_filename_to_utf8((gchar*)data, 908 gchar* filename_utf8 = g_filename_to_utf8((gchar*)data,
904 -1, NULL, NULL, NULL); 909 -1, NULL, NULL, &error);
905 picker->mFiles.push_back(std::string(filename_utf8)); 910 if (error)
906 lldebugs << "ADDED FILE " << filename_utf8 << llendl; 911 {
907 g_free(filename_utf8); 912 // This condition should really be notified to the user, e.g.,
913 // through a message box. Just logging it is inapropriate.
914 // FIXME.
915
916 // Ghhhh. g_filename_display_name is new to glib 2.6, and it
917 // is too new for SL! (Note that the latest glib as of this
918 // writing is 2.22. *sigh*) LL supplied *makeASCII family are
919 // also unsuitable since they allow control characters...
920
921 std::string display_name;
922 for (const gchar *str = (const gchar *)data; *str; str++)
923 {
924 display_name += (char)((*str >= 0x20 && *str <= 0x7E) ? *str : '?');
925 }
926 llwarns << "g_filename_to_utf8 failed on \"" << display_name << "\": " << error->message << llendl;
927 }
928
929 if (filename_utf8)
930 {
931 picker->mFiles.push_back(std::string(filename_utf8));
932 lldebugs << "ADDED FILE " << filename_utf8 << llendl;
933 g_free(filename_utf8);
934 }
935
936 setlocale(LC_ALL, old_locale.c_str());
908} 937}
909 938
910// static 939// static
diff --git a/linden/indra/newview/llfloaterchat.cpp b/linden/indra/newview/llfloaterchat.cpp
index a626ad3..14f9a10 100644
--- a/linden/indra/newview/llfloaterchat.cpp
+++ b/linden/indra/newview/llfloaterchat.cpp
@@ -385,7 +385,11 @@ void LLFloaterChat::addChat(const LLChat& chat,
385 text_color = gSavedSettings.getColor("IMChatColor"); 385 text_color = gSavedSettings.getColor("IMChatColor");
386 size = INSTANT_MSG_SIZE; 386 size = INSTANT_MSG_SIZE;
387 } 387 }
388 gConsole->addLine(chat.mText, size, text_color); 388 // We display anything if it's not an IM. If it's an IM, check pref...
389 if ( !from_instant_message || gSavedSettings.getBOOL("IMInChatHistory") )
390 {
391 gConsole->addLine(chat.mText, size, text_color);
392 }
389 } 393 }
390 394
391 if(from_instant_message && gSavedPerAccountSettings.getBOOL("LogChatIM")) 395 if(from_instant_message && gSavedPerAccountSettings.getBOOL("LogChatIM"))
diff --git a/linden/indra/newview/llviewercontrol.cpp b/linden/indra/newview/llviewercontrol.cpp
index 7a30b22..a8a3b16 100644
--- a/linden/indra/newview/llviewercontrol.cpp
+++ b/linden/indra/newview/llviewercontrol.cpp
@@ -70,6 +70,7 @@
70#include "llvosurfacepatch.h" 70#include "llvosurfacepatch.h"
71#include "llvowlsky.h" 71#include "llvowlsky.h"
72#include "llrender.h" 72#include "llrender.h"
73#include "llmediamanager.h"
73 74
74#ifdef TOGGLE_HACKED_GODLIKE_VIEWER 75#ifdef TOGGLE_HACKED_GODLIKE_VIEWER
75BOOL gHackGodmode = FALSE; 76BOOL gHackGodmode = FALSE;
@@ -429,6 +430,22 @@ bool handleVoiceClientPrefsChanged(const LLSD& newvalue)
429 return true; 430 return true;
430} 431}
431 432
433bool handleMediaDebugLevelChanged(const LLSD& newvalue)
434{
435 LLMediaManager *mgr = LLMediaManager::getInstance();
436 if (mgr)
437 {
438 LLMediaBase *impl =
439 mgr->createSourceFromMimeType("http", "audio/mpeg");
440
441 if (impl)
442 {
443 impl->setDebugLevel( (LLMediaBase::EDebugLevel)newvalue.asInteger() );
444 }
445 }
446 return true;
447}
448
432//////////////////////////////////////////////////////////////////////////// 449////////////////////////////////////////////////////////////////////////////
433 450
434void settings_setup_listeners() 451void settings_setup_listeners()
@@ -556,6 +573,7 @@ void settings_setup_listeners()
556 gSavedSettings.getControl("VoiceInputAudioDevice")->getSignal()->connect(boost::bind(&handleVoiceClientPrefsChanged, _1)); 573 gSavedSettings.getControl("VoiceInputAudioDevice")->getSignal()->connect(boost::bind(&handleVoiceClientPrefsChanged, _1));
557 gSavedSettings.getControl("VoiceOutputAudioDevice")->getSignal()->connect(boost::bind(&handleVoiceClientPrefsChanged, _1)); 574 gSavedSettings.getControl("VoiceOutputAudioDevice")->getSignal()->connect(boost::bind(&handleVoiceClientPrefsChanged, _1));
558 gSavedSettings.getControl("LipSyncEnabled")->getSignal()->connect(boost::bind(&handleVoiceClientPrefsChanged, _1)); 575 gSavedSettings.getControl("LipSyncEnabled")->getSignal()->connect(boost::bind(&handleVoiceClientPrefsChanged, _1));
576 gSavedSettings.getControl("MediaDebugLevel")->getSignal()->connect(boost::bind(&handleMediaDebugLevelChanged, _1));
559} 577}
560 578
561template <> eControlType get_control_type<U32>(const U32& in, LLSD& out) 579template <> eControlType get_control_type<U32>(const U32& in, LLSD& out)
diff --git a/linden/indra/newview/llviewermedia.cpp b/linden/indra/newview/llviewermedia.cpp
index e99c838..d8fcdcb 100644
--- a/linden/indra/newview/llviewermedia.cpp
+++ b/linden/indra/newview/llviewermedia.cpp
@@ -496,6 +496,13 @@ void LLViewerMedia::initClass()
496 const LLMIMETypes::LLMIMEInfo& info = it->second; 496 const LLMIMETypes::LLMIMEInfo& info = it->second;
497 mm->addMimeTypeImplNameMap( mime_type, info.mImpl ); 497 mm->addMimeTypeImplNameMap( mime_type, info.mImpl );
498 } 498 }
499
500 LLMediaBase *impl = mm->createSourceFromMimeType("http", "audio/mpeg");
501 if (impl)
502 {
503 U32 level = gSavedSettings.getU32("MediaDebugLevel");
504 impl->setDebugLevel( (LLMediaBase::EDebugLevel)level );
505 }
499} 506}
500 507
501////////////////////////////////////////////////////////////////////////////////////////// 508//////////////////////////////////////////////////////////////////////////////////////////
diff --git a/linden/indra/newview/llviewermenu.cpp b/linden/indra/newview/llviewermenu.cpp
index 31fdcc3..486e4f1 100644
--- a/linden/indra/newview/llviewermenu.cpp
+++ b/linden/indra/newview/llviewermenu.cpp
@@ -5963,8 +5963,9 @@ void queue_actions(LLFloaterScriptQueue* q, const std::string& noscriptmsg, cons
5963{ 5963{
5964 // Apply until an object fails 5964 // Apply until an object fails
5965 QueueObjects func(q); 5965 QueueObjects func(q);
5966 const bool firstonly = true; 5966 LLSelectMgr *mgr = LLSelectMgr::getInstance();
5967 bool fail = LLSelectMgr::getInstance()->getSelection()->applyToObjects(&func, firstonly); 5967 LLObjectSelectionHandle selectHandle = mgr->getSelection();
5968 bool fail = selectHandle->applyToObjects(&func);
5968 if(fail) 5969 if(fail)
5969 { 5970 {
5970 if ( !func.scripted ) 5971 if ( !func.scripted )
diff --git a/linden/indra/newview/llviewermessage.cpp b/linden/indra/newview/llviewermessage.cpp
index b947f95..221de59 100644
--- a/linden/indra/newview/llviewermessage.cpp
+++ b/linden/indra/newview/llviewermessage.cpp
@@ -3286,6 +3286,10 @@ void process_sound_trigger(LLMessageSystem *msg, void **)
3286 msg->getVector3Fast(_PREHASH_SoundData, _PREHASH_Position, pos_local); 3286 msg->getVector3Fast(_PREHASH_SoundData, _PREHASH_Position, pos_local);
3287 msg->getF32Fast(_PREHASH_SoundData, _PREHASH_Gain, gain); 3287 msg->getF32Fast(_PREHASH_SoundData, _PREHASH_Gain, gain);
3288 3288
3289 //If we have sounds muted, don't even try to load or trigger the sound.
3290 if(gSavedSettings.getBOOL("MuteSounds") || gain == 0.0)
3291 return;
3292
3289 // adjust sound location to true global coords 3293 // adjust sound location to true global coords
3290 LLVector3d pos_global = from_region_handle(region_handle); 3294 LLVector3d pos_global = from_region_handle(region_handle);
3291 pos_global.mdV[VX] += pos_local.mV[VX]; 3295 pos_global.mdV[VX] += pos_local.mV[VX];
@@ -3317,7 +3321,7 @@ void process_sound_trigger(LLMessageSystem *msg, void **)
3317 3321
3318void process_preload_sound(LLMessageSystem *msg, void **user_data) 3322void process_preload_sound(LLMessageSystem *msg, void **user_data)
3319{ 3323{
3320 if (!gAudiop) 3324 if (!gAudiop || gSavedSettings.getBOOL("MuteSounds"))
3321 { 3325 {
3322 return; 3326 return;
3323 } 3327 }
@@ -3363,6 +3367,9 @@ void process_attached_sound(LLMessageSystem *msg, void **user_data)
3363 msg->getF32Fast(_PREHASH_DataBlock, _PREHASH_Gain, gain); 3367 msg->getF32Fast(_PREHASH_DataBlock, _PREHASH_Gain, gain);
3364 msg->getU8Fast(_PREHASH_DataBlock, _PREHASH_Flags, flags); 3368 msg->getU8Fast(_PREHASH_DataBlock, _PREHASH_Flags, flags);
3365 3369
3370 if(gSavedSettings.getBOOL("MuteSounds") || gain == 0.0)
3371 return;
3372
3366 LLViewerObject *objectp = gObjectList.findObject(object_id); 3373 LLViewerObject *objectp = gObjectList.findObject(object_id);
3367 if (!objectp) 3374 if (!objectp)
3368 { 3375 {
diff --git a/linden/indra/newview/llvoavatar.cpp b/linden/indra/newview/llvoavatar.cpp
index fd2487b..7d70dd2 100644
--- a/linden/indra/newview/llvoavatar.cpp
+++ b/linden/indra/newview/llvoavatar.cpp
@@ -3096,69 +3096,62 @@ void LLVOAvatar::idleUpdateNameTag(const LLVector3& root_pos_last)
3096 (is_away != mNameAway || is_busy != mNameBusy || is_muted != mNameMute) 3096 (is_away != mNameAway || is_busy != mNameBusy || is_muted != mNameMute)
3097 || is_appearance != mNameAppearance) 3097 || is_appearance != mNameAppearance)
3098 { 3098 {
3099 char line[MAX_STRING]; /* Flawfinder: ignore */ 3099 std::string line;
3100 if (!sRenderGroupTitles) 3100 if (title && title->getString() && title->getString()[0] != '\0')
3101 {
3102 // If all group titles are turned off, stack first name
3103 // on a line above last name
3104 strncpy(line, firstname->getString(), MAX_STRING -1 ); /* Flawfinder: ignore */
3105 line[MAX_STRING -1] = '\0';
3106 strcat(line, "\n");
3107 }
3108 else if (title && title->getString() && title->getString()[0] != '\0')
3109 { 3101 {
3110 strncpy(line, title->getString(), MAX_STRING -1 ); /* Flawfinder: ignore */ 3102 line += title->getString();
3111 line[MAX_STRING -1] = '\0'; 3103 LLStringFn::replace_nonprintable(line,LL_UNKNOWN_CHAR);
3112 strcat(line, "\n"); /* Flawfinder: ignore */ 3104 line += "\n";
3113 strncat(line, firstname->getString(), MAX_STRING - strlen(line) -1 ); /* Flawfinder: ignore */ 3105 line += firstname->getString();
3114 } 3106 }
3115 else 3107 else
3116 { 3108 {
3117 strncpy(line, firstname->getString(), MAX_STRING -1 ); /* Flawfinder: ignore */ 3109 line += firstname->getString();
3118 line[MAX_STRING -1] = '\0';
3119 } 3110 }
3120 3111
3121 strcat(line, " "); /* Flawfinder: ignore */ 3112 line += " ";
3122 strncat(line, lastname->getString(), MAX_STRING - strlen(line) -1); /* Flawfinder: ignore */ 3113 line += lastname->getString();
3123 BOOL need_comma = FALSE; 3114 BOOL need_comma = FALSE;
3124 3115
3125 if (is_away || is_muted || is_busy) 3116 if (is_away || is_muted || is_busy)
3126 { 3117 {
3127 strcat(line, " ("); /* Flawfinder: ignore */ 3118 line += " (";
3128 if (is_away) 3119 if (is_away)
3129 { 3120 {
3130 strcat(line, "Away"); /* Flawfinder: ignore */ 3121 line += "Away";
3131 need_comma = TRUE; 3122 need_comma = TRUE;
3132 } 3123 }
3133 if (is_busy) 3124 if (is_busy)
3134 { 3125 {
3135 if (need_comma) 3126 if (need_comma)
3136 { 3127 {
3137 strcat(line, ", "); /* Flawfinder: ignore */ 3128 line += ", ";
3138 } 3129 }
3139 strcat(line, "Busy"); /* Flawfinder: ignore */ 3130 line += "Busy";
3140 need_comma = TRUE; 3131 need_comma = TRUE;
3141 } 3132 }
3142 if (is_muted) 3133 if (is_muted)
3143 { 3134 {
3144 if (need_comma) 3135 if (need_comma)
3145 { 3136 {
3146 strcat(line, ", "); /* Flawfinder: ignore */ 3137 line += ", ";
3147 } 3138 }
3148 strcat(line, "Muted"); /* Flawfinder: ignore */ 3139 line += "Muted";
3149 need_comma = TRUE; 3140 need_comma = TRUE;
3150 } 3141 }
3151 strcat(line,")"); /* Flawfinder: ignore */ 3142 line += ")";
3152 } 3143 }
3153 if (is_appearance) 3144 if (is_appearance)
3154 { 3145 {
3155 strcat(line, "\n(Editing Appearance)"); /* Flawfinder: ignore */ 3146 line += "\n";
3147 line += "(Editing Appearance)";
3156 } 3148 }
3157 mNameAway = is_away; 3149 mNameAway = is_away;
3158 mNameBusy = is_busy; 3150 mNameBusy = is_busy;
3159 mNameMute = is_muted; 3151 mNameMute = is_muted;
3160 mNameAppearance = is_appearance; 3152 mNameAppearance = is_appearance;
3161 mTitle = title ? title->getString() : ""; 3153 mTitle = title ? title->getString() : "";
3154 LLStringFn::replace_nonprintable(mTitle,LL_UNKNOWN_CHAR);
3162 mNameString = utf8str_to_wstring(line); 3155 mNameString = utf8str_to_wstring(line);
3163 new_name = TRUE; 3156 new_name = TRUE;
3164 } 3157 }
diff --git a/linden/indra/newview/pipeline.cpp b/linden/indra/newview/pipeline.cpp
index 3523635..e7db166 100644
--- a/linden/indra/newview/pipeline.cpp
+++ b/linden/indra/newview/pipeline.cpp
@@ -2665,6 +2665,18 @@ void LLPipeline::renderForSelect(std::set<LLViewerObject*>& objects, BOOL render
2665 LLGLState::checkClientArrays(); 2665 LLGLState::checkClientArrays();
2666 U32 last_type = 0; 2666 U32 last_type = 0;
2667 2667
2668 // If we don't do this, we crash something on changing graphics settings
2669 // from Medium -> Low, because we unload all the shaders and the
2670 // draw pools aren't aware. I don't know if this has to be a separate
2671 // loop before actual rendering. JC
2672 for (pool_set_t::iterator iter = mPools.begin(); iter != mPools.end(); ++iter)
2673 {
2674 LLDrawPool *poolp = *iter;
2675 if (poolp->isFacePool() && hasRenderType(poolp->getType()))
2676 {
2677 poolp->prerender();
2678 }
2679 }
2668 for (pool_set_t::iterator iter = mPools.begin(); iter != mPools.end(); ++iter) 2680 for (pool_set_t::iterator iter = mPools.begin(); iter != mPools.end(); ++iter)
2669 { 2681 {
2670 LLDrawPool *poolp = *iter; 2682 LLDrawPool *poolp = *iter;
diff --git a/linden/indra/newview/skins/default/xui/de/panel_media_controls.xml b/linden/indra/newview/skins/default/xui/de/panel_media_controls.xml
index eb47fa2..d15d186 100644
--- a/linden/indra/newview/skins/default/xui/de/panel_media_controls.xml
+++ b/linden/indra/newview/skins/default/xui/de/panel_media_controls.xml
@@ -2,7 +2,6 @@
2<panel name="media_controls"> 2<panel name="media_controls">
3 <icon name="music_icon" tool_tip="Streaming-Musik abspielen/unterbrechen" /> 3 <icon name="music_icon" tool_tip="Streaming-Musik abspielen/unterbrechen" />
4 <button name="music_play" tool_tip="Streaming-Musik spielen/unterbrechen" /> 4 <button name="music_play" tool_tip="Streaming-Musik spielen/unterbrechen" />
5 <button name="music_pause" tool_tip="Streaming-Musik unterbrechen" />
6 <button name="music_stop" tool_tip="Streaming-Musik beenden" /> 5 <button name="music_stop" tool_tip="Streaming-Musik beenden" />
7 <icon name="media_icon" tool_tip="Streaming-Musik abspielen/unterbrechen" /> 6 <icon name="media_icon" tool_tip="Streaming-Musik abspielen/unterbrechen" />
8 <button name="media_play" tool_tip="Streaming-Musik abspielen/unterbrechen" /> 7 <button name="media_play" tool_tip="Streaming-Musik abspielen/unterbrechen" />
diff --git a/linden/indra/newview/skins/default/xui/en-us/panel_preferences_im.xml b/linden/indra/newview/skins/default/xui/en-us/panel_preferences_im.xml
index d3174af..8020e9a 100644
--- a/linden/indra/newview/skins/default/xui/en-us/panel_preferences_im.xml
+++ b/linden/indra/newview/skins/default/xui/en-us/panel_preferences_im.xml
@@ -28,7 +28,7 @@
28 width="350" /> 28 width="350" />
29 <check_box bottom="-65" control_name="IMInChatHistory" enabled="true" follows="left|top" 29 <check_box bottom="-65" control_name="IMInChatHistory" enabled="true" follows="left|top"
30 font="SansSerifSmall" height="16" initial_value="false" 30 font="SansSerifSmall" height="16" initial_value="false"
31 label="Include IM in Local Chat" left="148" mouse_opaque="true" 31 label="Show IMs with Chat" left="148" mouse_opaque="true"
32 name="include_im_in_chat_history" radio_style="false" width="237" /> 32 name="include_im_in_chat_history" radio_style="false" width="237" />
33 <check_box bottom="-85" control_name="IMShowTimestamps" enabled="true" follows="left|top" 33 <check_box bottom="-85" control_name="IMShowTimestamps" enabled="true" follows="left|top"
34 font="SansSerifSmall" height="16" initial_value="false" 34 font="SansSerifSmall" height="16" initial_value="false"
diff --git a/linden/indra/newview/skins/default/xui/fr/panel_media_controls.xml b/linden/indra/newview/skins/default/xui/fr/panel_media_controls.xml
index ff498ba..7b601b4 100644
--- a/linden/indra/newview/skins/default/xui/fr/panel_media_controls.xml
+++ b/linden/indra/newview/skins/default/xui/fr/panel_media_controls.xml
@@ -2,7 +2,6 @@
2<panel name="media_controls"> 2<panel name="media_controls">
3 <icon name="music_icon" tool_tip="Jouer/pauser le flux musical" /> 3 <icon name="music_icon" tool_tip="Jouer/pauser le flux musical" />
4 <button name="music_play" tool_tip="Jouer/pauser le flux musical" /> 4 <button name="music_play" tool_tip="Jouer/pauser le flux musical" />
5 <button name="music_pause" tool_tip="Pauser le flux musical" />
6 <button name="music_stop" tool_tip="Arrêter le flux musical" /> 5 <button name="music_stop" tool_tip="Arrêter le flux musical" />
7 <icon name="media_icon" tool_tip="Jouer/pauser le flux musical" /> 6 <icon name="media_icon" tool_tip="Jouer/pauser le flux musical" />
8 <button name="media_play" tool_tip="Jouer/pauser le flux musical" /> 7 <button name="media_play" tool_tip="Jouer/pauser le flux musical" />
diff --git a/linden/indra/newview/skins/default/xui/ja/panel_media_controls.xml b/linden/indra/newview/skins/default/xui/ja/panel_media_controls.xml
index 7c87dc3..89791a5 100644
--- a/linden/indra/newview/skins/default/xui/ja/panel_media_controls.xml
+++ b/linden/indra/newview/skins/default/xui/ja/panel_media_controls.xml
@@ -4,8 +4,6 @@
4 tool_tip="ストリーミング・ミュージックの再生/一時停止" /> 4 tool_tip="ストリーミング・ミュージックの再生/一時停止" />
5 <button name="music_play" 5 <button name="music_play"
6 tool_tip="ストリーミング・ミュージックの再生/一時停止" /> 6 tool_tip="ストリーミング・ミュージックの再生/一時停止" />
7 <button name="music_pause"
8 tool_tip="ストリーミング・ミュージックの一時停止" />
9 <button name="music_stop" 7 <button name="music_stop"
10 tool_tip="ストリーミング・ミュージックの停止" /> 8 tool_tip="ストリーミング・ミュージックの停止" />
11 <icon name="media_icon" 9 <icon name="media_icon"
diff --git a/linden/indra/newview/skins/default/xui/ko/panel_media_controls.xml b/linden/indra/newview/skins/default/xui/ko/panel_media_controls.xml
index 3b3aa0e..fa5a010 100644
--- a/linden/indra/newview/skins/default/xui/ko/panel_media_controls.xml
+++ b/linden/indra/newview/skins/default/xui/ko/panel_media_controls.xml
@@ -2,7 +2,6 @@
2<panel name="media_controls"> 2<panel name="media_controls">
3 <icon name="music_icon" tool_tip="스트리밍 음악 재생/일시 정지" /> 3 <icon name="music_icon" tool_tip="스트리밍 음악 재생/일시 정지" />
4 <button name="music_play" tool_tip="스트리밍 음악 재생/일시 정지" /> 4 <button name="music_play" tool_tip="스트리밍 음악 재생/일시 정지" />
5 <button name="music_pause" tool_tip="스트리밍 음악 일시 정지" />
6 <button name="music_stop" tool_tip="스트리밍 음악 중지" /> 5 <button name="music_stop" tool_tip="스트리밍 음악 중지" />
7 <icon name="media_icon" tool_tip="스트리밍 음악 재생/일시 정지" /> 6 <icon name="media_icon" tool_tip="스트리밍 음악 재생/일시 정지" />
8 <button name="media_play" tool_tip="스트리밍 음악 재생/일시 정지" /> 7 <button name="media_play" tool_tip="스트리밍 음악 재생/일시 정지" />
diff --git a/linden/indra/newview/skins/silver/xui/en-us/panel_media_controls.xml b/linden/indra/newview/skins/silver/xui/en-us/panel_media_controls.xml
index d62fe09..466db79 100644
--- a/linden/indra/newview/skins/silver/xui/en-us/panel_media_controls.xml
+++ b/linden/indra/newview/skins/silver/xui/en-us/panel_media_controls.xml
@@ -4,7 +4,7 @@
4 name="media_controls" width="220"> 4 name="media_controls" width="220">
5 <icon bottom="-18" enabled="true" follows="left|top" height="14" 5 <icon bottom="-18" enabled="true" follows="left|top" height="14"
6 image_name="icn_label_music.tga" label="" left_delta="6" mouse_opaque="true" 6 image_name="icn_label_music.tga" label="" left_delta="6" mouse_opaque="true"
7 name="music_icon" scale_image="true" tool_tip="Play/pause streaming music" 7 name="music_icon" scale_image="true" tool_tip="Play streaming music"
8 width="14" /> 8 width="14" />
9 <button bottom="-21" enabled="true" follows="left|top" height="20" 9 <button bottom="-21" enabled="true" follows="left|top" height="20"
10 image_overlay="icn_play.tga" 10 image_overlay="icn_play.tga"
@@ -13,15 +13,7 @@
13 image_disabled="flyout_btn_left_disabled.tga" 13 image_disabled="flyout_btn_left_disabled.tga"
14 label="" left_delta="16" 14 label="" left_delta="16"
15 mouse_opaque="true" name="music_play" scale_image="true" toggle="false" 15 mouse_opaque="true" name="music_play" scale_image="true" toggle="false"
16 tool_tip="Play/pause streaming music" width="20" /> 16 tool_tip="Play streaming music" width="20" />
17 <button bottom="-21" enabled="false" follows="left|top" height="20"
18 image_overlay="icn_pause.tga"
19 image_unselected="flyout_btn_left.tga"
20 image_selected="flyout_btn_left_selected.tga"
21 image_disabled="flyout_btn_left_disabled.tga"
22 label="" left_delta="0"
23 mouse_opaque="true" name="music_pause" scale_image="true" toggle="false"
24 tool_tip="Pause streaming music" width="20" />
25 <button bottom="-21" enabled="true" follows="left|top" height="20" 17 <button bottom="-21" enabled="true" follows="left|top" height="20"
26 image_overlay="icn_stop.tga" 18 image_overlay="icn_stop.tga"
27 image_unselected="flyout_btn_right.tga" 19 image_unselected="flyout_btn_right.tga"
@@ -33,7 +25,7 @@
33 <icon bottom="-18" enabled="true" follows="left|top" height="14" 25 <icon bottom="-18" enabled="true" follows="left|top" height="14"
34 image_name="icn_label_media.tga" 26 image_name="icn_label_media.tga"
35 label="" left_delta="22" mouse_opaque="true" 27 label="" left_delta="22" mouse_opaque="true"
36 name="media_icon" scale_image="true" tool_tip="Play/pause streaming music" 28 name="media_icon" scale_image="true" tool_tip="Play/pause streaming media"
37 width="17" /> 29 width="17" />
38 <button bottom="-21" enabled="true" follows="left|top" height="20" 30 <button bottom="-21" enabled="true" follows="left|top" height="20"
39 image_overlay="icn_play.tga" 31 image_overlay="icn_play.tga"