aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/OpenSim/Region/ScriptEngine/Shared/Api/Implementation
diff options
context:
space:
mode:
Diffstat (limited to 'OpenSim/Region/ScriptEngine/Shared/Api/Implementation')
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/AsyncCommandManager.cs20
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/CM_Api.cs118
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs3174
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs116
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs9
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/Timer.cs32
6 files changed, 2465 insertions, 1004 deletions
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/AsyncCommandManager.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/AsyncCommandManager.cs
index 993d10f..3cbdde5 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/AsyncCommandManager.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/AsyncCommandManager.cs
@@ -301,6 +301,26 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
301 return null; 301 return null;
302 } 302 }
303 303
304 public static void StateChange(IScriptEngine engine, uint localID, UUID itemID)
305 {
306 // Remove a specific script
307
308 // Remove dataserver events
309 m_Dataserver[engine].RemoveEvents(localID, itemID);
310
311 IWorldComm comms = engine.World.RequestModuleInterface<IWorldComm>();
312 if (comms != null)
313 comms.DeleteListener(itemID);
314
315 IXMLRPC xmlrpc = engine.World.RequestModuleInterface<IXMLRPC>();
316 xmlrpc.DeleteChannels(itemID);
317 xmlrpc.CancelSRDRequests(itemID);
318
319 // Remove Sensors
320 m_SensorRepeat[engine].UnSetSenseRepeaterEvents(localID, itemID);
321
322 }
323
304 public static Object[] GetSerializationData(IScriptEngine engine, UUID itemID) 324 public static Object[] GetSerializationData(IScriptEngine engine, UUID itemID)
305 { 325 {
306 List<Object> data = new List<Object>(); 326 List<Object> data = new List<Object>();
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/CM_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/CM_Api.cs
new file mode 100644
index 0000000..489c1c6
--- /dev/null
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/CM_Api.cs
@@ -0,0 +1,118 @@
1/*
2 * Copyright (c) Contributors, http://opensimulator.org/
3 * See CONTRIBUTORS.TXT for a full list of copyright holders.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 * * Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * * Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 * * Neither the name of the OpenSimulator Project nor the
13 * names of its contributors may be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
17 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19 * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
20 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28using System;
29using System.Reflection;
30using System.Collections;
31using System.Collections.Generic;
32using System.Runtime.Remoting.Lifetime;
33using OpenMetaverse;
34using Nini.Config;
35using OpenSim;
36using OpenSim.Framework;
37using OpenSim.Region.CoreModules.World.LightShare;
38using OpenSim.Region.Framework.Interfaces;
39using OpenSim.Region.Framework.Scenes;
40using OpenSim.Region.ScriptEngine.Shared;
41using OpenSim.Region.ScriptEngine.Shared.Api.Plugins;
42using OpenSim.Region.ScriptEngine.Shared.ScriptBase;
43using OpenSim.Region.ScriptEngine.Interfaces;
44using OpenSim.Region.ScriptEngine.Shared.Api.Interfaces;
45using OpenSim.Services.Interfaces;
46
47using LSL_Float = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLFloat;
48using LSL_Integer = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLInteger;
49using LSL_Key = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
50using LSL_List = OpenSim.Region.ScriptEngine.Shared.LSL_Types.list;
51using LSL_Rotation = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Quaternion;
52using LSL_String = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
53using LSL_Vector = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Vector3;
54
55namespace OpenSim.Region.ScriptEngine.Shared.Api
56{
57 [Serializable]
58 public class CM_Api : MarshalByRefObject, ICM_Api, IScriptApi
59 {
60 internal IScriptEngine m_ScriptEngine;
61 internal SceneObjectPart m_host;
62 internal uint m_localID;
63 internal UUID m_itemID;
64 internal bool m_CMFunctionsEnabled = false;
65
66 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, uint localID, UUID itemID)
67 {
68 m_ScriptEngine = ScriptEngine;
69 m_host = host;
70 m_localID = localID;
71 m_itemID = itemID;
72
73 if (m_ScriptEngine.Config.GetBoolean("AllowCareminsterFunctions", false))
74 m_CMFunctionsEnabled = true;
75 }
76
77 public override Object InitializeLifetimeService()
78 {
79 ILease lease = (ILease)base.InitializeLifetimeService();
80
81 if (lease.CurrentState == LeaseState.Initial)
82 {
83 lease.InitialLeaseTime = TimeSpan.FromMinutes(0);
84 // lease.RenewOnCallTime = TimeSpan.FromSeconds(10.0);
85 // lease.SponsorshipTimeout = TimeSpan.FromMinutes(1.0);
86 }
87 return lease;
88 }
89
90 public Scene World
91 {
92 get { return m_ScriptEngine.World; }
93 }
94
95 public string cmDetectedCountry(int number)
96 {
97 m_host.AddScriptLPS(1);
98 DetectParams detectedParams = m_ScriptEngine.GetDetectParams(m_itemID, number);
99 if (detectedParams == null)
100 return String.Empty;
101 return detectedParams.Country;
102 }
103
104 public string cmGetAgentCountry(LSL_Key key)
105 {
106 if (!World.Permissions.IsGod(m_host.OwnerID))
107 return String.Empty;
108
109 UUID uuid;
110
111 if (!UUID.TryParse(key, out uuid))
112 return String.Empty;
113
114 UserAccount account = World.UserAccountService.GetUserAccount(World.RegionInfo.ScopeID, uuid);
115 return account.UserCountry;
116 }
117 }
118}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs
index 0003515..6d4072c 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs
@@ -28,10 +28,12 @@
28using System; 28using System;
29using System.Collections; 29using System.Collections;
30using System.Collections.Generic; 30using System.Collections.Generic;
31using System.Diagnostics; //for [DebuggerNonUserCode]
31using System.Runtime.Remoting.Lifetime; 32using System.Runtime.Remoting.Lifetime;
32using System.Text; 33using System.Text;
33using System.Threading; 34using System.Threading;
34using System.Text.RegularExpressions; 35using System.Text.RegularExpressions;
36using System.Timers;
35using Nini.Config; 37using Nini.Config;
36using log4net; 38using log4net;
37using OpenMetaverse; 39using OpenMetaverse;
@@ -44,6 +46,7 @@ using OpenSim.Region.CoreModules.World.Land;
44using OpenSim.Region.CoreModules.World.Terrain; 46using OpenSim.Region.CoreModules.World.Terrain;
45using OpenSim.Region.Framework.Interfaces; 47using OpenSim.Region.Framework.Interfaces;
46using OpenSim.Region.Framework.Scenes; 48using OpenSim.Region.Framework.Scenes;
49using OpenSim.Region.Framework.Scenes.Serialization;
47using OpenSim.Region.Framework.Scenes.Animation; 50using OpenSim.Region.Framework.Scenes.Animation;
48using OpenSim.Region.Physics.Manager; 51using OpenSim.Region.Physics.Manager;
49using OpenSim.Region.ScriptEngine.Shared; 52using OpenSim.Region.ScriptEngine.Shared;
@@ -65,6 +68,7 @@ using LSL_Rotation = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Quaternion;
65using LSL_String = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString; 68using LSL_String = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
66using LSL_Vector = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Vector3; 69using LSL_Vector = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Vector3;
67using System.Reflection; 70using System.Reflection;
71using Timer = System.Timers.Timer;
68 72
69namespace OpenSim.Region.ScriptEngine.Shared.Api 73namespace OpenSim.Region.ScriptEngine.Shared.Api
70{ 74{
@@ -99,16 +103,26 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
99 protected int m_notecardLineReadCharsMax = 255; 103 protected int m_notecardLineReadCharsMax = 255;
100 protected int m_scriptConsoleChannel = 0; 104 protected int m_scriptConsoleChannel = 0;
101 protected bool m_scriptConsoleChannelEnabled = false; 105 protected bool m_scriptConsoleChannelEnabled = false;
106 protected bool m_debuggerSafe = false;
102 protected IUrlModule m_UrlModule = null; 107 protected IUrlModule m_UrlModule = null;
103 protected Dictionary<UUID, UserInfoCacheEntry> m_userInfoCache = 108 protected Dictionary<UUID, UserInfoCacheEntry> m_userInfoCache =
104 new Dictionary<UUID, UserInfoCacheEntry>(); 109 new Dictionary<UUID, UserInfoCacheEntry>();
105 110
111 protected Timer m_ShoutSayTimer;
112 protected int m_SayShoutCount = 0;
113
106 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, uint localID, UUID itemID) 114 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, uint localID, UUID itemID)
107 { 115 {
116 m_ShoutSayTimer = new Timer(1000);
117 m_ShoutSayTimer.Elapsed += SayShoutTimerElapsed;
118 m_ShoutSayTimer.AutoReset = true;
119 m_ShoutSayTimer.Start();
120
108 m_ScriptEngine = ScriptEngine; 121 m_ScriptEngine = ScriptEngine;
109 m_host = host; 122 m_host = host;
110 m_localID = localID; 123 m_localID = localID;
111 m_itemID = itemID; 124 m_itemID = itemID;
125 m_debuggerSafe = m_ScriptEngine.Config.GetBoolean("DebuggerSafe", false);
112 126
113 m_ScriptDelayFactor = 127 m_ScriptDelayFactor =
114 m_ScriptEngine.Config.GetFloat("ScriptDelayFactor", 1.0f); 128 m_ScriptEngine.Config.GetFloat("ScriptDelayFactor", 1.0f);
@@ -156,6 +170,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
156 get { return m_ScriptEngine.World; } 170 get { return m_ScriptEngine.World; }
157 } 171 }
158 172
173 [DebuggerNonUserCode]
159 public void state(string newState) 174 public void state(string newState)
160 { 175 {
161 m_ScriptEngine.SetState(m_itemID, newState); 176 m_ScriptEngine.SetState(m_itemID, newState);
@@ -165,6 +180,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
165 /// Reset the named script. The script must be present 180 /// Reset the named script. The script must be present
166 /// in the same prim. 181 /// in the same prim.
167 /// </summary> 182 /// </summary>
183 [DebuggerNonUserCode]
168 public void llResetScript() 184 public void llResetScript()
169 { 185 {
170 m_host.AddScriptLPS(1); 186 m_host.AddScriptLPS(1);
@@ -221,9 +237,62 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
221 } 237 }
222 } 238 }
223 239
240 public List<ScenePresence> GetLinkAvatars(int linkType)
241 {
242 List<ScenePresence> ret = new List<ScenePresence>();
243 if (m_host == null || m_host.ParentGroup == null || m_host.ParentGroup.IsDeleted)
244 return ret;
245
246 List<ScenePresence> avs = m_host.ParentGroup.GetLinkedAvatars();
247
248 switch (linkType)
249 {
250 case ScriptBaseClass.LINK_SET:
251 return avs;
252
253 case ScriptBaseClass.LINK_ROOT:
254 return ret;
255
256 case ScriptBaseClass.LINK_ALL_OTHERS:
257 return avs;
258
259 case ScriptBaseClass.LINK_ALL_CHILDREN:
260 return avs;
261
262 case ScriptBaseClass.LINK_THIS:
263 return ret;
264
265 default:
266 if (linkType < 0)
267 return ret;
268
269 int partCount = m_host.ParentGroup.GetPartCount();
270
271 if (linkType <= partCount)
272 {
273 return ret;
274 }
275 else
276 {
277 linkType = linkType - partCount;
278 if (linkType > avs.Count)
279 {
280 return ret;
281 }
282 else
283 {
284 ret.Add(avs[linkType-1]);
285 return ret;
286 }
287 }
288 }
289 }
290
224 public List<SceneObjectPart> GetLinkParts(int linkType) 291 public List<SceneObjectPart> GetLinkParts(int linkType)
225 { 292 {
226 List<SceneObjectPart> ret = new List<SceneObjectPart>(); 293 List<SceneObjectPart> ret = new List<SceneObjectPart>();
294 if (m_host == null || m_host.ParentGroup == null || m_host.ParentGroup.IsDeleted)
295 return ret;
227 ret.Add(m_host); 296 ret.Add(m_host);
228 297
229 switch (linkType) 298 switch (linkType)
@@ -270,40 +339,48 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
270 protected UUID InventorySelf() 339 protected UUID InventorySelf()
271 { 340 {
272 UUID invItemID = new UUID(); 341 UUID invItemID = new UUID();
273 342 bool unlock = false;
274 lock (m_host.TaskInventory) 343 if (!m_host.TaskInventory.IsReadLockedByMe())
275 { 344 {
276 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 345 m_host.TaskInventory.LockItemsForRead(true);
346 unlock = true;
347 }
348 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
349 {
350 if (inv.Value.Type == 10 && inv.Value.ItemID == m_itemID)
277 { 351 {
278 if (inv.Value.Type == 10 && inv.Value.ItemID == m_itemID) 352 invItemID = inv.Key;
279 { 353 break;
280 invItemID = inv.Key;
281 break;
282 }
283 } 354 }
284 } 355 }
285 356 if (unlock)
357 {
358 m_host.TaskInventory.LockItemsForRead(false);
359 }
286 return invItemID; 360 return invItemID;
287 } 361 }
288 362
289 protected UUID InventoryKey(string name, int type) 363 protected UUID InventoryKey(string name, int type)
290 { 364 {
291 m_host.AddScriptLPS(1); 365 m_host.AddScriptLPS(1);
292 366 m_host.TaskInventory.LockItemsForRead(true);
293 lock (m_host.TaskInventory) 367
368 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
294 { 369 {
295 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 370 if (inv.Value.Name == name)
296 { 371 {
297 if (inv.Value.Name == name) 372 m_host.TaskInventory.LockItemsForRead(false);
373
374 if (inv.Value.Type != type)
298 { 375 {
299 if (inv.Value.Type != type) 376 return UUID.Zero;
300 return UUID.Zero;
301
302 return inv.Value.AssetID;
303 } 377 }
378
379 return inv.Value.AssetID;
304 } 380 }
305 } 381 }
306 382
383 m_host.TaskInventory.LockItemsForRead(false);
307 return UUID.Zero; 384 return UUID.Zero;
308 } 385 }
309 386
@@ -311,17 +388,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
311 { 388 {
312 m_host.AddScriptLPS(1); 389 m_host.AddScriptLPS(1);
313 390
314 lock (m_host.TaskInventory) 391
392 m_host.TaskInventory.LockItemsForRead(true);
393
394 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
315 { 395 {
316 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 396 if (inv.Value.Name == name)
317 { 397 {
318 if (inv.Value.Name == name) 398 m_host.TaskInventory.LockItemsForRead(false);
319 { 399 return inv.Value.AssetID;
320 return inv.Value.AssetID;
321 }
322 } 400 }
323 } 401 }
324 402
403 m_host.TaskInventory.LockItemsForRead(false);
404
405
325 return UUID.Zero; 406 return UUID.Zero;
326 } 407 }
327 408
@@ -463,31 +544,48 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
463 544
464 //Now we start getting into quaternions which means sin/cos, matrices and vectors. ckrinke 545 //Now we start getting into quaternions which means sin/cos, matrices and vectors. ckrinke
465 546
466 /// <summary> 547 // Utility function for llRot2Euler
467 /// Convert an LSL rotation to a Euler vector. 548
468 /// </summary> 549 // normalize an angle between -PI and PI (-180 to +180 degrees)
469 /// <remarks> 550 protected double NormalizeAngle(double angle)
470 /// Using algorithm based off http://www.euclideanspace.com/maths/geometry/rotations/conversions/quaternionToEuler/quat_2_euler_paper_ver2-1.pdf
471 /// to avoid issues with singularity and rounding with Y rotation of +/- PI/2
472 /// </remarks>
473 /// <param name="r"></param>
474 /// <returns></returns>
475 public LSL_Vector llRot2Euler(LSL_Rotation r)
476 { 551 {
477 m_host.AddScriptLPS(1); 552 if (angle > -Math.PI && angle < Math.PI)
553 return angle;
478 554
479 LSL_Vector v = new LSL_Vector(0.0, 0.0, 1.0) * r; // Z axis unit vector unaffected by Z rotation component of r. 555 int numPis = (int)(Math.PI / angle);
480 double m = LSL_Vector.Mag(v); // Just in case v isn't normalized, need magnitude for Asin() operation later. 556 double remainder = angle - Math.PI * numPis;
481 if (m == 0.0) return new LSL_Vector(); 557 if (numPis % 2 == 1)
482 double x = Math.Atan2(-v.y, v.z); 558 return Math.PI - angle;
483 double sin = v.x / m; 559 return remainder;
484 if (sin < -0.999999 || sin > 0.999999) x = 0.0; // Force X rotation to 0 at the singularities. 560 }
485 double y = Math.Asin(sin);
486 // Rotate X axis unit vector by r and unwind the X and Y rotations leaving only the Z rotation
487 v = new LSL_Vector(1.0, 0.0, 0.0) * ((r * new LSL_Rotation(Math.Sin(-x / 2.0), 0.0, 0.0, Math.Cos(-x / 2.0))) * new LSL_Rotation(0.0, Math.Sin(-y / 2.0), 0.0, Math.Cos(-y / 2.0)));
488 double z = Math.Atan2(v.y, v.x);
489 561
490 return new LSL_Vector(x, y, z); 562 public LSL_Vector llRot2Euler(LSL_Rotation q1)
563 {
564 m_host.AddScriptLPS(1);
565 LSL_Vector eul = new LSL_Vector();
566
567 double sqw = q1.s*q1.s;
568 double sqx = q1.x*q1.x;
569 double sqy = q1.z*q1.z;
570 double sqz = q1.y*q1.y;
571 double unit = sqx + sqy + sqz + sqw; // if normalised is one, otherwise is correction factor
572 double test = q1.x*q1.z + q1.y*q1.s;
573 if (test > 0.4999*unit) { // singularity at north pole
574 eul.z = 2 * Math.Atan2(q1.x,q1.s);
575 eul.y = Math.PI/2;
576 eul.x = 0;
577 return eul;
578 }
579 if (test < -0.4999*unit) { // singularity at south pole
580 eul.z = -2 * Math.Atan2(q1.x,q1.s);
581 eul.y = -Math.PI/2;
582 eul.x = 0;
583 return eul;
584 }
585 eul.z = Math.Atan2(2*q1.z*q1.s-2*q1.x*q1.y , sqx - sqy - sqz + sqw);
586 eul.y = Math.Asin(2*test/unit);
587 eul.x = Math.Atan2(2*q1.x*q1.s-2*q1.z*q1.y , -sqx + sqy - sqz + sqw);
588 return eul;
491 } 589 }
492 590
493 /* From wiki: 591 /* From wiki:
@@ -689,77 +787,76 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
689 { 787 {
690 //A and B should both be normalized 788 //A and B should both be normalized
691 m_host.AddScriptLPS(1); 789 m_host.AddScriptLPS(1);
692 LSL_Rotation rotBetween; 790 /* This method is more accurate than the SL one, and thus causes problems
693 // Check for zero vectors. If either is zero, return zero rotation. Otherwise, 791 for scripts that deal with the SL inaccuracy around 180-degrees -.- .._.
694 // continue calculation. 792
695 if (a == new LSL_Vector(0.0f, 0.0f, 0.0f) || b == new LSL_Vector(0.0f, 0.0f, 0.0f)) 793 double dotProduct = LSL_Vector.Dot(a, b);
794 LSL_Vector crossProduct = LSL_Vector.Cross(a, b);
795 double magProduct = LSL_Vector.Mag(a) * LSL_Vector.Mag(b);
796 double angle = Math.Acos(dotProduct / magProduct);
797 LSL_Vector axis = LSL_Vector.Norm(crossProduct);
798 double s = Math.Sin(angle / 2);
799
800 double x = axis.x * s;
801 double y = axis.y * s;
802 double z = axis.z * s;
803 double w = Math.Cos(angle / 2);
804
805 if (Double.IsNaN(x) || Double.IsNaN(y) || Double.IsNaN(z) || Double.IsNaN(w))
806 return new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
807
808 return new LSL_Rotation((float)x, (float)y, (float)z, (float)w);
809 */
810
811 // This method mimics the 180 errors found in SL
812 // See www.euclideanspace.com... angleBetween
813 LSL_Vector vec_a = a;
814 LSL_Vector vec_b = b;
815
816 // Eliminate zero length
817 LSL_Float vec_a_mag = LSL_Vector.Mag(vec_a);
818 LSL_Float vec_b_mag = LSL_Vector.Mag(vec_b);
819 if (vec_a_mag < 0.00001 ||
820 vec_b_mag < 0.00001)
696 { 821 {
697 rotBetween = new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f); 822 return new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
698 } 823 }
699 else 824
825 // Normalize
826 vec_a = llVecNorm(vec_a);
827 vec_b = llVecNorm(vec_b);
828
829 // Calculate axis and rotation angle
830 LSL_Vector axis = vec_a % vec_b;
831 LSL_Float cos_theta = vec_a * vec_b;
832
833 // Check if parallel
834 if (cos_theta > 0.99999)
700 { 835 {
701 a = LSL_Vector.Norm(a); 836 return new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
702 b = LSL_Vector.Norm(b); 837 }
703 double dotProduct = LSL_Vector.Dot(a, b); 838
704 // There are two degenerate cases possible. These are for vectors 180 or 839 // Check if anti-parallel
705 // 0 degrees apart. These have to be detected and handled individually. 840 else if (cos_theta < -0.99999)
706 // 841 {
707 // Check for vectors 180 degrees apart. 842 LSL_Vector orthog_axis = new LSL_Vector(1.0, 0.0, 0.0) - (vec_a.x / (vec_a * vec_a) * vec_a);
708 // A dot product of -1 would mean the angle between vectors is 180 degrees. 843 if (LSL_Vector.Mag(orthog_axis) < 0.000001) orthog_axis = new LSL_Vector(0.0, 0.0, 1.0);
709 if (dotProduct < -0.9999999f) 844 return new LSL_Rotation((float)orthog_axis.x, (float)orthog_axis.y, (float)orthog_axis.z, 0.0);
710 { 845 }
711 // First assume X axis is orthogonal to the vectors. 846 else // other rotation
712 LSL_Vector orthoVector = new LSL_Vector(1.0f, 0.0f, 0.0f); 847 {
713 orthoVector = orthoVector - a * (a.x / LSL_Vector.Dot(a, a)); 848 LSL_Float theta = (LSL_Float)Math.Acos(cos_theta) * 0.5f;
714 // Check for near zero vector. A very small non-zero number here will create 849 axis = llVecNorm(axis);
715 // a rotation in an undesired direction. 850 double x, y, z, s, t;
716 if (LSL_Vector.Mag(orthoVector) > 0.0001) 851 s = Math.Cos(theta);
717 { 852 t = Math.Sin(theta);
718 rotBetween = new LSL_Rotation(orthoVector.x, orthoVector.y, orthoVector.z, 0.0f); 853 x = axis.x * t;
719 } 854 y = axis.y * t;
720 // If the magnitude of the vector was near zero, then assume the X axis is not 855 z = axis.z * t;
721 // orthogonal and use the Z axis instead. 856 return new LSL_Rotation(x,y,z,s);
722 else
723 {
724 // Set 180 z rotation.
725 rotBetween = new LSL_Rotation(0.0f, 0.0f, 1.0f, 0.0f);
726 }
727 }
728 // Check for parallel vectors.
729 // A dot product of 1 would mean the angle between vectors is 0 degrees.
730 else if (dotProduct > 0.9999999f)
731 {
732 // Set zero rotation.
733 rotBetween = new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
734 }
735 else
736 {
737 // All special checks have been performed so get the axis of rotation.
738 LSL_Vector crossProduct = LSL_Vector.Cross(a, b);
739 // Quarternion s value is the length of the unit vector + dot product.
740 double qs = 1.0 + dotProduct;
741 rotBetween = new LSL_Rotation(crossProduct.x, crossProduct.y, crossProduct.z, qs);
742 // Normalize the rotation.
743 double mag = LSL_Rotation.Mag(rotBetween);
744 // We shouldn't have to worry about a divide by zero here. The qs value will be
745 // non-zero because we already know if we're here, then the dotProduct is not -1 so
746 // qs will not be zero. Also, we've already handled the input vectors being zero so the
747 // crossProduct vector should also not be zero.
748 rotBetween.x = rotBetween.x / mag;
749 rotBetween.y = rotBetween.y / mag;
750 rotBetween.z = rotBetween.z / mag;
751 rotBetween.s = rotBetween.s / mag;
752 // Check for undefined values and set zero rotation if any found. This code might not actually be required
753 // any longer since zero vectors are checked for at the top.
754 if (Double.IsNaN(rotBetween.x) || Double.IsNaN(rotBetween.y) || Double.IsNaN(rotBetween.z) || Double.IsNaN(rotBetween.s))
755 {
756 rotBetween = new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
757 }
758 }
759 } 857 }
760 return rotBetween;
761 } 858 }
762 859
763 public void llWhisper(int channelID, string text) 860 public void llWhisper(int channelID, string text)
764 { 861 {
765 m_host.AddScriptLPS(1); 862 m_host.AddScriptLPS(1);
@@ -779,6 +876,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
779 { 876 {
780 m_host.AddScriptLPS(1); 877 m_host.AddScriptLPS(1);
781 878
879 if (channelID == 0)
880 m_SayShoutCount++;
881
882 if (m_SayShoutCount >= 11)
883 ScriptSleep(2000);
884
782 if (m_scriptConsoleChannelEnabled && (channelID == m_scriptConsoleChannel)) 885 if (m_scriptConsoleChannelEnabled && (channelID == m_scriptConsoleChannel))
783 { 886 {
784 Console.WriteLine(text); 887 Console.WriteLine(text);
@@ -801,6 +904,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
801 { 904 {
802 m_host.AddScriptLPS(1); 905 m_host.AddScriptLPS(1);
803 906
907 if (channelID == 0)
908 m_SayShoutCount++;
909
910 if (m_SayShoutCount >= 11)
911 ScriptSleep(2000);
912
804 if (text.Length > 1023) 913 if (text.Length > 1023)
805 text = text.Substring(0, 1023); 914 text = text.Substring(0, 1023);
806 915
@@ -1101,10 +1210,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1101 return detectedParams.TouchUV; 1210 return detectedParams.TouchUV;
1102 } 1211 }
1103 1212
1213 [DebuggerNonUserCode]
1104 public virtual void llDie() 1214 public virtual void llDie()
1105 { 1215 {
1106 m_host.AddScriptLPS(1); 1216 m_host.AddScriptLPS(1);
1107 throw new SelfDeleteException(); 1217 if (!m_host.ParentGroup.IsAttachment) throw new SelfDeleteException();
1108 } 1218 }
1109 1219
1110 public LSL_Float llGround(LSL_Vector offset) 1220 public LSL_Float llGround(LSL_Vector offset)
@@ -1177,6 +1287,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1177 1287
1178 public void llSetStatus(int status, int value) 1288 public void llSetStatus(int status, int value)
1179 { 1289 {
1290 if (m_host == null || m_host.ParentGroup == null || m_host.ParentGroup.IsDeleted)
1291 return;
1180 m_host.AddScriptLPS(1); 1292 m_host.AddScriptLPS(1);
1181 1293
1182 int statusrotationaxis = 0; 1294 int statusrotationaxis = 0;
@@ -1406,6 +1518,48 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1406 { 1518 {
1407 m_host.AddScriptLPS(1); 1519 m_host.AddScriptLPS(1);
1408 1520
1521 SetColor(m_host, color, face);
1522 }
1523
1524 protected void SetColor(SceneObjectPart part, LSL_Vector color, int face)
1525 {
1526 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1527 return;
1528
1529 Primitive.TextureEntry tex = part.Shape.Textures;
1530 Color4 texcolor;
1531 if (face >= 0 && face < GetNumberOfSides(part))
1532 {
1533 texcolor = tex.CreateFace((uint)face).RGBA;
1534 texcolor.R = Util.Clip((float)color.x, 0.0f, 1.0f);
1535 texcolor.G = Util.Clip((float)color.y, 0.0f, 1.0f);
1536 texcolor.B = Util.Clip((float)color.z, 0.0f, 1.0f);
1537 tex.FaceTextures[face].RGBA = texcolor;
1538 part.UpdateTextureEntry(tex.GetBytes());
1539 return;
1540 }
1541 else if (face == ScriptBaseClass.ALL_SIDES)
1542 {
1543 for (uint i = 0; i < GetNumberOfSides(part); i++)
1544 {
1545 if (tex.FaceTextures[i] != null)
1546 {
1547 texcolor = tex.FaceTextures[i].RGBA;
1548 texcolor.R = Util.Clip((float)color.x, 0.0f, 1.0f);
1549 texcolor.G = Util.Clip((float)color.y, 0.0f, 1.0f);
1550 texcolor.B = Util.Clip((float)color.z, 0.0f, 1.0f);
1551 tex.FaceTextures[i].RGBA = texcolor;
1552 }
1553 texcolor = tex.DefaultTexture.RGBA;
1554 texcolor.R = Util.Clip((float)color.x, 0.0f, 1.0f);
1555 texcolor.G = Util.Clip((float)color.y, 0.0f, 1.0f);
1556 texcolor.B = Util.Clip((float)color.z, 0.0f, 1.0f);
1557 tex.DefaultTexture.RGBA = texcolor;
1558 }
1559 part.UpdateTextureEntry(tex.GetBytes());
1560 return;
1561 }
1562
1409 if (face == ScriptBaseClass.ALL_SIDES) 1563 if (face == ScriptBaseClass.ALL_SIDES)
1410 face = SceneObjectPart.ALL_SIDES; 1564 face = SceneObjectPart.ALL_SIDES;
1411 1565
@@ -1414,6 +1568,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1414 1568
1415 public void SetTexGen(SceneObjectPart part, int face,int style) 1569 public void SetTexGen(SceneObjectPart part, int face,int style)
1416 { 1570 {
1571 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1572 return;
1573
1417 Primitive.TextureEntry tex = part.Shape.Textures; 1574 Primitive.TextureEntry tex = part.Shape.Textures;
1418 MappingType textype; 1575 MappingType textype;
1419 textype = MappingType.Default; 1576 textype = MappingType.Default;
@@ -1444,6 +1601,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1444 1601
1445 public void SetGlow(SceneObjectPart part, int face, float glow) 1602 public void SetGlow(SceneObjectPart part, int face, float glow)
1446 { 1603 {
1604 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1605 return;
1606
1447 Primitive.TextureEntry tex = part.Shape.Textures; 1607 Primitive.TextureEntry tex = part.Shape.Textures;
1448 if (face >= 0 && face < GetNumberOfSides(part)) 1608 if (face >= 0 && face < GetNumberOfSides(part))
1449 { 1609 {
@@ -1469,6 +1629,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1469 1629
1470 public void SetShiny(SceneObjectPart part, int face, int shiny, Bumpiness bump) 1630 public void SetShiny(SceneObjectPart part, int face, int shiny, Bumpiness bump)
1471 { 1631 {
1632 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1633 return;
1472 1634
1473 Shininess sval = new Shininess(); 1635 Shininess sval = new Shininess();
1474 1636
@@ -1519,6 +1681,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1519 1681
1520 public void SetFullBright(SceneObjectPart part, int face, bool bright) 1682 public void SetFullBright(SceneObjectPart part, int face, bool bright)
1521 { 1683 {
1684 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1685 return;
1686
1522 Primitive.TextureEntry tex = part.Shape.Textures; 1687 Primitive.TextureEntry tex = part.Shape.Textures;
1523 if (face >= 0 && face < GetNumberOfSides(part)) 1688 if (face >= 0 && face < GetNumberOfSides(part))
1524 { 1689 {
@@ -1579,13 +1744,26 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1579 m_host.AddScriptLPS(1); 1744 m_host.AddScriptLPS(1);
1580 1745
1581 List<SceneObjectPart> parts = GetLinkParts(linknumber); 1746 List<SceneObjectPart> parts = GetLinkParts(linknumber);
1582 1747 if (parts.Count > 0)
1583 foreach (SceneObjectPart part in parts) 1748 {
1584 SetAlpha(part, alpha, face); 1749 try
1750 {
1751 parts[0].ParentGroup.areUpdatesSuspended = true;
1752 foreach (SceneObjectPart part in parts)
1753 SetAlpha(part, alpha, face);
1754 }
1755 finally
1756 {
1757 parts[0].ParentGroup.areUpdatesSuspended = false;
1758 }
1759 }
1585 } 1760 }
1586 1761
1587 protected void SetAlpha(SceneObjectPart part, double alpha, int face) 1762 protected void SetAlpha(SceneObjectPart part, double alpha, int face)
1588 { 1763 {
1764 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1765 return;
1766
1589 Primitive.TextureEntry tex = part.Shape.Textures; 1767 Primitive.TextureEntry tex = part.Shape.Textures;
1590 Color4 texcolor; 1768 Color4 texcolor;
1591 if (face >= 0 && face < GetNumberOfSides(part)) 1769 if (face >= 0 && face < GetNumberOfSides(part))
@@ -1638,7 +1816,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1638 protected void SetFlexi(SceneObjectPart part, bool flexi, int softness, float gravity, float friction, 1816 protected void SetFlexi(SceneObjectPart part, bool flexi, int softness, float gravity, float friction,
1639 float wind, float tension, LSL_Vector Force) 1817 float wind, float tension, LSL_Vector Force)
1640 { 1818 {
1641 if (part == null) 1819 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1642 return; 1820 return;
1643 1821
1644 if (flexi) 1822 if (flexi)
@@ -1672,7 +1850,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1672 /// <param name="falloff"></param> 1850 /// <param name="falloff"></param>
1673 protected void SetPointLight(SceneObjectPart part, bool light, LSL_Vector color, float intensity, float radius, float falloff) 1851 protected void SetPointLight(SceneObjectPart part, bool light, LSL_Vector color, float intensity, float radius, float falloff)
1674 { 1852 {
1675 if (part == null) 1853 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1676 return; 1854 return;
1677 1855
1678 if (light) 1856 if (light)
@@ -1749,15 +1927,27 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1749 m_host.AddScriptLPS(1); 1927 m_host.AddScriptLPS(1);
1750 1928
1751 List<SceneObjectPart> parts = GetLinkParts(linknumber); 1929 List<SceneObjectPart> parts = GetLinkParts(linknumber);
1752 1930 if (parts.Count > 0)
1753 foreach (SceneObjectPart part in parts) 1931 {
1754 SetTexture(part, texture, face); 1932 try
1755 1933 {
1934 parts[0].ParentGroup.areUpdatesSuspended = true;
1935 foreach (SceneObjectPart part in parts)
1936 SetTexture(part, texture, face);
1937 }
1938 finally
1939 {
1940 parts[0].ParentGroup.areUpdatesSuspended = false;
1941 }
1942 }
1756 ScriptSleep(200); 1943 ScriptSleep(200);
1757 } 1944 }
1758 1945
1759 protected void SetTexture(SceneObjectPart part, string texture, int face) 1946 protected void SetTexture(SceneObjectPart part, string texture, int face)
1760 { 1947 {
1948 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1949 return;
1950
1761 UUID textureID = new UUID(); 1951 UUID textureID = new UUID();
1762 1952
1763 textureID = InventoryKey(texture, (int)AssetType.Texture); 1953 textureID = InventoryKey(texture, (int)AssetType.Texture);
@@ -1802,6 +1992,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1802 1992
1803 protected void ScaleTexture(SceneObjectPart part, double u, double v, int face) 1993 protected void ScaleTexture(SceneObjectPart part, double u, double v, int face)
1804 { 1994 {
1995 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1996 return;
1997
1805 Primitive.TextureEntry tex = part.Shape.Textures; 1998 Primitive.TextureEntry tex = part.Shape.Textures;
1806 if (face >= 0 && face < GetNumberOfSides(part)) 1999 if (face >= 0 && face < GetNumberOfSides(part))
1807 { 2000 {
@@ -1838,6 +2031,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1838 2031
1839 protected void OffsetTexture(SceneObjectPart part, double u, double v, int face) 2032 protected void OffsetTexture(SceneObjectPart part, double u, double v, int face)
1840 { 2033 {
2034 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2035 return;
2036
1841 Primitive.TextureEntry tex = part.Shape.Textures; 2037 Primitive.TextureEntry tex = part.Shape.Textures;
1842 if (face >= 0 && face < GetNumberOfSides(part)) 2038 if (face >= 0 && face < GetNumberOfSides(part))
1843 { 2039 {
@@ -1874,6 +2070,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1874 2070
1875 protected void RotateTexture(SceneObjectPart part, double rotation, int face) 2071 protected void RotateTexture(SceneObjectPart part, double rotation, int face)
1876 { 2072 {
2073 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2074 return;
2075
1877 Primitive.TextureEntry tex = part.Shape.Textures; 2076 Primitive.TextureEntry tex = part.Shape.Textures;
1878 if (face >= 0 && face < GetNumberOfSides(part)) 2077 if (face >= 0 && face < GetNumberOfSides(part))
1879 { 2078 {
@@ -1978,26 +2177,76 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1978 return real_vec; 2177 return real_vec;
1979 } 2178 }
1980 2179
2180 public LSL_Integer llSetRegionPos(LSL_Vector pos)
2181 {
2182 return new LSL_Integer(SetRegionPos(m_host, pos));
2183 }
2184
2185 protected int SetRegionPos(SceneObjectPart part, LSL_Vector targetPos)
2186 {
2187 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2188 return 0;
2189
2190 SceneObjectGroup grp = part.ParentGroup;
2191
2192 if (grp.IsAttachment)
2193 return 0;
2194
2195 if (grp.RootPart.PhysActor != null && grp.RootPart.PhysActor.IsPhysical)
2196 return 0;
2197
2198 if (targetPos.x < -10.0f || targetPos.x >= (float)Constants.RegionSize || targetPos.y < -10.0f || targetPos.y >= (float)Constants.RegionSize || targetPos.z < 0 || targetPos.z >= 4096.0f)
2199 return 0;
2200
2201 float constrainedX = (float)targetPos.x;
2202 float constrainedY = (float)targetPos.y;
2203
2204 if (constrainedX < 0.0f)
2205 constrainedX = 0.0f;
2206 if (constrainedY < 0.0f)
2207 constrainedY = 0.0f;
2208 if (constrainedX >= (float)Constants.RegionSize)
2209 constrainedX = (float)Constants.RegionSize - 0.1f;
2210 if (constrainedY >= (float)Constants.RegionSize)
2211 constrainedY = (float)Constants.RegionSize -0.1f;
2212
2213 float ground = World.GetGroundHeight(constrainedX, constrainedY);
2214
2215 if (targetPos.z < ground)
2216 targetPos.z = ground;
2217
2218 Vector3 dest = new Vector3((float)targetPos.x, (float)targetPos.y, (float)targetPos.z);
2219
2220 if (!World.Permissions.CanObjectEntry(grp.UUID, false, dest))
2221 return 0;
2222
2223 grp.UpdateGroupPosition(dest);
2224
2225 return 1;
2226 }
2227
1981 protected void SetPos(SceneObjectPart part, LSL_Vector targetPos) 2228 protected void SetPos(SceneObjectPart part, LSL_Vector targetPos)
1982 { 2229 {
1983 // Capped movemment if distance > 10m (http://wiki.secondlife.com/wiki/LlSetPos) 2230 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2231 return;
2232
1984 LSL_Vector currentPos = GetPartLocalPos(part); 2233 LSL_Vector currentPos = GetPartLocalPos(part);
2234 LSL_Vector toPos = GetSetPosTarget(part, targetPos, currentPos);
1985 2235
1986 float ground = World.GetGroundHeight((float)targetPos.x, (float)targetPos.y);
1987 bool disable_underground_movement = m_ScriptEngine.Config.GetBoolean("DisableUndergroundMovement", true);
1988 2236
1989 if (part.ParentGroup.RootPart == part) 2237 if (part.ParentGroup.RootPart == part)
1990 { 2238 {
1991 if ((targetPos.z < ground) && disable_underground_movement && m_host.ParentGroup.AttachmentPoint == 0)
1992 targetPos.z = ground;
1993 SceneObjectGroup parent = part.ParentGroup; 2239 SceneObjectGroup parent = part.ParentGroup;
1994 LSL_Vector real_vec = SetPosAdjust(currentPos, targetPos); 2240 Vector3 dest = new Vector3((float)toPos.x, (float)toPos.y, (float)toPos.z);
1995 parent.UpdateGroupPosition(new Vector3((float)real_vec.x, (float)real_vec.y, (float)real_vec.z)); 2241 if (!World.Permissions.CanObjectEntry(parent.UUID, false, dest))
2242 return;
2243 Util.FireAndForget(delegate(object x) {
2244 parent.UpdateGroupPosition(dest);
2245 });
1996 } 2246 }
1997 else 2247 else
1998 { 2248 {
1999 LSL_Vector rel_vec = SetPosAdjust(currentPos, targetPos); 2249 part.OffsetPosition = new Vector3((float)toPos.x, (float)toPos.y, (float)toPos.z);
2000 part.OffsetPosition = new Vector3((float)rel_vec.x, (float)rel_vec.y, (float)rel_vec.z);
2001 SceneObjectGroup parent = part.ParentGroup; 2250 SceneObjectGroup parent = part.ParentGroup;
2002 parent.HasGroupChanged = true; 2251 parent.HasGroupChanged = true;
2003 parent.ScheduleGroupForTerseUpdate(); 2252 parent.ScheduleGroupForTerseUpdate();
@@ -2028,11 +2277,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2028 } 2277 }
2029 else 2278 else
2030 { 2279 {
2031 if (m_host.IsRoot) 2280 if (part.IsRoot)
2032 { 2281 {
2033 return new LSL_Vector(m_host.AttachedPos.X, 2282 return new LSL_Vector(part.AttachedPos.X,
2034 m_host.AttachedPos.Y, 2283 part.AttachedPos.Y,
2035 m_host.AttachedPos.Z); 2284 part.AttachedPos.Z);
2036 } 2285 }
2037 else 2286 else
2038 { 2287 {
@@ -2048,9 +2297,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2048 m_host.AddScriptLPS(1); 2297 m_host.AddScriptLPS(1);
2049 2298
2050 // try to let this work as in SL... 2299 // try to let this work as in SL...
2051 if (m_host.ParentID == 0) 2300 if (m_host.LinkNum < 2)
2052 { 2301 {
2053 // special case: If we are root, rotate complete SOG to new rotation 2302 // Special case: If we are root, rotate complete SOG to new
2303 // rotation.
2304 // We are root if the link number is 0 (single prim) or 1
2305 // (root prim). ParentID may be nonzero in attachments and
2306 // using it would cause attachments and HUDs to rotate
2307 // to the wrong positions.
2054 SetRot(m_host, Rot2Quaternion(rot)); 2308 SetRot(m_host, Rot2Quaternion(rot));
2055 } 2309 }
2056 else 2310 else
@@ -2075,6 +2329,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2075 2329
2076 protected void SetRot(SceneObjectPart part, Quaternion rot) 2330 protected void SetRot(SceneObjectPart part, Quaternion rot)
2077 { 2331 {
2332 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2333 return;
2334
2078 part.UpdateRotation(rot); 2335 part.UpdateRotation(rot);
2079 // Update rotation does not move the object in the physics scene if it's a linkset. 2336 // Update rotation does not move the object in the physics scene if it's a linkset.
2080 2337
@@ -2227,13 +2484,13 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2227 public void llApplyRotationalImpulse(LSL_Vector force, int local) 2484 public void llApplyRotationalImpulse(LSL_Vector force, int local)
2228 { 2485 {
2229 m_host.AddScriptLPS(1); 2486 m_host.AddScriptLPS(1);
2230 m_host.ApplyAngularImpulse(new Vector3((float)force.x, (float)force.y, (float)force.z), local != 0); 2487 m_host.ParentGroup.RootPart.ApplyAngularImpulse(new Vector3((float)force.x, (float)force.y, (float)force.z), local != 0);
2231 } 2488 }
2232 2489
2233 public void llSetTorque(LSL_Vector torque, int local) 2490 public void llSetTorque(LSL_Vector torque, int local)
2234 { 2491 {
2235 m_host.AddScriptLPS(1); 2492 m_host.AddScriptLPS(1);
2236 m_host.SetAngularImpulse(new Vector3((float)torque.x, (float)torque.y, (float)torque.z), local != 0); 2493 m_host.ParentGroup.RootPart.SetAngularImpulse(new Vector3((float)torque.x, (float)torque.y, (float)torque.z), local != 0);
2237 } 2494 }
2238 2495
2239 public LSL_Vector llGetTorque() 2496 public LSL_Vector llGetTorque()
@@ -2698,12 +2955,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2698 2955
2699 m_host.AddScriptLPS(1); 2956 m_host.AddScriptLPS(1);
2700 2957
2958 m_host.TaskInventory.LockItemsForRead(true);
2701 TaskInventoryItem item = m_host.TaskInventory[invItemID]; 2959 TaskInventoryItem item = m_host.TaskInventory[invItemID];
2702 2960 m_host.TaskInventory.LockItemsForRead(false);
2703 lock (m_host.TaskInventory)
2704 {
2705 item = m_host.TaskInventory[invItemID];
2706 }
2707 2961
2708 if (item.PermsGranter == UUID.Zero) 2962 if (item.PermsGranter == UUID.Zero)
2709 return 0; 2963 return 0;
@@ -2771,64 +3025,69 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2771 { 3025 {
2772 m_host.AddScriptLPS(1); 3026 m_host.AddScriptLPS(1);
2773 3027
2774 if (Double.IsNaN(rot.x) || Double.IsNaN(rot.y) || Double.IsNaN(rot.z) || Double.IsNaN(rot.s)) 3028 Util.FireAndForget(delegate (object x)
2775 return; 3029 {
2776 float dist = (float)llVecDist(llGetPos(), pos); 3030 if (Double.IsNaN(rot.x) || Double.IsNaN(rot.y) || Double.IsNaN(rot.z) || Double.IsNaN(rot.s))
3031 return;
3032 float dist = (float)llVecDist(llGetPos(), pos);
2777 3033
2778 if (dist > m_ScriptDistanceFactor * 10.0f) 3034 if (dist > m_ScriptDistanceFactor * 10.0f)
2779 return; 3035 return;
2780 3036
2781 TaskInventoryDictionary partInventory = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 3037 //Clone is thread-safe
3038 TaskInventoryDictionary partInventory = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
2782 3039
2783 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in partInventory) 3040 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in partInventory)
2784 {
2785 if (inv.Value.Name == inventory)
2786 { 3041 {
2787 // make sure we're an object. 3042 if (inv.Value.Name == inventory)
2788 if (inv.Value.InvType != (int)InventoryType.Object)
2789 { 3043 {
2790 llSay(0, "Unable to create requested object. Object is missing from database."); 3044 // make sure we're an object.
2791 return; 3045 if (inv.Value.InvType != (int)InventoryType.Object)
2792 } 3046 {
3047 llSay(0, "Unable to create requested object. Object is missing from database.");
3048 return;
3049 }
2793 3050
2794 Vector3 llpos = new Vector3((float)pos.x, (float)pos.y, (float)pos.z); 3051 Vector3 llpos = new Vector3((float)pos.x, (float)pos.y, (float)pos.z);
2795 Vector3 llvel = new Vector3((float)vel.x, (float)vel.y, (float)vel.z); 3052 Vector3 llvel = new Vector3((float)vel.x, (float)vel.y, (float)vel.z);
2796 3053
2797 // need the magnitude later 3054 // need the magnitude later
2798 float velmag = (float)Util.GetMagnitude(llvel); 3055 // float velmag = (float)Util.GetMagnitude(llvel);
2799 3056
2800 SceneObjectGroup new_group = World.RezObject(m_host, inv.Value, llpos, Rot2Quaternion(rot), llvel, param); 3057 SceneObjectGroup new_group = World.RezObject(m_host, inv.Value, llpos, Rot2Quaternion(rot), llvel, param);
2801 3058
2802 // If either of these are null, then there was an unknown error. 3059 // If either of these are null, then there was an unknown error.
2803 if (new_group == null) 3060 if (new_group == null)
2804 continue; 3061 continue;
2805 3062
2806 // objects rezzed with this method are die_at_edge by default. 3063 // objects rezzed with this method are die_at_edge by default.
2807 new_group.RootPart.SetDieAtEdge(true); 3064 new_group.RootPart.SetDieAtEdge(true);
2808 3065
2809 new_group.ResumeScripts(); 3066 new_group.ResumeScripts();
2810 3067
2811 m_ScriptEngine.PostObjectEvent(m_host.LocalId, new EventParams( 3068 m_ScriptEngine.PostObjectEvent(m_host.LocalId, new EventParams(
2812 "object_rez", new Object[] { 3069 "object_rez", new Object[] {
2813 new LSL_String( 3070 new LSL_String(
2814 new_group.RootPart.UUID.ToString()) }, 3071 new_group.RootPart.UUID.ToString()) },
2815 new DetectParams[0])); 3072 new DetectParams[0]));
2816 3073
2817 float groupmass = new_group.GetMass(); 3074 float groupmass = new_group.GetMass();
2818 3075
2819 if (new_group.RootPart.PhysActor != null && new_group.RootPart.PhysActor.IsPhysical && llvel != Vector3.Zero) 3076 if (new_group.RootPart.PhysActor != null && new_group.RootPart.PhysActor.IsPhysical && llvel != Vector3.Zero)
2820 { 3077 {
2821 //Recoil. 3078 //Recoil.
2822 llApplyImpulse(new LSL_Vector(llvel.X * groupmass, llvel.Y * groupmass, llvel.Z * groupmass), 0); 3079 llApplyImpulse(new LSL_Vector(llvel.X * groupmass, llvel.Y * groupmass, llvel.Z * groupmass), 0);
3080 }
3081 // Variable script delay? (see (http://wiki.secondlife.com/wiki/LSL_Delay)
3082 return;
2823 } 3083 }
2824 // Variable script delay? (see (http://wiki.secondlife.com/wiki/LSL_Delay)
2825 ScriptSleep((int)((groupmass * velmag) / 10));
2826 ScriptSleep(100);
2827 return;
2828 } 3084 }
2829 }
2830 3085
2831 llSay(0, "Could not find object " + inventory); 3086 llSay(0, "Could not find object " + inventory);
3087 });
3088
3089 //ScriptSleep((int)((groupmass * velmag) / 10));
3090 ScriptSleep(100);
2832 } 3091 }
2833 3092
2834 public void llRezObject(string inventory, LSL_Vector pos, LSL_Vector vel, LSL_Rotation rot, int param) 3093 public void llRezObject(string inventory, LSL_Vector pos, LSL_Vector vel, LSL_Rotation rot, int param)
@@ -2839,34 +3098,39 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2839 public void llLookAt(LSL_Vector target, double strength, double damping) 3098 public void llLookAt(LSL_Vector target, double strength, double damping)
2840 { 3099 {
2841 m_host.AddScriptLPS(1); 3100 m_host.AddScriptLPS(1);
2842 // Determine where we are looking from
2843 LSL_Vector from = llGetPos();
2844 3101
2845 // Work out the normalised vector from the source to the target 3102 // Get the normalized vector to the target
2846 LSL_Vector delta = llVecNorm(target - from); 3103 LSL_Vector d1 = llVecNorm(target - llGetPos());
2847 LSL_Vector angle = new LSL_Vector(0,0,0);
2848 3104
2849 // Calculate the yaw 3105 // Get the bearing (yaw)
2850 // subtracting PI_BY_TWO is required to compensate for the odd SL co-ordinate system 3106 LSL_Vector a1 = new LSL_Vector(0,0,0);
2851 angle.x = llAtan2(delta.z, delta.y) - ScriptBaseClass.PI_BY_TWO; 3107 a1.z = llAtan2(d1.y, d1.x);
2852 3108
2853 // Calculate pitch 3109 // Get the elevation (pitch)
2854 angle.y = llAtan2(delta.x, llSqrt((delta.y * delta.y) + (delta.z * delta.z))); 3110 LSL_Vector a2 = new LSL_Vector(0,0,0);
3111 a2.y= -llAtan2(d1.z, llSqrt((d1.x * d1.x) + (d1.y * d1.y)));
2855 3112
2856 // we need to convert from a vector describing 3113 LSL_Rotation r1 = llEuler2Rot(a1);
2857 // the angles of rotation in radians into rotation value 3114 LSL_Rotation r2 = llEuler2Rot(a2);
3115 LSL_Rotation r3 = new LSL_Rotation(0.000000, 0.707107, 0.000000, 0.707107);
2858 3116
2859 LSL_Rotation rot = llEuler2Rot(angle); 3117 if (m_host.PhysActor == null || !m_host.PhysActor.IsPhysical)
2860
2861 // Per discussion with Melanie, for non-physical objects llLookAt appears to simply
2862 // set the rotation of the object, copy that behavior
2863 if (strength == 0 || m_host.PhysActor == null || !m_host.PhysActor.IsPhysical)
2864 { 3118 {
2865 llSetRot(rot); 3119 // Do nothing if either value is 0 (this has been checked in SL)
3120 if (strength <= 0.0 || damping <= 0.0)
3121 return;
3122
3123 llSetRot(r3 * r2 * r1);
2866 } 3124 }
2867 else 3125 else
2868 { 3126 {
2869 m_host.StartLookAt(Rot2Quaternion(rot), (float)strength, (float)damping); 3127 if (strength == 0)
3128 {
3129 llSetRot(r3 * r2 * r1);
3130 return;
3131 }
3132
3133 m_host.StartLookAt(Rot2Quaternion(r3 * r2 * r1), (float)strength, (float)damping);
2870 } 3134 }
2871 } 3135 }
2872 3136
@@ -2916,13 +3180,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2916 { 3180 {
2917 TaskInventoryItem item; 3181 TaskInventoryItem item;
2918 3182
2919 lock (m_host.TaskInventory) 3183 m_host.TaskInventory.LockItemsForRead(true);
3184 if (!m_host.TaskInventory.ContainsKey(InventorySelf()))
2920 { 3185 {
2921 if (!m_host.TaskInventory.ContainsKey(InventorySelf())) 3186 m_host.TaskInventory.LockItemsForRead(false);
2922 return; 3187 return;
2923 else
2924 item = m_host.TaskInventory[InventorySelf()];
2925 } 3188 }
3189 else
3190 {
3191 item = m_host.TaskInventory[InventorySelf()];
3192 }
3193 m_host.TaskInventory.LockItemsForRead(false);
2926 3194
2927 if (item.PermsGranter != UUID.Zero) 3195 if (item.PermsGranter != UUID.Zero)
2928 { 3196 {
@@ -2944,13 +3212,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2944 { 3212 {
2945 TaskInventoryItem item; 3213 TaskInventoryItem item;
2946 3214
3215 m_host.TaskInventory.LockItemsForRead(true);
2947 lock (m_host.TaskInventory) 3216 lock (m_host.TaskInventory)
2948 { 3217 {
3218
2949 if (!m_host.TaskInventory.ContainsKey(InventorySelf())) 3219 if (!m_host.TaskInventory.ContainsKey(InventorySelf()))
3220 {
3221 m_host.TaskInventory.LockItemsForRead(false);
2950 return; 3222 return;
3223 }
2951 else 3224 else
3225 {
2952 item = m_host.TaskInventory[InventorySelf()]; 3226 item = m_host.TaskInventory[InventorySelf()];
3227 }
2953 } 3228 }
3229 m_host.TaskInventory.LockItemsForRead(false);
2954 3230
2955 m_host.AddScriptLPS(1); 3231 m_host.AddScriptLPS(1);
2956 3232
@@ -2982,18 +3258,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2982 { 3258 {
2983 m_host.AddScriptLPS(1); 3259 m_host.AddScriptLPS(1);
2984 3260
2985// if (m_host.ParentGroup.RootPart.AttachmentPoint == 0)
2986// return;
2987
2988 TaskInventoryItem item; 3261 TaskInventoryItem item;
2989 3262
2990 lock (m_host.TaskInventory) 3263 m_host.TaskInventory.LockItemsForRead(true);
3264
3265 if (!m_host.TaskInventory.ContainsKey(InventorySelf()))
2991 { 3266 {
2992 if (!m_host.TaskInventory.ContainsKey(InventorySelf())) 3267 m_host.TaskInventory.LockItemsForRead(false);
2993 return; 3268 return;
2994 else
2995 item = m_host.TaskInventory[InventorySelf()];
2996 } 3269 }
3270 else
3271 {
3272 item = m_host.TaskInventory[InventorySelf()];
3273 }
3274
3275 m_host.TaskInventory.LockItemsForRead(false);
2997 3276
2998 if (item.PermsGranter != m_host.OwnerID) 3277 if (item.PermsGranter != m_host.OwnerID)
2999 return; 3278 return;
@@ -3019,13 +3298,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3019 3298
3020 TaskInventoryItem item; 3299 TaskInventoryItem item;
3021 3300
3022 lock (m_host.TaskInventory) 3301 m_host.TaskInventory.LockItemsForRead(true);
3302
3303 if (!m_host.TaskInventory.ContainsKey(InventorySelf()))
3023 { 3304 {
3024 if (!m_host.TaskInventory.ContainsKey(InventorySelf())) 3305 m_host.TaskInventory.LockItemsForRead(false);
3025 return; 3306 return;
3026 else 3307 }
3027 item = m_host.TaskInventory[InventorySelf()]; 3308 else
3309 {
3310 item = m_host.TaskInventory[InventorySelf()];
3028 } 3311 }
3312 m_host.TaskInventory.LockItemsForRead(false);
3313
3029 3314
3030 if (item.PermsGranter != m_host.OwnerID) 3315 if (item.PermsGranter != m_host.OwnerID)
3031 return; 3316 return;
@@ -3072,6 +3357,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3072 3357
3073 public void llInstantMessage(string user, string message) 3358 public void llInstantMessage(string user, string message)
3074 { 3359 {
3360 UUID result;
3361 if (!UUID.TryParse(user, out result))
3362 {
3363 ShoutError("An invalid key was passed to llInstantMessage");
3364 ScriptSleep(2000);
3365 return;
3366 }
3367
3368
3075 m_host.AddScriptLPS(1); 3369 m_host.AddScriptLPS(1);
3076 3370
3077 // We may be able to use ClientView.SendInstantMessage here, but we need a client instance. 3371 // We may be able to use ClientView.SendInstantMessage here, but we need a client instance.
@@ -3086,14 +3380,34 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3086 UUID friendTransactionID = UUID.Random(); 3380 UUID friendTransactionID = UUID.Random();
3087 3381
3088 //m_pendingFriendRequests.Add(friendTransactionID, fromAgentID); 3382 //m_pendingFriendRequests.Add(friendTransactionID, fromAgentID);
3089 3383
3090 GridInstantMessage msg = new GridInstantMessage(); 3384 GridInstantMessage msg = new GridInstantMessage();
3091 msg.fromAgentID = new Guid(m_host.UUID.ToString()); // fromAgentID.Guid; 3385 msg.fromAgentID = new Guid(m_host.UUID.ToString()); // fromAgentID.Guid;
3092 msg.toAgentID = new Guid(user); // toAgentID.Guid; 3386 msg.toAgentID = new Guid(user); // toAgentID.Guid;
3093 msg.imSessionID = new Guid(friendTransactionID.ToString()); // This is the item we're mucking with here 3387 msg.imSessionID = new Guid(friendTransactionID.ToString()); // This is the item we're mucking with here
3094// m_log.Debug("[Scripting IM]: From:" + msg.fromAgentID.ToString() + " To: " + msg.toAgentID.ToString() + " Session:" + msg.imSessionID.ToString() + " Message:" + message); 3388// m_log.Debug("[Scripting IM]: From:" + msg.fromAgentID.ToString() + " To: " + msg.toAgentID.ToString() + " Session:" + msg.imSessionID.ToString() + " Message:" + message);
3095// m_log.Debug("[Scripting IM]: Filling Session: " + msg.imSessionID.ToString()); 3389// m_log.Debug("[Scripting IM]: Filling Session: " + msg.imSessionID.ToString());
3096 msg.timestamp = (uint)Util.UnixTimeSinceEpoch();// timestamp; 3390// DateTime dt = DateTime.UtcNow;
3391//
3392// // Ticks from UtcNow, but make it look like local. Evil, huh?
3393// dt = DateTime.SpecifyKind(dt, DateTimeKind.Local);
3394//
3395// try
3396// {
3397// // Convert that to the PST timezone
3398// TimeZoneInfo timeZoneInfo = TimeZoneInfo.FindSystemTimeZoneById("America/Los_Angeles");
3399// dt = TimeZoneInfo.ConvertTime(dt, timeZoneInfo);
3400// }
3401// catch
3402// {
3403// // No logging here, as it could be VERY spammy
3404// }
3405//
3406// // And make it look local again to fool the unix time util
3407// dt = DateTime.SpecifyKind(dt, DateTimeKind.Utc);
3408
3409 msg.timestamp = (uint)Util.UnixTimeSinceEpoch();
3410
3097 //if (client != null) 3411 //if (client != null)
3098 //{ 3412 //{
3099 msg.fromAgentName = m_host.Name;//client.FirstName + " " + client.LastName;// fromAgentName; 3413 msg.fromAgentName = m_host.Name;//client.FirstName + " " + client.LastName;// fromAgentName;
@@ -3107,12 +3421,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3107 msg.message = message.Substring(0, 1024); 3421 msg.message = message.Substring(0, 1024);
3108 else 3422 else
3109 msg.message = message; 3423 msg.message = message;
3110 msg.dialog = (byte)19; // messgage from script ??? // dialog; 3424 msg.dialog = (byte)19; // MessageFromObject
3111 msg.fromGroup = false;// fromGroup; 3425 msg.fromGroup = false;// fromGroup;
3112 msg.offline = (byte)0; //offline; 3426 msg.offline = (byte)0; //offline;
3113 msg.ParentEstateID = 0; //ParentEstateID; 3427 msg.ParentEstateID = World.RegionInfo.EstateSettings.EstateID;
3114 msg.Position = new Vector3(m_host.AbsolutePosition); 3428 msg.Position = new Vector3(m_host.AbsolutePosition);
3115 msg.RegionID = World.RegionInfo.RegionID.Guid;//RegionID.Guid; 3429 msg.RegionID = World.RegionInfo.RegionID.Guid;
3116 msg.binaryBucket 3430 msg.binaryBucket
3117 = Util.StringToBytes256( 3431 = Util.StringToBytes256(
3118 "{0}/{1}/{2}/{3}", 3432 "{0}/{1}/{2}/{3}",
@@ -3140,7 +3454,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3140 } 3454 }
3141 3455
3142 emailModule.SendEmail(m_host.UUID, address, subject, message); 3456 emailModule.SendEmail(m_host.UUID, address, subject, message);
3143 ScriptSleep(20000); 3457 ScriptSleep(15000);
3144 } 3458 }
3145 3459
3146 public void llGetNextEmail(string address, string subject) 3460 public void llGetNextEmail(string address, string subject)
@@ -3279,14 +3593,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3279 3593
3280 TaskInventoryItem item; 3594 TaskInventoryItem item;
3281 3595
3282 lock (m_host.TaskInventory) 3596 m_host.TaskInventory.LockItemsForRead(true);
3597 if (!m_host.TaskInventory.ContainsKey(InventorySelf()))
3283 { 3598 {
3284 if (!m_host.TaskInventory.ContainsKey(InventorySelf())) 3599 m_host.TaskInventory.LockItemsForRead(false);
3285 return; 3600 return;
3286 else
3287 item = m_host.TaskInventory[InventorySelf()];
3288 } 3601 }
3289 3602 else
3603 {
3604 item = m_host.TaskInventory[InventorySelf()];
3605 }
3606 m_host.TaskInventory.LockItemsForRead(false);
3290 if (item.PermsGranter == UUID.Zero) 3607 if (item.PermsGranter == UUID.Zero)
3291 return; 3608 return;
3292 3609
@@ -3316,13 +3633,18 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3316 3633
3317 TaskInventoryItem item; 3634 TaskInventoryItem item;
3318 3635
3319 lock (m_host.TaskInventory) 3636 m_host.TaskInventory.LockItemsForRead(true);
3637 if (!m_host.TaskInventory.ContainsKey(InventorySelf()))
3320 { 3638 {
3321 if (!m_host.TaskInventory.ContainsKey(InventorySelf())) 3639 m_host.TaskInventory.LockItemsForRead(false);
3322 return; 3640 return;
3323 else 3641 }
3324 item = m_host.TaskInventory[InventorySelf()]; 3642 else
3643 {
3644 item = m_host.TaskInventory[InventorySelf()];
3325 } 3645 }
3646 m_host.TaskInventory.LockItemsForRead(false);
3647
3326 3648
3327 if (item.PermsGranter == UUID.Zero) 3649 if (item.PermsGranter == UUID.Zero)
3328 return; 3650 return;
@@ -3389,10 +3711,18 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3389 3711
3390 TaskInventoryItem item; 3712 TaskInventoryItem item;
3391 3713
3392 lock (m_host.TaskInventory) 3714
3715 m_host.TaskInventory.LockItemsForRead(true);
3716 if (!m_host.TaskInventory.ContainsKey(invItemID))
3717 {
3718 m_host.TaskInventory.LockItemsForRead(false);
3719 return;
3720 }
3721 else
3393 { 3722 {
3394 item = m_host.TaskInventory[invItemID]; 3723 item = m_host.TaskInventory[invItemID];
3395 } 3724 }
3725 m_host.TaskInventory.LockItemsForRead(false);
3396 3726
3397 if (agentID == UUID.Zero || perm == 0) // Releasing permissions 3727 if (agentID == UUID.Zero || perm == 0) // Releasing permissions
3398 { 3728 {
@@ -3420,15 +3750,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3420 int implicitPerms = ScriptBaseClass.PERMISSION_TAKE_CONTROLS | 3750 int implicitPerms = ScriptBaseClass.PERMISSION_TAKE_CONTROLS |
3421 ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION | 3751 ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION |
3422 ScriptBaseClass.PERMISSION_CONTROL_CAMERA | 3752 ScriptBaseClass.PERMISSION_CONTROL_CAMERA |
3753 ScriptBaseClass.PERMISSION_TRACK_CAMERA |
3423 ScriptBaseClass.PERMISSION_ATTACH; 3754 ScriptBaseClass.PERMISSION_ATTACH;
3424 3755
3425 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms 3756 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms
3426 { 3757 {
3427 lock (m_host.TaskInventory) 3758 m_host.TaskInventory.LockItemsForWrite(true);
3428 { 3759 m_host.TaskInventory[invItemID].PermsGranter = agentID;
3429 m_host.TaskInventory[invItemID].PermsGranter = agentID; 3760 m_host.TaskInventory[invItemID].PermsMask = perm;
3430 m_host.TaskInventory[invItemID].PermsMask = perm; 3761 m_host.TaskInventory.LockItemsForWrite(false);
3431 }
3432 3762
3433 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams( 3763 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams(
3434 "run_time_permissions", new Object[] { 3764 "run_time_permissions", new Object[] {
@@ -3438,28 +3768,44 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3438 return; 3768 return;
3439 } 3769 }
3440 } 3770 }
3441 else if (m_host.SitTargetAvatar == agentID) // Sitting avatar 3771 else
3442 { 3772 {
3443 // When agent is sitting, certain permissions are implicit if requested from sitting agent 3773 bool sitting = false;
3444 int implicitPerms = ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION | 3774 if (m_host.SitTargetAvatar == agentID)
3445 ScriptBaseClass.PERMISSION_CONTROL_CAMERA | 3775 {
3446 ScriptBaseClass.PERMISSION_TRACK_CAMERA | 3776 sitting = true;
3447 ScriptBaseClass.PERMISSION_TAKE_CONTROLS; 3777 }
3778 else
3779 {
3780 foreach (SceneObjectPart p in m_host.ParentGroup.Parts)
3781 {
3782 if (p.SitTargetAvatar == agentID)
3783 sitting = true;
3784 }
3785 }
3448 3786
3449 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms 3787 if (sitting)
3450 { 3788 {
3451 lock (m_host.TaskInventory) 3789 // When agent is sitting, certain permissions are implicit if requested from sitting agent
3790 int implicitPerms = ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION |
3791 ScriptBaseClass.PERMISSION_CONTROL_CAMERA |
3792 ScriptBaseClass.PERMISSION_TRACK_CAMERA |
3793 ScriptBaseClass.PERMISSION_TAKE_CONTROLS;
3794
3795 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms
3452 { 3796 {
3797 m_host.TaskInventory.LockItemsForWrite(true);
3453 m_host.TaskInventory[invItemID].PermsGranter = agentID; 3798 m_host.TaskInventory[invItemID].PermsGranter = agentID;
3454 m_host.TaskInventory[invItemID].PermsMask = perm; 3799 m_host.TaskInventory[invItemID].PermsMask = perm;
3455 } 3800 m_host.TaskInventory.LockItemsForWrite(false);
3456 3801
3457 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams( 3802 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams(
3458 "run_time_permissions", new Object[] { 3803 "run_time_permissions", new Object[] {
3459 new LSL_Integer(perm) }, 3804 new LSL_Integer(perm) },
3460 new DetectParams[0])); 3805 new DetectParams[0]));
3461 3806
3462 return; 3807 return;
3808 }
3463 } 3809 }
3464 } 3810 }
3465 3811
@@ -3473,11 +3819,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3473 3819
3474 if (!m_waitingForScriptAnswer) 3820 if (!m_waitingForScriptAnswer)
3475 { 3821 {
3476 lock (m_host.TaskInventory) 3822 m_host.TaskInventory.LockItemsForWrite(true);
3477 { 3823 m_host.TaskInventory[invItemID].PermsGranter = agentID;
3478 m_host.TaskInventory[invItemID].PermsGranter = agentID; 3824 m_host.TaskInventory[invItemID].PermsMask = 0;
3479 m_host.TaskInventory[invItemID].PermsMask = 0; 3825 m_host.TaskInventory.LockItemsForWrite(false);
3480 }
3481 3826
3482 presence.ControllingClient.OnScriptAnswer += handleScriptAnswer; 3827 presence.ControllingClient.OnScriptAnswer += handleScriptAnswer;
3483 m_waitingForScriptAnswer=true; 3828 m_waitingForScriptAnswer=true;
@@ -3512,10 +3857,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3512 if ((answer & ScriptBaseClass.PERMISSION_TAKE_CONTROLS) == 0) 3857 if ((answer & ScriptBaseClass.PERMISSION_TAKE_CONTROLS) == 0)
3513 llReleaseControls(); 3858 llReleaseControls();
3514 3859
3515 lock (m_host.TaskInventory) 3860
3516 { 3861 m_host.TaskInventory.LockItemsForWrite(true);
3517 m_host.TaskInventory[invItemID].PermsMask = answer; 3862 m_host.TaskInventory[invItemID].PermsMask = answer;
3518 } 3863 m_host.TaskInventory.LockItemsForWrite(false);
3864
3519 3865
3520 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams( 3866 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams(
3521 "run_time_permissions", new Object[] { 3867 "run_time_permissions", new Object[] {
@@ -3527,16 +3873,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3527 { 3873 {
3528 m_host.AddScriptLPS(1); 3874 m_host.AddScriptLPS(1);
3529 3875
3530 lock (m_host.TaskInventory) 3876 m_host.TaskInventory.LockItemsForRead(true);
3877
3878 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
3531 { 3879 {
3532 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 3880 if (item.Type == 10 && item.ItemID == m_itemID)
3533 { 3881 {
3534 if (item.Type == 10 && item.ItemID == m_itemID) 3882 m_host.TaskInventory.LockItemsForRead(false);
3535 { 3883 return item.PermsGranter.ToString();
3536 return item.PermsGranter.ToString();
3537 }
3538 } 3884 }
3539 } 3885 }
3886 m_host.TaskInventory.LockItemsForRead(false);
3540 3887
3541 return UUID.Zero.ToString(); 3888 return UUID.Zero.ToString();
3542 } 3889 }
@@ -3545,19 +3892,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3545 { 3892 {
3546 m_host.AddScriptLPS(1); 3893 m_host.AddScriptLPS(1);
3547 3894
3548 lock (m_host.TaskInventory) 3895 m_host.TaskInventory.LockItemsForRead(true);
3896
3897 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
3549 { 3898 {
3550 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 3899 if (item.Type == 10 && item.ItemID == m_itemID)
3551 { 3900 {
3552 if (item.Type == 10 && item.ItemID == m_itemID) 3901 int perms = item.PermsMask;
3553 { 3902 if (m_automaticLinkPermission)
3554 int perms = item.PermsMask; 3903 perms |= ScriptBaseClass.PERMISSION_CHANGE_LINKS;
3555 if (m_automaticLinkPermission) 3904 m_host.TaskInventory.LockItemsForRead(false);
3556 perms |= ScriptBaseClass.PERMISSION_CHANGE_LINKS; 3905 return perms;
3557 return perms;
3558 }
3559 } 3906 }
3560 } 3907 }
3908 m_host.TaskInventory.LockItemsForRead(false);
3561 3909
3562 return 0; 3910 return 0;
3563 } 3911 }
@@ -3579,9 +3927,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3579 public void llSetLinkColor(int linknumber, LSL_Vector color, int face) 3927 public void llSetLinkColor(int linknumber, LSL_Vector color, int face)
3580 { 3928 {
3581 List<SceneObjectPart> parts = GetLinkParts(linknumber); 3929 List<SceneObjectPart> parts = GetLinkParts(linknumber);
3582 3930 if (parts.Count > 0)
3583 foreach (SceneObjectPart part in parts) 3931 {
3584 part.SetFaceColor(new Vector3((float)color.x, (float)color.y, (float)color.z), face); 3932 try
3933 {
3934 parts[0].ParentGroup.areUpdatesSuspended = true;
3935 foreach (SceneObjectPart part in parts)
3936 part.SetFaceColor(new Vector3((float)color.x, (float)color.y, (float)color.z), face);
3937 }
3938 finally
3939 {
3940 parts[0].ParentGroup.areUpdatesSuspended = false;
3941 }
3942 }
3585 } 3943 }
3586 3944
3587 public void llCreateLink(string target, int parent) 3945 public void llCreateLink(string target, int parent)
@@ -3594,11 +3952,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3594 return; 3952 return;
3595 3953
3596 TaskInventoryItem item; 3954 TaskInventoryItem item;
3597 lock (m_host.TaskInventory) 3955 m_host.TaskInventory.LockItemsForRead(true);
3598 { 3956 item = m_host.TaskInventory[invItemID];
3599 item = m_host.TaskInventory[invItemID]; 3957 m_host.TaskInventory.LockItemsForRead(false);
3600 } 3958
3601
3602 if ((item.PermsMask & ScriptBaseClass.PERMISSION_CHANGE_LINKS) == 0 3959 if ((item.PermsMask & ScriptBaseClass.PERMISSION_CHANGE_LINKS) == 0
3603 && !m_automaticLinkPermission) 3960 && !m_automaticLinkPermission)
3604 { 3961 {
@@ -3615,11 +3972,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3615 3972
3616 if (targetPart.ParentGroup.AttachmentPoint != 0) 3973 if (targetPart.ParentGroup.AttachmentPoint != 0)
3617 return; // Fail silently if attached 3974 return; // Fail silently if attached
3975
3976 if (targetPart.ParentGroup.RootPart.OwnerID != m_host.ParentGroup.RootPart.OwnerID)
3977 return;
3978
3618 SceneObjectGroup parentPrim = null, childPrim = null; 3979 SceneObjectGroup parentPrim = null, childPrim = null;
3619 3980
3620 if (targetPart != null) 3981 if (targetPart != null)
3621 { 3982 {
3622 if (parent != 0) { 3983 if (parent != 0)
3984 {
3623 parentPrim = m_host.ParentGroup; 3985 parentPrim = m_host.ParentGroup;
3624 childPrim = targetPart.ParentGroup; 3986 childPrim = targetPart.ParentGroup;
3625 } 3987 }
@@ -3631,7 +3993,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3631 3993
3632 // Required for linking 3994 // Required for linking
3633 childPrim.RootPart.ClearUpdateSchedule(); 3995 childPrim.RootPart.ClearUpdateSchedule();
3634 parentPrim.LinkToGroup(childPrim); 3996 parentPrim.LinkToGroup(childPrim, true);
3635 } 3997 }
3636 3998
3637 parentPrim.TriggerScriptChangedEvent(Changed.LINK); 3999 parentPrim.TriggerScriptChangedEvent(Changed.LINK);
@@ -3650,16 +4012,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3650 m_host.AddScriptLPS(1); 4012 m_host.AddScriptLPS(1);
3651 UUID invItemID = InventorySelf(); 4013 UUID invItemID = InventorySelf();
3652 4014
3653 lock (m_host.TaskInventory) 4015 m_host.TaskInventory.LockItemsForRead(true);
3654 {
3655 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_CHANGE_LINKS) == 0 4016 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_CHANGE_LINKS) == 0
3656 && !m_automaticLinkPermission) 4017 && !m_automaticLinkPermission)
3657 { 4018 {
3658 ShoutError("Script trying to link but PERMISSION_CHANGE_LINKS permission not set!"); 4019 ShoutError("Script trying to link but PERMISSION_CHANGE_LINKS permission not set!");
4020 m_host.TaskInventory.LockItemsForRead(false);
3659 return; 4021 return;
3660 } 4022 }
3661 } 4023 m_host.TaskInventory.LockItemsForRead(false);
3662 4024
3663 if (linknum < ScriptBaseClass.LINK_THIS) 4025 if (linknum < ScriptBaseClass.LINK_THIS)
3664 return; 4026 return;
3665 4027
@@ -3698,10 +4060,22 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3698 // Restructuring Multiple Prims. 4060 // Restructuring Multiple Prims.
3699 List<SceneObjectPart> parts = new List<SceneObjectPart>(parentPrim.Parts); 4061 List<SceneObjectPart> parts = new List<SceneObjectPart>(parentPrim.Parts);
3700 parts.Remove(parentPrim.RootPart); 4062 parts.Remove(parentPrim.RootPart);
3701 foreach (SceneObjectPart part in parts) 4063 if (parts.Count > 0)
3702 { 4064 {
3703 parentPrim.DelinkFromGroup(part.LocalId, true); 4065 try
4066 {
4067 parts[0].ParentGroup.areUpdatesSuspended = true;
4068 foreach (SceneObjectPart part in parts)
4069 {
4070 parentPrim.DelinkFromGroup(part.LocalId, true);
4071 }
4072 }
4073 finally
4074 {
4075 parts[0].ParentGroup.areUpdatesSuspended = false;
4076 }
3704 } 4077 }
4078
3705 parentPrim.HasGroupChanged = true; 4079 parentPrim.HasGroupChanged = true;
3706 parentPrim.ScheduleGroupForFullUpdate(); 4080 parentPrim.ScheduleGroupForFullUpdate();
3707 parentPrim.TriggerScriptChangedEvent(Changed.LINK); 4081 parentPrim.TriggerScriptChangedEvent(Changed.LINK);
@@ -3710,12 +4084,22 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3710 { 4084 {
3711 SceneObjectPart newRoot = parts[0]; 4085 SceneObjectPart newRoot = parts[0];
3712 parts.Remove(newRoot); 4086 parts.Remove(newRoot);
3713 foreach (SceneObjectPart part in parts) 4087
4088 try
3714 { 4089 {
3715 // Required for linking 4090 parts[0].ParentGroup.areUpdatesSuspended = true;
3716 part.ClearUpdateSchedule(); 4091 foreach (SceneObjectPart part in parts)
3717 newRoot.ParentGroup.LinkToGroup(part.ParentGroup); 4092 {
4093 part.ClearUpdateSchedule();
4094 newRoot.ParentGroup.LinkToGroup(part.ParentGroup);
4095 }
3718 } 4096 }
4097 finally
4098 {
4099 parts[0].ParentGroup.areUpdatesSuspended = false;
4100 }
4101
4102
3719 newRoot.ParentGroup.HasGroupChanged = true; 4103 newRoot.ParentGroup.HasGroupChanged = true;
3720 newRoot.ParentGroup.ScheduleGroupForFullUpdate(); 4104 newRoot.ParentGroup.ScheduleGroupForFullUpdate();
3721 } 4105 }
@@ -3735,6 +4119,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3735 public void llBreakAllLinks() 4119 public void llBreakAllLinks()
3736 { 4120 {
3737 m_host.AddScriptLPS(1); 4121 m_host.AddScriptLPS(1);
4122
4123 UUID invItemID = InventorySelf();
4124
4125 TaskInventoryItem item;
4126 m_host.TaskInventory.LockItemsForRead(true);
4127 item = m_host.TaskInventory[invItemID];
4128 m_host.TaskInventory.LockItemsForRead(false);
4129
4130 if ((item.PermsMask & ScriptBaseClass.PERMISSION_CHANGE_LINKS) == 0
4131 && !m_automaticLinkPermission)
4132 {
4133 ShoutError("Script trying to link but PERMISSION_CHANGE_LINKS permission not set!");
4134 return;
4135 }
4136
3738 SceneObjectGroup parentPrim = m_host.ParentGroup; 4137 SceneObjectGroup parentPrim = m_host.ParentGroup;
3739 if (parentPrim.AttachmentPoint != 0) 4138 if (parentPrim.AttachmentPoint != 0)
3740 return; // Fail silently if attached 4139 return; // Fail silently if attached
@@ -3754,25 +4153,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3754 public LSL_String llGetLinkKey(int linknum) 4153 public LSL_String llGetLinkKey(int linknum)
3755 { 4154 {
3756 m_host.AddScriptLPS(1); 4155 m_host.AddScriptLPS(1);
3757 List<UUID> keytable = new List<UUID>();
3758 // parse for sitting avatare-uuids
3759 World.ForEachRootScenePresence(delegate(ScenePresence presence)
3760 {
3761 if (presence.ParentID != 0 && m_host.ParentGroup.HasChildPrim(presence.ParentID))
3762 keytable.Add(presence.UUID);
3763 });
3764
3765 int totalprims = m_host.ParentGroup.PrimCount + keytable.Count;
3766 if (linknum > m_host.ParentGroup.PrimCount && linknum <= totalprims)
3767 {
3768 return keytable[totalprims - linknum].ToString();
3769 }
3770
3771 if (linknum == 1 && m_host.ParentGroup.PrimCount == 1 && keytable.Count == 1)
3772 {
3773 return m_host.UUID.ToString();
3774 }
3775
3776 SceneObjectPart part = m_host.ParentGroup.GetLinkNumPart(linknum); 4156 SceneObjectPart part = m_host.ParentGroup.GetLinkNumPart(linknum);
3777 if (part != null) 4157 if (part != null)
3778 { 4158 {
@@ -3780,6 +4160,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3780 } 4160 }
3781 else 4161 else
3782 { 4162 {
4163 if (linknum > m_host.ParentGroup.PrimCount || (linknum == 1 && m_host.ParentGroup.PrimCount == 1))
4164 {
4165 linknum -= (m_host.ParentGroup.PrimCount) + 1;
4166
4167 if (linknum < 0)
4168 return UUID.Zero.ToString();
4169
4170 List<ScenePresence> avatars = GetLinkAvatars(ScriptBaseClass.LINK_SET);
4171 if (avatars.Count > linknum)
4172 {
4173 return avatars[linknum].UUID.ToString();
4174 }
4175 }
3783 return UUID.Zero.ToString(); 4176 return UUID.Zero.ToString();
3784 } 4177 }
3785 } 4178 }
@@ -3878,17 +4271,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3878 m_host.AddScriptLPS(1); 4271 m_host.AddScriptLPS(1);
3879 int count = 0; 4272 int count = 0;
3880 4273
3881 lock (m_host.TaskInventory) 4274 m_host.TaskInventory.LockItemsForRead(true);
4275 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
3882 { 4276 {
3883 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 4277 if (inv.Value.Type == type || type == -1)
3884 { 4278 {
3885 if (inv.Value.Type == type || type == -1) 4279 count = count + 1;
3886 {
3887 count = count + 1;
3888 }
3889 } 4280 }
3890 } 4281 }
3891 4282
4283 m_host.TaskInventory.LockItemsForRead(false);
3892 return count; 4284 return count;
3893 } 4285 }
3894 4286
@@ -3897,16 +4289,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3897 m_host.AddScriptLPS(1); 4289 m_host.AddScriptLPS(1);
3898 ArrayList keys = new ArrayList(); 4290 ArrayList keys = new ArrayList();
3899 4291
3900 lock (m_host.TaskInventory) 4292 m_host.TaskInventory.LockItemsForRead(true);
4293 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
3901 { 4294 {
3902 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 4295 if (inv.Value.Type == type || type == -1)
3903 { 4296 {
3904 if (inv.Value.Type == type || type == -1) 4297 keys.Add(inv.Value.Name);
3905 {
3906 keys.Add(inv.Value.Name);
3907 }
3908 } 4298 }
3909 } 4299 }
4300 m_host.TaskInventory.LockItemsForRead(false);
3910 4301
3911 if (keys.Count == 0) 4302 if (keys.Count == 0)
3912 { 4303 {
@@ -3943,25 +4334,25 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3943 } 4334 }
3944 4335
3945 // move the first object found with this inventory name 4336 // move the first object found with this inventory name
3946 lock (m_host.TaskInventory) 4337 m_host.TaskInventory.LockItemsForRead(true);
4338 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
3947 { 4339 {
3948 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 4340 if (inv.Value.Name == inventory)
3949 { 4341 {
3950 if (inv.Value.Name == inventory) 4342 found = true;
3951 { 4343 objId = inv.Key;
3952 found = true; 4344 assetType = inv.Value.Type;
3953 objId = inv.Key; 4345 objName = inv.Value.Name;
3954 assetType = inv.Value.Type; 4346 break;
3955 objName = inv.Value.Name;
3956 break;
3957 }
3958 } 4347 }
3959 } 4348 }
4349 m_host.TaskInventory.LockItemsForRead(false);
3960 4350
3961 if (!found) 4351 if (!found)
3962 { 4352 {
3963 llSay(0, String.Format("Could not find object '{0}'", inventory)); 4353 llSay(0, String.Format("Could not find object '{0}'", inventory));
3964 throw new Exception(String.Format("The inventory object '{0}' could not be found", inventory)); 4354 return;
4355// throw new Exception(String.Format("The inventory object '{0}' could not be found", inventory));
3965 } 4356 }
3966 4357
3967 // check if destination is an object 4358 // check if destination is an object
@@ -3987,48 +4378,68 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3987 return; 4378 return;
3988 } 4379 }
3989 } 4380 }
4381
3990 // destination is an avatar 4382 // destination is an avatar
3991 InventoryItemBase agentItem = World.MoveTaskInventoryItem(destId, UUID.Zero, m_host, objId); 4383 InventoryItemBase agentItem = World.MoveTaskInventoryItem(destId, UUID.Zero, m_host, objId);
3992 4384
3993 if (agentItem == null) 4385 if (agentItem == null)
3994 return; 4386 return;
3995 4387
3996 byte[] bucket = new byte[17]; 4388 byte[] bucket = new byte[1];
3997 bucket[0] = (byte)assetType; 4389 bucket[0] = (byte)assetType;
3998 byte[] objBytes = agentItem.ID.GetBytes(); 4390 //byte[] objBytes = agentItem.ID.GetBytes();
3999 Array.Copy(objBytes, 0, bucket, 1, 16); 4391 //Array.Copy(objBytes, 0, bucket, 1, 16);
4000 4392
4001 GridInstantMessage msg = new GridInstantMessage(World, 4393 GridInstantMessage msg = new GridInstantMessage(World,
4002 m_host.UUID, m_host.Name+", an object owned by "+ 4394 m_host.OwnerID, m_host.Name, destId,
4003 resolveName(m_host.OwnerID)+",", destId,
4004 (byte)InstantMessageDialog.TaskInventoryOffered, 4395 (byte)InstantMessageDialog.TaskInventoryOffered,
4005 false, objName+"\n"+m_host.Name+" is located at "+ 4396 false, objName+". "+m_host.Name+" is located at "+
4006 World.RegionInfo.RegionName+" "+ 4397 World.RegionInfo.RegionName+" "+
4007 m_host.AbsolutePosition.ToString(), 4398 m_host.AbsolutePosition.ToString(),
4008 agentItem.ID, true, m_host.AbsolutePosition, 4399 agentItem.ID, true, m_host.AbsolutePosition,
4009 bucket); 4400 bucket);
4010 if (m_TransferModule != null) 4401
4011 m_TransferModule.SendInstantMessage(msg, delegate(bool success) {}); 4402 ScenePresence sp;
4403
4404 if (World.TryGetScenePresence(destId, out sp))
4405 {
4406 sp.ControllingClient.SendInstantMessage(msg);
4407 }
4408 else
4409 {
4410 if (m_TransferModule != null)
4411 m_TransferModule.SendInstantMessage(msg, delegate(bool success) {});
4412 }
4413
4414 //This delay should only occur when giving inventory to avatars.
4012 ScriptSleep(3000); 4415 ScriptSleep(3000);
4013 } 4416 }
4014 } 4417 }
4015 4418
4419 [DebuggerNonUserCode]
4016 public void llRemoveInventory(string name) 4420 public void llRemoveInventory(string name)
4017 { 4421 {
4018 m_host.AddScriptLPS(1); 4422 m_host.AddScriptLPS(1);
4019 4423
4020 lock (m_host.TaskInventory) 4424 List<TaskInventoryItem> inv;
4425 try
4021 { 4426 {
4022 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 4427 m_host.TaskInventory.LockItemsForRead(true);
4428 inv = new List<TaskInventoryItem>(m_host.TaskInventory.Values);
4429 }
4430 finally
4431 {
4432 m_host.TaskInventory.LockItemsForRead(false);
4433 }
4434 foreach (TaskInventoryItem item in inv)
4435 {
4436 if (item.Name == name)
4023 { 4437 {
4024 if (item.Name == name) 4438 if (item.ItemID == m_itemID)
4025 { 4439 throw new ScriptDeleteException();
4026 if (item.ItemID == m_itemID) 4440 else
4027 throw new ScriptDeleteException(); 4441 m_host.Inventory.RemoveInventoryItem(item.ItemID);
4028 else 4442 return;
4029 m_host.Inventory.RemoveInventoryItem(item.ItemID);
4030 return;
4031 }
4032 } 4443 }
4033 } 4444 }
4034 } 4445 }
@@ -4063,112 +4474,122 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4063 { 4474 {
4064 m_host.AddScriptLPS(1); 4475 m_host.AddScriptLPS(1);
4065 4476
4066 UUID uuid = (UUID)id; 4477 UUID uuid;
4067 PresenceInfo pinfo = null; 4478 if (UUID.TryParse(id, out uuid))
4068 UserAccount account;
4069
4070 UserInfoCacheEntry ce;
4071 if (!m_userInfoCache.TryGetValue(uuid, out ce))
4072 { 4479 {
4073 account = World.UserAccountService.GetUserAccount(World.RegionInfo.ScopeID, uuid); 4480 PresenceInfo pinfo = null;
4074 if (account == null) 4481 UserAccount account;
4482
4483 UserInfoCacheEntry ce;
4484 if (!m_userInfoCache.TryGetValue(uuid, out ce))
4075 { 4485 {
4076 m_userInfoCache[uuid] = null; // Cache negative 4486 account = World.UserAccountService.GetUserAccount(World.RegionInfo.ScopeID, uuid);
4077 return UUID.Zero.ToString(); 4487 if (account == null)
4078 } 4488 {
4489 m_userInfoCache[uuid] = null; // Cache negative
4490 return UUID.Zero.ToString();
4491 }
4079 4492
4080 4493
4081 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() }); 4494 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() });
4082 if (pinfos != null && pinfos.Length > 0) 4495 if (pinfos != null && pinfos.Length > 0)
4083 {
4084 foreach (PresenceInfo p in pinfos)
4085 { 4496 {
4086 if (p.RegionID != UUID.Zero) 4497 foreach (PresenceInfo p in pinfos)
4087 { 4498 {
4088 pinfo = p; 4499 if (p.RegionID != UUID.Zero)
4500 {
4501 pinfo = p;
4502 }
4089 } 4503 }
4090 } 4504 }
4091 }
4092 4505
4093 ce = new UserInfoCacheEntry(); 4506 ce = new UserInfoCacheEntry();
4094 ce.time = Util.EnvironmentTickCount(); 4507 ce.time = Util.EnvironmentTickCount();
4095 ce.account = account; 4508 ce.account = account;
4096 ce.pinfo = pinfo; 4509 ce.pinfo = pinfo;
4097 } 4510 m_userInfoCache[uuid] = ce;
4098 else 4511 }
4099 { 4512 else
4100 if (ce == null) 4513 {
4101 return UUID.Zero.ToString(); 4514 if (ce == null)
4515 return UUID.Zero.ToString();
4102 4516
4103 account = ce.account; 4517 account = ce.account;
4104 pinfo = ce.pinfo; 4518 pinfo = ce.pinfo;
4105 } 4519 }
4106 4520
4107 if (Util.EnvironmentTickCount() < ce.time || (Util.EnvironmentTickCount() - ce.time) >= 20000) 4521 if (Util.EnvironmentTickCount() < ce.time || (Util.EnvironmentTickCount() - ce.time) >= 20000)
4108 {
4109 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() });
4110 if (pinfos != null && pinfos.Length > 0)
4111 { 4522 {
4112 foreach (PresenceInfo p in pinfos) 4523 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() });
4524 if (pinfos != null && pinfos.Length > 0)
4113 { 4525 {
4114 if (p.RegionID != UUID.Zero) 4526 foreach (PresenceInfo p in pinfos)
4115 { 4527 {
4116 pinfo = p; 4528 if (p.RegionID != UUID.Zero)
4529 {
4530 pinfo = p;
4531 }
4117 } 4532 }
4118 } 4533 }
4119 } 4534 else
4120 else 4535 pinfo = null;
4121 pinfo = null;
4122 4536
4123 ce.time = Util.EnvironmentTickCount(); 4537 ce.time = Util.EnvironmentTickCount();
4124 ce.pinfo = pinfo; 4538 ce.pinfo = pinfo;
4125 } 4539 }
4126 4540
4127 string reply = String.Empty; 4541 string reply = String.Empty;
4128 4542
4129 switch (data) 4543 switch (data)
4130 { 4544 {
4131 case 1: // DATA_ONLINE (0|1) 4545 case 1: // DATA_ONLINE (0|1)
4132 if (pinfo != null && pinfo.RegionID != UUID.Zero) 4546 if (pinfo != null && pinfo.RegionID != UUID.Zero)
4133 reply = "1"; 4547 reply = "1";
4134 else 4548 else
4135 reply = "0"; 4549 reply = "0";
4136 break; 4550 break;
4137 case 2: // DATA_NAME (First Last) 4551 case 2: // DATA_NAME (First Last)
4138 reply = account.FirstName + " " + account.LastName; 4552 reply = account.FirstName + " " + account.LastName;
4139 break; 4553 break;
4140 case 3: // DATA_BORN (YYYY-MM-DD) 4554 case 3: // DATA_BORN (YYYY-MM-DD)
4141 DateTime born = new DateTime(1970, 1, 1, 0, 0, 0, 0); 4555 DateTime born = new DateTime(1970, 1, 1, 0, 0, 0, 0);
4142 born = born.AddSeconds(account.Created); 4556 born = born.AddSeconds(account.Created);
4143 reply = born.ToString("yyyy-MM-dd"); 4557 reply = born.ToString("yyyy-MM-dd");
4144 break; 4558 break;
4145 case 4: // DATA_RATING (0,0,0,0,0,0) 4559 case 4: // DATA_RATING (0,0,0,0,0,0)
4146 reply = "0,0,0,0,0,0"; 4560 reply = "0,0,0,0,0,0";
4147 break; 4561 break;
4148 case 8: // DATA_PAYINFO (0|1|2|3) 4562 case 8: // DATA_PAYINFO (0|1|2|3)
4149 reply = "0"; 4563 reply = "0";
4150 break; 4564 break;
4151 default: 4565 default:
4152 return UUID.Zero.ToString(); // Raise no event 4566 return UUID.Zero.ToString(); // Raise no event
4153 } 4567 }
4154 4568
4155 UUID rq = UUID.Random(); 4569 UUID rq = UUID.Random();
4156 4570
4157 UUID tid = AsyncCommands. 4571 UUID tid = AsyncCommands.
4158 DataserverPlugin.RegisterRequest(m_localID, 4572 DataserverPlugin.RegisterRequest(m_localID,
4159 m_itemID, rq.ToString()); 4573 m_itemID, rq.ToString());
4160 4574
4161 AsyncCommands. 4575 AsyncCommands.
4162 DataserverPlugin.DataserverReply(rq.ToString(), reply); 4576 DataserverPlugin.DataserverReply(rq.ToString(), reply);
4163 4577
4164 ScriptSleep(100); 4578 ScriptSleep(100);
4165 return tid.ToString(); 4579 return tid.ToString();
4580 }
4581 else
4582 {
4583 ShoutError("Invalid UUID passed to llRequestAgentData.");
4584 }
4585 return "";
4166 } 4586 }
4167 4587
4168 public LSL_String llRequestInventoryData(string name) 4588 public LSL_String llRequestInventoryData(string name)
4169 { 4589 {
4170 m_host.AddScriptLPS(1); 4590 m_host.AddScriptLPS(1);
4171 4591
4592 //Clone is thread safe
4172 TaskInventoryDictionary itemDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 4593 TaskInventoryDictionary itemDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
4173 4594
4174 foreach (TaskInventoryItem item in itemDictionary.Values) 4595 foreach (TaskInventoryItem item in itemDictionary.Values)
@@ -4220,19 +4641,54 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4220 if (UUID.TryParse(agent, out agentId)) 4641 if (UUID.TryParse(agent, out agentId))
4221 { 4642 {
4222 ScenePresence presence = World.GetScenePresence(agentId); 4643 ScenePresence presence = World.GetScenePresence(agentId);
4223 if (presence != null) 4644 if (presence != null && presence.PresenceType != PresenceType.Npc)
4224 { 4645 {
4646 // agent must not be a god
4647 if (presence.UserLevel >= 200) return;
4648
4225 // agent must be over the owners land 4649 // agent must be over the owners land
4226 if (m_host.OwnerID == World.LandChannel.GetLandObject( 4650 if (m_host.OwnerID == World.LandChannel.GetLandObject(
4227 presence.AbsolutePosition.X, presence.AbsolutePosition.Y).LandData.OwnerID) 4651 presence.AbsolutePosition.X, presence.AbsolutePosition.Y).LandData.OwnerID)
4228 { 4652 {
4229 World.TeleportClientHome(agentId, presence.ControllingClient); 4653 if (!World.TeleportClientHome(agentId, presence.ControllingClient))
4654 {
4655 // They can't be teleported home for some reason
4656 GridRegion regionInfo = World.GridService.GetRegionByUUID(UUID.Zero, new UUID("2b02daac-e298-42fa-9a75-f488d37896e6"));
4657 if (regionInfo != null)
4658 {
4659 World.RequestTeleportLocation(
4660 presence.ControllingClient, regionInfo.RegionHandle, new Vector3(128, 128, 23), Vector3.Zero,
4661 (uint)(Constants.TeleportFlags.SetLastToTarget | Constants.TeleportFlags.ViaHome));
4662 }
4663 }
4230 } 4664 }
4231 } 4665 }
4232 } 4666 }
4233 ScriptSleep(5000); 4667 ScriptSleep(5000);
4234 } 4668 }
4235 4669
4670 public void llTeleportAgent(string agent, string simname, LSL_Vector pos, LSL_Vector lookAt)
4671 {
4672 m_host.AddScriptLPS(1);
4673 UUID agentId = new UUID();
4674 if (UUID.TryParse(agent, out agentId))
4675 {
4676 ScenePresence presence = World.GetScenePresence(agentId);
4677 if (presence != null && presence.PresenceType != PresenceType.Npc)
4678 {
4679 // agent must not be a god
4680 if (presence.UserLevel >= 200) return;
4681
4682 // agent must be over the owners land
4683 if (m_host.OwnerID == World.LandChannel.GetLandObject(
4684 presence.AbsolutePosition.X, presence.AbsolutePosition.Y).LandData.OwnerID)
4685 {
4686 World.RequestTeleportLocation(presence.ControllingClient, simname, new Vector3((float)pos.x, (float)pos.y, (float)pos.z), new Vector3((float)lookAt.x, (float)lookAt.y, (float)lookAt.z), (uint)TeleportFlags.ViaLocation);
4687 }
4688 }
4689 }
4690 }
4691
4236 public void llTextBox(string agent, string message, int chatChannel) 4692 public void llTextBox(string agent, string message, int chatChannel)
4237 { 4693 {
4238 IDialogModule dm = World.RequestModuleInterface<IDialogModule>(); 4694 IDialogModule dm = World.RequestModuleInterface<IDialogModule>();
@@ -4244,7 +4700,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4244 UUID av = new UUID(); 4700 UUID av = new UUID();
4245 if (!UUID.TryParse(agent,out av)) 4701 if (!UUID.TryParse(agent,out av))
4246 { 4702 {
4247 LSLError("First parameter to llDialog needs to be a key"); 4703 //LSLError("First parameter to llDialog needs to be a key");
4248 return; 4704 return;
4249 } 4705 }
4250 4706
@@ -4281,17 +4737,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4281 UUID soundId = UUID.Zero; 4737 UUID soundId = UUID.Zero;
4282 if (!UUID.TryParse(impact_sound, out soundId)) 4738 if (!UUID.TryParse(impact_sound, out soundId))
4283 { 4739 {
4284 lock (m_host.TaskInventory) 4740 m_host.TaskInventory.LockItemsForRead(true);
4741 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
4285 { 4742 {
4286 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 4743 if (item.Type == (int)AssetType.Sound && item.Name == impact_sound)
4287 { 4744 {
4288 if (item.Type == (int)AssetType.Sound && item.Name == impact_sound) 4745 soundId = item.AssetID;
4289 { 4746 break;
4290 soundId = item.AssetID;
4291 break;
4292 }
4293 } 4747 }
4294 } 4748 }
4749 m_host.TaskInventory.LockItemsForRead(false);
4295 } 4750 }
4296 m_host.CollisionSound = soundId; 4751 m_host.CollisionSound = soundId;
4297 m_host.CollisionSoundVolume = (float)impact_volume; 4752 m_host.CollisionSoundVolume = (float)impact_volume;
@@ -4331,6 +4786,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4331 UUID partItemID; 4786 UUID partItemID;
4332 foreach (SceneObjectPart part in parts) 4787 foreach (SceneObjectPart part in parts)
4333 { 4788 {
4789 //Clone is thread safe
4334 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)part.TaskInventory.Clone(); 4790 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)part.TaskInventory.Clone();
4335 4791
4336 foreach (TaskInventoryItem item in itemsDictionary.Values) 4792 foreach (TaskInventoryItem item in itemsDictionary.Values)
@@ -4545,17 +5001,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4545 5001
4546 m_host.AddScriptLPS(1); 5002 m_host.AddScriptLPS(1);
4547 5003
4548 lock (m_host.TaskInventory) 5004 m_host.TaskInventory.LockItemsForRead(true);
5005 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
4549 { 5006 {
4550 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 5007 if (item.Type == 10 && item.ItemID == m_itemID)
4551 { 5008 {
4552 if (item.Type == 10 && item.ItemID == m_itemID) 5009 result = item.Name!=null?item.Name:String.Empty;
4553 { 5010 break;
4554 result = item.Name != null ? item.Name : String.Empty;
4555 break;
4556 }
4557 } 5011 }
4558 } 5012 }
5013 m_host.TaskInventory.LockItemsForRead(false);
4559 5014
4560 return result; 5015 return result;
4561 } 5016 }
@@ -4728,23 +5183,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4728 { 5183 {
4729 m_host.AddScriptLPS(1); 5184 m_host.AddScriptLPS(1);
4730 5185
4731 lock (m_host.TaskInventory) 5186 m_host.TaskInventory.LockItemsForRead(true);
5187 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
4732 { 5188 {
4733 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 5189 if (inv.Value.Name == name)
4734 { 5190 {
4735 if (inv.Value.Name == name) 5191 if ((inv.Value.CurrentPermissions & (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify)) == (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify))
4736 { 5192 {
4737 if ((inv.Value.CurrentPermissions & (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify)) == (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify)) 5193 m_host.TaskInventory.LockItemsForRead(false);
4738 { 5194 return inv.Value.AssetID.ToString();
4739 return inv.Value.AssetID.ToString(); 5195 }
4740 } 5196 else
4741 else 5197 {
4742 { 5198 m_host.TaskInventory.LockItemsForRead(false);
4743 return UUID.Zero.ToString(); 5199 return UUID.Zero.ToString();
4744 }
4745 } 5200 }
4746 } 5201 }
4747 } 5202 }
5203 m_host.TaskInventory.LockItemsForRead(false);
4748 5204
4749 return UUID.Zero.ToString(); 5205 return UUID.Zero.ToString();
4750 } 5206 }
@@ -4897,14 +5353,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4897 { 5353 {
4898 m_host.AddScriptLPS(1); 5354 m_host.AddScriptLPS(1);
4899 5355
4900 if (src == null) 5356 return src.Length;
4901 {
4902 return 0;
4903 }
4904 else
4905 {
4906 return src.Length;
4907 }
4908 } 5357 }
4909 5358
4910 public LSL_Integer llList2Integer(LSL_List src, int index) 5359 public LSL_Integer llList2Integer(LSL_List src, int index)
@@ -4950,7 +5399,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4950 else if (src.Data[index] is LSL_Float) 5399 else if (src.Data[index] is LSL_Float)
4951 return Convert.ToDouble(((LSL_Float) src.Data[index]).value); 5400 return Convert.ToDouble(((LSL_Float) src.Data[index]).value);
4952 else if (src.Data[index] is LSL_String) 5401 else if (src.Data[index] is LSL_String)
4953 return Convert.ToDouble(((LSL_String) src.Data[index]).m_string); 5402 {
5403 string str = ((LSL_String) src.Data[index]).m_string;
5404 Match m = Regex.Match(str, "^\\s*(-?\\+?[,0-9]+\\.?[0-9]*)");
5405 if (m != Match.Empty)
5406 {
5407 str = m.Value;
5408 double d = 0.0;
5409 if (!Double.TryParse(str, out d))
5410 return 0.0;
5411
5412 return d;
5413 }
5414 return 0.0;
5415 }
4954 return Convert.ToDouble(src.Data[index]); 5416 return Convert.ToDouble(src.Data[index]);
4955 } 5417 }
4956 catch (FormatException) 5418 catch (FormatException)
@@ -5223,7 +5685,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5223 } 5685 }
5224 } 5686 }
5225 } 5687 }
5226 else { 5688 else
5689 {
5227 object[] array = new object[src.Length]; 5690 object[] array = new object[src.Length];
5228 Array.Copy(src.Data, 0, array, 0, src.Length); 5691 Array.Copy(src.Data, 0, array, 0, src.Length);
5229 result = new LSL_List(array); 5692 result = new LSL_List(array);
@@ -5330,7 +5793,13 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5330 public LSL_Integer llGetRegionAgentCount() 5793 public LSL_Integer llGetRegionAgentCount()
5331 { 5794 {
5332 m_host.AddScriptLPS(1); 5795 m_host.AddScriptLPS(1);
5333 return new LSL_Integer(World.GetRootAgentCount()); 5796
5797 int count = 0;
5798 World.ForEachRootScenePresence(delegate(ScenePresence sp) {
5799 count++;
5800 });
5801
5802 return new LSL_Integer(count);
5334 } 5803 }
5335 5804
5336 public LSL_Vector llGetRegionCorner() 5805 public LSL_Vector llGetRegionCorner()
@@ -5672,10 +6141,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5672 m_host.AddScriptLPS(1); 6141 m_host.AddScriptLPS(1);
5673 6142
5674 List<SceneObjectPart> parts = GetLinkParts(linknumber); 6143 List<SceneObjectPart> parts = GetLinkParts(linknumber);
5675 6144 if (parts.Count > 0)
5676 foreach (var part in parts)
5677 { 6145 {
5678 SetTextureAnim(part, mode, face, sizex, sizey, start, length, rate); 6146 try
6147 {
6148 parts[0].ParentGroup.areUpdatesSuspended = true;
6149 foreach (var part in parts)
6150 {
6151 SetTextureAnim(part, mode, face, sizex, sizey, start, length, rate);
6152 }
6153 }
6154 finally
6155 {
6156 parts[0].ParentGroup.areUpdatesSuspended = false;
6157 }
5679 } 6158 }
5680 } 6159 }
5681 6160
@@ -5727,13 +6206,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5727 6206
5728 if (m_host.OwnerID == land.LandData.OwnerID) 6207 if (m_host.OwnerID == land.LandData.OwnerID)
5729 { 6208 {
5730 World.TeleportClientHome(agentID, presence.ControllingClient); 6209 Vector3 pos = World.GetNearestAllowedPosition(presence, land);
6210 presence.TeleportWithMomentum(pos);
6211 presence.ControllingClient.SendAlertMessage("You have been ejected from this land");
5731 } 6212 }
5732 } 6213 }
5733 } 6214 }
5734 ScriptSleep(5000); 6215 ScriptSleep(5000);
5735 } 6216 }
5736 6217
6218 public LSL_List llParseString2List(string str, LSL_List separators, LSL_List in_spacers)
6219 {
6220 return ParseString2List(str, separators, in_spacers, false);
6221 }
6222
5737 public LSL_Integer llOverMyLand(string id) 6223 public LSL_Integer llOverMyLand(string id)
5738 { 6224 {
5739 m_host.AddScriptLPS(1); 6225 m_host.AddScriptLPS(1);
@@ -5798,8 +6284,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5798 UUID agentId = new UUID(); 6284 UUID agentId = new UUID();
5799 if (!UUID.TryParse(agent, out agentId)) 6285 if (!UUID.TryParse(agent, out agentId))
5800 return new LSL_Integer(0); 6286 return new LSL_Integer(0);
6287 if (agentId == m_host.GroupID)
6288 return new LSL_Integer(1);
5801 ScenePresence presence = World.GetScenePresence(agentId); 6289 ScenePresence presence = World.GetScenePresence(agentId);
5802 if (presence == null || presence.IsChildAgent) // Return flase for child agents 6290 if (presence == null || presence.IsChildAgent) // Return false for child agents
5803 return new LSL_Integer(0); 6291 return new LSL_Integer(0);
5804 IClientAPI client = presence.ControllingClient; 6292 IClientAPI client = presence.ControllingClient;
5805 if (m_host.GroupID == client.ActiveGroupId) 6293 if (m_host.GroupID == client.ActiveGroupId)
@@ -5934,7 +6422,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5934 return m_host.ParentGroup.AttachmentPoint; 6422 return m_host.ParentGroup.AttachmentPoint;
5935 } 6423 }
5936 6424
5937 public LSL_Integer llGetFreeMemory() 6425 public virtual LSL_Integer llGetFreeMemory()
5938 { 6426 {
5939 m_host.AddScriptLPS(1); 6427 m_host.AddScriptLPS(1);
5940 // Make scripts designed for LSO happy 6428 // Make scripts designed for LSO happy
@@ -6051,7 +6539,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6051 SetParticleSystem(m_host, rules); 6539 SetParticleSystem(m_host, rules);
6052 } 6540 }
6053 6541
6054 private void SetParticleSystem(SceneObjectPart part, LSL_List rules) { 6542 private void SetParticleSystem(SceneObjectPart part, LSL_List rules)
6543 {
6055 6544
6056 6545
6057 if (rules.Length == 0) 6546 if (rules.Length == 0)
@@ -6245,14 +6734,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6245 6734
6246 protected UUID GetTaskInventoryItem(string name) 6735 protected UUID GetTaskInventoryItem(string name)
6247 { 6736 {
6248 lock (m_host.TaskInventory) 6737 m_host.TaskInventory.LockItemsForRead(true);
6738 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
6249 { 6739 {
6250 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 6740 if (inv.Value.Name == name)
6251 { 6741 {
6252 if (inv.Value.Name == name) 6742 m_host.TaskInventory.LockItemsForRead(false);
6253 return inv.Key; 6743 return inv.Key;
6254 } 6744 }
6255 } 6745 }
6746 m_host.TaskInventory.LockItemsForRead(false);
6256 6747
6257 return UUID.Zero; 6748 return UUID.Zero;
6258 } 6749 }
@@ -6290,16 +6781,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6290 if (folderID == UUID.Zero) 6781 if (folderID == UUID.Zero)
6291 return; 6782 return;
6292 6783
6293 byte[] bucket = new byte[17]; 6784 byte[] bucket = new byte[1];
6294 bucket[0] = (byte)AssetType.Folder; 6785 bucket[0] = (byte)AssetType.Folder;
6295 byte[] objBytes = folderID.GetBytes(); 6786 //byte[] objBytes = folderID.GetBytes();
6296 Array.Copy(objBytes, 0, bucket, 1, 16); 6787 //Array.Copy(objBytes, 0, bucket, 1, 16);
6297 6788
6298 GridInstantMessage msg = new GridInstantMessage(World, 6789 GridInstantMessage msg = new GridInstantMessage(World,
6299 m_host.UUID, m_host.Name+", an object owned by "+ 6790 m_host.OwnerID, m_host.Name, destID,
6300 resolveName(m_host.OwnerID)+",", destID, 6791 (byte)InstantMessageDialog.TaskInventoryOffered,
6301 (byte)InstantMessageDialog.InventoryOffered, 6792 false, category+". "+m_host.Name+" is located at "+
6302 false, category+"\n"+m_host.Name+" is located at "+
6303 World.RegionInfo.RegionName+" "+ 6793 World.RegionInfo.RegionName+" "+
6304 m_host.AbsolutePosition.ToString(), 6794 m_host.AbsolutePosition.ToString(),
6305 folderID, true, m_host.AbsolutePosition, 6795 folderID, true, m_host.AbsolutePosition,
@@ -6537,13 +7027,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6537 UUID av = new UUID(); 7027 UUID av = new UUID();
6538 if (!UUID.TryParse(avatar,out av)) 7028 if (!UUID.TryParse(avatar,out av))
6539 { 7029 {
6540 LSLError("First parameter to llDialog needs to be a key"); 7030 //LSLError("First parameter to llDialog needs to be a key");
6541 return; 7031 return;
6542 } 7032 }
6543 if (buttons.Length < 1) 7033 if (buttons.Length < 1)
6544 { 7034 {
6545 LSLError("No less than 1 button can be shown"); 7035 buttons.Add("OK");
6546 return;
6547 } 7036 }
6548 if (buttons.Length > 12) 7037 if (buttons.Length > 12)
6549 { 7038 {
@@ -6560,7 +7049,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6560 } 7049 }
6561 if (buttons.Data[i].ToString().Length > 24) 7050 if (buttons.Data[i].ToString().Length > 24)
6562 { 7051 {
6563 LSLError("button label cannot be longer than 24 characters"); 7052 llWhisper(ScriptBaseClass.DEBUG_CHANNEL, "button label cannot be longer than 24 characters");
6564 return; 7053 return;
6565 } 7054 }
6566 buts[i] = buttons.Data[i].ToString(); 7055 buts[i] = buttons.Data[i].ToString();
@@ -6619,22 +7108,23 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6619 } 7108 }
6620 7109
6621 // copy the first script found with this inventory name 7110 // copy the first script found with this inventory name
6622 lock (m_host.TaskInventory) 7111 TaskInventoryItem scriptItem = null;
7112 m_host.TaskInventory.LockItemsForRead(true);
7113 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
6623 { 7114 {
6624 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 7115 if (inv.Value.Name == name)
6625 { 7116 {
6626 if (inv.Value.Name == name) 7117 // make sure the object is a script
7118 if (10 == inv.Value.Type)
6627 { 7119 {
6628 // make sure the object is a script 7120 found = true;
6629 if (10 == inv.Value.Type) 7121 srcId = inv.Key;
6630 { 7122 scriptItem = inv.Value;
6631 found = true; 7123 break;
6632 srcId = inv.Key;
6633 break;
6634 }
6635 } 7124 }
6636 } 7125 }
6637 } 7126 }
7127 m_host.TaskInventory.LockItemsForRead(false);
6638 7128
6639 if (!found) 7129 if (!found)
6640 { 7130 {
@@ -6642,9 +7132,18 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6642 return; 7132 return;
6643 } 7133 }
6644 7134
6645 // the rest of the permission checks are done in RezScript, so check the pin there as well 7135 SceneObjectPart dest = World.GetSceneObjectPart(destId);
6646 World.RezScriptFromPrim(srcId, m_host, destId, pin, running, start_param); 7136 if (dest != null)
7137 {
7138 if ((scriptItem.BasePermissions & (uint)PermissionMask.Transfer) != 0 || dest.ParentGroup.RootPart.OwnerID == m_host.ParentGroup.RootPart.OwnerID)
7139 {
7140 // the rest of the permission checks are done in RezScript, so check the pin there as well
7141 World.RezScriptFromPrim(srcId, m_host, destId, pin, running, start_param);
6647 7142
7143 if ((scriptItem.BasePermissions & (uint)PermissionMask.Copy) == 0)
7144 m_host.Inventory.RemoveInventoryItem(srcId);
7145 }
7146 }
6648 // this will cause the delay even if the script pin or permissions were wrong - seems ok 7147 // this will cause the delay even if the script pin or permissions were wrong - seems ok
6649 ScriptSleep(3000); 7148 ScriptSleep(3000);
6650 } 7149 }
@@ -6707,19 +7206,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6707 public LSL_String llMD5String(string src, int nonce) 7206 public LSL_String llMD5String(string src, int nonce)
6708 { 7207 {
6709 m_host.AddScriptLPS(1); 7208 m_host.AddScriptLPS(1);
6710 return Util.Md5Hash(String.Format("{0}:{1}", src, nonce.ToString())); 7209 return Util.Md5Hash(String.Format("{0}:{1}", src, nonce.ToString()), Encoding.UTF8);
6711 } 7210 }
6712 7211
6713 public LSL_String llSHA1String(string src) 7212 public LSL_String llSHA1String(string src)
6714 { 7213 {
6715 m_host.AddScriptLPS(1); 7214 m_host.AddScriptLPS(1);
6716 return Util.SHA1Hash(src).ToLower(); 7215 return Util.SHA1Hash(src, Encoding.UTF8).ToLower();
6717 } 7216 }
6718 7217
6719 protected ObjectShapePacket.ObjectDataBlock SetPrimitiveBlockShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, byte profileshape, byte pathcurve) 7218 protected ObjectShapePacket.ObjectDataBlock SetPrimitiveBlockShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, byte profileshape, byte pathcurve)
6720 { 7219 {
6721 float tempFloat; // Use in float expressions below to avoid byte cast precision issues. 7220 float tempFloat; // Use in float expressions below to avoid byte cast precision issues.
6722 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock(); 7221 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock();
7222 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7223 return shapeBlock;
6723 7224
6724 if (holeshape != (int)ScriptBaseClass.PRIM_HOLE_DEFAULT && 7225 if (holeshape != (int)ScriptBaseClass.PRIM_HOLE_DEFAULT &&
6725 holeshape != (int)ScriptBaseClass.PRIM_HOLE_CIRCLE && 7226 holeshape != (int)ScriptBaseClass.PRIM_HOLE_CIRCLE &&
@@ -6824,6 +7325,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6824 // Prim type box, cylinder and prism. 7325 // Prim type box, cylinder and prism.
6825 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector taper_b, LSL_Vector topshear, byte profileshape, byte pathcurve) 7326 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector taper_b, LSL_Vector topshear, byte profileshape, byte pathcurve)
6826 { 7327 {
7328 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7329 return;
7330
6827 float tempFloat; // Use in float expressions below to avoid byte cast precision issues. 7331 float tempFloat; // Use in float expressions below to avoid byte cast precision issues.
6828 ObjectShapePacket.ObjectDataBlock shapeBlock; 7332 ObjectShapePacket.ObjectDataBlock shapeBlock;
6829 7333
@@ -6877,6 +7381,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6877 // Prim type sphere. 7381 // Prim type sphere.
6878 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector dimple, byte profileshape, byte pathcurve) 7382 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector dimple, byte profileshape, byte pathcurve)
6879 { 7383 {
7384 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7385 return;
7386
6880 ObjectShapePacket.ObjectDataBlock shapeBlock; 7387 ObjectShapePacket.ObjectDataBlock shapeBlock;
6881 7388
6882 shapeBlock = SetPrimitiveBlockShapeParams(part, holeshape, cut, hollow, twist, profileshape, pathcurve); 7389 shapeBlock = SetPrimitiveBlockShapeParams(part, holeshape, cut, hollow, twist, profileshape, pathcurve);
@@ -6918,6 +7425,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6918 // Prim type torus, tube and ring. 7425 // Prim type torus, tube and ring.
6919 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector holesize, LSL_Vector topshear, LSL_Vector profilecut, LSL_Vector taper_a, float revolutions, float radiusoffset, float skew, byte profileshape, byte pathcurve) 7426 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector holesize, LSL_Vector topshear, LSL_Vector profilecut, LSL_Vector taper_a, float revolutions, float radiusoffset, float skew, byte profileshape, byte pathcurve)
6920 { 7427 {
7428 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7429 return;
7430
6921 float tempFloat; // Use in float expressions below to avoid byte cast precision issues. 7431 float tempFloat; // Use in float expressions below to avoid byte cast precision issues.
6922 ObjectShapePacket.ObjectDataBlock shapeBlock; 7432 ObjectShapePacket.ObjectDataBlock shapeBlock;
6923 7433
@@ -7053,6 +7563,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7053 // Prim type sculpt. 7563 // Prim type sculpt.
7054 protected void SetPrimitiveShapeParams(SceneObjectPart part, string map, int type, byte pathcurve) 7564 protected void SetPrimitiveShapeParams(SceneObjectPart part, string map, int type, byte pathcurve)
7055 { 7565 {
7566 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7567 return;
7568
7056 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock(); 7569 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock();
7057 UUID sculptId; 7570 UUID sculptId;
7058 7571
@@ -7077,7 +7590,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7077 type != (ScriptBaseClass.PRIM_SCULPT_TYPE_TORUS | flag)) 7590 type != (ScriptBaseClass.PRIM_SCULPT_TYPE_TORUS | flag))
7078 { 7591 {
7079 // default 7592 // default
7080 type = (int)ScriptBaseClass.PRIM_SCULPT_TYPE_SPHERE; 7593 type = type | (int)ScriptBaseClass.PRIM_SCULPT_TYPE_SPHERE;
7081 } 7594 }
7082 7595
7083 part.Shape.SetSculptProperties((byte)type, sculptId); 7596 part.Shape.SetSculptProperties((byte)type, sculptId);
@@ -7093,32 +7606,119 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7093 ScriptSleep(200); 7606 ScriptSleep(200);
7094 } 7607 }
7095 7608
7096 public void llSetLinkPrimitiveParams(int linknumber, LSL_List rules) 7609 public void llSetLinkPrimitiveParamsFast(int linknumber, LSL_List rules)
7097 { 7610 {
7098 m_host.AddScriptLPS(1); 7611 m_host.AddScriptLPS(1);
7099 7612
7100 setLinkPrimParams(linknumber, rules); 7613 setLinkPrimParams(linknumber, rules);
7101
7102 ScriptSleep(200);
7103 } 7614 }
7104 7615
7105 public void llSetLinkPrimitiveParamsFast(int linknumber, LSL_List rules) 7616 private void setLinkPrimParams(int linknumber, LSL_List rules)
7106 { 7617 {
7107 m_host.AddScriptLPS(1); 7618 List<SceneObjectPart> parts = GetLinkParts(linknumber);
7619 List<ScenePresence> avatars = GetLinkAvatars(linknumber);
7620 if (parts.Count>0)
7621 {
7622 try
7623 {
7624 parts[0].ParentGroup.areUpdatesSuspended = true;
7625 foreach (SceneObjectPart part in parts)
7626 SetPrimParams(part, rules);
7627 }
7628 finally
7629 {
7630 parts[0].ParentGroup.areUpdatesSuspended = false;
7631 }
7632 }
7633 if (avatars.Count > 0)
7634 {
7635 foreach (ScenePresence avatar in avatars)
7636 SetPrimParams(avatar, rules);
7637 }
7638 }
7108 7639
7109 setLinkPrimParams(linknumber, rules); 7640 public void llSetLinkPrimitiveParams(int linknumber, LSL_List rules)
7641 {
7642 llSetLinkPrimitiveParamsFast(linknumber, rules);
7643 ScriptSleep(200);
7110 } 7644 }
7111 7645
7112 protected void setLinkPrimParams(int linknumber, LSL_List rules) 7646 protected void SetPrimParams(ScenePresence av, LSL_List rules)
7113 { 7647 {
7114 List<SceneObjectPart> parts = GetLinkParts(linknumber); 7648 //This is a special version of SetPrimParams to deal with avatars which are sat on the linkset.
7649 //We only support PRIM_POSITION and PRIM_ROTATION
7115 7650
7116 foreach (SceneObjectPart part in parts) 7651 int idx = 0;
7117 SetPrimParams(part, rules); 7652
7653 while (idx < rules.Length)
7654 {
7655 int code = rules.GetLSLIntegerItem(idx++);
7656
7657 int remain = rules.Length - idx;
7658
7659 switch (code)
7660 {
7661 case (int)ScriptBaseClass.PRIM_POSITION:
7662 {
7663 if (remain < 1)
7664 return;
7665 LSL_Vector v;
7666 v = rules.GetVector3Item(idx++);
7667
7668 SceneObjectPart part = World.GetSceneObjectPart(av.ParentID);
7669 if (part == null)
7670 break;
7671
7672 LSL_Rotation localRot = ScriptBaseClass.ZERO_ROTATION;
7673 LSL_Vector localPos = ScriptBaseClass.ZERO_VECTOR;
7674 if (llGetLinkNumber() > 1)
7675 {
7676 localRot = llGetLocalRot();
7677 localPos = llGetLocalPos();
7678 }
7679
7680 v -= localPos;
7681 v /= localRot;
7682
7683 LSL_Vector sitOffset = (llRot2Up(new LSL_Rotation(av.Rotation.X, av.Rotation.Y, av.Rotation.Z, av.Rotation.W)) * av.Appearance.AvatarHeight * 0.02638f);
7684
7685 v = v + 2 * sitOffset;
7686
7687 av.OffsetPosition = new Vector3((float)v.x, (float)v.y, (float)v.z);
7688 av.SendAvatarDataToAllAgents();
7689
7690 }
7691 break;
7692
7693 case (int)ScriptBaseClass.PRIM_ROTATION:
7694 {
7695 if (remain < 1)
7696 return;
7697
7698 LSL_Rotation localRot = ScriptBaseClass.ZERO_ROTATION;
7699 LSL_Vector localPos = ScriptBaseClass.ZERO_VECTOR;
7700 if (llGetLinkNumber() > 1)
7701 {
7702 localRot = llGetLocalRot();
7703 localPos = llGetLocalPos();
7704 }
7705
7706 LSL_Rotation r;
7707 r = rules.GetQuaternionItem(idx++);
7708 r = r * llGetRootRotation() / localRot;
7709 av.Rotation = new Quaternion((float)r.x, (float)r.y, (float)r.z, (float)r.s);
7710 av.SendAvatarDataToAllAgents();
7711 }
7712 break;
7713 }
7714 }
7118 } 7715 }
7119 7716
7120 protected void SetPrimParams(SceneObjectPart part, LSL_List rules) 7717 protected void SetPrimParams(SceneObjectPart part, LSL_List rules)
7121 { 7718 {
7719 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7720 return;
7721
7122 int idx = 0; 7722 int idx = 0;
7123 7723
7124 bool positionChanged = false; 7724 bool positionChanged = false;
@@ -7146,6 +7746,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7146 currentPosition = GetSetPosTarget(part, v, currentPosition); 7746 currentPosition = GetSetPosTarget(part, v, currentPosition);
7147 7747
7148 break; 7748 break;
7749 case (int)ScriptBaseClass.PRIM_POS_LOCAL:
7750 if (remain < 1)
7751 return;
7752
7753 v=rules.GetVector3Item(idx++);
7754 positionChanged = true;
7755 currentPosition = GetSetPosTarget(part, v, currentPosition);
7756
7757 break;
7149 case (int)ScriptBaseClass.PRIM_SIZE: 7758 case (int)ScriptBaseClass.PRIM_SIZE:
7150 if (remain < 1) 7759 if (remain < 1)
7151 return; 7760 return;
@@ -7512,7 +8121,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7512 if (part.ParentGroup.RootPart == part) 8121 if (part.ParentGroup.RootPart == part)
7513 { 8122 {
7514 SceneObjectGroup parent = part.ParentGroup; 8123 SceneObjectGroup parent = part.ParentGroup;
7515 parent.UpdateGroupPosition(new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z)); 8124 Util.FireAndForget(delegate(object x) {
8125 parent.UpdateGroupPosition(new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z));
8126 });
7516 } 8127 }
7517 else 8128 else
7518 { 8129 {
@@ -7523,6 +8134,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7523 } 8134 }
7524 } 8135 }
7525 } 8136 }
8137
8138 if (positionChanged)
8139 {
8140 if (part.ParentGroup.RootPart == part)
8141 {
8142 SceneObjectGroup parent = part.ParentGroup;
8143 Util.FireAndForget(delegate(object x) {
8144 parent.UpdateGroupPosition(new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z));
8145 });
8146 }
8147 else
8148 {
8149 part.OffsetPosition = new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z);
8150 SceneObjectGroup parent = part.ParentGroup;
8151 parent.HasGroupChanged = true;
8152 parent.ScheduleGroupForTerseUpdate();
8153 }
8154 }
7526 } 8155 }
7527 8156
7528 public LSL_String llStringToBase64(string str) 8157 public LSL_String llStringToBase64(string str)
@@ -7683,13 +8312,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7683 public LSL_Integer llGetNumberOfPrims() 8312 public LSL_Integer llGetNumberOfPrims()
7684 { 8313 {
7685 m_host.AddScriptLPS(1); 8314 m_host.AddScriptLPS(1);
7686 int avatarCount = 0; 8315 int avatarCount = m_host.ParentGroup.GetLinkedAvatars().Count;
7687 World.ForEachRootScenePresence(delegate(ScenePresence presence) 8316
7688 {
7689 if (presence.ParentID != 0 && m_host.ParentGroup.HasChildPrim(presence.ParentID))
7690 avatarCount++;
7691 });
7692
7693 return m_host.ParentGroup.PrimCount + avatarCount; 8317 return m_host.ParentGroup.PrimCount + avatarCount;
7694 } 8318 }
7695 8319
@@ -7705,55 +8329,98 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7705 m_host.AddScriptLPS(1); 8329 m_host.AddScriptLPS(1);
7706 UUID objID = UUID.Zero; 8330 UUID objID = UUID.Zero;
7707 LSL_List result = new LSL_List(); 8331 LSL_List result = new LSL_List();
8332
8333 // If the ID is not valid, return null result
7708 if (!UUID.TryParse(obj, out objID)) 8334 if (!UUID.TryParse(obj, out objID))
7709 { 8335 {
7710 result.Add(new LSL_Vector()); 8336 result.Add(new LSL_Vector());
7711 result.Add(new LSL_Vector()); 8337 result.Add(new LSL_Vector());
7712 return result; 8338 return result;
7713 } 8339 }
8340
8341 // Check if this is an attached prim. If so, replace
8342 // the UUID with the avatar UUID and report it's bounding box
8343 SceneObjectPart part = World.GetSceneObjectPart(objID);
8344 if (part != null && part.ParentGroup.IsAttachment)
8345 objID = part.ParentGroup.AttachedAvatar;
8346
8347 // Find out if this is an avatar ID. If so, return it's box
7714 ScenePresence presence = World.GetScenePresence(objID); 8348 ScenePresence presence = World.GetScenePresence(objID);
7715 if (presence != null) 8349 if (presence != null)
7716 { 8350 {
7717 if (presence.ParentID == 0) // not sat on an object 8351 // As per LSL Wiki, there is no difference between sitting
8352 // and standing avatar since server 1.36
8353 LSL_Vector lower;
8354 LSL_Vector upper;
8355 if (presence.Animator.Animations.DefaultAnimation.AnimID
8356 == AnimationSet.Animations.AnimsUUID["SIT_GROUND_CONSTRAINED"])
7718 { 8357 {
7719 LSL_Vector lower; 8358 // This is for ground sitting avatars
7720 LSL_Vector upper; 8359 float height = presence.Appearance.AvatarHeight / 2.66666667f;
7721 if (presence.Animator.Animations.DefaultAnimation.AnimID 8360 lower = new LSL_Vector(-0.3375f, -0.45f, height * -1.0f);
7722 == AnimationSet.Animations.AnimsUUID["SIT_GROUND_CONSTRAINED"]) 8361 upper = new LSL_Vector(0.3375f, 0.45f, 0.0f);
7723 {
7724 // This is for ground sitting avatars
7725 float height = presence.Appearance.AvatarHeight / 2.66666667f;
7726 lower = new LSL_Vector(-0.3375f, -0.45f, height * -1.0f);
7727 upper = new LSL_Vector(0.3375f, 0.45f, 0.0f);
7728 }
7729 else
7730 {
7731 // This is for standing/flying avatars
7732 float height = presence.Appearance.AvatarHeight / 2.0f;
7733 lower = new LSL_Vector(-0.225f, -0.3f, height * -1.0f);
7734 upper = new LSL_Vector(0.225f, 0.3f, height + 0.05f);
7735 }
7736 result.Add(lower);
7737 result.Add(upper);
7738 return result;
7739 } 8362 }
7740 else 8363 else
7741 { 8364 {
7742 // sitting on an object so we need the bounding box of that 8365 // This is for standing/flying avatars
7743 // which should include the avatar so set the UUID to the 8366 float height = presence.Appearance.AvatarHeight / 2.0f;
7744 // UUID of the object the avatar is sat on and allow it to fall through 8367 lower = new LSL_Vector(-0.225f, -0.3f, height * -1.0f);
7745 // to processing an object 8368 upper = new LSL_Vector(0.225f, 0.3f, height + 0.05f);
7746 SceneObjectPart p = World.GetSceneObjectPart(presence.ParentID);
7747 objID = p.UUID;
7748 } 8369 }
8370
8371 // Adjust to the documented error offsets (see LSL Wiki)
8372 lower += new LSL_Vector(0.05f, 0.05f, 0.05f);
8373 upper -= new LSL_Vector(0.05f, 0.05f, 0.05f);
8374
8375 if (lower.x > upper.x)
8376 lower.x = upper.x;
8377 if (lower.y > upper.y)
8378 lower.y = upper.y;
8379 if (lower.z > upper.z)
8380 lower.z = upper.z;
8381
8382 result.Add(lower);
8383 result.Add(upper);
8384 return result;
7749 } 8385 }
7750 SceneObjectPart part = World.GetSceneObjectPart(objID); 8386
8387 part = World.GetSceneObjectPart(objID);
7751 // Currently only works for single prims without a sitting avatar 8388 // Currently only works for single prims without a sitting avatar
7752 if (part != null) 8389 if (part != null)
7753 { 8390 {
7754 Vector3 halfSize = part.Scale / 2.0f; 8391 float minX;
7755 LSL_Vector lower = new LSL_Vector(halfSize.X * -1.0f, halfSize.Y * -1.0f, halfSize.Z * -1.0f); 8392 float maxX;
7756 LSL_Vector upper = new LSL_Vector(halfSize.X, halfSize.Y, halfSize.Z); 8393 float minY;
8394 float maxY;
8395 float minZ;
8396 float maxZ;
8397
8398 // This BBox is in sim coordinates, with the offset being
8399 // a contained point.
8400 Vector3[] offsets = Scene.GetCombinedBoundingBox(new List<SceneObjectGroup> { part.ParentGroup },
8401 out minX, out maxX, out minY, out maxY, out minZ, out maxZ);
8402
8403 minX -= offsets[0].X;
8404 maxX -= offsets[0].X;
8405 minY -= offsets[0].Y;
8406 maxY -= offsets[0].Y;
8407 minZ -= offsets[0].Z;
8408 maxZ -= offsets[0].Z;
8409
8410 LSL_Vector lower;
8411 LSL_Vector upper;
8412
8413 // Adjust to the documented error offsets (see LSL Wiki)
8414 lower = new LSL_Vector(minX + 0.05f, minY + 0.05f, minZ + 0.05f);
8415 upper = new LSL_Vector(maxX - 0.05f, maxY - 0.05f, maxZ - 0.05f);
8416
8417 if (lower.x > upper.x)
8418 lower.x = upper.x;
8419 if (lower.y > upper.y)
8420 lower.y = upper.y;
8421 if (lower.z > upper.z)
8422 lower.z = upper.z;
8423
7757 result.Add(lower); 8424 result.Add(lower);
7758 result.Add(upper); 8425 result.Add(upper);
7759 return result; 8426 return result;
@@ -7833,13 +8500,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7833 LSL_Vector v = new LSL_Vector(part.AbsolutePosition.X, 8500 LSL_Vector v = new LSL_Vector(part.AbsolutePosition.X,
7834 part.AbsolutePosition.Y, 8501 part.AbsolutePosition.Y,
7835 part.AbsolutePosition.Z); 8502 part.AbsolutePosition.Z);
7836 // For some reason, the part.AbsolutePosition.* values do not change if the
7837 // linkset is rotated; they always reflect the child prim's world position
7838 // as though the linkset is unrotated. This is incompatible behavior with SL's
7839 // implementation, so will break scripts imported from there (not to mention it
7840 // makes it more difficult to determine a child prim's actual inworld position).
7841 if (part.ParentID != 0)
7842 v = ((v - llGetRootPosition()) * llGetRootRotation()) + llGetRootPosition();
7843 res.Add(v); 8503 res.Add(v);
7844 break; 8504 break;
7845 8505
@@ -8010,56 +8670,92 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8010 case (int)ScriptBaseClass.PRIM_BUMP_SHINY: 8670 case (int)ScriptBaseClass.PRIM_BUMP_SHINY:
8011 if (remain < 1) 8671 if (remain < 1)
8012 return res; 8672 return res;
8013 8673 face = (int)rules.GetLSLIntegerItem(idx++);
8014 face=(int)rules.GetLSLIntegerItem(idx++);
8015 8674
8016 tex = part.Shape.Textures; 8675 tex = part.Shape.Textures;
8676 int shiny;
8017 if (face == ScriptBaseClass.ALL_SIDES) 8677 if (face == ScriptBaseClass.ALL_SIDES)
8018 { 8678 {
8019 for (face = 0; face < GetNumberOfSides(part); face++) 8679 for (face = 0; face < GetNumberOfSides(part); face++)
8020 { 8680 {
8021 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8681 Shininess shinyness = tex.GetFace((uint)face).Shiny;
8022 // Convert Shininess to PRIM_SHINY_* 8682 if (shinyness == Shininess.High)
8023 res.Add(new LSL_Integer((uint)texface.Shiny >> 6)); 8683 {
8024 // PRIM_BUMP_* 8684 shiny = ScriptBaseClass.PRIM_SHINY_HIGH;
8025 res.Add(new LSL_Integer((int)texface.Bump)); 8685 }
8686 else if (shinyness == Shininess.Medium)
8687 {
8688 shiny = ScriptBaseClass.PRIM_SHINY_MEDIUM;
8689 }
8690 else if (shinyness == Shininess.Low)
8691 {
8692 shiny = ScriptBaseClass.PRIM_SHINY_LOW;
8693 }
8694 else
8695 {
8696 shiny = ScriptBaseClass.PRIM_SHINY_NONE;
8697 }
8698 res.Add(new LSL_Integer(shiny));
8699 res.Add(new LSL_Integer((int)tex.GetFace((uint)face).Bump));
8026 } 8700 }
8027 } 8701 }
8028 else 8702 else
8029 { 8703 {
8030 if (face >= 0 && face < GetNumberOfSides(part)) 8704 Shininess shinyness = tex.GetFace((uint)face).Shiny;
8705 if (shinyness == Shininess.High)
8031 { 8706 {
8032 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8707 shiny = ScriptBaseClass.PRIM_SHINY_HIGH;
8033 // Convert Shininess to PRIM_SHINY_*
8034 res.Add(new LSL_Integer((uint)texface.Shiny >> 6));
8035 // PRIM_BUMP_*
8036 res.Add(new LSL_Integer((int)texface.Bump));
8037 } 8708 }
8709 else if (shinyness == Shininess.Medium)
8710 {
8711 shiny = ScriptBaseClass.PRIM_SHINY_MEDIUM;
8712 }
8713 else if (shinyness == Shininess.Low)
8714 {
8715 shiny = ScriptBaseClass.PRIM_SHINY_LOW;
8716 }
8717 else
8718 {
8719 shiny = ScriptBaseClass.PRIM_SHINY_NONE;
8720 }
8721 res.Add(new LSL_Integer(shiny));
8722 res.Add(new LSL_Integer((int)tex.GetFace((uint)face).Bump));
8038 } 8723 }
8039 break; 8724 break;
8040 8725
8041 case (int)ScriptBaseClass.PRIM_FULLBRIGHT: 8726 case (int)ScriptBaseClass.PRIM_FULLBRIGHT:
8042 if (remain < 1) 8727 if (remain < 1)
8043 return res; 8728 return res;
8044 8729 face = (int)rules.GetLSLIntegerItem(idx++);
8045 face=(int)rules.GetLSLIntegerItem(idx++);
8046 8730
8047 tex = part.Shape.Textures; 8731 tex = part.Shape.Textures;
8732 int fullbright;
8048 if (face == ScriptBaseClass.ALL_SIDES) 8733 if (face == ScriptBaseClass.ALL_SIDES)
8049 { 8734 {
8050 for (face = 0; face < GetNumberOfSides(part); face++) 8735 for (face = 0; face < GetNumberOfSides(part); face++)
8051 { 8736 {
8052 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8737 if (tex.GetFace((uint)face).Fullbright == true)
8053 res.Add(new LSL_Integer(texface.Fullbright ? 1 : 0)); 8738 {
8739 fullbright = ScriptBaseClass.TRUE;
8740 }
8741 else
8742 {
8743 fullbright = ScriptBaseClass.FALSE;
8744 }
8745 res.Add(new LSL_Integer(fullbright));
8054 } 8746 }
8055 } 8747 }
8056 else 8748 else
8057 { 8749 {
8058 if (face >= 0 && face < GetNumberOfSides(part)) 8750 if (tex.GetFace((uint)face).Fullbright == true)
8059 { 8751 {
8060 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8752 fullbright = ScriptBaseClass.TRUE;
8061 res.Add(new LSL_Integer(texface.Fullbright ? 1 : 0)); 8753 }
8754 else
8755 {
8756 fullbright = ScriptBaseClass.FALSE;
8062 } 8757 }
8758 res.Add(new LSL_Integer(fullbright));
8063 } 8759 }
8064 break; 8760 break;
8065 8761
@@ -8081,27 +8777,35 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8081 break; 8777 break;
8082 8778
8083 case (int)ScriptBaseClass.PRIM_TEXGEN: 8779 case (int)ScriptBaseClass.PRIM_TEXGEN:
8780 // (PRIM_TEXGEN_DEFAULT, PRIM_TEXGEN_PLANAR)
8084 if (remain < 1) 8781 if (remain < 1)
8085 return res; 8782 return res;
8086 8783 face = (int)rules.GetLSLIntegerItem(idx++);
8087 face=(int)rules.GetLSLIntegerItem(idx++);
8088 8784
8089 tex = part.Shape.Textures; 8785 tex = part.Shape.Textures;
8090 if (face == ScriptBaseClass.ALL_SIDES) 8786 if (face == ScriptBaseClass.ALL_SIDES)
8091 { 8787 {
8092 for (face = 0; face < GetNumberOfSides(part); face++) 8788 for (face = 0; face < GetNumberOfSides(part); face++)
8093 { 8789 {
8094 MappingType texgen = tex.GetFace((uint)face).TexMapType; 8790 if (tex.GetFace((uint)face).TexMapType == MappingType.Planar)
8095 // Convert MappingType to PRIM_TEXGEN_DEFAULT, PRIM_TEXGEN_PLANAR etc. 8791 {
8096 res.Add(new LSL_Integer((uint)texgen >> 1)); 8792 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_PLANAR));
8793 }
8794 else
8795 {
8796 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_DEFAULT));
8797 }
8097 } 8798 }
8098 } 8799 }
8099 else 8800 else
8100 { 8801 {
8101 if (face >= 0 && face < GetNumberOfSides(part)) 8802 if (tex.GetFace((uint)face).TexMapType == MappingType.Planar)
8803 {
8804 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_PLANAR));
8805 }
8806 else
8102 { 8807 {
8103 MappingType texgen = tex.GetFace((uint)face).TexMapType; 8808 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_DEFAULT));
8104 res.Add(new LSL_Integer((uint)texgen >> 1));
8105 } 8809 }
8106 } 8810 }
8107 break; 8811 break;
@@ -8124,28 +8828,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8124 case (int)ScriptBaseClass.PRIM_GLOW: 8828 case (int)ScriptBaseClass.PRIM_GLOW:
8125 if (remain < 1) 8829 if (remain < 1)
8126 return res; 8830 return res;
8127 8831 face = (int)rules.GetLSLIntegerItem(idx++);
8128 face=(int)rules.GetLSLIntegerItem(idx++);
8129 8832
8130 tex = part.Shape.Textures; 8833 tex = part.Shape.Textures;
8834 float primglow;
8131 if (face == ScriptBaseClass.ALL_SIDES) 8835 if (face == ScriptBaseClass.ALL_SIDES)
8132 { 8836 {
8133 for (face = 0; face < GetNumberOfSides(part); face++) 8837 for (face = 0; face < GetNumberOfSides(part); face++)
8134 { 8838 {
8135 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8839 primglow = tex.GetFace((uint)face).Glow;
8136 res.Add(new LSL_Float(texface.Glow)); 8840 res.Add(new LSL_Float(primglow));
8137 } 8841 }
8138 } 8842 }
8139 else 8843 else
8140 { 8844 {
8141 if (face >= 0 && face < GetNumberOfSides(part)) 8845 primglow = tex.GetFace((uint)face).Glow;
8142 { 8846 res.Add(new LSL_Float(primglow));
8143 Primitive.TextureEntryFace texface = tex.GetFace((uint)face);
8144 res.Add(new LSL_Float(texface.Glow));
8145 }
8146 } 8847 }
8147 break; 8848 break;
8148
8149 case (int)ScriptBaseClass.PRIM_TEXT: 8849 case (int)ScriptBaseClass.PRIM_TEXT:
8150 Color4 textColor = part.GetTextColor(); 8850 Color4 textColor = part.GetTextColor();
8151 res.Add(new LSL_String(part.Text)); 8851 res.Add(new LSL_String(part.Text));
@@ -8757,8 +9457,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8757 // The function returns an ordered list 9457 // The function returns an ordered list
8758 // representing the tokens found in the supplied 9458 // representing the tokens found in the supplied
8759 // sources string. If two successive tokenizers 9459 // sources string. If two successive tokenizers
8760 // are encountered, then a NULL entry is added 9460 // are encountered, then a null-string entry is
8761 // to the list. 9461 // added to the list.
8762 // 9462 //
8763 // It is a precondition that the source and 9463 // It is a precondition that the source and
8764 // toekizer lisst are non-null. If they are null, 9464 // toekizer lisst are non-null. If they are null,
@@ -8766,7 +9466,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8766 // while their lengths are being determined. 9466 // while their lengths are being determined.
8767 // 9467 //
8768 // A small amount of working memoryis required 9468 // A small amount of working memoryis required
8769 // of approximately 8*#tokenizers. 9469 // of approximately 8*#tokenizers + 8*srcstrlen.
8770 // 9470 //
8771 // There are many ways in which this function 9471 // There are many ways in which this function
8772 // can be implemented, this implementation is 9472 // can be implemented, this implementation is
@@ -8782,155 +9482,124 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8782 // and eliminates redundant tokenizers as soon 9482 // and eliminates redundant tokenizers as soon
8783 // as is possible. 9483 // as is possible.
8784 // 9484 //
8785 // The implementation tries to avoid any copying 9485 // The implementation tries to minimize temporary
8786 // of arrays or other objects. 9486 // garbage generation.
8787 // </remarks> 9487 // </remarks>
8788 9488
8789 private LSL_List ParseString(string src, LSL_List separators, LSL_List spacers, bool keepNulls) 9489 public LSL_List llParseStringKeepNulls(string src, LSL_List separators, LSL_List spacers)
8790 { 9490 {
8791 int beginning = 0; 9491 return ParseString2List(src, separators, spacers, true);
8792 int srclen = src.Length; 9492 }
8793 int seplen = separators.Length;
8794 object[] separray = separators.Data;
8795 int spclen = spacers.Length;
8796 object[] spcarray = spacers.Data;
8797 int mlen = seplen+spclen;
8798
8799 int[] offset = new int[mlen+1];
8800 bool[] active = new bool[mlen];
8801
8802 int best;
8803 int j;
8804
8805 // Initial capacity reduces resize cost
8806 9493
8807 LSL_List tokens = new LSL_List(); 9494 private LSL_List ParseString2List(string src, LSL_List separators, LSL_List spacers, bool keepNulls)
9495 {
9496 int srclen = src.Length;
9497 int seplen = separators.Length;
9498 object[] separray = separators.Data;
9499 int spclen = spacers.Length;
9500 object[] spcarray = spacers.Data;
9501 int dellen = 0;
9502 string[] delarray = new string[seplen+spclen];
8808 9503
8809 // All entries are initially valid 9504 int outlen = 0;
9505 string[] outarray = new string[srclen*2+1];
8810 9506
8811 for (int i = 0; i < mlen; i++) 9507 int i, j;
8812 active[i] = true; 9508 string d;
8813 9509
8814 offset[mlen] = srclen; 9510 m_host.AddScriptLPS(1);
8815 9511
8816 while (beginning < srclen) 9512 /*
9513 * Convert separator and spacer lists to C# strings.
9514 * Also filter out null strings so we don't hang.
9515 */
9516 for (i = 0; i < seplen; i ++)
8817 { 9517 {
9518 d = separray[i].ToString();
9519 if (d.Length > 0)
9520 {
9521 delarray[dellen++] = d;
9522 }
9523 }
9524 seplen = dellen;
8818 9525
8819 best = mlen; // as bad as it gets 9526 for (i = 0; i < spclen; i ++)
9527 {
9528 d = spcarray[i].ToString();
9529 if (d.Length > 0)
9530 {
9531 delarray[dellen++] = d;
9532 }
9533 }
8820 9534
8821 // Scan for separators 9535 /*
9536 * Scan through source string from beginning to end.
9537 */
9538 for (i = 0;;)
9539 {
8822 9540
8823 for (j = 0; j < seplen; j++) 9541 /*
9542 * Find earliest delimeter in src starting at i (if any).
9543 */
9544 int earliestDel = -1;
9545 int earliestSrc = srclen;
9546 string earliestStr = null;
9547 for (j = 0; j < dellen; j ++)
8824 { 9548 {
8825 if (separray[j].ToString() == String.Empty) 9549 d = delarray[j];
8826 active[j] = false; 9550 if (d != null)
8827
8828 if (active[j])
8829 { 9551 {
8830 // scan all of the markers 9552 int index = src.IndexOf(d, i);
8831 if ((offset[j] = src.IndexOf(separray[j].ToString(), beginning)) == -1) 9553 if (index < 0)
8832 { 9554 {
8833 // not present at all 9555 delarray[j] = null; // delim nowhere in src, don't check it anymore
8834 active[j] = false;
8835 } 9556 }
8836 else 9557 else if (index < earliestSrc)
8837 { 9558 {
8838 // present and correct 9559 earliestSrc = index; // where delimeter starts in source string
8839 if (offset[j] < offset[best]) 9560 earliestDel = j; // where delimeter is in delarray[]
8840 { 9561 earliestStr = d; // the delimeter string from delarray[]
8841 // closest so far 9562 if (index == i) break; // can't do any better than found at beg of string
8842 best = j;
8843 if (offset[best] == beginning)
8844 break;
8845 }
8846 } 9563 }
8847 } 9564 }
8848 } 9565 }
8849 9566
8850 // Scan for spacers 9567 /*
8851 9568 * Output source string starting at i through start of earliest delimeter.
8852 if (offset[best] != beginning) 9569 */
9570 if (keepNulls || (earliestSrc > i))
8853 { 9571 {
8854 for (j = seplen; (j < mlen) && (offset[best] > beginning); j++) 9572 outarray[outlen++] = src.Substring(i, earliestSrc - i);
8855 {
8856 if (spcarray[j-seplen].ToString() == String.Empty)
8857 active[j] = false;
8858
8859 if (active[j])
8860 {
8861 // scan all of the markers
8862 if ((offset[j] = src.IndexOf(spcarray[j-seplen].ToString(), beginning)) == -1)
8863 {
8864 // not present at all
8865 active[j] = false;
8866 }
8867 else
8868 {
8869 // present and correct
8870 if (offset[j] < offset[best])
8871 {
8872 // closest so far
8873 best = j;
8874 }
8875 }
8876 }
8877 }
8878 } 9573 }
8879 9574
8880 // This is the normal exit from the scanning loop 9575 /*
9576 * If no delimeter found at or after i, we're done scanning.
9577 */
9578 if (earliestDel < 0) break;
8881 9579
8882 if (best == mlen) 9580 /*
9581 * If delimeter was a spacer, output the spacer.
9582 */
9583 if (earliestDel >= seplen)
8883 { 9584 {
8884 // no markers were found on this pass 9585 outarray[outlen++] = earliestStr;
8885 // so we're pretty much done
8886 if ((keepNulls) || ((!keepNulls) && (srclen - beginning) > 0))
8887 tokens.Add(new LSL_String(src.Substring(beginning, srclen - beginning)));
8888 break;
8889 } 9586 }
8890 9587
8891 // Otherwise we just add the newly delimited token 9588 /*
8892 // and recalculate where the search should continue. 9589 * Look at rest of src string following delimeter.
8893 if ((keepNulls) || ((!keepNulls) && (offset[best] - beginning) > 0)) 9590 */
8894 tokens.Add(new LSL_String(src.Substring(beginning,offset[best]-beginning))); 9591 i = earliestSrc + earliestStr.Length;
8895
8896 if (best < seplen)
8897 {
8898 beginning = offset[best] + (separray[best].ToString()).Length;
8899 }
8900 else
8901 {
8902 beginning = offset[best] + (spcarray[best - seplen].ToString()).Length;
8903 string str = spcarray[best - seplen].ToString();
8904 if ((keepNulls) || ((!keepNulls) && (str.Length > 0)))
8905 tokens.Add(new LSL_String(str));
8906 }
8907 } 9592 }
8908 9593
8909 // This an awkward an not very intuitive boundary case. If the 9594 /*
8910 // last substring is a tokenizer, then there is an implied trailing 9595 * Make up an exact-sized output array suitable for an LSL_List object.
8911 // null list entry. Hopefully the single comparison will not be too 9596 */
8912 // arduous. Alternatively the 'break' could be replced with a return 9597 object[] outlist = new object[outlen];
8913 // but that's shabby programming. 9598 for (i = 0; i < outlen; i ++)
8914
8915 if ((beginning == srclen) && (keepNulls))
8916 { 9599 {
8917 if (srclen != 0) 9600 outlist[i] = new LSL_String(outarray[i]);
8918 tokens.Add(new LSL_String(""));
8919 } 9601 }
8920 9602 return new LSL_List(outlist);
8921 return tokens;
8922 }
8923
8924 public LSL_List llParseString2List(string src, LSL_List separators, LSL_List spacers)
8925 {
8926 m_host.AddScriptLPS(1);
8927 return this.ParseString(src, separators, spacers, false);
8928 }
8929
8930 public LSL_List llParseStringKeepNulls(string src, LSL_List separators, LSL_List spacers)
8931 {
8932 m_host.AddScriptLPS(1);
8933 return this.ParseString(src, separators, spacers, true);
8934 } 9603 }
8935 9604
8936 public LSL_Integer llGetObjectPermMask(int mask) 9605 public LSL_Integer llGetObjectPermMask(int mask)
@@ -9007,28 +9676,28 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9007 { 9676 {
9008 m_host.AddScriptLPS(1); 9677 m_host.AddScriptLPS(1);
9009 9678
9010 lock (m_host.TaskInventory) 9679 m_host.TaskInventory.LockItemsForRead(true);
9680 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
9011 { 9681 {
9012 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 9682 if (inv.Value.Name == item)
9013 { 9683 {
9014 if (inv.Value.Name == item) 9684 m_host.TaskInventory.LockItemsForRead(false);
9685 switch (mask)
9015 { 9686 {
9016 switch (mask) 9687 case 0:
9017 { 9688 return (int)inv.Value.BasePermissions;
9018 case 0: 9689 case 1:
9019 return (int)inv.Value.BasePermissions; 9690 return (int)inv.Value.CurrentPermissions;
9020 case 1: 9691 case 2:
9021 return (int)inv.Value.CurrentPermissions; 9692 return (int)inv.Value.GroupPermissions;
9022 case 2: 9693 case 3:
9023 return (int)inv.Value.GroupPermissions; 9694 return (int)inv.Value.EveryonePermissions;
9024 case 3: 9695 case 4:
9025 return (int)inv.Value.EveryonePermissions; 9696 return (int)inv.Value.NextPermissions;
9026 case 4:
9027 return (int)inv.Value.NextPermissions;
9028 }
9029 } 9697 }
9030 } 9698 }
9031 } 9699 }
9700 m_host.TaskInventory.LockItemsForRead(false);
9032 9701
9033 return -1; 9702 return -1;
9034 } 9703 }
@@ -9075,16 +9744,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9075 { 9744 {
9076 m_host.AddScriptLPS(1); 9745 m_host.AddScriptLPS(1);
9077 9746
9078 lock (m_host.TaskInventory) 9747 m_host.TaskInventory.LockItemsForRead(true);
9748 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
9079 { 9749 {
9080 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 9750 if (inv.Value.Name == item)
9081 { 9751 {
9082 if (inv.Value.Name == item) 9752 m_host.TaskInventory.LockItemsForRead(false);
9083 { 9753 return inv.Value.CreatorID.ToString();
9084 return inv.Value.CreatorID.ToString();
9085 }
9086 } 9754 }
9087 } 9755 }
9756 m_host.TaskInventory.LockItemsForRead(false);
9088 9757
9089 llSay(0, "No item name '" + item + "'"); 9758 llSay(0, "No item name '" + item + "'");
9090 9759
@@ -9232,7 +9901,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9232 } 9901 }
9233 9902
9234 /// <summary> 9903 /// <summary>
9235 /// illListReplaceList removes the sub-list defined by the inclusive indices 9904 /// llListReplaceList removes the sub-list defined by the inclusive indices
9236 /// start and end and inserts the src list in its place. The inclusive 9905 /// start and end and inserts the src list in its place. The inclusive
9237 /// nature of the indices means that at least one element must be deleted 9906 /// nature of the indices means that at least one element must be deleted
9238 /// if the indices are within the bounds of the existing list. I.e. 2,2 9907 /// if the indices are within the bounds of the existing list. I.e. 2,2
@@ -9289,16 +9958,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9289 // based upon end. Note that if end exceeds the upper 9958 // based upon end. Note that if end exceeds the upper
9290 // bound in this case, the entire destination list 9959 // bound in this case, the entire destination list
9291 // is removed. 9960 // is removed.
9292 else 9961 else if (start == 0)
9293 { 9962 {
9294 if (end + 1 < dest.Length) 9963 if (end + 1 < dest.Length)
9295 {
9296 return src + dest.GetSublist(end + 1, -1); 9964 return src + dest.GetSublist(end + 1, -1);
9297 }
9298 else 9965 else
9299 {
9300 return src; 9966 return src;
9301 } 9967 }
9968 else // Start < 0
9969 {
9970 if (end + 1 < dest.Length)
9971 return dest.GetSublist(end + 1, -1);
9972 else
9973 return new LSL_List();
9302 } 9974 }
9303 } 9975 }
9304 // Finally, if start > end, we strip away a prefix and 9976 // Finally, if start > end, we strip away a prefix and
@@ -9349,17 +10021,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9349 int width = 0; 10021 int width = 0;
9350 int height = 0; 10022 int height = 0;
9351 10023
9352 ParcelMediaCommandEnum? commandToSend = null; 10024 uint commandToSend = 0;
9353 float time = 0.0f; // default is from start 10025 float time = 0.0f; // default is from start
9354 10026
9355 ScenePresence presence = null; 10027 ScenePresence presence = null;
9356 10028
9357 for (int i = 0; i < commandList.Data.Length; i++) 10029 for (int i = 0; i < commandList.Data.Length; i++)
9358 { 10030 {
9359 ParcelMediaCommandEnum command = (ParcelMediaCommandEnum)commandList.Data[i]; 10031 uint command = (uint)(commandList.GetLSLIntegerItem(i));
9360 switch (command) 10032 switch (command)
9361 { 10033 {
9362 case ParcelMediaCommandEnum.Agent: 10034 case (uint)ParcelMediaCommandEnum.Agent:
9363 // we send only to one agent 10035 // we send only to one agent
9364 if ((i + 1) < commandList.Length) 10036 if ((i + 1) < commandList.Length)
9365 { 10037 {
@@ -9376,25 +10048,25 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9376 } 10048 }
9377 break; 10049 break;
9378 10050
9379 case ParcelMediaCommandEnum.Loop: 10051 case (uint)ParcelMediaCommandEnum.Loop:
9380 loop = 1; 10052 loop = 1;
9381 commandToSend = command; 10053 commandToSend = command;
9382 update = true; //need to send the media update packet to set looping 10054 update = true; //need to send the media update packet to set looping
9383 break; 10055 break;
9384 10056
9385 case ParcelMediaCommandEnum.Play: 10057 case (uint)ParcelMediaCommandEnum.Play:
9386 loop = 0; 10058 loop = 0;
9387 commandToSend = command; 10059 commandToSend = command;
9388 update = true; //need to send the media update packet to make sure it doesn't loop 10060 update = true; //need to send the media update packet to make sure it doesn't loop
9389 break; 10061 break;
9390 10062
9391 case ParcelMediaCommandEnum.Pause: 10063 case (uint)ParcelMediaCommandEnum.Pause:
9392 case ParcelMediaCommandEnum.Stop: 10064 case (uint)ParcelMediaCommandEnum.Stop:
9393 case ParcelMediaCommandEnum.Unload: 10065 case (uint)ParcelMediaCommandEnum.Unload:
9394 commandToSend = command; 10066 commandToSend = command;
9395 break; 10067 break;
9396 10068
9397 case ParcelMediaCommandEnum.Url: 10069 case (uint)ParcelMediaCommandEnum.Url:
9398 if ((i + 1) < commandList.Length) 10070 if ((i + 1) < commandList.Length)
9399 { 10071 {
9400 if (commandList.Data[i + 1] is LSL_String) 10072 if (commandList.Data[i + 1] is LSL_String)
@@ -9407,7 +10079,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9407 } 10079 }
9408 break; 10080 break;
9409 10081
9410 case ParcelMediaCommandEnum.Texture: 10082 case (uint)ParcelMediaCommandEnum.Texture:
9411 if ((i + 1) < commandList.Length) 10083 if ((i + 1) < commandList.Length)
9412 { 10084 {
9413 if (commandList.Data[i + 1] is LSL_String) 10085 if (commandList.Data[i + 1] is LSL_String)
@@ -9420,7 +10092,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9420 } 10092 }
9421 break; 10093 break;
9422 10094
9423 case ParcelMediaCommandEnum.Time: 10095 case (uint)ParcelMediaCommandEnum.Time:
9424 if ((i + 1) < commandList.Length) 10096 if ((i + 1) < commandList.Length)
9425 { 10097 {
9426 if (commandList.Data[i + 1] is LSL_Float) 10098 if (commandList.Data[i + 1] is LSL_Float)
@@ -9432,7 +10104,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9432 } 10104 }
9433 break; 10105 break;
9434 10106
9435 case ParcelMediaCommandEnum.AutoAlign: 10107 case (uint)ParcelMediaCommandEnum.AutoAlign:
9436 if ((i + 1) < commandList.Length) 10108 if ((i + 1) < commandList.Length)
9437 { 10109 {
9438 if (commandList.Data[i + 1] is LSL_Integer) 10110 if (commandList.Data[i + 1] is LSL_Integer)
@@ -9446,7 +10118,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9446 } 10118 }
9447 break; 10119 break;
9448 10120
9449 case ParcelMediaCommandEnum.Type: 10121 case (uint)ParcelMediaCommandEnum.Type:
9450 if ((i + 1) < commandList.Length) 10122 if ((i + 1) < commandList.Length)
9451 { 10123 {
9452 if (commandList.Data[i + 1] is LSL_String) 10124 if (commandList.Data[i + 1] is LSL_String)
@@ -9459,7 +10131,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9459 } 10131 }
9460 break; 10132 break;
9461 10133
9462 case ParcelMediaCommandEnum.Desc: 10134 case (uint)ParcelMediaCommandEnum.Desc:
9463 if ((i + 1) < commandList.Length) 10135 if ((i + 1) < commandList.Length)
9464 { 10136 {
9465 if (commandList.Data[i + 1] is LSL_String) 10137 if (commandList.Data[i + 1] is LSL_String)
@@ -9472,7 +10144,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9472 } 10144 }
9473 break; 10145 break;
9474 10146
9475 case ParcelMediaCommandEnum.Size: 10147 case (uint)ParcelMediaCommandEnum.Size:
9476 if ((i + 2) < commandList.Length) 10148 if ((i + 2) < commandList.Length)
9477 { 10149 {
9478 if (commandList.Data[i + 1] is LSL_Integer) 10150 if (commandList.Data[i + 1] is LSL_Integer)
@@ -9542,7 +10214,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9542 } 10214 }
9543 } 10215 }
9544 10216
9545 if (commandToSend != null) 10217 if (commandToSend != 0)
9546 { 10218 {
9547 // the commandList contained a start/stop/... command, too 10219 // the commandList contained a start/stop/... command, too
9548 if (presence == null) 10220 if (presence == null)
@@ -9579,7 +10251,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9579 10251
9580 if (aList.Data[i] != null) 10252 if (aList.Data[i] != null)
9581 { 10253 {
9582 switch ((ParcelMediaCommandEnum) aList.Data[i]) 10254 switch ((ParcelMediaCommandEnum) Convert.ToInt32(aList.Data[i].ToString()))
9583 { 10255 {
9584 case ParcelMediaCommandEnum.Url: 10256 case ParcelMediaCommandEnum.Url:
9585 list.Add(new LSL_String(World.GetLandData(m_host.AbsolutePosition.X, m_host.AbsolutePosition.Y).MediaURL)); 10257 list.Add(new LSL_String(World.GetLandData(m_host.AbsolutePosition.X, m_host.AbsolutePosition.Y).MediaURL));
@@ -9622,16 +10294,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9622 { 10294 {
9623 m_host.AddScriptLPS(1); 10295 m_host.AddScriptLPS(1);
9624 10296
9625 lock (m_host.TaskInventory) 10297 m_host.TaskInventory.LockItemsForRead(true);
10298 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
9626 { 10299 {
9627 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 10300 if (inv.Value.Name == name)
9628 { 10301 {
9629 if (inv.Value.Name == name) 10302 m_host.TaskInventory.LockItemsForRead(false);
9630 { 10303 return inv.Value.Type;
9631 return inv.Value.Type;
9632 }
9633 } 10304 }
9634 } 10305 }
10306 m_host.TaskInventory.LockItemsForRead(false);
9635 10307
9636 return -1; 10308 return -1;
9637 } 10309 }
@@ -9642,15 +10314,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9642 10314
9643 if (quick_pay_buttons.Data.Length < 4) 10315 if (quick_pay_buttons.Data.Length < 4)
9644 { 10316 {
9645 LSLError("List must have at least 4 elements"); 10317 int x;
9646 return; 10318 for (x=quick_pay_buttons.Data.Length; x<= 4; x++)
10319 {
10320 quick_pay_buttons.Add(ScriptBaseClass.PAY_HIDE);
10321 }
9647 } 10322 }
9648 m_host.ParentGroup.RootPart.PayPrice[0]=price; 10323 int[] nPrice = new int[5];
9649 10324 nPrice[0] = price;
9650 m_host.ParentGroup.RootPart.PayPrice[1]=(LSL_Integer)quick_pay_buttons.Data[0]; 10325 nPrice[1] = quick_pay_buttons.GetLSLIntegerItem(0);
9651 m_host.ParentGroup.RootPart.PayPrice[2]=(LSL_Integer)quick_pay_buttons.Data[1]; 10326 nPrice[2] = quick_pay_buttons.GetLSLIntegerItem(1);
9652 m_host.ParentGroup.RootPart.PayPrice[3]=(LSL_Integer)quick_pay_buttons.Data[2]; 10327 nPrice[3] = quick_pay_buttons.GetLSLIntegerItem(2);
9653 m_host.ParentGroup.RootPart.PayPrice[4]=(LSL_Integer)quick_pay_buttons.Data[3]; 10328 nPrice[4] = quick_pay_buttons.GetLSLIntegerItem(3);
10329 m_host.ParentGroup.RootPart.PayPrice = nPrice;
9654 m_host.ParentGroup.HasGroupChanged = true; 10330 m_host.ParentGroup.HasGroupChanged = true;
9655 } 10331 }
9656 10332
@@ -9662,17 +10338,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9662 if (invItemID == UUID.Zero) 10338 if (invItemID == UUID.Zero)
9663 return new LSL_Vector(); 10339 return new LSL_Vector();
9664 10340
9665 lock (m_host.TaskInventory) 10341 m_host.TaskInventory.LockItemsForRead(true);
10342 if (m_host.TaskInventory[invItemID].PermsGranter == UUID.Zero)
9666 { 10343 {
9667 if (m_host.TaskInventory[invItemID].PermsGranter == UUID.Zero) 10344 m_host.TaskInventory.LockItemsForRead(false);
9668 return new LSL_Vector(); 10345 return new LSL_Vector();
10346 }
9669 10347
9670 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_TRACK_CAMERA) == 0) 10348 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_TRACK_CAMERA) == 0)
9671 { 10349 {
9672 ShoutError("No permissions to track the camera"); 10350 ShoutError("No permissions to track the camera");
9673 return new LSL_Vector(); 10351 m_host.TaskInventory.LockItemsForRead(false);
9674 } 10352 return new LSL_Vector();
9675 } 10353 }
10354 m_host.TaskInventory.LockItemsForRead(false);
9676 10355
9677 ScenePresence presence = World.GetScenePresence(m_host.OwnerID); 10356 ScenePresence presence = World.GetScenePresence(m_host.OwnerID);
9678 if (presence != null) 10357 if (presence != null)
@@ -9690,17 +10369,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9690 if (invItemID == UUID.Zero) 10369 if (invItemID == UUID.Zero)
9691 return new LSL_Rotation(); 10370 return new LSL_Rotation();
9692 10371
9693 lock (m_host.TaskInventory) 10372 m_host.TaskInventory.LockItemsForRead(true);
10373 if (m_host.TaskInventory[invItemID].PermsGranter == UUID.Zero)
9694 { 10374 {
9695 if (m_host.TaskInventory[invItemID].PermsGranter == UUID.Zero) 10375 m_host.TaskInventory.LockItemsForRead(false);
9696 return new LSL_Rotation(); 10376 return new LSL_Rotation();
9697 10377 }
9698 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_TRACK_CAMERA) == 0) 10378 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_TRACK_CAMERA) == 0)
9699 { 10379 {
9700 ShoutError("No permissions to track the camera"); 10380 ShoutError("No permissions to track the camera");
9701 return new LSL_Rotation(); 10381 m_host.TaskInventory.LockItemsForRead(false);
9702 } 10382 return new LSL_Rotation();
9703 } 10383 }
10384 m_host.TaskInventory.LockItemsForRead(false);
9704 10385
9705 ScenePresence presence = World.GetScenePresence(m_host.OwnerID); 10386 ScenePresence presence = World.GetScenePresence(m_host.OwnerID);
9706 if (presence != null) 10387 if (presence != null)
@@ -9762,8 +10443,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9762 { 10443 {
9763 m_host.AddScriptLPS(1); 10444 m_host.AddScriptLPS(1);
9764 DetectParams detectedParams = m_ScriptEngine.GetDetectParams(m_itemID, 0); 10445 DetectParams detectedParams = m_ScriptEngine.GetDetectParams(m_itemID, 0);
9765 if (detectedParams == null) return; // only works on the first detected avatar 10446 if (detectedParams == null)
9766 10447 {
10448 if (m_host.ParentGroup.IsAttachment == true)
10449 {
10450 detectedParams = new DetectParams();
10451 detectedParams.Key = m_host.OwnerID;
10452 }
10453 else
10454 {
10455 return;
10456 }
10457 }
10458
9767 ScenePresence avatar = World.GetScenePresence(detectedParams.Key); 10459 ScenePresence avatar = World.GetScenePresence(detectedParams.Key);
9768 if (avatar != null) 10460 if (avatar != null)
9769 { 10461 {
@@ -9771,6 +10463,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9771 new Vector3((float)pos.x, (float)pos.y, (float)pos.z), 10463 new Vector3((float)pos.x, (float)pos.y, (float)pos.z),
9772 new Vector3((float)lookAt.x, (float)lookAt.y, (float)lookAt.z)); 10464 new Vector3((float)lookAt.x, (float)lookAt.y, (float)lookAt.z));
9773 } 10465 }
10466
9774 ScriptSleep(1000); 10467 ScriptSleep(1000);
9775 } 10468 }
9776 10469
@@ -9882,14 +10575,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9882 if (objectID == UUID.Zero) return; 10575 if (objectID == UUID.Zero) return;
9883 10576
9884 UUID agentID; 10577 UUID agentID;
9885 lock (m_host.TaskInventory) 10578 m_host.TaskInventory.LockItemsForRead(true);
9886 { 10579 // we need the permission first, to know which avatar we want to set the camera for
9887 // we need the permission first, to know which avatar we want to set the camera for 10580 agentID = m_host.TaskInventory[invItemID].PermsGranter;
9888 agentID = m_host.TaskInventory[invItemID].PermsGranter;
9889 10581
9890 if (agentID == UUID.Zero) return; 10582 if (agentID == UUID.Zero)
9891 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_CONTROL_CAMERA) == 0) return; 10583 {
10584 m_host.TaskInventory.LockItemsForRead(false);
10585 return;
10586 }
10587 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_CONTROL_CAMERA) == 0)
10588 {
10589 m_host.TaskInventory.LockItemsForRead(false);
10590 return;
9892 } 10591 }
10592 m_host.TaskInventory.LockItemsForRead(false);
9893 10593
9894 ScenePresence presence = World.GetScenePresence(agentID); 10594 ScenePresence presence = World.GetScenePresence(agentID);
9895 10595
@@ -9898,12 +10598,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9898 10598
9899 SortedDictionary<int, float> parameters = new SortedDictionary<int, float>(); 10599 SortedDictionary<int, float> parameters = new SortedDictionary<int, float>();
9900 object[] data = rules.Data; 10600 object[] data = rules.Data;
9901 for (int i = 0; i < data.Length; ++i) { 10601 for (int i = 0; i < data.Length; ++i)
10602 {
9902 int type = Convert.ToInt32(data[i++].ToString()); 10603 int type = Convert.ToInt32(data[i++].ToString());
9903 if (i >= data.Length) break; // odd number of entries => ignore the last 10604 if (i >= data.Length) break; // odd number of entries => ignore the last
9904 10605
9905 // some special cases: Vector parameters are split into 3 float parameters (with type+1, type+2, type+3) 10606 // some special cases: Vector parameters are split into 3 float parameters (with type+1, type+2, type+3)
9906 switch (type) { 10607 switch (type)
10608 {
9907 case ScriptBaseClass.CAMERA_FOCUS: 10609 case ScriptBaseClass.CAMERA_FOCUS:
9908 case ScriptBaseClass.CAMERA_FOCUS_OFFSET: 10610 case ScriptBaseClass.CAMERA_FOCUS_OFFSET:
9909 case ScriptBaseClass.CAMERA_POSITION: 10611 case ScriptBaseClass.CAMERA_POSITION:
@@ -9939,12 +10641,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9939 10641
9940 // we need the permission first, to know which avatar we want to clear the camera for 10642 // we need the permission first, to know which avatar we want to clear the camera for
9941 UUID agentID; 10643 UUID agentID;
9942 lock (m_host.TaskInventory) 10644 m_host.TaskInventory.LockItemsForRead(true);
10645 agentID = m_host.TaskInventory[invItemID].PermsGranter;
10646 if (agentID == UUID.Zero)
9943 { 10647 {
9944 agentID = m_host.TaskInventory[invItemID].PermsGranter; 10648 m_host.TaskInventory.LockItemsForRead(false);
9945 if (agentID == UUID.Zero) return; 10649 return;
9946 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_CONTROL_CAMERA) == 0) return;
9947 } 10650 }
10651 if ((m_host.TaskInventory[invItemID].PermsMask & ScriptBaseClass.PERMISSION_CONTROL_CAMERA) == 0)
10652 {
10653 m_host.TaskInventory.LockItemsForRead(false);
10654 return;
10655 }
10656 m_host.TaskInventory.LockItemsForRead(false);
9948 10657
9949 ScenePresence presence = World.GetScenePresence(agentID); 10658 ScenePresence presence = World.GetScenePresence(agentID);
9950 10659
@@ -10011,19 +10720,65 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10011 public LSL_String llXorBase64StringsCorrect(string str1, string str2) 10720 public LSL_String llXorBase64StringsCorrect(string str1, string str2)
10012 { 10721 {
10013 m_host.AddScriptLPS(1); 10722 m_host.AddScriptLPS(1);
10014 string ret = String.Empty; 10723
10015 string src1 = llBase64ToString(str1); 10724 if (str1 == String.Empty)
10016 string src2 = llBase64ToString(str2); 10725 return String.Empty;
10017 int c = 0; 10726 if (str2 == String.Empty)
10018 for (int i = 0; i < src1.Length; i++) 10727 return str1;
10728
10729 int len = str2.Length;
10730 if ((len % 4) != 0) // LL is EVIL!!!!
10019 { 10731 {
10020 ret += (char) (src1[i] ^ src2[c]); 10732 while (str2.EndsWith("="))
10733 str2 = str2.Substring(0, str2.Length - 1);
10021 10734
10022 c++; 10735 len = str2.Length;
10023 if (c >= src2.Length) 10736 int mod = len % 4;
10024 c = 0; 10737
10738 if (mod == 1)
10739 str2 = str2.Substring(0, str2.Length - 1);
10740 else if (mod == 2)
10741 str2 += "==";
10742 else if (mod == 3)
10743 str2 += "=";
10025 } 10744 }
10026 return llStringToBase64(ret); 10745
10746 byte[] data1;
10747 byte[] data2;
10748 try
10749 {
10750 data1 = Convert.FromBase64String(str1);
10751 data2 = Convert.FromBase64String(str2);
10752 }
10753 catch (Exception)
10754 {
10755 return new LSL_String(String.Empty);
10756 }
10757
10758 byte[] d2 = new Byte[data1.Length];
10759 int pos = 0;
10760
10761 if (data1.Length <= data2.Length)
10762 {
10763 Array.Copy(data2, 0, d2, 0, data1.Length);
10764 }
10765 else
10766 {
10767 while (pos < data1.Length)
10768 {
10769 len = data1.Length - pos;
10770 if (len > data2.Length)
10771 len = data2.Length;
10772
10773 Array.Copy(data2, 0, d2, pos, len);
10774 pos += len;
10775 }
10776 }
10777
10778 for (pos = 0 ; pos < data1.Length ; pos++ )
10779 data1[pos] ^= d2[pos];
10780
10781 return Convert.ToBase64String(data1);
10027 } 10782 }
10028 10783
10029 public LSL_String llHTTPRequest(string url, LSL_List parameters, string body) 10784 public LSL_String llHTTPRequest(string url, LSL_List parameters, string body)
@@ -10080,12 +10835,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10080 Regex r = new Regex(authregex); 10835 Regex r = new Regex(authregex);
10081 int[] gnums = r.GetGroupNumbers(); 10836 int[] gnums = r.GetGroupNumbers();
10082 Match m = r.Match(url); 10837 Match m = r.Match(url);
10083 if (m.Success) { 10838 if (m.Success)
10084 for (int i = 1; i < gnums.Length; i++) { 10839 {
10840 for (int i = 1; i < gnums.Length; i++)
10841 {
10085 //System.Text.RegularExpressions.Group g = m.Groups[gnums[i]]; 10842 //System.Text.RegularExpressions.Group g = m.Groups[gnums[i]];
10086 //CaptureCollection cc = g.Captures; 10843 //CaptureCollection cc = g.Captures;
10087 } 10844 }
10088 if (m.Groups.Count == 5) { 10845 if (m.Groups.Count == 5)
10846 {
10089 httpHeaders["Authorization"] = String.Format("Basic {0}", Convert.ToBase64String(System.Text.ASCIIEncoding.ASCII.GetBytes(m.Groups[2].ToString() + ":" + m.Groups[3].ToString()))); 10847 httpHeaders["Authorization"] = String.Format("Basic {0}", Convert.ToBase64String(System.Text.ASCIIEncoding.ASCII.GetBytes(m.Groups[2].ToString() + ":" + m.Groups[3].ToString())));
10090 url = m.Groups[1].ToString() + m.Groups[4].ToString(); 10848 url = m.Groups[1].ToString() + m.Groups[4].ToString();
10091 } 10849 }
@@ -10371,15 +11129,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10371 11129
10372 internal UUID ScriptByName(string name) 11130 internal UUID ScriptByName(string name)
10373 { 11131 {
10374 lock (m_host.TaskInventory) 11132 m_host.TaskInventory.LockItemsForRead(true);
11133
11134 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
10375 { 11135 {
10376 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 11136 if (item.Type == 10 && item.Name == name)
10377 { 11137 {
10378 if (item.Type == 10 && item.Name == name) 11138 m_host.TaskInventory.LockItemsForRead(false);
10379 return item.ItemID; 11139 return item.ItemID;
10380 } 11140 }
10381 } 11141 }
10382 11142
11143 m_host.TaskInventory.LockItemsForRead(false);
11144
10383 return UUID.Zero; 11145 return UUID.Zero;
10384 } 11146 }
10385 11147
@@ -10420,6 +11182,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10420 { 11182 {
10421 m_host.AddScriptLPS(1); 11183 m_host.AddScriptLPS(1);
10422 11184
11185 //Clone is thread safe
10423 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 11186 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
10424 11187
10425 UUID assetID = UUID.Zero; 11188 UUID assetID = UUID.Zero;
@@ -10482,6 +11245,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10482 { 11245 {
10483 m_host.AddScriptLPS(1); 11246 m_host.AddScriptLPS(1);
10484 11247
11248 //Clone is thread safe
10485 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 11249 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
10486 11250
10487 UUID assetID = UUID.Zero; 11251 UUID assetID = UUID.Zero;
@@ -10562,15 +11326,13 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10562 return GetLinkPrimitiveParams(obj, rules); 11326 return GetLinkPrimitiveParams(obj, rules);
10563 } 11327 }
10564 11328
10565 public void print(string str) 11329 public LSL_Integer llGetLinkNumberOfSides(LSL_Integer link)
10566 { 11330 {
10567 // yes, this is a real LSL function. See: http://wiki.secondlife.com/wiki/Print 11331 List<SceneObjectPart> parts = GetLinkParts(link);
10568 IOSSL_Api ossl = (IOSSL_Api)m_ScriptEngine.GetApi(m_itemID, "OSSL"); 11332 if (parts.Count < 1)
10569 if (ossl != null) 11333 return 0;
10570 { 11334
10571 ossl.CheckThreatLevel(ThreatLevel.High, "print"); 11335 return GetNumberOfSides(parts[0]);
10572 m_log.Info("LSL print():" + str);
10573 }
10574 } 11336 }
10575 11337
10576 private string Name2Username(string name) 11338 private string Name2Username(string name)
@@ -10616,153 +11378,397 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10616 return rq.ToString(); 11378 return rq.ToString();
10617 } 11379 }
10618 11380
11381 private void SayShoutTimerElapsed(Object sender, ElapsedEventArgs args)
11382 {
11383 m_SayShoutCount = 0;
11384 }
11385
11386 private struct Tri
11387 {
11388 public Vector3 p1;
11389 public Vector3 p2;
11390 public Vector3 p3;
11391 }
11392
11393 private bool InBoundingBox(ScenePresence avatar, Vector3 point)
11394 {
11395 float height = avatar.Appearance.AvatarHeight;
11396 Vector3 b1 = avatar.AbsolutePosition + new Vector3(-0.22f, -0.22f, -height/2);
11397 Vector3 b2 = avatar.AbsolutePosition + new Vector3(0.22f, 0.22f, height/2);
11398
11399 if (point.X > b1.X && point.X < b2.X &&
11400 point.Y > b1.Y && point.Y < b2.Y &&
11401 point.Z > b1.Z && point.Z < b2.Z)
11402 return true;
11403 return false;
11404 }
11405
11406 private ContactResult[] AvatarIntersection(Vector3 rayStart, Vector3 rayEnd)
11407 {
11408 List<ContactResult> contacts = new List<ContactResult>();
11409
11410 Vector3 ab = rayEnd - rayStart;
11411
11412 World.ForEachScenePresence(delegate(ScenePresence sp)
11413 {
11414 Vector3 ac = sp.AbsolutePosition - rayStart;
11415 Vector3 bc = sp.AbsolutePosition - rayEnd;
11416
11417 double d = Math.Abs(Vector3.Mag(Vector3.Cross(ab, ac)) / Vector3.Distance(rayStart, rayEnd));
11418
11419 if (d > 1.5)
11420 return;
11421
11422 double d2 = Vector3.Dot(Vector3.Negate(ab), ac);
11423
11424 if (d2 > 0)
11425 return;
11426
11427 double dp = Math.Sqrt(Vector3.Mag(ac) * Vector3.Mag(ac) - d * d);
11428 Vector3 p = rayStart + Vector3.Divide(Vector3.Multiply(ab, (float)dp), (float)Vector3.Mag(ab));
11429
11430 if (!InBoundingBox(sp, p))
11431 return;
11432
11433 ContactResult result = new ContactResult ();
11434 result.ConsumerID = sp.LocalId;
11435 result.Depth = Vector3.Distance(rayStart, p);
11436 result.Normal = Vector3.Zero;
11437 result.Pos = p;
11438
11439 contacts.Add(result);
11440 });
11441
11442 return contacts.ToArray();
11443 }
11444
11445 private ContactResult[] ObjectIntersection(Vector3 rayStart, Vector3 rayEnd, bool includePhysical, bool includeNonPhysical, bool includePhantom)
11446 {
11447 Ray ray = new Ray(rayStart, Vector3.Normalize(rayEnd - rayStart));
11448 List<ContactResult> contacts = new List<ContactResult>();
11449
11450 Vector3 ab = rayEnd - rayStart;
11451
11452 World.ForEachSOG(delegate(SceneObjectGroup group)
11453 {
11454 if (m_host.ParentGroup == group)
11455 return;
11456
11457 if (group.IsAttachment)
11458 return;
11459
11460 if (group.RootPart.PhysActor == null)
11461 {
11462 if (!includePhantom)
11463 return;
11464 }
11465 else
11466 {
11467 if (group.RootPart.PhysActor.IsPhysical)
11468 {
11469 if (!includePhysical)
11470 return;
11471 }
11472 else
11473 {
11474 if (!includeNonPhysical)
11475 return;
11476 }
11477 }
11478
11479 // Find the radius ouside of which we don't even need to hit test
11480 float minX;
11481 float maxX;
11482 float minY;
11483 float maxY;
11484 float minZ;
11485 float maxZ;
11486
11487 float radius = 0.0f;
11488
11489 group.GetAxisAlignedBoundingBoxRaw(out minX, out maxX, out minY, out maxY, out minZ, out maxZ);
11490
11491 if (Math.Abs(minX) > radius)
11492 radius = Math.Abs(minX);
11493 if (Math.Abs(minY) > radius)
11494 radius = Math.Abs(minY);
11495 if (Math.Abs(minZ) > radius)
11496 radius = Math.Abs(minZ);
11497 if (Math.Abs(maxX) > radius)
11498 radius = Math.Abs(maxX);
11499 if (Math.Abs(maxY) > radius)
11500 radius = Math.Abs(maxY);
11501 if (Math.Abs(maxZ) > radius)
11502 radius = Math.Abs(maxZ);
11503
11504 Vector3 ac = group.AbsolutePosition - rayStart;
11505 Vector3 bc = group.AbsolutePosition - rayEnd;
11506
11507 double d = Math.Abs(Vector3.Mag(Vector3.Cross(ab, ac)) / Vector3.Distance(rayStart, rayEnd));
11508
11509 // Too far off ray, don't bother
11510 if (d > radius)
11511 return;
11512
11513 // Behind ray, drop
11514 double d2 = Vector3.Dot(Vector3.Negate(ab), ac);
11515 if (d2 > 0)
11516 return;
11517
11518 EntityIntersection intersection = group.TestIntersection(ray, true, false);
11519 // Miss.
11520 if (!intersection.HitTF)
11521 return;
11522
11523 ContactResult result = new ContactResult ();
11524 result.ConsumerID = group.LocalId;
11525 result.Depth = intersection.distance;
11526 result.Normal = intersection.normal;
11527 result.Pos = intersection.ipoint;
11528
11529 contacts.Add(result);
11530 });
11531
11532 return contacts.ToArray();
11533 }
11534
11535 private ContactResult? GroundIntersection(Vector3 rayStart, Vector3 rayEnd)
11536 {
11537 double[,] heightfield = World.Heightmap.GetDoubles();
11538 List<ContactResult> contacts = new List<ContactResult>();
11539
11540 double min = 2048.0;
11541 double max = 0.0;
11542
11543 // Find the min and max of the heightfield
11544 for (int x = 0 ; x < World.Heightmap.Width ; x++)
11545 {
11546 for (int y = 0 ; y < World.Heightmap.Height ; y++)
11547 {
11548 if (heightfield[x, y] > max)
11549 max = heightfield[x, y];
11550 if (heightfield[x, y] < min)
11551 min = heightfield[x, y];
11552 }
11553 }
11554
11555
11556 // A ray extends past rayEnd, but doesn't go back before
11557 // rayStart. If the start is above the highest point of the ground
11558 // and the ray goes up, we can't hit the ground. Ever.
11559 if (rayStart.Z > max && rayEnd.Z >= rayStart.Z)
11560 return null;
11561
11562 // Same for going down
11563 if (rayStart.Z < min && rayEnd.Z <= rayStart.Z)
11564 return null;
11565
11566 List<Tri> trilist = new List<Tri>();
11567
11568 // Create our triangle list
11569 for (int x = 1 ; x < World.Heightmap.Width ; x++)
11570 {
11571 for (int y = 1 ; y < World.Heightmap.Height ; y++)
11572 {
11573 Tri t1 = new Tri();
11574 Tri t2 = new Tri();
11575
11576 Vector3 p1 = new Vector3(x-1, y-1, (float)heightfield[x-1, y-1]);
11577 Vector3 p2 = new Vector3(x, y-1, (float)heightfield[x, y-1]);
11578 Vector3 p3 = new Vector3(x, y, (float)heightfield[x, y]);
11579 Vector3 p4 = new Vector3(x-1, y, (float)heightfield[x-1, y]);
11580
11581 t1.p1 = p1;
11582 t1.p2 = p2;
11583 t1.p3 = p3;
11584
11585 t2.p1 = p3;
11586 t2.p2 = p4;
11587 t2.p3 = p1;
11588
11589 trilist.Add(t1);
11590 trilist.Add(t2);
11591 }
11592 }
11593
11594 // Ray direction
11595 Vector3 rayDirection = rayEnd - rayStart;
11596
11597 foreach (Tri t in trilist)
11598 {
11599 // Compute triangle plane normal and edges
11600 Vector3 u = t.p2 - t.p1;
11601 Vector3 v = t.p3 - t.p1;
11602 Vector3 n = Vector3.Cross(u, v);
11603
11604 if (n == Vector3.Zero)
11605 continue;
11606
11607 Vector3 w0 = rayStart - t.p1;
11608 double a = -Vector3.Dot(n, w0);
11609 double b = Vector3.Dot(n, rayDirection);
11610
11611 // Not intersecting the plane, or in plane (same thing)
11612 // Ignoring this MAY cause the ground to not be detected
11613 // sometimes
11614 if (Math.Abs(b) < 0.000001)
11615 continue;
11616
11617 double r = a / b;
11618
11619 // ray points away from plane
11620 if (r < 0.0)
11621 continue;
11622
11623 Vector3 ip = rayStart + Vector3.Multiply(rayDirection, (float)r);
11624
11625 float uu = Vector3.Dot(u, u);
11626 float uv = Vector3.Dot(u, v);
11627 float vv = Vector3.Dot(v, v);
11628 Vector3 w = ip - t.p1;
11629 float wu = Vector3.Dot(w, u);
11630 float wv = Vector3.Dot(w, v);
11631 float d = uv * uv - uu * vv;
11632
11633 float cs = (uv * wv - vv * wu) / d;
11634 if (cs < 0 || cs > 1.0)
11635 continue;
11636 float ct = (uv * wu - uu * wv) / d;
11637 if (ct < 0 || (cs + ct) > 1.0)
11638 continue;
11639
11640 // Add contact point
11641 ContactResult result = new ContactResult ();
11642 result.ConsumerID = 0;
11643 result.Depth = Vector3.Distance(rayStart, ip);
11644 result.Normal = n;
11645 result.Pos = ip;
11646
11647 contacts.Add(result);
11648 }
11649
11650 if (contacts.Count == 0)
11651 return null;
11652
11653 contacts.Sort(delegate(ContactResult a, ContactResult b)
11654 {
11655 return (int)(a.Depth - b.Depth);
11656 });
11657
11658 return contacts[0];
11659 }
11660
10619 public LSL_List llCastRay(LSL_Vector start, LSL_Vector end, LSL_List options) 11661 public LSL_List llCastRay(LSL_Vector start, LSL_Vector end, LSL_List options)
10620 { 11662 {
11663 LSL_List list = new LSL_List();
11664
10621 m_host.AddScriptLPS(1); 11665 m_host.AddScriptLPS(1);
10622 11666
10623 Vector3 dir = new Vector3((float)(end-start).x, (float)(end-start).y, (float)(end-start).z); 11667 Vector3 rayStart = new Vector3((float)start.x, (float)start.y, (float)start.z);
10624 Vector3 startvector = new Vector3((float)start.x, (float)start.y, (float)start.z); 11668 Vector3 rayEnd = new Vector3((float)end.x, (float)end.y, (float)end.z);
10625 Vector3 endvector = new Vector3((float)end.x, (float)end.y, (float)end.z); 11669 Vector3 dir = rayEnd - rayStart;
10626 11670
10627 int count = 0; 11671 float dist = Vector3.Mag(dir);
10628// int detectPhantom = 0; 11672
11673 int count = 1;
11674 bool detectPhantom = false;
10629 int dataFlags = 0; 11675 int dataFlags = 0;
10630 int rejectTypes = 0; 11676 int rejectTypes = 0;
10631 11677
10632 for (int i = 0; i < options.Length; i += 2) 11678 for (int i = 0; i < options.Length; i += 2)
10633 { 11679 {
10634 if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_MAX_HITS) 11680 if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_MAX_HITS)
10635 {
10636 count = options.GetLSLIntegerItem(i + 1); 11681 count = options.GetLSLIntegerItem(i + 1);
10637 } 11682 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DETECT_PHANTOM)
10638// else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DETECT_PHANTOM) 11683 detectPhantom = (options.GetLSLIntegerItem(i + 1) > 0);
10639// {
10640// detectPhantom = options.GetLSLIntegerItem(i + 1);
10641// }
10642 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DATA_FLAGS) 11684 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DATA_FLAGS)
10643 {
10644 dataFlags = options.GetLSLIntegerItem(i + 1); 11685 dataFlags = options.GetLSLIntegerItem(i + 1);
10645 }
10646 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_REJECT_TYPES) 11686 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_REJECT_TYPES)
10647 {
10648 rejectTypes = options.GetLSLIntegerItem(i + 1); 11687 rejectTypes = options.GetLSLIntegerItem(i + 1);
10649 }
10650 } 11688 }
10651 11689
10652 LSL_List list = new LSL_List(); 11690 if (count > 16)
10653 List<ContactResult> results = World.PhysicsScene.RaycastWorld(startvector, dir, dir.Length(), count); 11691 count = 16;
10654
10655 double distance = Util.GetDistanceTo(startvector, endvector);
10656 11692
10657 if (distance == 0) 11693 List<ContactResult> results = new List<ContactResult>();
10658 distance = 0.001;
10659
10660 Vector3 posToCheck = startvector;
10661 ITerrainChannel channel = World.RequestModuleInterface<ITerrainChannel>();
10662 11694
10663 bool checkTerrain = !((rejectTypes & ScriptBaseClass.RC_REJECT_LAND) == ScriptBaseClass.RC_REJECT_LAND); 11695 bool checkTerrain = !((rejectTypes & ScriptBaseClass.RC_REJECT_LAND) == ScriptBaseClass.RC_REJECT_LAND);
10664 bool checkAgents = !((rejectTypes & ScriptBaseClass.RC_REJECT_AGENTS) == ScriptBaseClass.RC_REJECT_AGENTS); 11696 bool checkAgents = !((rejectTypes & ScriptBaseClass.RC_REJECT_AGENTS) == ScriptBaseClass.RC_REJECT_AGENTS);
10665 bool checkNonPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_NONPHYSICAL) == ScriptBaseClass.RC_REJECT_NONPHYSICAL); 11697 bool checkNonPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_NONPHYSICAL) == ScriptBaseClass.RC_REJECT_NONPHYSICAL);
10666 bool checkPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_PHYSICAL) == ScriptBaseClass.RC_REJECT_PHYSICAL); 11698 bool checkPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_PHYSICAL) == ScriptBaseClass.RC_REJECT_PHYSICAL);
10667 11699
10668 for (float i = 0; i <= distance; i += 0.1f) 11700 if (checkTerrain)
10669 { 11701 {
10670 posToCheck = startvector + (dir * (i / (float)distance)); 11702 ContactResult? groundContact = GroundIntersection(rayStart, rayEnd);
11703 if (groundContact != null)
11704 results.Add((ContactResult)groundContact);
11705 }
10671 11706
10672 if (checkTerrain && channel[(int)(posToCheck.X + startvector.X), (int)(posToCheck.Y + startvector.Y)] < posToCheck.Z) 11707 if (checkAgents)
10673 { 11708 {
10674 ContactResult result = new ContactResult(); 11709 ContactResult[] agentHits = AvatarIntersection(rayStart, rayEnd);
10675 result.ConsumerID = 0; 11710 foreach (ContactResult r in agentHits)
10676 result.Depth = 0; 11711 results.Add(r);
10677 result.Normal = Vector3.Zero; 11712 }
10678 result.Pos = posToCheck;
10679 results.Add(result);
10680 checkTerrain = false;
10681 }
10682 11713
10683 if (checkAgents) 11714 if (checkPhysical || checkNonPhysical)
10684 { 11715 {
10685 World.ForEachRootScenePresence(delegate(ScenePresence sp) 11716 ContactResult[] objectHits = ObjectIntersection(rayStart, rayEnd, checkPhysical, checkNonPhysical, detectPhantom);
10686 { 11717 foreach (ContactResult r in objectHits)
10687 if (sp.AbsolutePosition.ApproxEquals(posToCheck, sp.PhysicsActor.Size.X)) 11718 results.Add(r);
10688 {
10689 ContactResult result = new ContactResult ();
10690 result.ConsumerID = sp.LocalId;
10691 result.Depth = 0;
10692 result.Normal = Vector3.Zero;
10693 result.Pos = posToCheck;
10694 results.Add(result);
10695 }
10696 });
10697 }
10698 } 11719 }
10699 11720
10700 int refcount = 0; 11721 results.Sort(delegate(ContactResult a, ContactResult b)
11722 {
11723 return (int)(a.Depth - b.Depth);
11724 });
11725
11726 int values = 0;
10701 foreach (ContactResult result in results) 11727 foreach (ContactResult result in results)
10702 { 11728 {
10703 if ((rejectTypes & ScriptBaseClass.RC_REJECT_LAND) 11729 if (result.Depth > dist)
10704 == ScriptBaseClass.RC_REJECT_LAND && result.ConsumerID == 0)
10705 continue; 11730 continue;
10706 11731
10707 ISceneEntity entity = World.GetSceneObjectPart(result.ConsumerID); 11732 UUID itemID = UUID.Zero;
11733 int linkNum = 0;
10708 11734
10709 if (entity == null && (rejectTypes & ScriptBaseClass.RC_REJECT_AGENTS) != ScriptBaseClass.RC_REJECT_AGENTS) 11735 SceneObjectPart part = World.GetSceneObjectPart(result.ConsumerID);
10710 entity = World.GetScenePresence(result.ConsumerID); //Only check if we should be looking for agents 11736 // It's a prim!
10711 11737 if (part != null)
10712 if (entity == null)
10713 { 11738 {
10714 list.Add(UUID.Zero); 11739 if ((dataFlags & ScriptBaseClass.RC_GET_ROOT_KEY) == ScriptBaseClass.RC_GET_ROOT_KEY)
10715 11740 itemID = part.ParentGroup.UUID;
10716 if ((dataFlags & ScriptBaseClass.RC_GET_LINK_NUM) == ScriptBaseClass.RC_GET_LINK_NUM) 11741 else
10717 list.Add(0); 11742 itemID = part.UUID;
10718
10719 list.Add(result.Pos);
10720
10721 if ((dataFlags & ScriptBaseClass.RC_GET_NORMAL) == ScriptBaseClass.RC_GET_NORMAL)
10722 list.Add(result.Normal);
10723 11743
10724 continue; //Can't find it, so add UUID.Zero 11744 linkNum = part.LinkNum;
10725 } 11745 }
10726 11746 else
10727 /*if (detectPhantom == 0 && intersection.obj is ISceneChildEntity &&
10728 ((ISceneChildEntity)intersection.obj).PhysActor == null)
10729 continue;*/ //Can't do this ATM, physics engine knows only of non phantom objects
10730
10731 if (entity is SceneObjectPart)
10732 { 11747 {
10733 if (((SceneObjectPart)entity).PhysActor != null && ((SceneObjectPart)entity).PhysActor.IsPhysical) 11748 ScenePresence sp = World.GetScenePresence(result.ConsumerID);
10734 { 11749 /// It it a boy? a girl?
10735 if (!checkPhysical) 11750 if (sp != null)
10736 continue; 11751 itemID = sp.UUID;
10737 }
10738 else
10739 {
10740 if (!checkNonPhysical)
10741 continue;
10742 }
10743 } 11752 }
10744 11753
10745 refcount++; 11754 list.Add(new LSL_String(itemID.ToString()));
10746 if ((dataFlags & ScriptBaseClass.RC_GET_ROOT_KEY) == ScriptBaseClass.RC_GET_ROOT_KEY && entity is SceneObjectPart) 11755 list.Add(new LSL_String(result.Pos.ToString()));
10747 list.Add(((SceneObjectPart)entity).ParentGroup.UUID);
10748 else
10749 list.Add(entity.UUID);
10750 11756
10751 if ((dataFlags & ScriptBaseClass.RC_GET_LINK_NUM) == ScriptBaseClass.RC_GET_LINK_NUM) 11757 if ((dataFlags & ScriptBaseClass.RC_GET_LINK_NUM) == ScriptBaseClass.RC_GET_LINK_NUM)
10752 { 11758 list.Add(new LSL_Integer(linkNum));
10753 if (entity is SceneObjectPart)
10754 list.Add(((SceneObjectPart)entity).LinkNum);
10755 else
10756 list.Add(0);
10757 }
10758 11759
10759 list.Add(result.Pos);
10760 11760
10761 if ((dataFlags & ScriptBaseClass.RC_GET_NORMAL) == ScriptBaseClass.RC_GET_NORMAL) 11761 if ((dataFlags & ScriptBaseClass.RC_GET_NORMAL) == ScriptBaseClass.RC_GET_NORMAL)
10762 list.Add(result.Normal); 11762 list.Add(new LSL_Vector(result.Normal.X, result.Normal.Y, result.Normal.Z));
11763
11764 values++;
11765 count--;
11766
11767 if (count == 0)
11768 break;
10763 } 11769 }
10764 11770
10765 list.Add(refcount); //The status code, either the # of contacts, RCERR_SIM_PERF_LOW, or RCERR_CAST_TIME_EXCEEDED 11771 list.Add(new LSL_Integer(values));
10766 11772
10767 return list; 11773 return list;
10768 } 11774 }
@@ -10802,7 +11808,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10802 case ScriptBaseClass.ESTATE_ACCESS_ALLOWED_AGENT_ADD: 11808 case ScriptBaseClass.ESTATE_ACCESS_ALLOWED_AGENT_ADD:
10803 if (!isAccount) return 0; 11809 if (!isAccount) return 0;
10804 if (estate.HasAccess(id)) return 1; 11810 if (estate.HasAccess(id)) return 1;
10805 if (estate.IsBanned(id)) 11811 if (estate.IsBanned(id, World.GetUserFlags(id)))
10806 estate.RemoveBan(id); 11812 estate.RemoveBan(id);
10807 estate.AddEstateUser(id); 11813 estate.AddEstateUser(id);
10808 break; 11814 break;
@@ -10821,14 +11827,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10821 break; 11827 break;
10822 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_ADD: 11828 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_ADD:
10823 if (!isAccount) return 0; 11829 if (!isAccount) return 0;
10824 if (estate.IsBanned(id)) return 1; 11830 if (estate.IsBanned(id, World.GetUserFlags(id))) return 1;
10825 EstateBan ban = new EstateBan(); 11831 EstateBan ban = new EstateBan();
10826 ban.EstateID = estate.EstateID; 11832 ban.EstateID = estate.EstateID;
10827 ban.BannedUserID = id; 11833 ban.BannedUserID = id;
10828 estate.AddBan(ban); 11834 estate.AddBan(ban);
10829 break; 11835 break;
10830 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_REMOVE: 11836 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_REMOVE:
10831 if (!isAccount || !estate.IsBanned(id)) return 0; 11837 if (!isAccount || !estate.IsBanned(id, World.GetUserFlags(id))) return 0;
10832 estate.RemoveBan(id); 11838 estate.RemoveBan(id);
10833 break; 11839 break;
10834 default: return 0; 11840 default: return 0;
@@ -10854,22 +11860,22 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10854 NotImplemented("llGetSPMaxMemory"); 11860 NotImplemented("llGetSPMaxMemory");
10855 } 11861 }
10856 11862
10857 public void llGetUsedMemory() 11863 public virtual LSL_Integer llGetUsedMemory()
10858 { 11864 {
10859 m_host.AddScriptLPS(1); 11865 m_host.AddScriptLPS(1);
10860 NotImplemented("llGetUsedMemory"); 11866 NotImplemented("llGetUsedMemory");
11867 return 0;
10861 } 11868 }
10862 11869
10863 public void llScriptProfiler(LSL_Integer flags) 11870 public void llScriptProfiler(LSL_Integer flags)
10864 { 11871 {
10865 m_host.AddScriptLPS(1); 11872 m_host.AddScriptLPS(1);
10866 NotImplemented("llScriptProfiler"); 11873 //NotImplemented("llScriptProfiler");
10867 } 11874 }
10868 11875
10869 public void llSetSoundQueueing(int queue) 11876 public void llSetSoundQueueing(int queue)
10870 { 11877 {
10871 m_host.AddScriptLPS(1); 11878 m_host.AddScriptLPS(1);
10872 NotImplemented("llSetSoundQueueing");
10873 } 11879 }
10874 11880
10875 public void llCollisionSprite(string impact_sprite) 11881 public void llCollisionSprite(string impact_sprite)
@@ -10881,10 +11887,274 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10881 public void llGodLikeRezObject(string inventory, LSL_Vector pos) 11887 public void llGodLikeRezObject(string inventory, LSL_Vector pos)
10882 { 11888 {
10883 m_host.AddScriptLPS(1); 11889 m_host.AddScriptLPS(1);
10884 NotImplemented("llGodLikeRezObject"); 11890
11891 if (!World.Permissions.IsGod(m_host.OwnerID))
11892 NotImplemented("llGodLikeRezObject");
11893
11894 AssetBase rezAsset = World.AssetService.Get(inventory);
11895 if (rezAsset == null)
11896 {
11897 llSay(0, "Asset not found");
11898 return;
11899 }
11900
11901 SceneObjectGroup group = null;
11902
11903 try
11904 {
11905 string xmlData = Utils.BytesToString(rezAsset.Data);
11906 group = SceneObjectSerializer.FromOriginalXmlFormat(xmlData);
11907 }
11908 catch
11909 {
11910 llSay(0, "Asset not found");
11911 return;
11912 }
11913
11914 if (group == null)
11915 {
11916 llSay(0, "Asset not found");
11917 return;
11918 }
11919
11920 group.RootPart.AttachPoint = group.RootPart.Shape.State;
11921 group.RootPart.AttachOffset = group.AbsolutePosition;
11922
11923 group.ResetIDs();
11924
11925 Vector3 llpos = new Vector3((float)pos.x, (float)pos.y, (float)pos.z);
11926 World.AddNewSceneObject(group, true, llpos, Quaternion.Identity, Vector3.Zero);
11927 group.CreateScriptInstances(0, true, World.DefaultScriptEngine, 3);
11928 group.ScheduleGroupForFullUpdate();
11929
11930 // objects rezzed with this method are die_at_edge by default.
11931 group.RootPart.SetDieAtEdge(true);
11932
11933 group.ResumeScripts();
11934
11935 m_ScriptEngine.PostObjectEvent(m_host.LocalId, new EventParams(
11936 "object_rez", new Object[] {
11937 new LSL_String(
11938 group.RootPart.UUID.ToString()) },
11939 new DetectParams[0]));
11940 }
11941
11942 public LSL_String llTransferLindenDollars(string destination, int amount)
11943 {
11944 UUID txn = UUID.Random();
11945
11946 Util.FireAndForget(delegate(object x)
11947 {
11948 int replycode = 0;
11949 string replydata = destination + "," + amount.ToString();
11950
11951 try
11952 {
11953 UUID invItemID=InventorySelf();
11954 if (invItemID == UUID.Zero)
11955 {
11956 replydata = "SERVICE_ERROR";
11957 return;
11958 }
11959
11960 m_host.AddScriptLPS(1);
11961
11962 m_host.TaskInventory.LockItemsForRead(true);
11963 TaskInventoryItem item = m_host.TaskInventory[invItemID];
11964 m_host.TaskInventory.LockItemsForRead(false);
11965
11966 if (item.PermsGranter == UUID.Zero)
11967 {
11968 replydata = "MISSING_PERMISSION_DEBIT";
11969 return;
11970 }
11971
11972 if ((item.PermsMask & ScriptBaseClass.PERMISSION_DEBIT) == 0)
11973 {
11974 replydata = "MISSING_PERMISSION_DEBIT";
11975 return;
11976 }
11977
11978 UUID toID = new UUID();
11979
11980 if (!UUID.TryParse(destination, out toID))
11981 {
11982 replydata = "INVALID_AGENT";
11983 return;
11984 }
11985
11986 IMoneyModule money = World.RequestModuleInterface<IMoneyModule>();
11987
11988 if (money == null)
11989 {
11990 replydata = "TRANSFERS_DISABLED";
11991 return;
11992 }
11993
11994 bool result = money.ObjectGiveMoney(
11995 m_host.ParentGroup.RootPart.UUID, m_host.ParentGroup.RootPart.OwnerID, toID, amount);
11996
11997 if (result)
11998 {
11999 replycode = 1;
12000 return;
12001 }
12002
12003 replydata = "LINDENDOLLAR_INSUFFICIENTFUNDS";
12004 }
12005 finally
12006 {
12007 m_ScriptEngine.PostScriptEvent(m_itemID, new EventParams(
12008 "transaction_result", new Object[] {
12009 new LSL_String(txn.ToString()),
12010 new LSL_Integer(replycode),
12011 new LSL_String(replydata) },
12012 new DetectParams[0]));
12013 }
12014 });
12015
12016 return txn.ToString();
10885 } 12017 }
10886 12018
10887 #endregion 12019 #endregion
12020
12021 public void llSetKeyframedMotion(LSL_List frames, LSL_List options)
12022 {
12023 SceneObjectGroup group = m_host.ParentGroup;
12024
12025 if (group.RootPart.PhysActor != null && group.RootPart.PhysActor.IsPhysical)
12026 return;
12027 if (group.IsAttachment)
12028 return;
12029
12030 if (frames.Data.Length > 0) // We are getting a new motion
12031 {
12032 if (group.RootPart.KeyframeMotion != null)
12033 group.RootPart.KeyframeMotion.Stop();
12034 group.RootPart.KeyframeMotion = null;
12035
12036 int idx = 0;
12037
12038 KeyframeMotion.PlayMode mode = KeyframeMotion.PlayMode.Forward;
12039 KeyframeMotion.DataFormat data = KeyframeMotion.DataFormat.Translation | KeyframeMotion.DataFormat.Rotation;
12040
12041 while (idx < options.Data.Length)
12042 {
12043 int option = (int)options.GetLSLIntegerItem(idx++);
12044 int remain = options.Data.Length - idx;
12045
12046 switch (option)
12047 {
12048 case ScriptBaseClass.KFM_MODE:
12049 if (remain < 1)
12050 break;
12051 int modeval = (int)options.GetLSLIntegerItem(idx++);
12052 switch(modeval)
12053 {
12054 case ScriptBaseClass.KFM_FORWARD:
12055 mode = KeyframeMotion.PlayMode.Forward;
12056 break;
12057 case ScriptBaseClass.KFM_REVERSE:
12058 mode = KeyframeMotion.PlayMode.Reverse;
12059 break;
12060 case ScriptBaseClass.KFM_LOOP:
12061 mode = KeyframeMotion.PlayMode.Loop;
12062 break;
12063 case ScriptBaseClass.KFM_PING_PONG:
12064 mode = KeyframeMotion.PlayMode.PingPong;
12065 break;
12066 }
12067 break;
12068 case ScriptBaseClass.KFM_DATA:
12069 if (remain < 1)
12070 break;
12071 int dataval = (int)options.GetLSLIntegerItem(idx++);
12072 data = (KeyframeMotion.DataFormat)dataval;
12073 break;
12074 }
12075 }
12076
12077 group.RootPart.KeyframeMotion = new KeyframeMotion(group, mode, data);
12078
12079 idx = 0;
12080
12081 int elemLength = 2;
12082 if (data == (KeyframeMotion.DataFormat.Translation | KeyframeMotion.DataFormat.Rotation))
12083 elemLength = 3;
12084
12085 List<KeyframeMotion.Keyframe> keyframes = new List<KeyframeMotion.Keyframe>();
12086 while (idx < frames.Data.Length)
12087 {
12088 int remain = frames.Data.Length - idx;
12089
12090 if (remain < elemLength)
12091 break;
12092
12093 KeyframeMotion.Keyframe frame = new KeyframeMotion.Keyframe();
12094 frame.Position = null;
12095 frame.Rotation = null;
12096
12097 if ((data & KeyframeMotion.DataFormat.Translation) != 0)
12098 {
12099 LSL_Types.Vector3 tempv = frames.GetVector3Item(idx++);
12100 frame.Position = new Vector3((float)tempv.x, (float)tempv.y, (float)tempv.z);
12101 }
12102 if ((data & KeyframeMotion.DataFormat.Rotation) != 0)
12103 {
12104 LSL_Types.Quaternion tempq = frames.GetQuaternionItem(idx++);
12105 frame.Rotation = new Quaternion((float)tempq.x, (float)tempq.y, (float)tempq.z, (float)tempq.s);
12106 }
12107
12108 float tempf = (float)frames.GetLSLFloatItem(idx++);
12109 frame.TimeMS = (int)(tempf * 1000.0f);
12110
12111 keyframes.Add(frame);
12112 }
12113
12114 group.RootPart.KeyframeMotion.SetKeyframes(keyframes.ToArray());
12115 group.RootPart.KeyframeMotion.Start();
12116 }
12117 else
12118 {
12119 if (group.RootPart.KeyframeMotion == null)
12120 return;
12121
12122 if (options.Data.Length == 0)
12123 {
12124 group.RootPart.KeyframeMotion.Stop();
12125 return;
12126 }
12127
12128 int code = (int)options.GetLSLIntegerItem(0);
12129
12130 int idx = 0;
12131
12132 while (idx < options.Data.Length)
12133 {
12134 int option = (int)options.GetLSLIntegerItem(idx++);
12135 int remain = options.Data.Length - idx;
12136
12137 switch (option)
12138 {
12139 case ScriptBaseClass.KFM_COMMAND:
12140 int cmd = (int)options.GetLSLIntegerItem(idx++);
12141 switch (cmd)
12142 {
12143 case ScriptBaseClass.KFM_CMD_PLAY:
12144 group.RootPart.KeyframeMotion.Start();
12145 break;
12146 case ScriptBaseClass.KFM_CMD_STOP:
12147 group.RootPart.KeyframeMotion.Stop();
12148 break;
12149 case ScriptBaseClass.KFM_CMD_PAUSE:
12150 group.RootPart.KeyframeMotion.Pause();
12151 break;
12152 }
12153 break;
12154 }
12155 }
12156 }
12157 }
10888 } 12158 }
10889 12159
10890 public class NotecardCache 12160 public class NotecardCache
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs
index 8edd146..ecc5fb5 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs
@@ -137,6 +137,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
137 internal ThreatLevel m_MaxThreatLevel = ThreatLevel.VeryLow; 137 internal ThreatLevel m_MaxThreatLevel = ThreatLevel.VeryLow;
138 internal float m_ScriptDelayFactor = 1.0f; 138 internal float m_ScriptDelayFactor = 1.0f;
139 internal float m_ScriptDistanceFactor = 1.0f; 139 internal float m_ScriptDistanceFactor = 1.0f;
140 internal bool m_debuggerSafe = false;
140 internal Dictionary<string, FunctionPerms > m_FunctionPerms = new Dictionary<string, FunctionPerms >(); 141 internal Dictionary<string, FunctionPerms > m_FunctionPerms = new Dictionary<string, FunctionPerms >();
141 142
142 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, uint localID, UUID itemID) 143 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, uint localID, UUID itemID)
@@ -145,6 +146,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
145 m_host = host; 146 m_host = host;
146 m_localID = localID; 147 m_localID = localID;
147 m_itemID = itemID; 148 m_itemID = itemID;
149 m_debuggerSafe = m_ScriptEngine.Config.GetBoolean("DebuggerSafe", false);
148 150
149 if (m_ScriptEngine.Config.GetBoolean("AllowOSFunctions", false)) 151 if (m_ScriptEngine.Config.GetBoolean("AllowOSFunctions", false))
150 m_OSFunctionsEnabled = true; 152 m_OSFunctionsEnabled = true;
@@ -206,7 +208,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
206 208
207 internal void OSSLError(string msg) 209 internal void OSSLError(string msg)
208 { 210 {
209 throw new Exception("OSSL Runtime Error: " + msg); 211 if (m_debuggerSafe)
212 {
213 OSSLShoutError(msg);
214 }
215 else
216 {
217 throw new Exception("OSSL Runtime Error: " + msg);
218 }
210 } 219 }
211 220
212 private void InitLSL() 221 private void InitLSL()
@@ -917,18 +926,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
917 if (target != null) 926 if (target != null)
918 { 927 {
919 UUID animID=UUID.Zero; 928 UUID animID=UUID.Zero;
920 lock (m_host.TaskInventory) 929 m_host.TaskInventory.LockItemsForRead(true);
930 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
921 { 931 {
922 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 932 if (inv.Value.Name == animation)
923 { 933 {
924 if (inv.Value.Name == animation) 934 if (inv.Value.Type == (int)AssetType.Animation)
925 { 935 animID = inv.Value.AssetID;
926 if (inv.Value.Type == (int)AssetType.Animation) 936 continue;
927 animID = inv.Value.AssetID;
928 continue;
929 }
930 } 937 }
931 } 938 }
939 m_host.TaskInventory.LockItemsForRead(false);
932 if (animID == UUID.Zero) 940 if (animID == UUID.Zero)
933 target.Animator.AddAnimation(animation, m_host.UUID); 941 target.Animator.AddAnimation(animation, m_host.UUID);
934 else 942 else
@@ -955,18 +963,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
955 if (target != null) 963 if (target != null)
956 { 964 {
957 UUID animID = UUID.Zero; 965 UUID animID = UUID.Zero;
958 lock (m_host.TaskInventory) 966 m_host.TaskInventory.LockItemsForRead(true);
967 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
959 { 968 {
960 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 969 if (inv.Value.Name == animation)
961 { 970 {
962 if (inv.Value.Name == animation) 971 if (inv.Value.Type == (int)AssetType.Animation)
963 { 972 animID = inv.Value.AssetID;
964 if (inv.Value.Type == (int)AssetType.Animation) 973 continue;
965 animID = inv.Value.AssetID;
966 continue;
967 }
968 } 974 }
969 } 975 }
976 m_host.TaskInventory.LockItemsForRead(false);
970 977
971 if (animID == UUID.Zero) 978 if (animID == UUID.Zero)
972 target.Animator.RemoveAnimation(animation); 979 target.Animator.RemoveAnimation(animation);
@@ -1798,6 +1805,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1798 1805
1799 if (!UUID.TryParse(notecardNameOrUuid, out assetID)) 1806 if (!UUID.TryParse(notecardNameOrUuid, out assetID))
1800 { 1807 {
1808 m_host.TaskInventory.LockItemsForRead(true);
1801 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 1809 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
1802 { 1810 {
1803 if (item.Type == 7 && item.Name == notecardNameOrUuid) 1811 if (item.Type == 7 && item.Name == notecardNameOrUuid)
@@ -1805,6 +1813,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1805 assetID = item.AssetID; 1813 assetID = item.AssetID;
1806 } 1814 }
1807 } 1815 }
1816 m_host.TaskInventory.LockItemsForRead(false);
1808 } 1817 }
1809 1818
1810 if (assetID == UUID.Zero) 1819 if (assetID == UUID.Zero)
@@ -2258,7 +2267,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2258 CheckThreatLevel(ThreatLevel.High, "osNpcCreate"); 2267 CheckThreatLevel(ThreatLevel.High, "osNpcCreate");
2259 m_host.AddScriptLPS(1); 2268 m_host.AddScriptLPS(1);
2260 2269
2261 return NpcCreate(firstname, lastname, position, notecard, false, false); 2270 return NpcCreate(firstname, lastname, position, notecard, true, false);
2262 } 2271 }
2263 2272
2264 public LSL_Key osNpcCreate(string firstname, string lastname, LSL_Vector position, string notecard, int options) 2273 public LSL_Key osNpcCreate(string firstname, string lastname, LSL_Vector position, string notecard, int options)
@@ -2269,24 +2278,39 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2269 return NpcCreate( 2278 return NpcCreate(
2270 firstname, lastname, position, notecard, 2279 firstname, lastname, position, notecard,
2271 (options & ScriptBaseClass.OS_NPC_NOT_OWNED) == 0, 2280 (options & ScriptBaseClass.OS_NPC_NOT_OWNED) == 0,
2272 (options & ScriptBaseClass.OS_NPC_SENSE_AS_AGENT) != 0); 2281 false);
2282// (options & ScriptBaseClass.OS_NPC_SENSE_AS_AGENT) != 0);
2273 } 2283 }
2274 2284
2275 private LSL_Key NpcCreate( 2285 private LSL_Key NpcCreate(
2276 string firstname, string lastname, LSL_Vector position, string notecard, bool owned, bool senseAsAgent) 2286 string firstname, string lastname, LSL_Vector position, string notecard, bool owned, bool senseAsAgent)
2277 { 2287 {
2288 if (!owned)
2289 OSSLError("Unowned NPCs are unsupported");
2290
2291 string groupTitle = String.Empty;
2292
2293 if (!World.Permissions.CanRezObject(1, m_host.OwnerID, new Vector3((float)position.x, (float)position.y, (float)position.z)))
2294 return new LSL_Key(UUID.Zero.ToString());
2295
2296 if (firstname != String.Empty || lastname != String.Empty)
2297 {
2298 if (firstname != "Shown outfit:")
2299 groupTitle = "- NPC -";
2300 }
2301
2278 INPCModule module = World.RequestModuleInterface<INPCModule>(); 2302 INPCModule module = World.RequestModuleInterface<INPCModule>();
2279 if (module != null) 2303 if (module != null)
2280 { 2304 {
2281 AvatarAppearance appearance = null; 2305 AvatarAppearance appearance = null;
2282 2306
2283 UUID id; 2307// UUID id;
2284 if (UUID.TryParse(notecard, out id)) 2308// if (UUID.TryParse(notecard, out id))
2285 { 2309// {
2286 ScenePresence clonePresence = World.GetScenePresence(id); 2310// ScenePresence clonePresence = World.GetScenePresence(id);
2287 if (clonePresence != null) 2311// if (clonePresence != null)
2288 appearance = clonePresence.Appearance; 2312// appearance = clonePresence.Appearance;
2289 } 2313// }
2290 2314
2291 if (appearance == null) 2315 if (appearance == null)
2292 { 2316 {
@@ -2314,6 +2338,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2314 World, 2338 World,
2315 appearance); 2339 appearance);
2316 2340
2341 ScenePresence sp;
2342 if (World.TryGetScenePresence(x, out sp))
2343 {
2344 sp.Grouptitle = groupTitle;
2345 sp.SendAvatarDataToAllAgents();
2346 }
2317 return new LSL_Key(x.ToString()); 2347 return new LSL_Key(x.ToString());
2318 } 2348 }
2319 2349
@@ -2582,16 +2612,32 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2582 CheckThreatLevel(ThreatLevel.High, "osNpcRemove"); 2612 CheckThreatLevel(ThreatLevel.High, "osNpcRemove");
2583 m_host.AddScriptLPS(1); 2613 m_host.AddScriptLPS(1);
2584 2614
2585 INPCModule module = World.RequestModuleInterface<INPCModule>(); 2615 ManualResetEvent ev = new ManualResetEvent(false);
2586 if (module != null)
2587 {
2588 UUID npcId = new UUID(npc.m_string);
2589 2616
2590 if (!module.CheckPermissions(npcId, m_host.OwnerID)) 2617 Util.FireAndForget(delegate(object x) {
2591 return; 2618 try
2619 {
2620 INPCModule module = World.RequestModuleInterface<INPCModule>();
2621 if (module != null)
2622 {
2623 UUID npcId = new UUID(npc.m_string);
2592 2624
2593 module.DeleteNPC(npcId, World); 2625 ILandObject l = World.LandChannel.GetLandObject(m_host.GroupPosition.X, m_host.GroupPosition.Y);
2594 } 2626 if (l == null || m_host.OwnerID != l.LandData.OwnerID)
2627 {
2628 if (!module.CheckPermissions(npcId, m_host.OwnerID))
2629 return;
2630 }
2631
2632 module.DeleteNPC(npcId, World);
2633 }
2634 }
2635 finally
2636 {
2637 ev.Set();
2638 }
2639 });
2640 ev.WaitOne();
2595 } 2641 }
2596 2642
2597 public void osNpcPlayAnimation(LSL_Key npc, string animation) 2643 public void osNpcPlayAnimation(LSL_Key npc, string animation)
@@ -2944,4 +2990,4 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2944 return date.ToString("yyyy-MM-ddTHH:mm:ss.fffffffZ"); 2990 return date.ToString("yyyy-MM-ddTHH:mm:ss.fffffffZ");
2945 } 2991 }
2946 } 2992 }
2947} \ No newline at end of file 2993}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs
index 1c272f8..1373971 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs
@@ -222,7 +222,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Plugins
222 // Is the sensor type is AGENT and not SCRIPTED then include agents 222 // Is the sensor type is AGENT and not SCRIPTED then include agents
223 if ((ts.type & (AGENT | AGENT_BY_USERNAME | NPC)) != 0 && (ts.type & SCRIPTED) == 0) 223 if ((ts.type & (AGENT | AGENT_BY_USERNAME | NPC)) != 0 && (ts.type & SCRIPTED) == 0)
224 { 224 {
225 sensedEntities.AddRange(doAgentSensor(ts)); 225 sensedEntities.AddRange(doAgentSensor(ts));
226 } 226 }
227 227
228 // If SCRIPTED or PASSIVE or ACTIVE check objects 228 // If SCRIPTED or PASSIVE or ACTIVE check objects
@@ -319,13 +319,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Plugins
319 float dy; 319 float dy;
320 float dz; 320 float dz;
321 321
322 Quaternion q = SensePoint.RotationOffset; 322// Quaternion q = SensePoint.RotationOffset;
323 Quaternion q = SensePoint.GetWorldRotation(); // non-attached prim Sensor *always* uses World rotation!
323 if (SensePoint.ParentGroup.IsAttachment) 324 if (SensePoint.ParentGroup.IsAttachment)
324 { 325 {
325 // In attachments, the sensor cone always orients with the 326 // In attachments, the sensor cone always orients with the
326 // avatar rotation. This may include a nonzero elevation if 327 // avatar rotation. This may include a nonzero elevation if
327 // in mouselook. 328 // in mouselook.
328 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar); 329 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar);
330 fromRegionPos = avatar.AbsolutePosition;
329 q = avatar.Rotation; 331 q = avatar.Rotation;
330 } 332 }
331 LSL_Types.Quaternion r = new LSL_Types.Quaternion(q.X, q.Y, q.Z, q.W); 333 LSL_Types.Quaternion r = new LSL_Types.Quaternion(q.X, q.Y, q.Z, q.W);
@@ -448,6 +450,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Plugins
448 // avatar rotation. This may include a nonzero elevation if 450 // avatar rotation. This may include a nonzero elevation if
449 // in mouselook. 451 // in mouselook.
450 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar); 452 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar);
453 if (avatar == null)
454 return sensedEntities;
455 fromRegionPos = avatar.AbsolutePosition;
451 q = avatar.Rotation; 456 q = avatar.Rotation;
452 } 457 }
453 458
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/Timer.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/Timer.cs
index bc63030..9ee6946 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/Timer.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/Timer.cs
@@ -118,25 +118,27 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Plugins
118 if (Timers.Count == 0) 118 if (Timers.Count == 0)
119 return; 119 return;
120 120
121 Dictionary<string, TimerClass>.ValueCollection tvals;
121 lock (TimerListLock) 122 lock (TimerListLock)
122 { 123 {
123 // Go through all timers 124 // Go through all timers
124 Dictionary<string, TimerClass>.ValueCollection tvals = Timers.Values; 125 tvals = Timers.Values;
125 foreach (TimerClass ts in tvals) 126 }
127
128 foreach (TimerClass ts in tvals)
129 {
130 // Time has passed?
131 if (ts.next < DateTime.Now.Ticks)
126 { 132 {
127 // Time has passed? 133 //m_log.Debug("Time has passed: Now: " + DateTime.Now.Ticks + ", Passed: " + ts.next);
128 if (ts.next < DateTime.Now.Ticks) 134 // Add it to queue
129 { 135 m_CmdManager.m_ScriptEngine.PostScriptEvent(ts.itemID,
130 //m_log.Debug("Time has passed: Now: " + DateTime.Now.Ticks + ", Passed: " + ts.next); 136 new EventParams("timer", new Object[0],
131 // Add it to queue 137 new DetectParams[0]));
132 m_CmdManager.m_ScriptEngine.PostScriptEvent(ts.itemID, 138 // set next interval
133 new EventParams("timer", new Object[0], 139
134 new DetectParams[0])); 140 //ts.next = DateTime.Now.ToUniversalTime().AddSeconds(ts.interval);
135 // set next interval 141 ts.next = DateTime.Now.Ticks + ts.interval;
136
137 //ts.next = DateTime.Now.ToUniversalTime().AddSeconds(ts.interval);
138 ts.next = DateTime.Now.Ticks + ts.interval;
139 }
140 } 142 }
141 } 143 }
142 } 144 }