aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/OpenSim/Region/ScriptEngine/Shared/Api
diff options
context:
space:
mode:
Diffstat (limited to 'OpenSim/Region/ScriptEngine/Shared/Api')
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/AsyncCommandManager.cs20
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/CM_Api.cs116
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs3032
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs102
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs11
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/Timer.cs32
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Interface/ICM_Api.cs46
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Interface/ILSL_Api.cs10
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Interface/IOSSL_Api.cs2
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Runtime/CM_Stub.cs71
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Runtime/Executor.cs2
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Constants.cs26
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Stub.cs51
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LS_Stub.cs21
-rw-r--r--OpenSim/Region/ScriptEngine/Shared/Api/Runtime/ScriptBase.cs2
15 files changed, 2662 insertions, 882 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..b5fa6de
--- /dev/null
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/CM_Api.cs
@@ -0,0 +1,116 @@
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 TaskInventoryItem m_item;
63 internal bool m_CMFunctionsEnabled = false;
64
65 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, TaskInventoryItem item)
66 {
67 m_ScriptEngine = ScriptEngine;
68 m_host = host;
69 m_item = item;
70
71 if (m_ScriptEngine.Config.GetBoolean("AllowCareminsterFunctions", false))
72 m_CMFunctionsEnabled = true;
73 }
74
75 public override Object InitializeLifetimeService()
76 {
77 ILease lease = (ILease)base.InitializeLifetimeService();
78
79 if (lease.CurrentState == LeaseState.Initial)
80 {
81 lease.InitialLeaseTime = TimeSpan.FromMinutes(0);
82 // lease.RenewOnCallTime = TimeSpan.FromSeconds(10.0);
83 // lease.SponsorshipTimeout = TimeSpan.FromMinutes(1.0);
84 }
85 return lease;
86 }
87
88 public Scene World
89 {
90 get { return m_ScriptEngine.World; }
91 }
92
93 public string cmDetectedCountry(int number)
94 {
95 m_host.AddScriptLPS(1);
96 DetectParams detectedParams = m_ScriptEngine.GetDetectParams(m_item.ItemID, number);
97 if (detectedParams == null)
98 return String.Empty;
99 return detectedParams.Country;
100 }
101
102 public string cmGetAgentCountry(LSL_Key key)
103 {
104 if (!World.Permissions.IsGod(m_host.OwnerID))
105 return String.Empty;
106
107 UUID uuid;
108
109 if (!UUID.TryParse(key, out uuid))
110 return String.Empty;
111
112 UserAccount account = World.UserAccountService.GetUserAccount(World.RegionInfo.ScopeID, uuid);
113 return account.UserCountry;
114 }
115 }
116}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/LSL_Api.cs
index fb0fdc9..e0b24dc 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{
@@ -103,15 +107,48 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
103 protected int m_notecardLineReadCharsMax = 255; 107 protected int m_notecardLineReadCharsMax = 255;
104 protected int m_scriptConsoleChannel = 0; 108 protected int m_scriptConsoleChannel = 0;
105 protected bool m_scriptConsoleChannelEnabled = false; 109 protected bool m_scriptConsoleChannelEnabled = false;
110 protected bool m_debuggerSafe = false;
106 protected IUrlModule m_UrlModule = null; 111 protected IUrlModule m_UrlModule = null;
107 protected Dictionary<UUID, UserInfoCacheEntry> m_userInfoCache = new Dictionary<UUID, UserInfoCacheEntry>(); 112 protected Dictionary<UUID, UserInfoCacheEntry> m_userInfoCache =
108 protected int EMAIL_PAUSE_TIME = 20; // documented delay value for smtp. 113 new Dictionary<UUID, UserInfoCacheEntry>();
114 protected int EMAIL_PAUSE_TIME = 20; // documented delay value for smtp.
115
116 protected Timer m_ShoutSayTimer;
117 protected int m_SayShoutCount = 0;
118
119 private Dictionary<string, string> MovementAnimationsForLSL =
120 new Dictionary<string, string> {
121 {"FLY", "Flying"},
122 {"FLYSLOW", "FlyingSlow"},
123 {"HOVER_UP", "Hovering Up"},
124 {"HOVER_DOWN", "Hovering Down"},
125 {"HOVER", "Hovering"},
126 {"LAND", "Landing"},
127 {"FALLDOWN", "Falling Down"},
128 {"PREJUMP", "PreJumping"},
129 {"JUMP", "Jumping"},
130 {"STANDUP", "Standing Up"},
131 {"SOFT_LAND", "Soft Landing"},
132 {"STAND", "Standing"},
133 {"CROUCHWALK", "CrouchWalking"},
134 {"RUN", "Running"},
135 {"WALK", "Walking"},
136 {"CROUCH", "Crouching"},
137 {"TURNLEFT", "Turning Left"},
138 {"TURNRIGHT", "Turning Right"}
139 };
109 140
110 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, TaskInventoryItem item) 141 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, TaskInventoryItem item)
111 { 142 {
143 m_ShoutSayTimer = new Timer(1000);
144 m_ShoutSayTimer.Elapsed += SayShoutTimerElapsed;
145 m_ShoutSayTimer.AutoReset = true;
146 m_ShoutSayTimer.Start();
147
112 m_ScriptEngine = ScriptEngine; 148 m_ScriptEngine = ScriptEngine;
113 m_host = host; 149 m_host = host;
114 m_item = item; 150 m_item = item;
151 m_debuggerSafe = m_ScriptEngine.Config.GetBoolean("DebuggerSafe", false);
115 152
116 LoadLimits(); // read script limits from config. 153 LoadLimits(); // read script limits from config.
117 154
@@ -171,6 +208,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
171 get { return m_ScriptEngine.World; } 208 get { return m_ScriptEngine.World; }
172 } 209 }
173 210
211 [DebuggerNonUserCode]
174 public void state(string newState) 212 public void state(string newState)
175 { 213 {
176 m_ScriptEngine.SetState(m_item.ItemID, newState); 214 m_ScriptEngine.SetState(m_item.ItemID, newState);
@@ -180,6 +218,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
180 /// Reset the named script. The script must be present 218 /// Reset the named script. The script must be present
181 /// in the same prim. 219 /// in the same prim.
182 /// </summary> 220 /// </summary>
221 [DebuggerNonUserCode]
183 public void llResetScript() 222 public void llResetScript()
184 { 223 {
185 m_host.AddScriptLPS(1); 224 m_host.AddScriptLPS(1);
@@ -236,9 +275,62 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
236 } 275 }
237 } 276 }
238 277
278 public List<ScenePresence> GetLinkAvatars(int linkType)
279 {
280 List<ScenePresence> ret = new List<ScenePresence>();
281 if (m_host == null || m_host.ParentGroup == null || m_host.ParentGroup.IsDeleted)
282 return ret;
283
284 List<ScenePresence> avs = m_host.ParentGroup.GetLinkedAvatars();
285
286 switch (linkType)
287 {
288 case ScriptBaseClass.LINK_SET:
289 return avs;
290
291 case ScriptBaseClass.LINK_ROOT:
292 return ret;
293
294 case ScriptBaseClass.LINK_ALL_OTHERS:
295 return avs;
296
297 case ScriptBaseClass.LINK_ALL_CHILDREN:
298 return avs;
299
300 case ScriptBaseClass.LINK_THIS:
301 return ret;
302
303 default:
304 if (linkType < 0)
305 return ret;
306
307 int partCount = m_host.ParentGroup.GetPartCount();
308
309 if (linkType <= partCount)
310 {
311 return ret;
312 }
313 else
314 {
315 linkType = linkType - partCount;
316 if (linkType > avs.Count)
317 {
318 return ret;
319 }
320 else
321 {
322 ret.Add(avs[linkType-1]);
323 return ret;
324 }
325 }
326 }
327 }
328
239 public List<SceneObjectPart> GetLinkParts(int linkType) 329 public List<SceneObjectPart> GetLinkParts(int linkType)
240 { 330 {
241 List<SceneObjectPart> ret = new List<SceneObjectPart>(); 331 List<SceneObjectPart> ret = new List<SceneObjectPart>();
332 if (m_host == null || m_host.ParentGroup == null || m_host.ParentGroup.IsDeleted)
333 return ret;
242 ret.Add(m_host); 334 ret.Add(m_host);
243 335
244 switch (linkType) 336 switch (linkType)
@@ -432,31 +524,48 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
432 524
433 //Now we start getting into quaternions which means sin/cos, matrices and vectors. ckrinke 525 //Now we start getting into quaternions which means sin/cos, matrices and vectors. ckrinke
434 526
435 /// <summary> 527 // Utility function for llRot2Euler
436 /// Convert an LSL rotation to a Euler vector. 528
437 /// </summary> 529 // normalize an angle between -PI and PI (-180 to +180 degrees)
438 /// <remarks> 530 protected double NormalizeAngle(double angle)
439 /// Using algorithm based off http://www.euclideanspace.com/maths/geometry/rotations/conversions/quaternionToEuler/quat_2_euler_paper_ver2-1.pdf
440 /// to avoid issues with singularity and rounding with Y rotation of +/- PI/2
441 /// </remarks>
442 /// <param name="r"></param>
443 /// <returns></returns>
444 public LSL_Vector llRot2Euler(LSL_Rotation r)
445 { 531 {
446 m_host.AddScriptLPS(1); 532 if (angle > -Math.PI && angle < Math.PI)
533 return angle;
534
535 int numPis = (int)(Math.PI / angle);
536 double remainder = angle - Math.PI * numPis;
537 if (numPis % 2 == 1)
538 return Math.PI - angle;
539 return remainder;
540 }
447 541
448 LSL_Vector v = new LSL_Vector(0.0, 0.0, 1.0) * r; // Z axis unit vector unaffected by Z rotation component of r. 542 public LSL_Vector llRot2Euler(LSL_Rotation q1)
449 double m = LSL_Vector.Mag(v); // Just in case v isn't normalized, need magnitude for Asin() operation later. 543 {
450 if (m == 0.0) return new LSL_Vector(); 544 m_host.AddScriptLPS(1);
451 double x = Math.Atan2(-v.y, v.z); 545 LSL_Vector eul = new LSL_Vector();
452 double sin = v.x / m;
453 if (sin < -0.999999 || sin > 0.999999) x = 0.0; // Force X rotation to 0 at the singularities.
454 double y = Math.Asin(sin);
455 // Rotate X axis unit vector by r and unwind the X and Y rotations leaving only the Z rotation
456 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)));
457 double z = Math.Atan2(v.y, v.x);
458 546
459 return new LSL_Vector(x, y, z); 547 double sqw = q1.s*q1.s;
548 double sqx = q1.x*q1.x;
549 double sqy = q1.z*q1.z;
550 double sqz = q1.y*q1.y;
551 double unit = sqx + sqy + sqz + sqw; // if normalised is one, otherwise is correction factor
552 double test = q1.x*q1.z + q1.y*q1.s;
553 if (test > 0.4999*unit) { // singularity at north pole
554 eul.z = 2 * Math.Atan2(q1.x,q1.s);
555 eul.y = Math.PI/2;
556 eul.x = 0;
557 return eul;
558 }
559 if (test < -0.4999*unit) { // singularity at south pole
560 eul.z = -2 * Math.Atan2(q1.x,q1.s);
561 eul.y = -Math.PI/2;
562 eul.x = 0;
563 return eul;
564 }
565 eul.z = Math.Atan2(2*q1.z*q1.s-2*q1.x*q1.y , sqx - sqy - sqz + sqw);
566 eul.y = Math.Asin(2*test/unit);
567 eul.x = Math.Atan2(2*q1.x*q1.s-2*q1.z*q1.y , -sqx + sqy - sqz + sqw);
568 return eul;
460 } 569 }
461 570
462 /* From wiki: 571 /* From wiki:
@@ -658,77 +767,76 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
658 { 767 {
659 //A and B should both be normalized 768 //A and B should both be normalized
660 m_host.AddScriptLPS(1); 769 m_host.AddScriptLPS(1);
661 LSL_Rotation rotBetween; 770 /* This method is more accurate than the SL one, and thus causes problems
662 // Check for zero vectors. If either is zero, return zero rotation. Otherwise, 771 for scripts that deal with the SL inaccuracy around 180-degrees -.- .._.
663 // continue calculation. 772
664 if (a == new LSL_Vector(0.0f, 0.0f, 0.0f) || b == new LSL_Vector(0.0f, 0.0f, 0.0f)) 773 double dotProduct = LSL_Vector.Dot(a, b);
774 LSL_Vector crossProduct = LSL_Vector.Cross(a, b);
775 double magProduct = LSL_Vector.Mag(a) * LSL_Vector.Mag(b);
776 double angle = Math.Acos(dotProduct / magProduct);
777 LSL_Vector axis = LSL_Vector.Norm(crossProduct);
778 double s = Math.Sin(angle / 2);
779
780 double x = axis.x * s;
781 double y = axis.y * s;
782 double z = axis.z * s;
783 double w = Math.Cos(angle / 2);
784
785 if (Double.IsNaN(x) || Double.IsNaN(y) || Double.IsNaN(z) || Double.IsNaN(w))
786 return new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
787
788 return new LSL_Rotation((float)x, (float)y, (float)z, (float)w);
789 */
790
791 // This method mimics the 180 errors found in SL
792 // See www.euclideanspace.com... angleBetween
793 LSL_Vector vec_a = a;
794 LSL_Vector vec_b = b;
795
796 // Eliminate zero length
797 LSL_Float vec_a_mag = LSL_Vector.Mag(vec_a);
798 LSL_Float vec_b_mag = LSL_Vector.Mag(vec_b);
799 if (vec_a_mag < 0.00001 ||
800 vec_b_mag < 0.00001)
665 { 801 {
666 rotBetween = new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f); 802 return new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
667 } 803 }
668 else 804
805 // Normalize
806 vec_a = llVecNorm(vec_a);
807 vec_b = llVecNorm(vec_b);
808
809 // Calculate axis and rotation angle
810 LSL_Vector axis = vec_a % vec_b;
811 LSL_Float cos_theta = vec_a * vec_b;
812
813 // Check if parallel
814 if (cos_theta > 0.99999)
669 { 815 {
670 a = LSL_Vector.Norm(a); 816 return new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
671 b = LSL_Vector.Norm(b); 817 }
672 double dotProduct = LSL_Vector.Dot(a, b); 818
673 // There are two degenerate cases possible. These are for vectors 180 or 819 // Check if anti-parallel
674 // 0 degrees apart. These have to be detected and handled individually. 820 else if (cos_theta < -0.99999)
675 // 821 {
676 // Check for vectors 180 degrees apart. 822 LSL_Vector orthog_axis = new LSL_Vector(1.0, 0.0, 0.0) - (vec_a.x / (vec_a * vec_a) * vec_a);
677 // A dot product of -1 would mean the angle between vectors is 180 degrees. 823 if (LSL_Vector.Mag(orthog_axis) < 0.000001) orthog_axis = new LSL_Vector(0.0, 0.0, 1.0);
678 if (dotProduct < -0.9999999f) 824 return new LSL_Rotation((float)orthog_axis.x, (float)orthog_axis.y, (float)orthog_axis.z, 0.0);
679 { 825 }
680 // First assume X axis is orthogonal to the vectors. 826 else // other rotation
681 LSL_Vector orthoVector = new LSL_Vector(1.0f, 0.0f, 0.0f); 827 {
682 orthoVector = orthoVector - a * (a.x / LSL_Vector.Dot(a, a)); 828 LSL_Float theta = (LSL_Float)Math.Acos(cos_theta) * 0.5f;
683 // Check for near zero vector. A very small non-zero number here will create 829 axis = llVecNorm(axis);
684 // a rotation in an undesired direction. 830 double x, y, z, s, t;
685 if (LSL_Vector.Mag(orthoVector) > 0.0001) 831 s = Math.Cos(theta);
686 { 832 t = Math.Sin(theta);
687 rotBetween = new LSL_Rotation(orthoVector.x, orthoVector.y, orthoVector.z, 0.0f); 833 x = axis.x * t;
688 } 834 y = axis.y * t;
689 // If the magnitude of the vector was near zero, then assume the X axis is not 835 z = axis.z * t;
690 // orthogonal and use the Z axis instead. 836 return new LSL_Rotation(x,y,z,s);
691 else
692 {
693 // Set 180 z rotation.
694 rotBetween = new LSL_Rotation(0.0f, 0.0f, 1.0f, 0.0f);
695 }
696 }
697 // Check for parallel vectors.
698 // A dot product of 1 would mean the angle between vectors is 0 degrees.
699 else if (dotProduct > 0.9999999f)
700 {
701 // Set zero rotation.
702 rotBetween = new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
703 }
704 else
705 {
706 // All special checks have been performed so get the axis of rotation.
707 LSL_Vector crossProduct = LSL_Vector.Cross(a, b);
708 // Quarternion s value is the length of the unit vector + dot product.
709 double qs = 1.0 + dotProduct;
710 rotBetween = new LSL_Rotation(crossProduct.x, crossProduct.y, crossProduct.z, qs);
711 // Normalize the rotation.
712 double mag = LSL_Rotation.Mag(rotBetween);
713 // We shouldn't have to worry about a divide by zero here. The qs value will be
714 // non-zero because we already know if we're here, then the dotProduct is not -1 so
715 // qs will not be zero. Also, we've already handled the input vectors being zero so the
716 // crossProduct vector should also not be zero.
717 rotBetween.x = rotBetween.x / mag;
718 rotBetween.y = rotBetween.y / mag;
719 rotBetween.z = rotBetween.z / mag;
720 rotBetween.s = rotBetween.s / mag;
721 // Check for undefined values and set zero rotation if any found. This code might not actually be required
722 // any longer since zero vectors are checked for at the top.
723 if (Double.IsNaN(rotBetween.x) || Double.IsNaN(rotBetween.y) || Double.IsNaN(rotBetween.z) || Double.IsNaN(rotBetween.s))
724 {
725 rotBetween = new LSL_Rotation(0.0f, 0.0f, 0.0f, 1.0f);
726 }
727 }
728 } 837 }
729 return rotBetween;
730 } 838 }
731 839
732 public void llWhisper(int channelID, string text) 840 public void llWhisper(int channelID, string text)
733 { 841 {
734 m_host.AddScriptLPS(1); 842 m_host.AddScriptLPS(1);
@@ -748,6 +856,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
748 { 856 {
749 m_host.AddScriptLPS(1); 857 m_host.AddScriptLPS(1);
750 858
859 if (channelID == 0)
860 m_SayShoutCount++;
861
862 if (m_SayShoutCount >= 11)
863 ScriptSleep(2000);
864
751 if (m_scriptConsoleChannelEnabled && (channelID == m_scriptConsoleChannel)) 865 if (m_scriptConsoleChannelEnabled && (channelID == m_scriptConsoleChannel))
752 { 866 {
753 Console.WriteLine(text); 867 Console.WriteLine(text);
@@ -770,6 +884,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
770 { 884 {
771 m_host.AddScriptLPS(1); 885 m_host.AddScriptLPS(1);
772 886
887 if (channelID == 0)
888 m_SayShoutCount++;
889
890 if (m_SayShoutCount >= 11)
891 ScriptSleep(2000);
892
773 if (text.Length > 1023) 893 if (text.Length > 1023)
774 text = text.Substring(0, 1023); 894 text = text.Substring(0, 1023);
775 895
@@ -1072,10 +1192,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1072 return detectedParams.TouchUV; 1192 return detectedParams.TouchUV;
1073 } 1193 }
1074 1194
1195 [DebuggerNonUserCode]
1075 public virtual void llDie() 1196 public virtual void llDie()
1076 { 1197 {
1077 m_host.AddScriptLPS(1); 1198 m_host.AddScriptLPS(1);
1078 throw new SelfDeleteException(); 1199 if (!m_host.ParentGroup.IsAttachment) throw new SelfDeleteException();
1079 } 1200 }
1080 1201
1081 public LSL_Float llGround(LSL_Vector offset) 1202 public LSL_Float llGround(LSL_Vector offset)
@@ -1148,6 +1269,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1148 1269
1149 public void llSetStatus(int status, int value) 1270 public void llSetStatus(int status, int value)
1150 { 1271 {
1272 if (m_host == null || m_host.ParentGroup == null || m_host.ParentGroup.IsDeleted)
1273 return;
1151 m_host.AddScriptLPS(1); 1274 m_host.AddScriptLPS(1);
1152 1275
1153 int statusrotationaxis = 0; 1276 int statusrotationaxis = 0;
@@ -1171,6 +1294,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1171 if (!allow) 1294 if (!allow)
1172 return; 1295 return;
1173 1296
1297 if (m_host.ParentGroup.RootPart.PhysActor != null &&
1298 m_host.ParentGroup.RootPart.PhysActor.IsPhysical)
1299 return;
1300
1174 m_host.ScriptSetPhysicsStatus(true); 1301 m_host.ScriptSetPhysicsStatus(true);
1175 } 1302 }
1176 else 1303 else
@@ -1379,6 +1506,48 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1379 { 1506 {
1380 m_host.AddScriptLPS(1); 1507 m_host.AddScriptLPS(1);
1381 1508
1509 SetColor(m_host, color, face);
1510 }
1511
1512 protected void SetColor(SceneObjectPart part, LSL_Vector color, int face)
1513 {
1514 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1515 return;
1516
1517 Primitive.TextureEntry tex = part.Shape.Textures;
1518 Color4 texcolor;
1519 if (face >= 0 && face < GetNumberOfSides(part))
1520 {
1521 texcolor = tex.CreateFace((uint)face).RGBA;
1522 texcolor.R = Util.Clip((float)color.x, 0.0f, 1.0f);
1523 texcolor.G = Util.Clip((float)color.y, 0.0f, 1.0f);
1524 texcolor.B = Util.Clip((float)color.z, 0.0f, 1.0f);
1525 tex.FaceTextures[face].RGBA = texcolor;
1526 part.UpdateTextureEntry(tex.GetBytes());
1527 return;
1528 }
1529 else if (face == ScriptBaseClass.ALL_SIDES)
1530 {
1531 for (uint i = 0; i < GetNumberOfSides(part); i++)
1532 {
1533 if (tex.FaceTextures[i] != null)
1534 {
1535 texcolor = tex.FaceTextures[i].RGBA;
1536 texcolor.R = Util.Clip((float)color.x, 0.0f, 1.0f);
1537 texcolor.G = Util.Clip((float)color.y, 0.0f, 1.0f);
1538 texcolor.B = Util.Clip((float)color.z, 0.0f, 1.0f);
1539 tex.FaceTextures[i].RGBA = texcolor;
1540 }
1541 texcolor = tex.DefaultTexture.RGBA;
1542 texcolor.R = Util.Clip((float)color.x, 0.0f, 1.0f);
1543 texcolor.G = Util.Clip((float)color.y, 0.0f, 1.0f);
1544 texcolor.B = Util.Clip((float)color.z, 0.0f, 1.0f);
1545 tex.DefaultTexture.RGBA = texcolor;
1546 }
1547 part.UpdateTextureEntry(tex.GetBytes());
1548 return;
1549 }
1550
1382 if (face == ScriptBaseClass.ALL_SIDES) 1551 if (face == ScriptBaseClass.ALL_SIDES)
1383 face = SceneObjectPart.ALL_SIDES; 1552 face = SceneObjectPart.ALL_SIDES;
1384 1553
@@ -1387,6 +1556,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1387 1556
1388 public void SetTexGen(SceneObjectPart part, int face,int style) 1557 public void SetTexGen(SceneObjectPart part, int face,int style)
1389 { 1558 {
1559 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1560 return;
1561
1390 Primitive.TextureEntry tex = part.Shape.Textures; 1562 Primitive.TextureEntry tex = part.Shape.Textures;
1391 MappingType textype; 1563 MappingType textype;
1392 textype = MappingType.Default; 1564 textype = MappingType.Default;
@@ -1417,6 +1589,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1417 1589
1418 public void SetGlow(SceneObjectPart part, int face, float glow) 1590 public void SetGlow(SceneObjectPart part, int face, float glow)
1419 { 1591 {
1592 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1593 return;
1594
1420 Primitive.TextureEntry tex = part.Shape.Textures; 1595 Primitive.TextureEntry tex = part.Shape.Textures;
1421 if (face >= 0 && face < GetNumberOfSides(part)) 1596 if (face >= 0 && face < GetNumberOfSides(part))
1422 { 1597 {
@@ -1442,6 +1617,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1442 1617
1443 public void SetShiny(SceneObjectPart part, int face, int shiny, Bumpiness bump) 1618 public void SetShiny(SceneObjectPart part, int face, int shiny, Bumpiness bump)
1444 { 1619 {
1620 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1621 return;
1445 1622
1446 Shininess sval = new Shininess(); 1623 Shininess sval = new Shininess();
1447 1624
@@ -1492,6 +1669,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1492 1669
1493 public void SetFullBright(SceneObjectPart part, int face, bool bright) 1670 public void SetFullBright(SceneObjectPart part, int face, bool bright)
1494 { 1671 {
1672 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1673 return;
1674
1495 Primitive.TextureEntry tex = part.Shape.Textures; 1675 Primitive.TextureEntry tex = part.Shape.Textures;
1496 if (face >= 0 && face < GetNumberOfSides(part)) 1676 if (face >= 0 && face < GetNumberOfSides(part))
1497 { 1677 {
@@ -1552,13 +1732,26 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1552 m_host.AddScriptLPS(1); 1732 m_host.AddScriptLPS(1);
1553 1733
1554 List<SceneObjectPart> parts = GetLinkParts(linknumber); 1734 List<SceneObjectPart> parts = GetLinkParts(linknumber);
1555 1735 if (parts.Count > 0)
1556 foreach (SceneObjectPart part in parts) 1736 {
1557 SetAlpha(part, alpha, face); 1737 try
1738 {
1739 parts[0].ParentGroup.areUpdatesSuspended = true;
1740 foreach (SceneObjectPart part in parts)
1741 SetAlpha(part, alpha, face);
1742 }
1743 finally
1744 {
1745 parts[0].ParentGroup.areUpdatesSuspended = false;
1746 }
1747 }
1558 } 1748 }
1559 1749
1560 protected void SetAlpha(SceneObjectPart part, double alpha, int face) 1750 protected void SetAlpha(SceneObjectPart part, double alpha, int face)
1561 { 1751 {
1752 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1753 return;
1754
1562 Primitive.TextureEntry tex = part.Shape.Textures; 1755 Primitive.TextureEntry tex = part.Shape.Textures;
1563 Color4 texcolor; 1756 Color4 texcolor;
1564 if (face >= 0 && face < GetNumberOfSides(part)) 1757 if (face >= 0 && face < GetNumberOfSides(part))
@@ -1611,7 +1804,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1611 protected void SetFlexi(SceneObjectPart part, bool flexi, int softness, float gravity, float friction, 1804 protected void SetFlexi(SceneObjectPart part, bool flexi, int softness, float gravity, float friction,
1612 float wind, float tension, LSL_Vector Force) 1805 float wind, float tension, LSL_Vector Force)
1613 { 1806 {
1614 if (part == null) 1807 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1615 return; 1808 return;
1616 1809
1617 if (flexi) 1810 if (flexi)
@@ -1645,7 +1838,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1645 /// <param name="falloff"></param> 1838 /// <param name="falloff"></param>
1646 protected void SetPointLight(SceneObjectPart part, bool light, LSL_Vector color, float intensity, float radius, float falloff) 1839 protected void SetPointLight(SceneObjectPart part, bool light, LSL_Vector color, float intensity, float radius, float falloff)
1647 { 1840 {
1648 if (part == null) 1841 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1649 return; 1842 return;
1650 1843
1651 if (light) 1844 if (light)
@@ -1724,15 +1917,27 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1724 m_host.AddScriptLPS(1); 1917 m_host.AddScriptLPS(1);
1725 1918
1726 List<SceneObjectPart> parts = GetLinkParts(linknumber); 1919 List<SceneObjectPart> parts = GetLinkParts(linknumber);
1727 1920 if (parts.Count > 0)
1728 foreach (SceneObjectPart part in parts) 1921 {
1729 SetTexture(part, texture, face); 1922 try
1730 1923 {
1924 parts[0].ParentGroup.areUpdatesSuspended = true;
1925 foreach (SceneObjectPart part in parts)
1926 SetTexture(part, texture, face);
1927 }
1928 finally
1929 {
1930 parts[0].ParentGroup.areUpdatesSuspended = false;
1931 }
1932 }
1731 ScriptSleep(200); 1933 ScriptSleep(200);
1732 } 1934 }
1733 1935
1734 protected void SetTexture(SceneObjectPart part, string texture, int face) 1936 protected void SetTexture(SceneObjectPart part, string texture, int face)
1735 { 1937 {
1938 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1939 return;
1940
1736 UUID textureID = new UUID(); 1941 UUID textureID = new UUID();
1737 1942
1738 textureID = InventoryKey(texture, (int)AssetType.Texture); 1943 textureID = InventoryKey(texture, (int)AssetType.Texture);
@@ -1777,6 +1982,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1777 1982
1778 protected void ScaleTexture(SceneObjectPart part, double u, double v, int face) 1983 protected void ScaleTexture(SceneObjectPart part, double u, double v, int face)
1779 { 1984 {
1985 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
1986 return;
1987
1780 Primitive.TextureEntry tex = part.Shape.Textures; 1988 Primitive.TextureEntry tex = part.Shape.Textures;
1781 if (face >= 0 && face < GetNumberOfSides(part)) 1989 if (face >= 0 && face < GetNumberOfSides(part))
1782 { 1990 {
@@ -1813,6 +2021,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1813 2021
1814 protected void OffsetTexture(SceneObjectPart part, double u, double v, int face) 2022 protected void OffsetTexture(SceneObjectPart part, double u, double v, int face)
1815 { 2023 {
2024 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2025 return;
2026
1816 Primitive.TextureEntry tex = part.Shape.Textures; 2027 Primitive.TextureEntry tex = part.Shape.Textures;
1817 if (face >= 0 && face < GetNumberOfSides(part)) 2028 if (face >= 0 && face < GetNumberOfSides(part))
1818 { 2029 {
@@ -1849,6 +2060,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1849 2060
1850 protected void RotateTexture(SceneObjectPart part, double rotation, int face) 2061 protected void RotateTexture(SceneObjectPart part, double rotation, int face)
1851 { 2062 {
2063 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2064 return;
2065
1852 Primitive.TextureEntry tex = part.Shape.Textures; 2066 Primitive.TextureEntry tex = part.Shape.Textures;
1853 if (face >= 0 && face < GetNumberOfSides(part)) 2067 if (face >= 0 && face < GetNumberOfSides(part))
1854 { 2068 {
@@ -1953,26 +2167,76 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1953 return real_vec; 2167 return real_vec;
1954 } 2168 }
1955 2169
2170 public LSL_Integer llSetRegionPos(LSL_Vector pos)
2171 {
2172 return new LSL_Integer(SetRegionPos(m_host, pos));
2173 }
2174
2175 protected int SetRegionPos(SceneObjectPart part, LSL_Vector targetPos)
2176 {
2177 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2178 return 0;
2179
2180 SceneObjectGroup grp = part.ParentGroup;
2181
2182 if (grp.IsAttachment)
2183 return 0;
2184
2185 if (grp.RootPart.PhysActor != null && grp.RootPart.PhysActor.IsPhysical)
2186 return 0;
2187
2188 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)
2189 return 0;
2190
2191 float constrainedX = (float)targetPos.x;
2192 float constrainedY = (float)targetPos.y;
2193
2194 if (constrainedX < 0.0f)
2195 constrainedX = 0.0f;
2196 if (constrainedY < 0.0f)
2197 constrainedY = 0.0f;
2198 if (constrainedX >= (float)Constants.RegionSize)
2199 constrainedX = (float)Constants.RegionSize - 0.1f;
2200 if (constrainedY >= (float)Constants.RegionSize)
2201 constrainedY = (float)Constants.RegionSize -0.1f;
2202
2203 float ground = World.GetGroundHeight(constrainedX, constrainedY);
2204
2205 if (targetPos.z < ground)
2206 targetPos.z = ground;
2207
2208 Vector3 dest = new Vector3((float)targetPos.x, (float)targetPos.y, (float)targetPos.z);
2209
2210 if (!World.Permissions.CanObjectEntry(grp.UUID, false, dest))
2211 return 0;
2212
2213 grp.UpdateGroupPosition(dest);
2214
2215 return 1;
2216 }
2217
1956 protected void SetPos(SceneObjectPart part, LSL_Vector targetPos) 2218 protected void SetPos(SceneObjectPart part, LSL_Vector targetPos)
1957 { 2219 {
1958 // Capped movemment if distance > 10m (http://wiki.secondlife.com/wiki/LlSetPos) 2220 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2221 return;
2222
1959 LSL_Vector currentPos = GetPartLocalPos(part); 2223 LSL_Vector currentPos = GetPartLocalPos(part);
2224 LSL_Vector toPos = GetSetPosTarget(part, targetPos, currentPos);
1960 2225
1961 float ground = World.GetGroundHeight((float)targetPos.x, (float)targetPos.y);
1962 bool disable_underground_movement = m_ScriptEngine.Config.GetBoolean("DisableUndergroundMovement", true);
1963 2226
1964 if (part.ParentGroup.RootPart == part) 2227 if (part.ParentGroup.RootPart == part)
1965 { 2228 {
1966 if ((targetPos.z < ground) && disable_underground_movement && m_host.ParentGroup.AttachmentPoint == 0)
1967 targetPos.z = ground;
1968 SceneObjectGroup parent = part.ParentGroup; 2229 SceneObjectGroup parent = part.ParentGroup;
1969 LSL_Vector real_vec = SetPosAdjust(currentPos, targetPos); 2230 Vector3 dest = new Vector3((float)toPos.x, (float)toPos.y, (float)toPos.z);
1970 parent.UpdateGroupPosition(new Vector3((float)real_vec.x, (float)real_vec.y, (float)real_vec.z)); 2231 if (!World.Permissions.CanObjectEntry(parent.UUID, false, dest))
2232 return;
2233 Util.FireAndForget(delegate(object x) {
2234 parent.UpdateGroupPosition(dest);
2235 });
1971 } 2236 }
1972 else 2237 else
1973 { 2238 {
1974 LSL_Vector rel_vec = SetPosAdjust(currentPos, targetPos); 2239 part.OffsetPosition = new Vector3((float)toPos.x, (float)toPos.y, (float)toPos.z);
1975 part.OffsetPosition = new Vector3((float)rel_vec.x, (float)rel_vec.y, (float)rel_vec.z);
1976 SceneObjectGroup parent = part.ParentGroup; 2240 SceneObjectGroup parent = part.ParentGroup;
1977 parent.HasGroupChanged = true; 2241 parent.HasGroupChanged = true;
1978 parent.ScheduleGroupForTerseUpdate(); 2242 parent.ScheduleGroupForTerseUpdate();
@@ -2005,17 +2269,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2005 else 2269 else
2006 { 2270 {
2007 if (part.ParentGroup.IsAttachment) 2271 if (part.ParentGroup.IsAttachment)
2008 {
2009 pos = part.AttachedPos; 2272 pos = part.AttachedPos;
2010 }
2011 else 2273 else
2012 {
2013 pos = part.AbsolutePosition; 2274 pos = part.AbsolutePosition;
2014 }
2015 } 2275 }
2016 2276
2017// m_log.DebugFormat("[LSL API]: Returning {0} in GetPartLocalPos()", pos);
2018
2019 return new LSL_Vector(pos.X, pos.Y, pos.Z); 2277 return new LSL_Vector(pos.X, pos.Y, pos.Z);
2020 } 2278 }
2021 2279
@@ -2024,9 +2282,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2024 m_host.AddScriptLPS(1); 2282 m_host.AddScriptLPS(1);
2025 2283
2026 // try to let this work as in SL... 2284 // try to let this work as in SL...
2027 if (m_host.ParentID == 0) 2285 if (m_host.LinkNum < 2)
2028 { 2286 {
2029 // special case: If we are root, rotate complete SOG to new rotation 2287 // Special case: If we are root, rotate complete SOG to new
2288 // rotation.
2289 // We are root if the link number is 0 (single prim) or 1
2290 // (root prim). ParentID may be nonzero in attachments and
2291 // using it would cause attachments and HUDs to rotate
2292 // to the wrong positions.
2030 SetRot(m_host, Rot2Quaternion(rot)); 2293 SetRot(m_host, Rot2Quaternion(rot));
2031 } 2294 }
2032 else 2295 else
@@ -2051,6 +2314,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2051 2314
2052 protected void SetRot(SceneObjectPart part, Quaternion rot) 2315 protected void SetRot(SceneObjectPart part, Quaternion rot)
2053 { 2316 {
2317 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
2318 return;
2319
2054 part.UpdateRotation(rot); 2320 part.UpdateRotation(rot);
2055 // Update rotation does not move the object in the physics scene if it's a linkset. 2321 // Update rotation does not move the object in the physics scene if it's a linkset.
2056 2322
@@ -2205,13 +2471,13 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2205 public void llApplyRotationalImpulse(LSL_Vector force, int local) 2471 public void llApplyRotationalImpulse(LSL_Vector force, int local)
2206 { 2472 {
2207 m_host.AddScriptLPS(1); 2473 m_host.AddScriptLPS(1);
2208 m_host.ApplyAngularImpulse(new Vector3((float)force.x, (float)force.y, (float)force.z), local != 0); 2474 m_host.ParentGroup.RootPart.ApplyAngularImpulse(new Vector3((float)force.x, (float)force.y, (float)force.z), local != 0);
2209 } 2475 }
2210 2476
2211 public void llSetTorque(LSL_Vector torque, int local) 2477 public void llSetTorque(LSL_Vector torque, int local)
2212 { 2478 {
2213 m_host.AddScriptLPS(1); 2479 m_host.AddScriptLPS(1);
2214 m_host.SetAngularImpulse(new Vector3((float)torque.x, (float)torque.y, (float)torque.z), local != 0); 2480 m_host.ParentGroup.RootPart.SetAngularImpulse(new Vector3((float)torque.x, (float)torque.y, (float)torque.z), local != 0);
2215 } 2481 }
2216 2482
2217 public LSL_Vector llGetTorque() 2483 public LSL_Vector llGetTorque()
@@ -2232,16 +2498,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2232 { 2498 {
2233 m_host.AddScriptLPS(1); 2499 m_host.AddScriptLPS(1);
2234 2500
2235 Vector3 vel; 2501 Vector3 vel = Vector3.Zero;
2236 2502
2237 if (m_host.ParentGroup.IsAttachment) 2503 if (m_host.ParentGroup.IsAttachment)
2238 { 2504 {
2239 ScenePresence avatar = m_host.ParentGroup.Scene.GetScenePresence(m_host.ParentGroup.AttachedAvatar); 2505 ScenePresence avatar = m_host.ParentGroup.Scene.GetScenePresence(m_host.ParentGroup.AttachedAvatar);
2240 vel = avatar.Velocity; 2506 if (avatar != null)
2507 vel = avatar.Velocity;
2241 } 2508 }
2242 else 2509 else
2243 { 2510 {
2244 vel = m_host.Velocity; 2511 vel = m_host.ParentGroup.RootPart.Velocity;
2245 } 2512 }
2246 2513
2247 return new LSL_Vector(vel.X, vel.Y, vel.Z); 2514 return new LSL_Vector(vel.X, vel.Y, vel.Z);
@@ -2790,8 +3057,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2790 3057
2791 if (pa != null && pa.IsPhysical && llvel != Vector3.Zero) 3058 if (pa != null && pa.IsPhysical && llvel != Vector3.Zero)
2792 { 3059 {
2793 //Recoil. 3060 // recoil
2794 llApplyImpulse(new LSL_Vector(llvel.X * groupmass, llvel.Y * groupmass, llvel.Z * groupmass), 0); 3061 llvel *= -groupmass;
3062 llApplyImpulse(new LSL_Vector(llvel.X, llvel.Y,llvel.Z), 0);
2795 } 3063 }
2796 // Variable script delay? (see (http://wiki.secondlife.com/wiki/LSL_Delay) 3064 // Variable script delay? (see (http://wiki.secondlife.com/wiki/LSL_Delay)
2797 return; 3065 return;
@@ -2813,35 +3081,39 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2813 public void llLookAt(LSL_Vector target, double strength, double damping) 3081 public void llLookAt(LSL_Vector target, double strength, double damping)
2814 { 3082 {
2815 m_host.AddScriptLPS(1); 3083 m_host.AddScriptLPS(1);
2816 // Determine where we are looking from
2817 LSL_Vector from = llGetPos();
2818 3084
2819 // Work out the normalised vector from the source to the target 3085 // Get the normalized vector to the target
2820 LSL_Vector delta = llVecNorm(target - from); 3086 LSL_Vector d1 = llVecNorm(target - llGetPos());
2821 LSL_Vector angle = new LSL_Vector(0,0,0);
2822 3087
2823 // Calculate the yaw 3088 // Get the bearing (yaw)
2824 // subtracting PI_BY_TWO is required to compensate for the odd SL co-ordinate system 3089 LSL_Vector a1 = new LSL_Vector(0,0,0);
2825 angle.x = llAtan2(delta.z, delta.y) - ScriptBaseClass.PI_BY_TWO; 3090 a1.z = llAtan2(d1.y, d1.x);
2826 3091
2827 // Calculate pitch 3092 // Get the elevation (pitch)
2828 angle.y = llAtan2(delta.x, llSqrt((delta.y * delta.y) + (delta.z * delta.z))); 3093 LSL_Vector a2 = new LSL_Vector(0,0,0);
3094 a2.y= -llAtan2(d1.z, llSqrt((d1.x * d1.x) + (d1.y * d1.y)));
2829 3095
2830 // we need to convert from a vector describing 3096 LSL_Rotation r1 = llEuler2Rot(a1);
2831 // the angles of rotation in radians into rotation value 3097 LSL_Rotation r2 = llEuler2Rot(a2);
2832 LSL_Rotation rot = llEuler2Rot(angle); 3098 LSL_Rotation r3 = new LSL_Rotation(0.000000, 0.707107, 0.000000, 0.707107);
2833
2834 // Per discussion with Melanie, for non-physical objects llLookAt appears to simply
2835 // set the rotation of the object, copy that behavior
2836 PhysicsActor pa = m_host.PhysActor;
2837 3099
2838 if (strength == 0 || pa == null || !pa.IsPhysical) 3100 if (m_host.PhysActor == null || !m_host.PhysActor.IsPhysical)
2839 { 3101 {
2840 llSetRot(rot); 3102 // Do nothing if either value is 0 (this has been checked in SL)
3103 if (strength <= 0.0 || damping <= 0.0)
3104 return;
3105
3106 llSetRot(r3 * r2 * r1);
2841 } 3107 }
2842 else 3108 else
2843 { 3109 {
2844 m_host.StartLookAt(Rot2Quaternion(rot), (float)strength, (float)damping); 3110 if (strength == 0)
3111 {
3112 llSetRot(r3 * r2 * r1);
3113 return;
3114 }
3115
3116 m_host.StartLookAt(Rot2Quaternion(r3 * r2 * r1), (float)strength, (float)damping);
2845 } 3117 }
2846 } 3118 }
2847 3119
@@ -2887,17 +3159,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2887 } 3159 }
2888 else 3160 else
2889 { 3161 {
2890 if (m_host.IsRoot) 3162 // new SL always returns object mass
2891 { 3163// if (m_host.IsRoot)
3164// {
2892 return m_host.ParentGroup.GetMass(); 3165 return m_host.ParentGroup.GetMass();
2893 } 3166// }
2894 else 3167// else
2895 { 3168// {
2896 return m_host.GetMass(); 3169// return m_host.GetMass();
2897 } 3170// }
2898 } 3171 }
2899 } 3172 }
2900 3173
3174
3175 public LSL_Float llGetMassMKS()
3176 {
3177 return 100f * llGetMass();
3178 }
3179
2901 public void llCollisionFilter(string name, string id, int accept) 3180 public void llCollisionFilter(string name, string id, int accept)
2902 { 3181 {
2903 m_host.AddScriptLPS(1); 3182 m_host.AddScriptLPS(1);
@@ -2972,7 +3251,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2972 IAttachmentsModule attachmentsModule = m_ScriptEngine.World.AttachmentsModule; 3251 IAttachmentsModule attachmentsModule = m_ScriptEngine.World.AttachmentsModule;
2973 3252
2974 if (attachmentsModule != null) 3253 if (attachmentsModule != null)
2975 return attachmentsModule.AttachObject(presence, grp, (uint)attachmentPoint, false); 3254 return attachmentsModule.AttachObject(presence, grp, (uint)attachmentPoint, false, true);
2976 else 3255 else
2977 return false; 3256 return false;
2978 } 3257 }
@@ -3005,9 +3284,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3005 { 3284 {
3006 m_host.AddScriptLPS(1); 3285 m_host.AddScriptLPS(1);
3007 3286
3008// if (m_host.ParentGroup.RootPart.AttachmentPoint == 0)
3009// return;
3010
3011 if (m_item.PermsGranter != m_host.OwnerID) 3287 if (m_item.PermsGranter != m_host.OwnerID)
3012 return; 3288 return;
3013 3289
@@ -3050,6 +3326,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3050 3326
3051 public void llInstantMessage(string user, string message) 3327 public void llInstantMessage(string user, string message)
3052 { 3328 {
3329 UUID result;
3330 if (!UUID.TryParse(user, out result))
3331 {
3332 ShoutError("An invalid key was passed to llInstantMessage");
3333 ScriptSleep(2000);
3334 return;
3335 }
3336
3337
3053 m_host.AddScriptLPS(1); 3338 m_host.AddScriptLPS(1);
3054 3339
3055 // We may be able to use ClientView.SendInstantMessage here, but we need a client instance. 3340 // We may be able to use ClientView.SendInstantMessage here, but we need a client instance.
@@ -3064,14 +3349,34 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3064 UUID friendTransactionID = UUID.Random(); 3349 UUID friendTransactionID = UUID.Random();
3065 3350
3066 //m_pendingFriendRequests.Add(friendTransactionID, fromAgentID); 3351 //m_pendingFriendRequests.Add(friendTransactionID, fromAgentID);
3067 3352
3068 GridInstantMessage msg = new GridInstantMessage(); 3353 GridInstantMessage msg = new GridInstantMessage();
3069 msg.fromAgentID = new Guid(m_host.UUID.ToString()); // fromAgentID.Guid; 3354 msg.fromAgentID = new Guid(m_host.UUID.ToString()); // fromAgentID.Guid;
3070 msg.toAgentID = new Guid(user); // toAgentID.Guid; 3355 msg.toAgentID = new Guid(user); // toAgentID.Guid;
3071 msg.imSessionID = new Guid(friendTransactionID.ToString()); // This is the item we're mucking with here 3356 msg.imSessionID = new Guid(friendTransactionID.ToString()); // This is the item we're mucking with here
3072// m_log.Debug("[Scripting IM]: From:" + msg.fromAgentID.ToString() + " To: " + msg.toAgentID.ToString() + " Session:" + msg.imSessionID.ToString() + " Message:" + message); 3357// m_log.Debug("[Scripting IM]: From:" + msg.fromAgentID.ToString() + " To: " + msg.toAgentID.ToString() + " Session:" + msg.imSessionID.ToString() + " Message:" + message);
3073// m_log.Debug("[Scripting IM]: Filling Session: " + msg.imSessionID.ToString()); 3358// m_log.Debug("[Scripting IM]: Filling Session: " + msg.imSessionID.ToString());
3074 msg.timestamp = (uint)Util.UnixTimeSinceEpoch();// timestamp; 3359// DateTime dt = DateTime.UtcNow;
3360//
3361// // Ticks from UtcNow, but make it look like local. Evil, huh?
3362// dt = DateTime.SpecifyKind(dt, DateTimeKind.Local);
3363//
3364// try
3365// {
3366// // Convert that to the PST timezone
3367// TimeZoneInfo timeZoneInfo = TimeZoneInfo.FindSystemTimeZoneById("America/Los_Angeles");
3368// dt = TimeZoneInfo.ConvertTime(dt, timeZoneInfo);
3369// }
3370// catch
3371// {
3372// // No logging here, as it could be VERY spammy
3373// }
3374//
3375// // And make it look local again to fool the unix time util
3376// dt = DateTime.SpecifyKind(dt, DateTimeKind.Utc);
3377
3378 msg.timestamp = (uint)Util.UnixTimeSinceEpoch();
3379
3075 //if (client != null) 3380 //if (client != null)
3076 //{ 3381 //{
3077 msg.fromAgentName = m_host.Name;//client.FirstName + " " + client.LastName;// fromAgentName; 3382 msg.fromAgentName = m_host.Name;//client.FirstName + " " + client.LastName;// fromAgentName;
@@ -3085,12 +3390,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3085 msg.message = message.Substring(0, 1024); 3390 msg.message = message.Substring(0, 1024);
3086 else 3391 else
3087 msg.message = message; 3392 msg.message = message;
3088 msg.dialog = (byte)19; // messgage from script ??? // dialog; 3393 msg.dialog = (byte)19; // MessageFromObject
3089 msg.fromGroup = false;// fromGroup; 3394 msg.fromGroup = false;// fromGroup;
3090 msg.offline = (byte)0; //offline; 3395 msg.offline = (byte)0; //offline;
3091 msg.ParentEstateID = 0; //ParentEstateID; 3396 msg.ParentEstateID = World.RegionInfo.EstateSettings.EstateID;
3092 msg.Position = new Vector3(m_host.AbsolutePosition); 3397 msg.Position = new Vector3(m_host.AbsolutePosition);
3093 msg.RegionID = World.RegionInfo.RegionID.Guid;//RegionID.Guid; 3398 msg.RegionID = World.RegionInfo.RegionID.Guid;
3094 msg.binaryBucket 3399 msg.binaryBucket
3095 = Util.StringToBytes256( 3400 = Util.StringToBytes256(
3096 "{0}/{1}/{2}/{3}", 3401 "{0}/{1}/{2}/{3}",
@@ -3118,7 +3423,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3118 } 3423 }
3119 3424
3120 emailModule.SendEmail(m_host.UUID, address, subject, message); 3425 emailModule.SendEmail(m_host.UUID, address, subject, message);
3121 llSleep(EMAIL_PAUSE_TIME); 3426 ScriptSleep(EMAIL_PAUSE_TIME * 1000);
3122 } 3427 }
3123 3428
3124 public void llGetNextEmail(string address, string subject) 3429 public void llGetNextEmail(string address, string subject)
@@ -3362,15 +3667,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3362 int implicitPerms = ScriptBaseClass.PERMISSION_TAKE_CONTROLS | 3667 int implicitPerms = ScriptBaseClass.PERMISSION_TAKE_CONTROLS |
3363 ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION | 3668 ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION |
3364 ScriptBaseClass.PERMISSION_CONTROL_CAMERA | 3669 ScriptBaseClass.PERMISSION_CONTROL_CAMERA |
3670 ScriptBaseClass.PERMISSION_TRACK_CAMERA |
3365 ScriptBaseClass.PERMISSION_ATTACH; 3671 ScriptBaseClass.PERMISSION_ATTACH;
3366 3672
3367 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms 3673 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms
3368 { 3674 {
3369 lock (m_host.TaskInventory) 3675 m_host.TaskInventory.LockItemsForWrite(true);
3370 { 3676 m_host.TaskInventory[m_item.ItemID].PermsGranter = agentID;
3371 m_host.TaskInventory[m_item.ItemID].PermsGranter = agentID; 3677 m_host.TaskInventory[m_item.ItemID].PermsMask = perm;
3372 m_host.TaskInventory[m_item.ItemID].PermsMask = perm; 3678 m_host.TaskInventory.LockItemsForWrite(false);
3373 }
3374 3679
3375 m_ScriptEngine.PostScriptEvent(m_item.ItemID, new EventParams( 3680 m_ScriptEngine.PostScriptEvent(m_item.ItemID, new EventParams(
3376 "run_time_permissions", new Object[] { 3681 "run_time_permissions", new Object[] {
@@ -3380,28 +3685,44 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3380 return; 3685 return;
3381 } 3686 }
3382 } 3687 }
3383 else if (m_host.SitTargetAvatar == agentID) // Sitting avatar 3688 else
3384 { 3689 {
3385 // When agent is sitting, certain permissions are implicit if requested from sitting agent 3690 bool sitting = false;
3386 int implicitPerms = ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION | 3691 if (m_host.SitTargetAvatar == agentID)
3387 ScriptBaseClass.PERMISSION_CONTROL_CAMERA | 3692 {
3388 ScriptBaseClass.PERMISSION_TRACK_CAMERA | 3693 sitting = true;
3389 ScriptBaseClass.PERMISSION_TAKE_CONTROLS; 3694 }
3695 else
3696 {
3697 foreach (SceneObjectPart p in m_host.ParentGroup.Parts)
3698 {
3699 if (p.SitTargetAvatar == agentID)
3700 sitting = true;
3701 }
3702 }
3390 3703
3391 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms 3704 if (sitting)
3392 { 3705 {
3393 lock (m_host.TaskInventory) 3706 // When agent is sitting, certain permissions are implicit if requested from sitting agent
3707 int implicitPerms = ScriptBaseClass.PERMISSION_TRIGGER_ANIMATION |
3708 ScriptBaseClass.PERMISSION_CONTROL_CAMERA |
3709 ScriptBaseClass.PERMISSION_TRACK_CAMERA |
3710 ScriptBaseClass.PERMISSION_TAKE_CONTROLS;
3711
3712 if ((perm & (~implicitPerms)) == 0) // Requested only implicit perms
3394 { 3713 {
3714 m_host.TaskInventory.LockItemsForWrite(true);
3395 m_host.TaskInventory[m_item.ItemID].PermsGranter = agentID; 3715 m_host.TaskInventory[m_item.ItemID].PermsGranter = agentID;
3396 m_host.TaskInventory[m_item.ItemID].PermsMask = perm; 3716 m_host.TaskInventory[m_item.ItemID].PermsMask = perm;
3397 } 3717 m_host.TaskInventory.LockItemsForWrite(false);
3398 3718
3399 m_ScriptEngine.PostScriptEvent(m_item.ItemID, new EventParams( 3719 m_ScriptEngine.PostScriptEvent(m_item.ItemID, new EventParams(
3400 "run_time_permissions", new Object[] { 3720 "run_time_permissions", new Object[] {
3401 new LSL_Integer(perm) }, 3721 new LSL_Integer(perm) },
3402 new DetectParams[0])); 3722 new DetectParams[0]));
3403 3723
3404 return; 3724 return;
3725 }
3405 } 3726 }
3406 } 3727 }
3407 3728
@@ -3438,11 +3759,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3438 3759
3439 if (!m_waitingForScriptAnswer) 3760 if (!m_waitingForScriptAnswer)
3440 { 3761 {
3441 lock (m_host.TaskInventory) 3762 m_host.TaskInventory.LockItemsForWrite(true);
3442 { 3763 m_host.TaskInventory[m_item.ItemID].PermsGranter = agentID;
3443 m_host.TaskInventory[m_item.ItemID].PermsGranter = agentID; 3764 m_host.TaskInventory[m_item.ItemID].PermsMask = 0;
3444 m_host.TaskInventory[m_item.ItemID].PermsMask = 0; 3765 m_host.TaskInventory.LockItemsForWrite(false);
3445 }
3446 3766
3447 presence.ControllingClient.OnScriptAnswer += handleScriptAnswer; 3767 presence.ControllingClient.OnScriptAnswer += handleScriptAnswer;
3448 m_waitingForScriptAnswer=true; 3768 m_waitingForScriptAnswer=true;
@@ -3471,14 +3791,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3471 if ((answer & ScriptBaseClass.PERMISSION_TAKE_CONTROLS) == 0) 3791 if ((answer & ScriptBaseClass.PERMISSION_TAKE_CONTROLS) == 0)
3472 llReleaseControls(); 3792 llReleaseControls();
3473 3793
3474 lock (m_host.TaskInventory) 3794 m_host.TaskInventory.LockItemsForWrite(true);
3475 { 3795 m_host.TaskInventory[m_item.ItemID].PermsMask = answer;
3476 m_host.TaskInventory[m_item.ItemID].PermsMask = answer; 3796 m_host.TaskInventory.LockItemsForWrite(false);
3477 } 3797
3478 3798 m_ScriptEngine.PostScriptEvent(m_item.ItemID, new EventParams(
3479 m_ScriptEngine.PostScriptEvent( 3799 "run_time_permissions", new Object[] {
3480 m_item.ItemID, 3800 new LSL_Integer(answer) },
3481 new EventParams("run_time_permissions", new Object[] { new LSL_Integer(answer) }, new DetectParams[0])); 3801 new DetectParams[0]));
3482 } 3802 }
3483 3803
3484 public LSL_String llGetPermissionsKey() 3804 public LSL_String llGetPermissionsKey()
@@ -3517,14 +3837,25 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3517 public void llSetLinkColor(int linknumber, LSL_Vector color, int face) 3837 public void llSetLinkColor(int linknumber, LSL_Vector color, int face)
3518 { 3838 {
3519 List<SceneObjectPart> parts = GetLinkParts(linknumber); 3839 List<SceneObjectPart> parts = GetLinkParts(linknumber);
3520 3840 if (parts.Count > 0)
3521 foreach (SceneObjectPart part in parts) 3841 {
3522 part.SetFaceColor(new Vector3((float)color.x, (float)color.y, (float)color.z), face); 3842 try
3843 {
3844 parts[0].ParentGroup.areUpdatesSuspended = true;
3845 foreach (SceneObjectPart part in parts)
3846 part.SetFaceColor(new Vector3((float)color.x, (float)color.y, (float)color.z), face);
3847 }
3848 finally
3849 {
3850 parts[0].ParentGroup.areUpdatesSuspended = false;
3851 }
3852 }
3523 } 3853 }
3524 3854
3525 public void llCreateLink(string target, int parent) 3855 public void llCreateLink(string target, int parent)
3526 { 3856 {
3527 m_host.AddScriptLPS(1); 3857 m_host.AddScriptLPS(1);
3858
3528 UUID targetID; 3859 UUID targetID;
3529 3860
3530 if (!UUID.TryParse(target, out targetID)) 3861 if (!UUID.TryParse(target, out targetID))
@@ -3630,10 +3961,22 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3630 // Restructuring Multiple Prims. 3961 // Restructuring Multiple Prims.
3631 List<SceneObjectPart> parts = new List<SceneObjectPart>(parentPrim.Parts); 3962 List<SceneObjectPart> parts = new List<SceneObjectPart>(parentPrim.Parts);
3632 parts.Remove(parentPrim.RootPart); 3963 parts.Remove(parentPrim.RootPart);
3633 foreach (SceneObjectPart part in parts) 3964 if (parts.Count > 0)
3634 { 3965 {
3635 parentPrim.DelinkFromGroup(part.LocalId, true); 3966 try
3967 {
3968 parts[0].ParentGroup.areUpdatesSuspended = true;
3969 foreach (SceneObjectPart part in parts)
3970 {
3971 parentPrim.DelinkFromGroup(part.LocalId, true);
3972 }
3973 }
3974 finally
3975 {
3976 parts[0].ParentGroup.areUpdatesSuspended = false;
3977 }
3636 } 3978 }
3979
3637 parentPrim.HasGroupChanged = true; 3980 parentPrim.HasGroupChanged = true;
3638 parentPrim.ScheduleGroupForFullUpdate(); 3981 parentPrim.ScheduleGroupForFullUpdate();
3639 parentPrim.TriggerScriptChangedEvent(Changed.LINK); 3982 parentPrim.TriggerScriptChangedEvent(Changed.LINK);
@@ -3642,12 +3985,22 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3642 { 3985 {
3643 SceneObjectPart newRoot = parts[0]; 3986 SceneObjectPart newRoot = parts[0];
3644 parts.Remove(newRoot); 3987 parts.Remove(newRoot);
3645 foreach (SceneObjectPart part in parts) 3988
3989 try
3646 { 3990 {
3647 // Required for linking 3991 parts[0].ParentGroup.areUpdatesSuspended = true;
3648 part.ClearUpdateSchedule(); 3992 foreach (SceneObjectPart part in parts)
3649 newRoot.ParentGroup.LinkToGroup(part.ParentGroup); 3993 {
3994 part.ClearUpdateSchedule();
3995 newRoot.ParentGroup.LinkToGroup(part.ParentGroup);
3996 }
3650 } 3997 }
3998 finally
3999 {
4000 parts[0].ParentGroup.areUpdatesSuspended = false;
4001 }
4002
4003
3651 newRoot.ParentGroup.HasGroupChanged = true; 4004 newRoot.ParentGroup.HasGroupChanged = true;
3652 newRoot.ParentGroup.ScheduleGroupForFullUpdate(); 4005 newRoot.ParentGroup.ScheduleGroupForFullUpdate();
3653 } 4006 }
@@ -3667,6 +4020,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3667 public void llBreakAllLinks() 4020 public void llBreakAllLinks()
3668 { 4021 {
3669 m_host.AddScriptLPS(1); 4022 m_host.AddScriptLPS(1);
4023
4024 TaskInventoryItem item = m_item;
4025
4026 if ((item.PermsMask & ScriptBaseClass.PERMISSION_CHANGE_LINKS) == 0
4027 && !m_automaticLinkPermission)
4028 {
4029 ShoutError("Script trying to link but PERMISSION_CHANGE_LINKS permission not set!");
4030 return;
4031 }
4032
3670 SceneObjectGroup parentPrim = m_host.ParentGroup; 4033 SceneObjectGroup parentPrim = m_host.ParentGroup;
3671 if (parentPrim.AttachmentPoint != 0) 4034 if (parentPrim.AttachmentPoint != 0)
3672 return; // Fail silently if attached 4035 return; // Fail silently if attached
@@ -3686,25 +4049,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3686 public LSL_String llGetLinkKey(int linknum) 4049 public LSL_String llGetLinkKey(int linknum)
3687 { 4050 {
3688 m_host.AddScriptLPS(1); 4051 m_host.AddScriptLPS(1);
3689 List<UUID> keytable = new List<UUID>();
3690 // parse for sitting avatare-uuids
3691 World.ForEachRootScenePresence(delegate(ScenePresence presence)
3692 {
3693 if (presence.ParentID != 0 && m_host.ParentGroup.ContainsPart(presence.ParentID))
3694 keytable.Add(presence.UUID);
3695 });
3696
3697 int totalprims = m_host.ParentGroup.PrimCount + keytable.Count;
3698 if (linknum > m_host.ParentGroup.PrimCount && linknum <= totalprims)
3699 {
3700 return keytable[totalprims - linknum].ToString();
3701 }
3702
3703 if (linknum == 1 && m_host.ParentGroup.PrimCount == 1 && keytable.Count == 1)
3704 {
3705 return m_host.UUID.ToString();
3706 }
3707
3708 SceneObjectPart part = m_host.ParentGroup.GetLinkNumPart(linknum); 4052 SceneObjectPart part = m_host.ParentGroup.GetLinkNumPart(linknum);
3709 if (part != null) 4053 if (part != null)
3710 { 4054 {
@@ -3712,6 +4056,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3712 } 4056 }
3713 else 4057 else
3714 { 4058 {
4059 if (linknum > m_host.ParentGroup.PrimCount || (linknum == 1 && m_host.ParentGroup.PrimCount == 1))
4060 {
4061 linknum -= (m_host.ParentGroup.PrimCount) + 1;
4062
4063 if (linknum < 0)
4064 return UUID.Zero.ToString();
4065
4066 List<ScenePresence> avatars = GetLinkAvatars(ScriptBaseClass.LINK_SET);
4067 if (avatars.Count > linknum)
4068 {
4069 return avatars[linknum].UUID.ToString();
4070 }
4071 }
3715 return UUID.Zero.ToString(); 4072 return UUID.Zero.ToString();
3716 } 4073 }
3717 } 4074 }
@@ -3811,17 +4168,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3811 m_host.AddScriptLPS(1); 4168 m_host.AddScriptLPS(1);
3812 int count = 0; 4169 int count = 0;
3813 4170
3814 lock (m_host.TaskInventory) 4171 m_host.TaskInventory.LockItemsForRead(true);
4172 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
3815 { 4173 {
3816 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 4174 if (inv.Value.Type == type || type == -1)
3817 { 4175 {
3818 if (inv.Value.Type == type || type == -1) 4176 count = count + 1;
3819 {
3820 count = count + 1;
3821 }
3822 } 4177 }
3823 } 4178 }
3824 4179
4180 m_host.TaskInventory.LockItemsForRead(false);
3825 return count; 4181 return count;
3826 } 4182 }
3827 4183
@@ -3830,16 +4186,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3830 m_host.AddScriptLPS(1); 4186 m_host.AddScriptLPS(1);
3831 ArrayList keys = new ArrayList(); 4187 ArrayList keys = new ArrayList();
3832 4188
3833 lock (m_host.TaskInventory) 4189 m_host.TaskInventory.LockItemsForRead(true);
4190 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
3834 { 4191 {
3835 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 4192 if (inv.Value.Type == type || type == -1)
3836 { 4193 {
3837 if (inv.Value.Type == type || type == -1) 4194 keys.Add(inv.Value.Name);
3838 {
3839 keys.Add(inv.Value.Name);
3840 }
3841 } 4195 }
3842 } 4196 }
4197 m_host.TaskInventory.LockItemsForRead(false);
3843 4198
3844 if (keys.Count == 0) 4199 if (keys.Count == 0)
3845 { 4200 {
@@ -3876,25 +4231,25 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3876 } 4231 }
3877 4232
3878 // move the first object found with this inventory name 4233 // move the first object found with this inventory name
3879 lock (m_host.TaskInventory) 4234 m_host.TaskInventory.LockItemsForRead(true);
4235 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
3880 { 4236 {
3881 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 4237 if (inv.Value.Name == inventory)
3882 { 4238 {
3883 if (inv.Value.Name == inventory) 4239 found = true;
3884 { 4240 objId = inv.Key;
3885 found = true; 4241 assetType = inv.Value.Type;
3886 objId = inv.Key; 4242 objName = inv.Value.Name;
3887 assetType = inv.Value.Type; 4243 break;
3888 objName = inv.Value.Name;
3889 break;
3890 }
3891 } 4244 }
3892 } 4245 }
4246 m_host.TaskInventory.LockItemsForRead(false);
3893 4247
3894 if (!found) 4248 if (!found)
3895 { 4249 {
3896 llSay(0, String.Format("Could not find object '{0}'", inventory)); 4250 llSay(0, String.Format("Could not find object '{0}'", inventory));
3897 throw new Exception(String.Format("The inventory object '{0}' could not be found", inventory)); 4251 return;
4252// throw new Exception(String.Format("The inventory object '{0}' could not be found", inventory));
3898 } 4253 }
3899 4254
3900 // check if destination is an object 4255 // check if destination is an object
@@ -3920,48 +4275,68 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3920 return; 4275 return;
3921 } 4276 }
3922 } 4277 }
4278
3923 // destination is an avatar 4279 // destination is an avatar
3924 InventoryItemBase agentItem = World.MoveTaskInventoryItem(destId, UUID.Zero, m_host, objId); 4280 InventoryItemBase agentItem = World.MoveTaskInventoryItem(destId, UUID.Zero, m_host, objId);
3925 4281
3926 if (agentItem == null) 4282 if (agentItem == null)
3927 return; 4283 return;
3928 4284
3929 byte[] bucket = new byte[17]; 4285 byte[] bucket = new byte[1];
3930 bucket[0] = (byte)assetType; 4286 bucket[0] = (byte)assetType;
3931 byte[] objBytes = agentItem.ID.GetBytes(); 4287 //byte[] objBytes = agentItem.ID.GetBytes();
3932 Array.Copy(objBytes, 0, bucket, 1, 16); 4288 //Array.Copy(objBytes, 0, bucket, 1, 16);
3933 4289
3934 GridInstantMessage msg = new GridInstantMessage(World, 4290 GridInstantMessage msg = new GridInstantMessage(World,
3935 m_host.UUID, m_host.Name+", an object owned by "+ 4291 m_host.OwnerID, m_host.Name, destId,
3936 resolveName(m_host.OwnerID)+",", destId,
3937 (byte)InstantMessageDialog.TaskInventoryOffered, 4292 (byte)InstantMessageDialog.TaskInventoryOffered,
3938 false, objName+"\n"+m_host.Name+" is located at "+ 4293 false, objName+". "+m_host.Name+" is located at "+
3939 World.RegionInfo.RegionName+" "+ 4294 World.RegionInfo.RegionName+" "+
3940 m_host.AbsolutePosition.ToString(), 4295 m_host.AbsolutePosition.ToString(),
3941 agentItem.ID, true, m_host.AbsolutePosition, 4296 agentItem.ID, true, m_host.AbsolutePosition,
3942 bucket); 4297 bucket);
3943 if (m_TransferModule != null) 4298
3944 m_TransferModule.SendInstantMessage(msg, delegate(bool success) {}); 4299 ScenePresence sp;
4300
4301 if (World.TryGetScenePresence(destId, out sp))
4302 {
4303 sp.ControllingClient.SendInstantMessage(msg);
4304 }
4305 else
4306 {
4307 if (m_TransferModule != null)
4308 m_TransferModule.SendInstantMessage(msg, delegate(bool success) {});
4309 }
4310
4311 //This delay should only occur when giving inventory to avatars.
3945 ScriptSleep(3000); 4312 ScriptSleep(3000);
3946 } 4313 }
3947 } 4314 }
3948 4315
4316 [DebuggerNonUserCode]
3949 public void llRemoveInventory(string name) 4317 public void llRemoveInventory(string name)
3950 { 4318 {
3951 m_host.AddScriptLPS(1); 4319 m_host.AddScriptLPS(1);
3952 4320
3953 lock (m_host.TaskInventory) 4321 List<TaskInventoryItem> inv;
4322 try
3954 { 4323 {
3955 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 4324 m_host.TaskInventory.LockItemsForRead(true);
4325 inv = new List<TaskInventoryItem>(m_host.TaskInventory.Values);
4326 }
4327 finally
4328 {
4329 m_host.TaskInventory.LockItemsForRead(false);
4330 }
4331 foreach (TaskInventoryItem item in inv)
4332 {
4333 if (item.Name == name)
3956 { 4334 {
3957 if (item.Name == name) 4335 if (item.ItemID == m_item.ItemID)
3958 { 4336 throw new ScriptDeleteException();
3959 if (item.ItemID == m_item.ItemID) 4337 else
3960 throw new ScriptDeleteException(); 4338 m_host.Inventory.RemoveInventoryItem(item.ItemID);
3961 else 4339 return;
3962 m_host.Inventory.RemoveInventoryItem(item.ItemID);
3963 return;
3964 }
3965 } 4340 }
3966 } 4341 }
3967 } 4342 }
@@ -3996,115 +4371,122 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3996 { 4371 {
3997 m_host.AddScriptLPS(1); 4372 m_host.AddScriptLPS(1);
3998 4373
3999 UUID uuid = (UUID)id; 4374 UUID uuid;
4000 PresenceInfo pinfo = null; 4375 if (UUID.TryParse(id, out uuid))
4001 UserAccount account;
4002
4003 UserInfoCacheEntry ce;
4004 if (!m_userInfoCache.TryGetValue(uuid, out ce))
4005 { 4376 {
4006 account = World.UserAccountService.GetUserAccount(World.RegionInfo.ScopeID, uuid); 4377 PresenceInfo pinfo = null;
4007 if (account == null) 4378 UserAccount account;
4379
4380 UserInfoCacheEntry ce;
4381 if (!m_userInfoCache.TryGetValue(uuid, out ce))
4008 { 4382 {
4009 m_userInfoCache[uuid] = null; // Cache negative 4383 account = World.UserAccountService.GetUserAccount(World.RegionInfo.ScopeID, uuid);
4010 return UUID.Zero.ToString(); 4384 if (account == null)
4011 } 4385 {
4386 m_userInfoCache[uuid] = null; // Cache negative
4387 return UUID.Zero.ToString();
4388 }
4012 4389
4013 4390
4014 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() }); 4391 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() });
4015 if (pinfos != null && pinfos.Length > 0) 4392 if (pinfos != null && pinfos.Length > 0)
4016 {
4017 foreach (PresenceInfo p in pinfos)
4018 { 4393 {
4019 if (p.RegionID != UUID.Zero) 4394 foreach (PresenceInfo p in pinfos)
4020 { 4395 {
4021 pinfo = p; 4396 if (p.RegionID != UUID.Zero)
4397 {
4398 pinfo = p;
4399 }
4022 } 4400 }
4023 } 4401 }
4024 }
4025 4402
4026 ce = new UserInfoCacheEntry(); 4403 ce = new UserInfoCacheEntry();
4027 ce.time = Util.EnvironmentTickCount(); 4404 ce.time = Util.EnvironmentTickCount();
4028 ce.account = account; 4405 ce.account = account;
4029 ce.pinfo = pinfo; 4406 ce.pinfo = pinfo;
4030 } 4407 m_userInfoCache[uuid] = ce;
4031 else 4408 }
4032 { 4409 else
4033 if (ce == null) 4410 {
4034 return UUID.Zero.ToString(); 4411 if (ce == null)
4412 return UUID.Zero.ToString();
4035 4413
4036 account = ce.account; 4414 account = ce.account;
4037 pinfo = ce.pinfo; 4415 pinfo = ce.pinfo;
4038 } 4416 }
4039 4417
4040 if (Util.EnvironmentTickCount() < ce.time || (Util.EnvironmentTickCount() - ce.time) >= 20000) 4418 if (Util.EnvironmentTickCount() < ce.time || (Util.EnvironmentTickCount() - ce.time) >= 20000)
4041 {
4042 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() });
4043 if (pinfos != null && pinfos.Length > 0)
4044 { 4419 {
4045 foreach (PresenceInfo p in pinfos) 4420 PresenceInfo[] pinfos = World.PresenceService.GetAgents(new string[] { uuid.ToString() });
4421 if (pinfos != null && pinfos.Length > 0)
4046 { 4422 {
4047 if (p.RegionID != UUID.Zero) 4423 foreach (PresenceInfo p in pinfos)
4048 { 4424 {
4049 pinfo = p; 4425 if (p.RegionID != UUID.Zero)
4426 {
4427 pinfo = p;
4428 }
4050 } 4429 }
4051 } 4430 }
4052 } 4431 else
4053 else 4432 pinfo = null;
4054 pinfo = null;
4055 4433
4056 ce.time = Util.EnvironmentTickCount(); 4434 ce.time = Util.EnvironmentTickCount();
4057 ce.pinfo = pinfo; 4435 ce.pinfo = pinfo;
4058 } 4436 }
4059 4437
4060 string reply = String.Empty; 4438 string reply = String.Empty;
4061 4439
4062 switch (data) 4440 switch (data)
4063 { 4441 {
4064 case 1: // DATA_ONLINE (0|1) 4442 case 1: // DATA_ONLINE (0|1)
4065 if (pinfo != null && pinfo.RegionID != UUID.Zero) 4443 if (pinfo != null && pinfo.RegionID != UUID.Zero)
4066 reply = "1"; 4444 reply = "1";
4067 else 4445 else
4068 reply = "0"; 4446 reply = "0";
4069 break; 4447 break;
4070 case 2: // DATA_NAME (First Last) 4448 case 2: // DATA_NAME (First Last)
4071 reply = account.FirstName + " " + account.LastName; 4449 reply = account.FirstName + " " + account.LastName;
4072 break; 4450 break;
4073 case 3: // DATA_BORN (YYYY-MM-DD) 4451 case 3: // DATA_BORN (YYYY-MM-DD)
4074 DateTime born = new DateTime(1970, 1, 1, 0, 0, 0, 0); 4452 DateTime born = new DateTime(1970, 1, 1, 0, 0, 0, 0);
4075 born = born.AddSeconds(account.Created); 4453 born = born.AddSeconds(account.Created);
4076 reply = born.ToString("yyyy-MM-dd"); 4454 reply = born.ToString("yyyy-MM-dd");
4077 break; 4455 break;
4078 case 4: // DATA_RATING (0,0,0,0,0,0) 4456 case 4: // DATA_RATING (0,0,0,0,0,0)
4079 reply = "0,0,0,0,0,0"; 4457 reply = "0,0,0,0,0,0";
4080 break; 4458 break;
4081 case 7: // DATA_USERLEVEL (integer) 4459 case 8: // DATA_PAYINFO (0|1|2|3)
4082 reply = account.UserLevel.ToString(); 4460 reply = "0";
4083 break; 4461 break;
4084 case 8: // DATA_PAYINFO (0|1|2|3) 4462 default:
4085 reply = "0"; 4463 return UUID.Zero.ToString(); // Raise no event
4086 break; 4464 }
4087 default:
4088 return UUID.Zero.ToString(); // Raise no event
4089 }
4090 4465
4091 UUID rq = UUID.Random(); 4466 UUID rq = UUID.Random();
4092 4467
4093 UUID tid = AsyncCommands. 4468 UUID tid = AsyncCommands.
4094 DataserverPlugin.RegisterRequest(m_host.LocalId, 4469 DataserverPlugin.RegisterRequest(m_host.LocalId,
4095 m_item.ItemID, rq.ToString()); 4470 m_item.ItemID, rq.ToString());
4096 4471
4097 AsyncCommands. 4472 AsyncCommands.
4098 DataserverPlugin.DataserverReply(rq.ToString(), reply); 4473 DataserverPlugin.DataserverReply(rq.ToString(), reply);
4099 4474
4100 ScriptSleep(100); 4475 ScriptSleep(100);
4101 return tid.ToString(); 4476 return tid.ToString();
4477 }
4478 else
4479 {
4480 ShoutError("Invalid UUID passed to llRequestAgentData.");
4481 }
4482 return "";
4102 } 4483 }
4103 4484
4104 public LSL_String llRequestInventoryData(string name) 4485 public LSL_String llRequestInventoryData(string name)
4105 { 4486 {
4106 m_host.AddScriptLPS(1); 4487 m_host.AddScriptLPS(1);
4107 4488
4489 //Clone is thread safe
4108 TaskInventoryDictionary itemDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 4490 TaskInventoryDictionary itemDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
4109 4491
4110 foreach (TaskInventoryItem item in itemDictionary.Values) 4492 foreach (TaskInventoryItem item in itemDictionary.Values)
@@ -4156,19 +4538,64 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4156 if (UUID.TryParse(agent, out agentId)) 4538 if (UUID.TryParse(agent, out agentId))
4157 { 4539 {
4158 ScenePresence presence = World.GetScenePresence(agentId); 4540 ScenePresence presence = World.GetScenePresence(agentId);
4159 if (presence != null) 4541 if (presence != null && presence.PresenceType != PresenceType.Npc)
4160 { 4542 {
4543 // agent must not be a god
4544 if (presence.UserLevel >= 200) return;
4545
4161 // agent must be over the owners land 4546 // agent must be over the owners land
4162 if (m_host.OwnerID == World.LandChannel.GetLandObject( 4547 if (m_host.OwnerID == World.LandChannel.GetLandObject(
4163 presence.AbsolutePosition.X, presence.AbsolutePosition.Y).LandData.OwnerID) 4548 presence.AbsolutePosition.X, presence.AbsolutePosition.Y).LandData.OwnerID)
4164 { 4549 {
4165 World.TeleportClientHome(agentId, presence.ControllingClient); 4550 if (!World.TeleportClientHome(agentId, presence.ControllingClient))
4551 {
4552 // They can't be teleported home for some reason
4553 GridRegion regionInfo = World.GridService.GetRegionByUUID(UUID.Zero, new UUID("2b02daac-e298-42fa-9a75-f488d37896e6"));
4554 if (regionInfo != null)
4555 {
4556 World.RequestTeleportLocation(
4557 presence.ControllingClient, regionInfo.RegionHandle, new Vector3(128, 128, 23), Vector3.Zero,
4558 (uint)(Constants.TeleportFlags.SetLastToTarget | Constants.TeleportFlags.ViaHome));
4559 }
4560 }
4166 } 4561 }
4167 } 4562 }
4168 } 4563 }
4169 ScriptSleep(5000); 4564 ScriptSleep(5000);
4170 } 4565 }
4171 4566
4567 public void llTeleportAgent(string agent, string simname, LSL_Vector pos, LSL_Vector lookAt)
4568 {
4569 m_host.AddScriptLPS(1);
4570 UUID agentId = new UUID();
4571 if (UUID.TryParse(agent, out agentId))
4572 {
4573 ScenePresence presence = World.GetScenePresence(agentId);
4574 if (presence != null && presence.PresenceType != PresenceType.Npc)
4575 {
4576 // agent must not be a god
4577 if (presence.GodLevel >= 200) return;
4578
4579 if (simname == String.Empty)
4580 simname = World.RegionInfo.RegionName;
4581
4582 // agent must be over the owners land
4583 if (m_host.OwnerID == World.LandChannel.GetLandObject(
4584 presence.AbsolutePosition.X, presence.AbsolutePosition.Y).LandData.OwnerID)
4585 {
4586 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);
4587 }
4588 else // or must be wearing the prim
4589 {
4590 if (m_host.ParentGroup.AttachmentPoint != 0 && m_host.OwnerID == presence.UUID)
4591 {
4592 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);
4593 }
4594 }
4595 }
4596 }
4597 }
4598
4172 public void llTextBox(string agent, string message, int chatChannel) 4599 public void llTextBox(string agent, string message, int chatChannel)
4173 { 4600 {
4174 IDialogModule dm = World.RequestModuleInterface<IDialogModule>(); 4601 IDialogModule dm = World.RequestModuleInterface<IDialogModule>();
@@ -4180,7 +4607,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4180 UUID av = new UUID(); 4607 UUID av = new UUID();
4181 if (!UUID.TryParse(agent,out av)) 4608 if (!UUID.TryParse(agent,out av))
4182 { 4609 {
4183 LSLError("First parameter to llDialog needs to be a key"); 4610 //LSLError("First parameter to llDialog needs to be a key");
4184 return; 4611 return;
4185 } 4612 }
4186 4613
@@ -4212,25 +4639,30 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4212 public void llCollisionSound(string impact_sound, double impact_volume) 4639 public void llCollisionSound(string impact_sound, double impact_volume)
4213 { 4640 {
4214 m_host.AddScriptLPS(1); 4641 m_host.AddScriptLPS(1);
4215 4642
4643 if(impact_sound == "")
4644 {
4645 m_host.CollisionSoundVolume = (float)impact_volume;
4646 m_host.CollisionSound = m_host.invalidCollisionSoundUUID;
4647 return;
4648 }
4216 // TODO: Parameter check logic required. 4649 // TODO: Parameter check logic required.
4217 UUID soundId = UUID.Zero; 4650 UUID soundId = UUID.Zero;
4218 if (!UUID.TryParse(impact_sound, out soundId)) 4651 if (!UUID.TryParse(impact_sound, out soundId))
4219 { 4652 {
4220 lock (m_host.TaskInventory) 4653 m_host.TaskInventory.LockItemsForRead(true);
4654 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
4221 { 4655 {
4222 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 4656 if (item.Type == (int)AssetType.Sound && item.Name == impact_sound)
4223 { 4657 {
4224 if (item.Type == (int)AssetType.Sound && item.Name == impact_sound) 4658 soundId = item.AssetID;
4225 { 4659 break;
4226 soundId = item.AssetID;
4227 break;
4228 }
4229 } 4660 }
4230 } 4661 }
4662 m_host.TaskInventory.LockItemsForRead(false);
4231 } 4663 }
4232 m_host.CollisionSound = soundId;
4233 m_host.CollisionSoundVolume = (float)impact_volume; 4664 m_host.CollisionSoundVolume = (float)impact_volume;
4665 m_host.CollisionSound = soundId;
4234 } 4666 }
4235 4667
4236 public LSL_String llGetAnimation(string id) 4668 public LSL_String llGetAnimation(string id)
@@ -4244,14 +4676,18 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4244 4676
4245 if (m_host.RegionHandle == presence.RegionHandle) 4677 if (m_host.RegionHandle == presence.RegionHandle)
4246 { 4678 {
4247 Dictionary<UUID, string> animationstateNames = DefaultAvatarAnimations.AnimStateNames;
4248
4249 if (presence != null) 4679 if (presence != null)
4250 { 4680 {
4251 AnimationSet currentAnims = presence.Animator.Animations; 4681 if (presence.SitGround)
4252 string currentAnimationState = String.Empty; 4682 return "Sitting on Ground";
4253 if (animationstateNames.TryGetValue(currentAnims.DefaultAnimation.AnimID, out currentAnimationState)) 4683 if (presence.ParentID != 0 || presence.ParentUUID != UUID.Zero)
4254 return currentAnimationState; 4684 return "Sitting";
4685
4686 string movementAnimation = presence.Animator.CurrentMovementAnimation;
4687 string lslMovementAnimation;
4688
4689 if (MovementAnimationsForLSL.TryGetValue(movementAnimation, out lslMovementAnimation))
4690 return lslMovementAnimation;
4255 } 4691 }
4256 } 4692 }
4257 4693
@@ -4267,6 +4703,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4267 UUID partItemID; 4703 UUID partItemID;
4268 foreach (SceneObjectPart part in parts) 4704 foreach (SceneObjectPart part in parts)
4269 { 4705 {
4706 //Clone is thread safe
4270 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)part.TaskInventory.Clone(); 4707 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)part.TaskInventory.Clone();
4271 4708
4272 foreach (TaskInventoryItem item in itemsDictionary.Values) 4709 foreach (TaskInventoryItem item in itemsDictionary.Values)
@@ -4400,7 +4837,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4400 { 4837 {
4401 float distance = (PusheePos - m_host.AbsolutePosition).Length(); 4838 float distance = (PusheePos - m_host.AbsolutePosition).Length();
4402 float distance_term = distance * distance * distance; // Script Energy 4839 float distance_term = distance * distance * distance; // Script Energy
4403 float pusher_mass = m_host.GetMass(); 4840 // use total object mass and not part
4841 float pusher_mass = m_host.ParentGroup.GetMass();
4404 4842
4405 float PUSH_ATTENUATION_DISTANCE = 17f; 4843 float PUSH_ATTENUATION_DISTANCE = 17f;
4406 float PUSH_ATTENUATION_SCALE = 5f; 4844 float PUSH_ATTENUATION_SCALE = 5f;
@@ -4639,23 +5077,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4639 { 5077 {
4640 m_host.AddScriptLPS(1); 5078 m_host.AddScriptLPS(1);
4641 5079
4642 lock (m_host.TaskInventory) 5080 m_host.TaskInventory.LockItemsForRead(true);
5081 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
4643 { 5082 {
4644 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 5083 if (inv.Value.Name == name)
4645 { 5084 {
4646 if (inv.Value.Name == name) 5085 if ((inv.Value.CurrentPermissions & (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify)) == (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify))
4647 { 5086 {
4648 if ((inv.Value.CurrentPermissions & (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify)) == (uint)(PermissionMask.Copy | PermissionMask.Transfer | PermissionMask.Modify)) 5087 m_host.TaskInventory.LockItemsForRead(false);
4649 { 5088 return inv.Value.AssetID.ToString();
4650 return inv.Value.AssetID.ToString(); 5089 }
4651 } 5090 else
4652 else 5091 {
4653 { 5092 m_host.TaskInventory.LockItemsForRead(false);
4654 return UUID.Zero.ToString(); 5093 return UUID.Zero.ToString();
4655 }
4656 } 5094 }
4657 } 5095 }
4658 } 5096 }
5097 m_host.TaskInventory.LockItemsForRead(false);
4659 5098
4660 return UUID.Zero.ToString(); 5099 return UUID.Zero.ToString();
4661 } 5100 }
@@ -4789,7 +5228,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4789 public LSL_Vector llGetCenterOfMass() 5228 public LSL_Vector llGetCenterOfMass()
4790 { 5229 {
4791 m_host.AddScriptLPS(1); 5230 m_host.AddScriptLPS(1);
4792 Vector3 center = m_host.GetGeometricCenter(); 5231 Vector3 center = m_host.GetCenterOfMass();
4793 return new LSL_Vector(center.X,center.Y,center.Z); 5232 return new LSL_Vector(center.X,center.Y,center.Z);
4794 } 5233 }
4795 5234
@@ -4808,14 +5247,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4808 { 5247 {
4809 m_host.AddScriptLPS(1); 5248 m_host.AddScriptLPS(1);
4810 5249
4811 if (src == null) 5250 return src.Length;
4812 {
4813 return 0;
4814 }
4815 else
4816 {
4817 return src.Length;
4818 }
4819 } 5251 }
4820 5252
4821 public LSL_Integer llList2Integer(LSL_List src, int index) 5253 public LSL_Integer llList2Integer(LSL_List src, int index)
@@ -4861,7 +5293,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
4861 else if (src.Data[index] is LSL_Float) 5293 else if (src.Data[index] is LSL_Float)
4862 return Convert.ToDouble(((LSL_Float) src.Data[index]).value); 5294 return Convert.ToDouble(((LSL_Float) src.Data[index]).value);
4863 else if (src.Data[index] is LSL_String) 5295 else if (src.Data[index] is LSL_String)
4864 return Convert.ToDouble(((LSL_String) src.Data[index]).m_string); 5296 {
5297 string str = ((LSL_String) src.Data[index]).m_string;
5298 Match m = Regex.Match(str, "^\\s*(-?\\+?[,0-9]+\\.?[0-9]*)");
5299 if (m != Match.Empty)
5300 {
5301 str = m.Value;
5302 double d = 0.0;
5303 if (!Double.TryParse(str, out d))
5304 return 0.0;
5305
5306 return d;
5307 }
5308 return 0.0;
5309 }
4865 return Convert.ToDouble(src.Data[index]); 5310 return Convert.ToDouble(src.Data[index]);
4866 } 5311 }
4867 catch (FormatException) 5312 catch (FormatException)
@@ -5134,7 +5579,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5134 } 5579 }
5135 } 5580 }
5136 } 5581 }
5137 else { 5582 else
5583 {
5138 object[] array = new object[src.Length]; 5584 object[] array = new object[src.Length];
5139 Array.Copy(src.Data, 0, array, 0, src.Length); 5585 Array.Copy(src.Data, 0, array, 0, src.Length);
5140 result = new LSL_List(array); 5586 result = new LSL_List(array);
@@ -5241,7 +5687,13 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5241 public LSL_Integer llGetRegionAgentCount() 5687 public LSL_Integer llGetRegionAgentCount()
5242 { 5688 {
5243 m_host.AddScriptLPS(1); 5689 m_host.AddScriptLPS(1);
5244 return new LSL_Integer(World.GetRootAgentCount()); 5690
5691 int count = 0;
5692 World.ForEachRootScenePresence(delegate(ScenePresence sp) {
5693 count++;
5694 });
5695
5696 return new LSL_Integer(count);
5245 } 5697 }
5246 5698
5247 public LSL_Vector llGetRegionCorner() 5699 public LSL_Vector llGetRegionCorner()
@@ -5521,6 +5973,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5521 flags |= ScriptBaseClass.AGENT_SITTING; 5973 flags |= ScriptBaseClass.AGENT_SITTING;
5522 } 5974 }
5523 5975
5976 if (agent.Appearance.VisualParams[(int)AvatarAppearance.VPElement.SHAPE_MALE] > 0)
5977 {
5978 flags |= ScriptBaseClass.AGENT_MALE;
5979 }
5980
5524 return flags; 5981 return flags;
5525 } 5982 }
5526 5983
@@ -5667,10 +6124,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5667 m_host.AddScriptLPS(1); 6124 m_host.AddScriptLPS(1);
5668 6125
5669 List<SceneObjectPart> parts = GetLinkParts(linknumber); 6126 List<SceneObjectPart> parts = GetLinkParts(linknumber);
5670 6127 if (parts.Count > 0)
5671 foreach (var part in parts)
5672 { 6128 {
5673 SetTextureAnim(part, mode, face, sizex, sizey, start, length, rate); 6129 try
6130 {
6131 parts[0].ParentGroup.areUpdatesSuspended = true;
6132 foreach (var part in parts)
6133 {
6134 SetTextureAnim(part, mode, face, sizex, sizey, start, length, rate);
6135 }
6136 }
6137 finally
6138 {
6139 parts[0].ParentGroup.areUpdatesSuspended = false;
6140 }
5674 } 6141 }
5675 } 6142 }
5676 6143
@@ -5722,13 +6189,20 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5722 6189
5723 if (m_host.OwnerID == land.LandData.OwnerID) 6190 if (m_host.OwnerID == land.LandData.OwnerID)
5724 { 6191 {
5725 World.TeleportClientHome(agentID, presence.ControllingClient); 6192 Vector3 pos = World.GetNearestAllowedPosition(presence, land);
6193 presence.TeleportWithMomentum(pos, null);
6194 presence.ControllingClient.SendAlertMessage("You have been ejected from this land");
5726 } 6195 }
5727 } 6196 }
5728 } 6197 }
5729 ScriptSleep(5000); 6198 ScriptSleep(5000);
5730 } 6199 }
5731 6200
6201 public LSL_List llParseString2List(string str, LSL_List separators, LSL_List in_spacers)
6202 {
6203 return ParseString2List(str, separators, in_spacers, false);
6204 }
6205
5732 public LSL_Integer llOverMyLand(string id) 6206 public LSL_Integer llOverMyLand(string id)
5733 { 6207 {
5734 m_host.AddScriptLPS(1); 6208 m_host.AddScriptLPS(1);
@@ -5793,8 +6267,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5793 UUID agentId = new UUID(); 6267 UUID agentId = new UUID();
5794 if (!UUID.TryParse(agent, out agentId)) 6268 if (!UUID.TryParse(agent, out agentId))
5795 return new LSL_Integer(0); 6269 return new LSL_Integer(0);
6270 if (agentId == m_host.GroupID)
6271 return new LSL_Integer(1);
5796 ScenePresence presence = World.GetScenePresence(agentId); 6272 ScenePresence presence = World.GetScenePresence(agentId);
5797 if (presence == null || presence.IsChildAgent) // Return flase for child agents 6273 if (presence == null || presence.IsChildAgent) // Return false for child agents
5798 return new LSL_Integer(0); 6274 return new LSL_Integer(0);
5799 IClientAPI client = presence.ControllingClient; 6275 IClientAPI client = presence.ControllingClient;
5800 if (m_host.GroupID == client.ActiveGroupId) 6276 if (m_host.GroupID == client.ActiveGroupId)
@@ -5929,7 +6405,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
5929 return m_host.ParentGroup.AttachmentPoint; 6405 return m_host.ParentGroup.AttachmentPoint;
5930 } 6406 }
5931 6407
5932 public LSL_Integer llGetFreeMemory() 6408 public virtual LSL_Integer llGetFreeMemory()
5933 { 6409 {
5934 m_host.AddScriptLPS(1); 6410 m_host.AddScriptLPS(1);
5935 // Make scripts designed for LSO happy 6411 // Make scripts designed for LSO happy
@@ -6046,7 +6522,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6046 SetParticleSystem(m_host, rules); 6522 SetParticleSystem(m_host, rules);
6047 } 6523 }
6048 6524
6049 private void SetParticleSystem(SceneObjectPart part, LSL_List rules) { 6525 private void SetParticleSystem(SceneObjectPart part, LSL_List rules)
6526 {
6050 6527
6051 6528
6052 if (rules.Length == 0) 6529 if (rules.Length == 0)
@@ -6240,14 +6717,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6240 6717
6241 protected UUID GetTaskInventoryItem(string name) 6718 protected UUID GetTaskInventoryItem(string name)
6242 { 6719 {
6243 lock (m_host.TaskInventory) 6720 m_host.TaskInventory.LockItemsForRead(true);
6721 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
6244 { 6722 {
6245 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 6723 if (inv.Value.Name == name)
6246 { 6724 {
6247 if (inv.Value.Name == name) 6725 m_host.TaskInventory.LockItemsForRead(false);
6248 return inv.Key; 6726 return inv.Key;
6249 } 6727 }
6250 } 6728 }
6729 m_host.TaskInventory.LockItemsForRead(false);
6251 6730
6252 return UUID.Zero; 6731 return UUID.Zero;
6253 } 6732 }
@@ -6285,16 +6764,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6285 if (folderID == UUID.Zero) 6764 if (folderID == UUID.Zero)
6286 return; 6765 return;
6287 6766
6288 byte[] bucket = new byte[17]; 6767 byte[] bucket = new byte[1];
6289 bucket[0] = (byte)AssetType.Folder; 6768 bucket[0] = (byte)AssetType.Folder;
6290 byte[] objBytes = folderID.GetBytes(); 6769 //byte[] objBytes = folderID.GetBytes();
6291 Array.Copy(objBytes, 0, bucket, 1, 16); 6770 //Array.Copy(objBytes, 0, bucket, 1, 16);
6292 6771
6293 GridInstantMessage msg = new GridInstantMessage(World, 6772 GridInstantMessage msg = new GridInstantMessage(World,
6294 m_host.UUID, m_host.Name+", an object owned by "+ 6773 m_host.OwnerID, m_host.Name, destID,
6295 resolveName(m_host.OwnerID)+",", destID, 6774 (byte)InstantMessageDialog.TaskInventoryOffered,
6296 (byte)InstantMessageDialog.InventoryOffered, 6775 false, category+". "+m_host.Name+" is located at "+
6297 false, category+"\n"+m_host.Name+" is located at "+
6298 World.RegionInfo.RegionName+" "+ 6776 World.RegionInfo.RegionName+" "+
6299 m_host.AbsolutePosition.ToString(), 6777 m_host.AbsolutePosition.ToString(),
6300 folderID, true, m_host.AbsolutePosition, 6778 folderID, true, m_host.AbsolutePosition,
@@ -6532,13 +7010,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6532 UUID av = new UUID(); 7010 UUID av = new UUID();
6533 if (!UUID.TryParse(avatar,out av)) 7011 if (!UUID.TryParse(avatar,out av))
6534 { 7012 {
6535 LSLError("First parameter to llDialog needs to be a key"); 7013 //LSLError("First parameter to llDialog needs to be a key");
6536 return; 7014 return;
6537 } 7015 }
6538 if (buttons.Length < 1) 7016 if (buttons.Length < 1)
6539 { 7017 {
6540 LSLError("No less than 1 button can be shown"); 7018 buttons.Add("OK");
6541 return;
6542 } 7019 }
6543 if (buttons.Length > 12) 7020 if (buttons.Length > 12)
6544 { 7021 {
@@ -6555,7 +7032,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6555 } 7032 }
6556 if (buttons.Data[i].ToString().Length > 24) 7033 if (buttons.Data[i].ToString().Length > 24)
6557 { 7034 {
6558 LSLError("button label cannot be longer than 24 characters"); 7035 llWhisper(ScriptBaseClass.DEBUG_CHANNEL, "button label cannot be longer than 24 characters");
6559 return; 7036 return;
6560 } 7037 }
6561 buts[i] = buttons.Data[i].ToString(); 7038 buts[i] = buttons.Data[i].ToString();
@@ -6614,22 +7091,23 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6614 } 7091 }
6615 7092
6616 // copy the first script found with this inventory name 7093 // copy the first script found with this inventory name
6617 lock (m_host.TaskInventory) 7094 TaskInventoryItem scriptItem = null;
7095 m_host.TaskInventory.LockItemsForRead(true);
7096 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
6618 { 7097 {
6619 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 7098 if (inv.Value.Name == name)
6620 { 7099 {
6621 if (inv.Value.Name == name) 7100 // make sure the object is a script
7101 if (10 == inv.Value.Type)
6622 { 7102 {
6623 // make sure the object is a script 7103 found = true;
6624 if (10 == inv.Value.Type) 7104 srcId = inv.Key;
6625 { 7105 scriptItem = inv.Value;
6626 found = true; 7106 break;
6627 srcId = inv.Key;
6628 break;
6629 }
6630 } 7107 }
6631 } 7108 }
6632 } 7109 }
7110 m_host.TaskInventory.LockItemsForRead(false);
6633 7111
6634 if (!found) 7112 if (!found)
6635 { 7113 {
@@ -6637,9 +7115,18 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6637 return; 7115 return;
6638 } 7116 }
6639 7117
6640 // the rest of the permission checks are done in RezScript, so check the pin there as well 7118 SceneObjectPart dest = World.GetSceneObjectPart(destId);
6641 World.RezScriptFromPrim(srcId, m_host, destId, pin, running, start_param); 7119 if (dest != null)
7120 {
7121 if ((scriptItem.BasePermissions & (uint)PermissionMask.Transfer) != 0 || dest.ParentGroup.RootPart.OwnerID == m_host.ParentGroup.RootPart.OwnerID)
7122 {
7123 // the rest of the permission checks are done in RezScript, so check the pin there as well
7124 World.RezScriptFromPrim(srcId, m_host, destId, pin, running, start_param);
6642 7125
7126 if ((scriptItem.BasePermissions & (uint)PermissionMask.Copy) == 0)
7127 m_host.Inventory.RemoveInventoryItem(srcId);
7128 }
7129 }
6643 // this will cause the delay even if the script pin or permissions were wrong - seems ok 7130 // this will cause the delay even if the script pin or permissions were wrong - seems ok
6644 ScriptSleep(3000); 7131 ScriptSleep(3000);
6645 } 7132 }
@@ -6702,19 +7189,21 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6702 public LSL_String llMD5String(string src, int nonce) 7189 public LSL_String llMD5String(string src, int nonce)
6703 { 7190 {
6704 m_host.AddScriptLPS(1); 7191 m_host.AddScriptLPS(1);
6705 return Util.Md5Hash(String.Format("{0}:{1}", src, nonce.ToString())); 7192 return Util.Md5Hash(String.Format("{0}:{1}", src, nonce.ToString()), Encoding.UTF8);
6706 } 7193 }
6707 7194
6708 public LSL_String llSHA1String(string src) 7195 public LSL_String llSHA1String(string src)
6709 { 7196 {
6710 m_host.AddScriptLPS(1); 7197 m_host.AddScriptLPS(1);
6711 return Util.SHA1Hash(src).ToLower(); 7198 return Util.SHA1Hash(src, Encoding.UTF8).ToLower();
6712 } 7199 }
6713 7200
6714 protected ObjectShapePacket.ObjectDataBlock SetPrimitiveBlockShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, byte profileshape, byte pathcurve) 7201 protected ObjectShapePacket.ObjectDataBlock SetPrimitiveBlockShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, byte profileshape, byte pathcurve)
6715 { 7202 {
6716 float tempFloat; // Use in float expressions below to avoid byte cast precision issues. 7203 float tempFloat; // Use in float expressions below to avoid byte cast precision issues.
6717 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock(); 7204 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock();
7205 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7206 return shapeBlock;
6718 7207
6719 if (holeshape != (int)ScriptBaseClass.PRIM_HOLE_DEFAULT && 7208 if (holeshape != (int)ScriptBaseClass.PRIM_HOLE_DEFAULT &&
6720 holeshape != (int)ScriptBaseClass.PRIM_HOLE_CIRCLE && 7209 holeshape != (int)ScriptBaseClass.PRIM_HOLE_CIRCLE &&
@@ -6819,6 +7308,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6819 // Prim type box, cylinder and prism. 7308 // Prim type box, cylinder and prism.
6820 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) 7309 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)
6821 { 7310 {
7311 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7312 return;
7313
6822 float tempFloat; // Use in float expressions below to avoid byte cast precision issues. 7314 float tempFloat; // Use in float expressions below to avoid byte cast precision issues.
6823 ObjectShapePacket.ObjectDataBlock shapeBlock; 7315 ObjectShapePacket.ObjectDataBlock shapeBlock;
6824 7316
@@ -6872,6 +7364,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6872 // Prim type sphere. 7364 // Prim type sphere.
6873 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector dimple, byte profileshape, byte pathcurve) 7365 protected void SetPrimitiveShapeParams(SceneObjectPart part, int holeshape, LSL_Vector cut, float hollow, LSL_Vector twist, LSL_Vector dimple, byte profileshape, byte pathcurve)
6874 { 7366 {
7367 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7368 return;
7369
6875 ObjectShapePacket.ObjectDataBlock shapeBlock; 7370 ObjectShapePacket.ObjectDataBlock shapeBlock;
6876 7371
6877 shapeBlock = SetPrimitiveBlockShapeParams(part, holeshape, cut, hollow, twist, profileshape, pathcurve); 7372 shapeBlock = SetPrimitiveBlockShapeParams(part, holeshape, cut, hollow, twist, profileshape, pathcurve);
@@ -6913,6 +7408,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
6913 // Prim type torus, tube and ring. 7408 // Prim type torus, tube and ring.
6914 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) 7409 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)
6915 { 7410 {
7411 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7412 return;
7413
6916 float tempFloat; // Use in float expressions below to avoid byte cast precision issues. 7414 float tempFloat; // Use in float expressions below to avoid byte cast precision issues.
6917 ObjectShapePacket.ObjectDataBlock shapeBlock; 7415 ObjectShapePacket.ObjectDataBlock shapeBlock;
6918 7416
@@ -7048,6 +7546,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7048 // Prim type sculpt. 7546 // Prim type sculpt.
7049 protected void SetPrimitiveShapeParams(SceneObjectPart part, string map, int type, byte pathcurve) 7547 protected void SetPrimitiveShapeParams(SceneObjectPart part, string map, int type, byte pathcurve)
7050 { 7548 {
7549 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7550 return;
7551
7051 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock(); 7552 ObjectShapePacket.ObjectDataBlock shapeBlock = new ObjectShapePacket.ObjectDataBlock();
7052 UUID sculptId; 7553 UUID sculptId;
7053 7554
@@ -7072,7 +7573,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7072 type != (ScriptBaseClass.PRIM_SCULPT_TYPE_TORUS | flag)) 7573 type != (ScriptBaseClass.PRIM_SCULPT_TYPE_TORUS | flag))
7073 { 7574 {
7074 // default 7575 // default
7075 type = (int)ScriptBaseClass.PRIM_SCULPT_TYPE_SPHERE; 7576 type = type | (int)ScriptBaseClass.PRIM_SCULPT_TYPE_SPHERE;
7076 } 7577 }
7077 7578
7078 part.Shape.SetSculptProperties((byte)type, sculptId); 7579 part.Shape.SetSculptProperties((byte)type, sculptId);
@@ -7088,32 +7589,149 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7088 ScriptSleep(200); 7589 ScriptSleep(200);
7089 } 7590 }
7090 7591
7091 public void llSetLinkPrimitiveParams(int linknumber, LSL_List rules) 7592 public void llSetLinkPrimitiveParamsFast(int linknumber, LSL_List rules)
7092 { 7593 {
7093 m_host.AddScriptLPS(1); 7594 m_host.AddScriptLPS(1);
7094 7595
7095 setLinkPrimParams(linknumber, rules); 7596 setLinkPrimParams(linknumber, rules);
7597 }
7096 7598
7097 ScriptSleep(200); 7599 private void setLinkPrimParams(int linknumber, LSL_List rules)
7600 {
7601 List<SceneObjectPart> parts = GetLinkParts(linknumber);
7602 List<ScenePresence> avatars = GetLinkAvatars(linknumber);
7603 if (parts.Count>0)
7604 {
7605 try
7606 {
7607 parts[0].ParentGroup.areUpdatesSuspended = true;
7608 foreach (SceneObjectPart part in parts)
7609 SetPrimParams(part, rules);
7610 }
7611 finally
7612 {
7613 parts[0].ParentGroup.areUpdatesSuspended = false;
7614 }
7615 }
7616 if (avatars.Count > 0)
7617 {
7618 foreach (ScenePresence avatar in avatars)
7619 SetPrimParams(avatar, rules);
7620 }
7098 } 7621 }
7099 7622
7100 public void llSetLinkPrimitiveParamsFast(int linknumber, LSL_List rules) 7623 private void SetPhysicsMaterial(SceneObjectPart part, int material_bits,
7624 float material_density, float material_friction,
7625 float material_restitution, float material_gravity_modifier)
7101 { 7626 {
7102 m_host.AddScriptLPS(1); 7627 ExtraPhysicsData physdata = new ExtraPhysicsData();
7628 physdata.PhysShapeType = (PhysShapeType)part.PhysicsShapeType;
7629 physdata.Density = part.Density;
7630 physdata.Friction = part.Friction;
7631 physdata.Bounce = part.Bounciness;
7632 physdata.GravitationModifier = part.GravityModifier;
7103 7633
7104 setLinkPrimParams(linknumber, rules); 7634 if ((material_bits & (int)ScriptBaseClass.DENSITY) != 0)
7635 physdata.Density = material_density;
7636 if ((material_bits & (int)ScriptBaseClass.FRICTION) != 0)
7637 physdata.Friction = material_friction;
7638 if ((material_bits & (int)ScriptBaseClass.RESTITUTION) != 0)
7639 physdata.Bounce = material_restitution;
7640 if ((material_bits & (int)ScriptBaseClass.GRAVITY_MULTIPLIER) != 0)
7641 physdata.GravitationModifier = material_gravity_modifier;
7642
7643 part.UpdateExtraPhysics(physdata);
7105 } 7644 }
7106 7645
7107 protected void setLinkPrimParams(int linknumber, LSL_List rules) 7646 public void llSetPhysicsMaterial(int material_bits,
7647 float material_gravity_modifier, float material_restitution,
7648 float material_friction, float material_density)
7108 { 7649 {
7109 List<SceneObjectPart> parts = GetLinkParts(linknumber); 7650 SetPhysicsMaterial(m_host, material_bits, material_density, material_friction, material_restitution, material_gravity_modifier);
7651 }
7110 7652
7111 foreach (SceneObjectPart part in parts) 7653 public void llSetLinkPrimitiveParams(int linknumber, LSL_List rules)
7112 SetPrimParams(part, rules); 7654 {
7655 llSetLinkPrimitiveParamsFast(linknumber, rules);
7656 ScriptSleep(200);
7657 }
7658
7659 protected void SetPrimParams(ScenePresence av, LSL_List rules)
7660 {
7661 //This is a special version of SetPrimParams to deal with avatars which are sat on the linkset.
7662 //We only support PRIM_POSITION and PRIM_ROTATION
7663
7664 int idx = 0;
7665
7666 while (idx < rules.Length)
7667 {
7668 int code = rules.GetLSLIntegerItem(idx++);
7669
7670 int remain = rules.Length - idx;
7671
7672 switch (code)
7673 {
7674 case (int)ScriptBaseClass.PRIM_POSITION:
7675 {
7676 if (remain < 1)
7677 return;
7678 LSL_Vector v;
7679 v = rules.GetVector3Item(idx++);
7680
7681 SceneObjectPart part = World.GetSceneObjectPart(av.ParentID);
7682 if (part == null)
7683 break;
7684
7685 LSL_Rotation localRot = ScriptBaseClass.ZERO_ROTATION;
7686 LSL_Vector localPos = ScriptBaseClass.ZERO_VECTOR;
7687 if (llGetLinkNumber() > 1)
7688 {
7689 localRot = llGetLocalRot();
7690 localPos = llGetLocalPos();
7691 }
7692
7693 v -= localPos;
7694 v /= localRot;
7695
7696 LSL_Vector sitOffset = (llRot2Up(new LSL_Rotation(av.Rotation.X, av.Rotation.Y, av.Rotation.Z, av.Rotation.W)) * av.Appearance.AvatarHeight * 0.02638f);
7697
7698 v = v + 2 * sitOffset;
7699
7700 av.OffsetPosition = new Vector3((float)v.x, (float)v.y, (float)v.z);
7701 av.SendAvatarDataToAllAgents();
7702
7703 }
7704 break;
7705
7706 case (int)ScriptBaseClass.PRIM_ROTATION:
7707 {
7708 if (remain < 1)
7709 return;
7710
7711 LSL_Rotation localRot = ScriptBaseClass.ZERO_ROTATION;
7712 LSL_Vector localPos = ScriptBaseClass.ZERO_VECTOR;
7713 if (llGetLinkNumber() > 1)
7714 {
7715 localRot = llGetLocalRot();
7716 localPos = llGetLocalPos();
7717 }
7718
7719 LSL_Rotation r;
7720 r = rules.GetQuaternionItem(idx++);
7721 r = r * llGetRootRotation() / localRot;
7722 av.Rotation = new Quaternion((float)r.x, (float)r.y, (float)r.z, (float)r.s);
7723 av.SendAvatarDataToAllAgents();
7724 }
7725 break;
7726 }
7727 }
7113 } 7728 }
7114 7729
7115 protected void SetPrimParams(SceneObjectPart part, LSL_List rules) 7730 protected void SetPrimParams(SceneObjectPart part, LSL_List rules)
7116 { 7731 {
7732 if (part == null || part.ParentGroup == null || part.ParentGroup.IsDeleted)
7733 return;
7734
7117 int idx = 0; 7735 int idx = 0;
7118 7736
7119 bool positionChanged = false; 7737 bool positionChanged = false;
@@ -7435,6 +8053,36 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7435 part.ScriptSetPhysicsStatus(physics); 8053 part.ScriptSetPhysicsStatus(physics);
7436 break; 8054 break;
7437 8055
8056 case (int)ScriptBaseClass.PRIM_PHYSICS_SHAPE_TYPE:
8057 if (remain < 1)
8058 return;
8059
8060 int shape_type = rules.GetLSLIntegerItem(idx++);
8061
8062 ExtraPhysicsData physdata = new ExtraPhysicsData();
8063 physdata.Density = part.Density;
8064 physdata.Bounce = part.Bounciness;
8065 physdata.GravitationModifier = part.GravityModifier;
8066 physdata.PhysShapeType = (PhysShapeType)shape_type;
8067
8068 part.UpdateExtraPhysics(physdata);
8069
8070 break;
8071
8072 case (int)ScriptBaseClass.PRIM_PHYSICS_MATERIAL:
8073 if (remain < 5)
8074 return;
8075
8076 int material_bits = rules.GetLSLIntegerItem(idx++);
8077 float material_density = (float)rules.GetLSLFloatItem(idx++);
8078 float material_friction = (float)rules.GetLSLFloatItem(idx++);
8079 float material_restitution = (float)rules.GetLSLFloatItem(idx++);
8080 float material_gravity_modifier = (float)rules.GetLSLFloatItem(idx++);
8081
8082 SetPhysicsMaterial(part, material_bits, material_density, material_friction, material_restitution, material_gravity_modifier);
8083
8084 break;
8085
7438 case (int)ScriptBaseClass.PRIM_TEMP_ON_REZ: 8086 case (int)ScriptBaseClass.PRIM_TEMP_ON_REZ:
7439 if (remain < 1) 8087 if (remain < 1)
7440 return; 8088 return;
@@ -7508,7 +8156,9 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7508 if (part.ParentGroup.RootPart == part) 8156 if (part.ParentGroup.RootPart == part)
7509 { 8157 {
7510 SceneObjectGroup parent = part.ParentGroup; 8158 SceneObjectGroup parent = part.ParentGroup;
7511 parent.UpdateGroupPosition(new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z)); 8159 Util.FireAndForget(delegate(object x) {
8160 parent.UpdateGroupPosition(new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z));
8161 });
7512 } 8162 }
7513 else 8163 else
7514 { 8164 {
@@ -7519,6 +8169,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7519 } 8169 }
7520 } 8170 }
7521 } 8171 }
8172
8173 if (positionChanged)
8174 {
8175 if (part.ParentGroup.RootPart == part)
8176 {
8177 SceneObjectGroup parent = part.ParentGroup;
8178 Util.FireAndForget(delegate(object x) {
8179 parent.UpdateGroupPosition(new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z));
8180 });
8181 }
8182 else
8183 {
8184 part.OffsetPosition = new Vector3((float)currentPosition.x, (float)currentPosition.y, (float)currentPosition.z);
8185 SceneObjectGroup parent = part.ParentGroup;
8186 parent.HasGroupChanged = true;
8187 parent.ScheduleGroupForTerseUpdate();
8188 }
8189 }
7522 } 8190 }
7523 8191
7524 public LSL_String llStringToBase64(string str) 8192 public LSL_String llStringToBase64(string str)
@@ -7679,13 +8347,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7679 public LSL_Integer llGetNumberOfPrims() 8347 public LSL_Integer llGetNumberOfPrims()
7680 { 8348 {
7681 m_host.AddScriptLPS(1); 8349 m_host.AddScriptLPS(1);
7682 int avatarCount = 0; 8350 int avatarCount = m_host.ParentGroup.GetLinkedAvatars().Count;
7683 World.ForEachRootScenePresence(delegate(ScenePresence presence) 8351
7684 {
7685 if (presence.ParentID != 0 && m_host.ParentGroup.ContainsPart(presence.ParentID))
7686 avatarCount++;
7687 });
7688
7689 return m_host.ParentGroup.PrimCount + avatarCount; 8352 return m_host.ParentGroup.PrimCount + avatarCount;
7690 } 8353 }
7691 8354
@@ -7701,55 +8364,98 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7701 m_host.AddScriptLPS(1); 8364 m_host.AddScriptLPS(1);
7702 UUID objID = UUID.Zero; 8365 UUID objID = UUID.Zero;
7703 LSL_List result = new LSL_List(); 8366 LSL_List result = new LSL_List();
8367
8368 // If the ID is not valid, return null result
7704 if (!UUID.TryParse(obj, out objID)) 8369 if (!UUID.TryParse(obj, out objID))
7705 { 8370 {
7706 result.Add(new LSL_Vector()); 8371 result.Add(new LSL_Vector());
7707 result.Add(new LSL_Vector()); 8372 result.Add(new LSL_Vector());
7708 return result; 8373 return result;
7709 } 8374 }
8375
8376 // Check if this is an attached prim. If so, replace
8377 // the UUID with the avatar UUID and report it's bounding box
8378 SceneObjectPart part = World.GetSceneObjectPart(objID);
8379 if (part != null && part.ParentGroup.IsAttachment)
8380 objID = part.ParentGroup.AttachedAvatar;
8381
8382 // Find out if this is an avatar ID. If so, return it's box
7710 ScenePresence presence = World.GetScenePresence(objID); 8383 ScenePresence presence = World.GetScenePresence(objID);
7711 if (presence != null) 8384 if (presence != null)
7712 { 8385 {
7713 if (presence.ParentID == 0) // not sat on an object 8386 // As per LSL Wiki, there is no difference between sitting
8387 // and standing avatar since server 1.36
8388 LSL_Vector lower;
8389 LSL_Vector upper;
8390 if (presence.Animator.Animations.DefaultAnimation.AnimID
8391 == DefaultAvatarAnimations.AnimsUUID["SIT_GROUND_CONSTRAINED"])
7714 { 8392 {
7715 LSL_Vector lower; 8393 // This is for ground sitting avatars
7716 LSL_Vector upper; 8394 float height = presence.Appearance.AvatarHeight / 2.66666667f;
7717 if (presence.Animator.Animations.DefaultAnimation.AnimID 8395 lower = new LSL_Vector(-0.3375f, -0.45f, height * -1.0f);
7718 == DefaultAvatarAnimations.AnimsUUID["SIT_GROUND_CONSTRAINED"]) 8396 upper = new LSL_Vector(0.3375f, 0.45f, 0.0f);
7719 {
7720 // This is for ground sitting avatars
7721 float height = presence.Appearance.AvatarHeight / 2.66666667f;
7722 lower = new LSL_Vector(-0.3375f, -0.45f, height * -1.0f);
7723 upper = new LSL_Vector(0.3375f, 0.45f, 0.0f);
7724 }
7725 else
7726 {
7727 // This is for standing/flying avatars
7728 float height = presence.Appearance.AvatarHeight / 2.0f;
7729 lower = new LSL_Vector(-0.225f, -0.3f, height * -1.0f);
7730 upper = new LSL_Vector(0.225f, 0.3f, height + 0.05f);
7731 }
7732 result.Add(lower);
7733 result.Add(upper);
7734 return result;
7735 } 8397 }
7736 else 8398 else
7737 { 8399 {
7738 // sitting on an object so we need the bounding box of that 8400 // This is for standing/flying avatars
7739 // which should include the avatar so set the UUID to the 8401 float height = presence.Appearance.AvatarHeight / 2.0f;
7740 // UUID of the object the avatar is sat on and allow it to fall through 8402 lower = new LSL_Vector(-0.225f, -0.3f, height * -1.0f);
7741 // to processing an object 8403 upper = new LSL_Vector(0.225f, 0.3f, height + 0.05f);
7742 SceneObjectPart p = World.GetSceneObjectPart(presence.ParentID);
7743 objID = p.UUID;
7744 } 8404 }
8405
8406 // Adjust to the documented error offsets (see LSL Wiki)
8407 lower += new LSL_Vector(0.05f, 0.05f, 0.05f);
8408 upper -= new LSL_Vector(0.05f, 0.05f, 0.05f);
8409
8410 if (lower.x > upper.x)
8411 lower.x = upper.x;
8412 if (lower.y > upper.y)
8413 lower.y = upper.y;
8414 if (lower.z > upper.z)
8415 lower.z = upper.z;
8416
8417 result.Add(lower);
8418 result.Add(upper);
8419 return result;
7745 } 8420 }
7746 SceneObjectPart part = World.GetSceneObjectPart(objID); 8421
8422 part = World.GetSceneObjectPart(objID);
7747 // Currently only works for single prims without a sitting avatar 8423 // Currently only works for single prims without a sitting avatar
7748 if (part != null) 8424 if (part != null)
7749 { 8425 {
7750 Vector3 halfSize = part.Scale / 2.0f; 8426 float minX;
7751 LSL_Vector lower = new LSL_Vector(halfSize.X * -1.0f, halfSize.Y * -1.0f, halfSize.Z * -1.0f); 8427 float maxX;
7752 LSL_Vector upper = new LSL_Vector(halfSize.X, halfSize.Y, halfSize.Z); 8428 float minY;
8429 float maxY;
8430 float minZ;
8431 float maxZ;
8432
8433 // This BBox is in sim coordinates, with the offset being
8434 // a contained point.
8435 Vector3[] offsets = Scene.GetCombinedBoundingBox(new List<SceneObjectGroup> { part.ParentGroup },
8436 out minX, out maxX, out minY, out maxY, out minZ, out maxZ);
8437
8438 minX -= offsets[0].X;
8439 maxX -= offsets[0].X;
8440 minY -= offsets[0].Y;
8441 maxY -= offsets[0].Y;
8442 minZ -= offsets[0].Z;
8443 maxZ -= offsets[0].Z;
8444
8445 LSL_Vector lower;
8446 LSL_Vector upper;
8447
8448 // Adjust to the documented error offsets (see LSL Wiki)
8449 lower = new LSL_Vector(minX + 0.05f, minY + 0.05f, minZ + 0.05f);
8450 upper = new LSL_Vector(maxX - 0.05f, maxY - 0.05f, maxZ - 0.05f);
8451
8452 if (lower.x > upper.x)
8453 lower.x = upper.x;
8454 if (lower.y > upper.y)
8455 lower.y = upper.y;
8456 if (lower.z > upper.z)
8457 lower.z = upper.z;
8458
7753 result.Add(lower); 8459 result.Add(lower);
7754 result.Add(upper); 8460 result.Add(upper);
7755 return result; 8461 return result;
@@ -7763,7 +8469,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7763 8469
7764 public LSL_Vector llGetGeometricCenter() 8470 public LSL_Vector llGetGeometricCenter()
7765 { 8471 {
7766 return new LSL_Vector(m_host.GetGeometricCenter().X, m_host.GetGeometricCenter().Y, m_host.GetGeometricCenter().Z); 8472 Vector3 tmp = m_host.GetGeometricCenter();
8473 return new LSL_Vector(tmp.X, tmp.Y, tmp.Z);
7767 } 8474 }
7768 8475
7769 public LSL_List llGetPrimitiveParams(LSL_List rules) 8476 public LSL_List llGetPrimitiveParams(LSL_List rules)
@@ -7829,13 +8536,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
7829 LSL_Vector v = new LSL_Vector(part.AbsolutePosition.X, 8536 LSL_Vector v = new LSL_Vector(part.AbsolutePosition.X,
7830 part.AbsolutePosition.Y, 8537 part.AbsolutePosition.Y,
7831 part.AbsolutePosition.Z); 8538 part.AbsolutePosition.Z);
7832 // For some reason, the part.AbsolutePosition.* values do not change if the
7833 // linkset is rotated; they always reflect the child prim's world position
7834 // as though the linkset is unrotated. This is incompatible behavior with SL's
7835 // implementation, so will break scripts imported from there (not to mention it
7836 // makes it more difficult to determine a child prim's actual inworld position).
7837 if (part.ParentID != 0)
7838 v = ((v - llGetRootPosition()) * llGetRootRotation()) + llGetRootPosition();
7839 res.Add(v); 8539 res.Add(v);
7840 break; 8540 break;
7841 8541
@@ -8006,56 +8706,92 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8006 case (int)ScriptBaseClass.PRIM_BUMP_SHINY: 8706 case (int)ScriptBaseClass.PRIM_BUMP_SHINY:
8007 if (remain < 1) 8707 if (remain < 1)
8008 return res; 8708 return res;
8009 8709 face = (int)rules.GetLSLIntegerItem(idx++);
8010 face=(int)rules.GetLSLIntegerItem(idx++);
8011 8710
8012 tex = part.Shape.Textures; 8711 tex = part.Shape.Textures;
8712 int shiny;
8013 if (face == ScriptBaseClass.ALL_SIDES) 8713 if (face == ScriptBaseClass.ALL_SIDES)
8014 { 8714 {
8015 for (face = 0; face < GetNumberOfSides(part); face++) 8715 for (face = 0; face < GetNumberOfSides(part); face++)
8016 { 8716 {
8017 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8717 Shininess shinyness = tex.GetFace((uint)face).Shiny;
8018 // Convert Shininess to PRIM_SHINY_* 8718 if (shinyness == Shininess.High)
8019 res.Add(new LSL_Integer((uint)texface.Shiny >> 6)); 8719 {
8020 // PRIM_BUMP_* 8720 shiny = ScriptBaseClass.PRIM_SHINY_HIGH;
8021 res.Add(new LSL_Integer((int)texface.Bump)); 8721 }
8722 else if (shinyness == Shininess.Medium)
8723 {
8724 shiny = ScriptBaseClass.PRIM_SHINY_MEDIUM;
8725 }
8726 else if (shinyness == Shininess.Low)
8727 {
8728 shiny = ScriptBaseClass.PRIM_SHINY_LOW;
8729 }
8730 else
8731 {
8732 shiny = ScriptBaseClass.PRIM_SHINY_NONE;
8733 }
8734 res.Add(new LSL_Integer(shiny));
8735 res.Add(new LSL_Integer((int)tex.GetFace((uint)face).Bump));
8022 } 8736 }
8023 } 8737 }
8024 else 8738 else
8025 { 8739 {
8026 if (face >= 0 && face < GetNumberOfSides(part)) 8740 Shininess shinyness = tex.GetFace((uint)face).Shiny;
8741 if (shinyness == Shininess.High)
8027 { 8742 {
8028 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8743 shiny = ScriptBaseClass.PRIM_SHINY_HIGH;
8029 // Convert Shininess to PRIM_SHINY_* 8744 }
8030 res.Add(new LSL_Integer((uint)texface.Shiny >> 6)); 8745 else if (shinyness == Shininess.Medium)
8031 // PRIM_BUMP_* 8746 {
8032 res.Add(new LSL_Integer((int)texface.Bump)); 8747 shiny = ScriptBaseClass.PRIM_SHINY_MEDIUM;
8033 } 8748 }
8749 else if (shinyness == Shininess.Low)
8750 {
8751 shiny = ScriptBaseClass.PRIM_SHINY_LOW;
8752 }
8753 else
8754 {
8755 shiny = ScriptBaseClass.PRIM_SHINY_NONE;
8756 }
8757 res.Add(new LSL_Integer(shiny));
8758 res.Add(new LSL_Integer((int)tex.GetFace((uint)face).Bump));
8034 } 8759 }
8035 break; 8760 break;
8036 8761
8037 case (int)ScriptBaseClass.PRIM_FULLBRIGHT: 8762 case (int)ScriptBaseClass.PRIM_FULLBRIGHT:
8038 if (remain < 1) 8763 if (remain < 1)
8039 return res; 8764 return res;
8040 8765 face = (int)rules.GetLSLIntegerItem(idx++);
8041 face=(int)rules.GetLSLIntegerItem(idx++);
8042 8766
8043 tex = part.Shape.Textures; 8767 tex = part.Shape.Textures;
8768 int fullbright;
8044 if (face == ScriptBaseClass.ALL_SIDES) 8769 if (face == ScriptBaseClass.ALL_SIDES)
8045 { 8770 {
8046 for (face = 0; face < GetNumberOfSides(part); face++) 8771 for (face = 0; face < GetNumberOfSides(part); face++)
8047 { 8772 {
8048 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8773 if (tex.GetFace((uint)face).Fullbright == true)
8049 res.Add(new LSL_Integer(texface.Fullbright ? 1 : 0)); 8774 {
8775 fullbright = ScriptBaseClass.TRUE;
8776 }
8777 else
8778 {
8779 fullbright = ScriptBaseClass.FALSE;
8780 }
8781 res.Add(new LSL_Integer(fullbright));
8050 } 8782 }
8051 } 8783 }
8052 else 8784 else
8053 { 8785 {
8054 if (face >= 0 && face < GetNumberOfSides(part)) 8786 if (tex.GetFace((uint)face).Fullbright == true)
8055 { 8787 {
8056 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8788 fullbright = ScriptBaseClass.TRUE;
8057 res.Add(new LSL_Integer(texface.Fullbright ? 1 : 0));
8058 } 8789 }
8790 else
8791 {
8792 fullbright = ScriptBaseClass.FALSE;
8793 }
8794 res.Add(new LSL_Integer(fullbright));
8059 } 8795 }
8060 break; 8796 break;
8061 8797
@@ -8077,27 +8813,35 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8077 break; 8813 break;
8078 8814
8079 case (int)ScriptBaseClass.PRIM_TEXGEN: 8815 case (int)ScriptBaseClass.PRIM_TEXGEN:
8816 // (PRIM_TEXGEN_DEFAULT, PRIM_TEXGEN_PLANAR)
8080 if (remain < 1) 8817 if (remain < 1)
8081 return res; 8818 return res;
8082 8819 face = (int)rules.GetLSLIntegerItem(idx++);
8083 face=(int)rules.GetLSLIntegerItem(idx++);
8084 8820
8085 tex = part.Shape.Textures; 8821 tex = part.Shape.Textures;
8086 if (face == ScriptBaseClass.ALL_SIDES) 8822 if (face == ScriptBaseClass.ALL_SIDES)
8087 { 8823 {
8088 for (face = 0; face < GetNumberOfSides(part); face++) 8824 for (face = 0; face < GetNumberOfSides(part); face++)
8089 { 8825 {
8090 MappingType texgen = tex.GetFace((uint)face).TexMapType; 8826 if (tex.GetFace((uint)face).TexMapType == MappingType.Planar)
8091 // Convert MappingType to PRIM_TEXGEN_DEFAULT, PRIM_TEXGEN_PLANAR etc. 8827 {
8092 res.Add(new LSL_Integer((uint)texgen >> 1)); 8828 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_PLANAR));
8829 }
8830 else
8831 {
8832 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_DEFAULT));
8833 }
8093 } 8834 }
8094 } 8835 }
8095 else 8836 else
8096 { 8837 {
8097 if (face >= 0 && face < GetNumberOfSides(part)) 8838 if (tex.GetFace((uint)face).TexMapType == MappingType.Planar)
8839 {
8840 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_PLANAR));
8841 }
8842 else
8098 { 8843 {
8099 MappingType texgen = tex.GetFace((uint)face).TexMapType; 8844 res.Add(new LSL_Integer(ScriptBaseClass.PRIM_TEXGEN_DEFAULT));
8100 res.Add(new LSL_Integer((uint)texgen >> 1));
8101 } 8845 }
8102 } 8846 }
8103 break; 8847 break;
@@ -8120,28 +8864,24 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8120 case (int)ScriptBaseClass.PRIM_GLOW: 8864 case (int)ScriptBaseClass.PRIM_GLOW:
8121 if (remain < 1) 8865 if (remain < 1)
8122 return res; 8866 return res;
8123 8867 face = (int)rules.GetLSLIntegerItem(idx++);
8124 face=(int)rules.GetLSLIntegerItem(idx++);
8125 8868
8126 tex = part.Shape.Textures; 8869 tex = part.Shape.Textures;
8870 float primglow;
8127 if (face == ScriptBaseClass.ALL_SIDES) 8871 if (face == ScriptBaseClass.ALL_SIDES)
8128 { 8872 {
8129 for (face = 0; face < GetNumberOfSides(part); face++) 8873 for (face = 0; face < GetNumberOfSides(part); face++)
8130 { 8874 {
8131 Primitive.TextureEntryFace texface = tex.GetFace((uint)face); 8875 primglow = tex.GetFace((uint)face).Glow;
8132 res.Add(new LSL_Float(texface.Glow)); 8876 res.Add(new LSL_Float(primglow));
8133 } 8877 }
8134 } 8878 }
8135 else 8879 else
8136 { 8880 {
8137 if (face >= 0 && face < GetNumberOfSides(part)) 8881 primglow = tex.GetFace((uint)face).Glow;
8138 { 8882 res.Add(new LSL_Float(primglow));
8139 Primitive.TextureEntryFace texface = tex.GetFace((uint)face);
8140 res.Add(new LSL_Float(texface.Glow));
8141 }
8142 } 8883 }
8143 break; 8884 break;
8144
8145 case (int)ScriptBaseClass.PRIM_TEXT: 8885 case (int)ScriptBaseClass.PRIM_TEXT:
8146 Color4 textColor = part.GetTextColor(); 8886 Color4 textColor = part.GetTextColor();
8147 res.Add(new LSL_String(part.Text)); 8887 res.Add(new LSL_String(part.Text));
@@ -8754,8 +9494,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8754 // The function returns an ordered list 9494 // The function returns an ordered list
8755 // representing the tokens found in the supplied 9495 // representing the tokens found in the supplied
8756 // sources string. If two successive tokenizers 9496 // sources string. If two successive tokenizers
8757 // are encountered, then a NULL entry is added 9497 // are encountered, then a null-string entry is
8758 // to the list. 9498 // added to the list.
8759 // 9499 //
8760 // It is a precondition that the source and 9500 // It is a precondition that the source and
8761 // toekizer lisst are non-null. If they are null, 9501 // toekizer lisst are non-null. If they are null,
@@ -8763,7 +9503,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8763 // while their lengths are being determined. 9503 // while their lengths are being determined.
8764 // 9504 //
8765 // A small amount of working memoryis required 9505 // A small amount of working memoryis required
8766 // of approximately 8*#tokenizers. 9506 // of approximately 8*#tokenizers + 8*srcstrlen.
8767 // 9507 //
8768 // There are many ways in which this function 9508 // There are many ways in which this function
8769 // can be implemented, this implementation is 9509 // can be implemented, this implementation is
@@ -8779,155 +9519,124 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
8779 // and eliminates redundant tokenizers as soon 9519 // and eliminates redundant tokenizers as soon
8780 // as is possible. 9520 // as is possible.
8781 // 9521 //
8782 // The implementation tries to avoid any copying 9522 // The implementation tries to minimize temporary
8783 // of arrays or other objects. 9523 // garbage generation.
8784 // </remarks> 9524 // </remarks>
8785 9525
8786 private LSL_List ParseString(string src, LSL_List separators, LSL_List spacers, bool keepNulls) 9526 public LSL_List llParseStringKeepNulls(string src, LSL_List separators, LSL_List spacers)
8787 { 9527 {
8788 int beginning = 0; 9528 return ParseString2List(src, separators, spacers, true);
8789 int srclen = src.Length; 9529 }
8790 int seplen = separators.Length;
8791 object[] separray = separators.Data;
8792 int spclen = spacers.Length;
8793 object[] spcarray = spacers.Data;
8794 int mlen = seplen+spclen;
8795
8796 int[] offset = new int[mlen+1];
8797 bool[] active = new bool[mlen];
8798
8799 int best;
8800 int j;
8801
8802 // Initial capacity reduces resize cost
8803 9530
8804 LSL_List tokens = new LSL_List(); 9531 private LSL_List ParseString2List(string src, LSL_List separators, LSL_List spacers, bool keepNulls)
9532 {
9533 int srclen = src.Length;
9534 int seplen = separators.Length;
9535 object[] separray = separators.Data;
9536 int spclen = spacers.Length;
9537 object[] spcarray = spacers.Data;
9538 int dellen = 0;
9539 string[] delarray = new string[seplen+spclen];
8805 9540
8806 // All entries are initially valid 9541 int outlen = 0;
9542 string[] outarray = new string[srclen*2+1];
8807 9543
8808 for (int i = 0; i < mlen; i++) 9544 int i, j;
8809 active[i] = true; 9545 string d;
8810 9546
8811 offset[mlen] = srclen; 9547 m_host.AddScriptLPS(1);
8812 9548
8813 while (beginning < srclen) 9549 /*
9550 * Convert separator and spacer lists to C# strings.
9551 * Also filter out null strings so we don't hang.
9552 */
9553 for (i = 0; i < seplen; i ++)
8814 { 9554 {
9555 d = separray[i].ToString();
9556 if (d.Length > 0)
9557 {
9558 delarray[dellen++] = d;
9559 }
9560 }
9561 seplen = dellen;
8815 9562
8816 best = mlen; // as bad as it gets 9563 for (i = 0; i < spclen; i ++)
9564 {
9565 d = spcarray[i].ToString();
9566 if (d.Length > 0)
9567 {
9568 delarray[dellen++] = d;
9569 }
9570 }
8817 9571
8818 // Scan for separators 9572 /*
9573 * Scan through source string from beginning to end.
9574 */
9575 for (i = 0;;)
9576 {
8819 9577
8820 for (j = 0; j < seplen; j++) 9578 /*
9579 * Find earliest delimeter in src starting at i (if any).
9580 */
9581 int earliestDel = -1;
9582 int earliestSrc = srclen;
9583 string earliestStr = null;
9584 for (j = 0; j < dellen; j ++)
8821 { 9585 {
8822 if (separray[j].ToString() == String.Empty) 9586 d = delarray[j];
8823 active[j] = false; 9587 if (d != null)
8824
8825 if (active[j])
8826 { 9588 {
8827 // scan all of the markers 9589 int index = src.IndexOf(d, i);
8828 if ((offset[j] = src.IndexOf(separray[j].ToString(), beginning)) == -1) 9590 if (index < 0)
8829 { 9591 {
8830 // not present at all 9592 delarray[j] = null; // delim nowhere in src, don't check it anymore
8831 active[j] = false;
8832 } 9593 }
8833 else 9594 else if (index < earliestSrc)
8834 { 9595 {
8835 // present and correct 9596 earliestSrc = index; // where delimeter starts in source string
8836 if (offset[j] < offset[best]) 9597 earliestDel = j; // where delimeter is in delarray[]
8837 { 9598 earliestStr = d; // the delimeter string from delarray[]
8838 // closest so far 9599 if (index == i) break; // can't do any better than found at beg of string
8839 best = j;
8840 if (offset[best] == beginning)
8841 break;
8842 }
8843 } 9600 }
8844 } 9601 }
8845 } 9602 }
8846 9603
8847 // Scan for spacers 9604 /*
8848 9605 * Output source string starting at i through start of earliest delimeter.
8849 if (offset[best] != beginning) 9606 */
9607 if (keepNulls || (earliestSrc > i))
8850 { 9608 {
8851 for (j = seplen; (j < mlen) && (offset[best] > beginning); j++) 9609 outarray[outlen++] = src.Substring(i, earliestSrc - i);
8852 {
8853 if (spcarray[j-seplen].ToString() == String.Empty)
8854 active[j] = false;
8855
8856 if (active[j])
8857 {
8858 // scan all of the markers
8859 if ((offset[j] = src.IndexOf(spcarray[j-seplen].ToString(), beginning)) == -1)
8860 {
8861 // not present at all
8862 active[j] = false;
8863 }
8864 else
8865 {
8866 // present and correct
8867 if (offset[j] < offset[best])
8868 {
8869 // closest so far
8870 best = j;
8871 }
8872 }
8873 }
8874 }
8875 } 9610 }
8876 9611
8877 // This is the normal exit from the scanning loop 9612 /*
9613 * If no delimeter found at or after i, we're done scanning.
9614 */
9615 if (earliestDel < 0) break;
8878 9616
8879 if (best == mlen) 9617 /*
9618 * If delimeter was a spacer, output the spacer.
9619 */
9620 if (earliestDel >= seplen)
8880 { 9621 {
8881 // no markers were found on this pass 9622 outarray[outlen++] = earliestStr;
8882 // so we're pretty much done
8883 if ((keepNulls) || ((!keepNulls) && (srclen - beginning) > 0))
8884 tokens.Add(new LSL_String(src.Substring(beginning, srclen - beginning)));
8885 break;
8886 } 9623 }
8887 9624
8888 // Otherwise we just add the newly delimited token 9625 /*
8889 // and recalculate where the search should continue. 9626 * Look at rest of src string following delimeter.
8890 if ((keepNulls) || ((!keepNulls) && (offset[best] - beginning) > 0)) 9627 */
8891 tokens.Add(new LSL_String(src.Substring(beginning,offset[best]-beginning))); 9628 i = earliestSrc + earliestStr.Length;
8892
8893 if (best < seplen)
8894 {
8895 beginning = offset[best] + (separray[best].ToString()).Length;
8896 }
8897 else
8898 {
8899 beginning = offset[best] + (spcarray[best - seplen].ToString()).Length;
8900 string str = spcarray[best - seplen].ToString();
8901 if ((keepNulls) || ((!keepNulls) && (str.Length > 0)))
8902 tokens.Add(new LSL_String(str));
8903 }
8904 } 9629 }
8905 9630
8906 // This an awkward an not very intuitive boundary case. If the 9631 /*
8907 // last substring is a tokenizer, then there is an implied trailing 9632 * Make up an exact-sized output array suitable for an LSL_List object.
8908 // null list entry. Hopefully the single comparison will not be too 9633 */
8909 // arduous. Alternatively the 'break' could be replced with a return 9634 object[] outlist = new object[outlen];
8910 // but that's shabby programming. 9635 for (i = 0; i < outlen; i ++)
8911
8912 if ((beginning == srclen) && (keepNulls))
8913 { 9636 {
8914 if (srclen != 0) 9637 outlist[i] = new LSL_String(outarray[i]);
8915 tokens.Add(new LSL_String(""));
8916 } 9638 }
8917 9639 return new LSL_List(outlist);
8918 return tokens;
8919 }
8920
8921 public LSL_List llParseString2List(string src, LSL_List separators, LSL_List spacers)
8922 {
8923 m_host.AddScriptLPS(1);
8924 return this.ParseString(src, separators, spacers, false);
8925 }
8926
8927 public LSL_List llParseStringKeepNulls(string src, LSL_List separators, LSL_List spacers)
8928 {
8929 m_host.AddScriptLPS(1);
8930 return this.ParseString(src, separators, spacers, true);
8931 } 9640 }
8932 9641
8933 public LSL_Integer llGetObjectPermMask(int mask) 9642 public LSL_Integer llGetObjectPermMask(int mask)
@@ -9004,28 +9713,28 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9004 { 9713 {
9005 m_host.AddScriptLPS(1); 9714 m_host.AddScriptLPS(1);
9006 9715
9007 lock (m_host.TaskInventory) 9716 m_host.TaskInventory.LockItemsForRead(true);
9717 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
9008 { 9718 {
9009 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 9719 if (inv.Value.Name == item)
9010 { 9720 {
9011 if (inv.Value.Name == item) 9721 m_host.TaskInventory.LockItemsForRead(false);
9722 switch (mask)
9012 { 9723 {
9013 switch (mask) 9724 case 0:
9014 { 9725 return (int)inv.Value.BasePermissions;
9015 case 0: 9726 case 1:
9016 return (int)inv.Value.BasePermissions; 9727 return (int)inv.Value.CurrentPermissions;
9017 case 1: 9728 case 2:
9018 return (int)inv.Value.CurrentPermissions; 9729 return (int)inv.Value.GroupPermissions;
9019 case 2: 9730 case 3:
9020 return (int)inv.Value.GroupPermissions; 9731 return (int)inv.Value.EveryonePermissions;
9021 case 3: 9732 case 4:
9022 return (int)inv.Value.EveryonePermissions; 9733 return (int)inv.Value.NextPermissions;
9023 case 4:
9024 return (int)inv.Value.NextPermissions;
9025 }
9026 } 9734 }
9027 } 9735 }
9028 } 9736 }
9737 m_host.TaskInventory.LockItemsForRead(false);
9029 9738
9030 return -1; 9739 return -1;
9031 } 9740 }
@@ -9072,16 +9781,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9072 { 9781 {
9073 m_host.AddScriptLPS(1); 9782 m_host.AddScriptLPS(1);
9074 9783
9075 lock (m_host.TaskInventory) 9784 m_host.TaskInventory.LockItemsForRead(true);
9785 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
9076 { 9786 {
9077 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 9787 if (inv.Value.Name == item)
9078 { 9788 {
9079 if (inv.Value.Name == item) 9789 m_host.TaskInventory.LockItemsForRead(false);
9080 { 9790 return inv.Value.CreatorID.ToString();
9081 return inv.Value.CreatorID.ToString();
9082 }
9083 } 9791 }
9084 } 9792 }
9793 m_host.TaskInventory.LockItemsForRead(false);
9085 9794
9086 llSay(0, "No item name '" + item + "'"); 9795 llSay(0, "No item name '" + item + "'");
9087 9796
@@ -9223,9 +9932,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9223 { 9932 {
9224 try 9933 try
9225 { 9934 {
9935 /*
9226 SceneObjectPart obj = World.GetSceneObjectPart(World.Entities[key].LocalId); 9936 SceneObjectPart obj = World.GetSceneObjectPart(World.Entities[key].LocalId);
9227 if (obj != null) 9937 if (obj != null)
9228 return (double)obj.GetMass(); 9938 return (double)obj.GetMass();
9939 */
9940 // return total object mass
9941 SceneObjectGroup obj = World.GetGroupByPrim(World.Entities[key].LocalId);
9942 if (obj != null)
9943 return obj.GetMass();
9944
9229 // the object is null so the key is for an avatar 9945 // the object is null so the key is for an avatar
9230 ScenePresence avatar = World.GetScenePresence(key); 9946 ScenePresence avatar = World.GetScenePresence(key);
9231 if (avatar != null) 9947 if (avatar != null)
@@ -9245,7 +9961,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9245 } 9961 }
9246 9962
9247 /// <summary> 9963 /// <summary>
9248 /// illListReplaceList removes the sub-list defined by the inclusive indices 9964 /// llListReplaceList removes the sub-list defined by the inclusive indices
9249 /// start and end and inserts the src list in its place. The inclusive 9965 /// start and end and inserts the src list in its place. The inclusive
9250 /// nature of the indices means that at least one element must be deleted 9966 /// nature of the indices means that at least one element must be deleted
9251 /// if the indices are within the bounds of the existing list. I.e. 2,2 9967 /// if the indices are within the bounds of the existing list. I.e. 2,2
@@ -9302,16 +10018,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9302 // based upon end. Note that if end exceeds the upper 10018 // based upon end. Note that if end exceeds the upper
9303 // bound in this case, the entire destination list 10019 // bound in this case, the entire destination list
9304 // is removed. 10020 // is removed.
9305 else 10021 else if (start == 0)
9306 { 10022 {
9307 if (end + 1 < dest.Length) 10023 if (end + 1 < dest.Length)
9308 {
9309 return src + dest.GetSublist(end + 1, -1); 10024 return src + dest.GetSublist(end + 1, -1);
9310 }
9311 else 10025 else
9312 {
9313 return src; 10026 return src;
9314 } 10027 }
10028 else // Start < 0
10029 {
10030 if (end + 1 < dest.Length)
10031 return dest.GetSublist(end + 1, -1);
10032 else
10033 return new LSL_List();
9315 } 10034 }
9316 } 10035 }
9317 // Finally, if start > end, we strip away a prefix and 10036 // Finally, if start > end, we strip away a prefix and
@@ -9362,17 +10081,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9362 int width = 0; 10081 int width = 0;
9363 int height = 0; 10082 int height = 0;
9364 10083
9365 ParcelMediaCommandEnum? commandToSend = null; 10084 uint commandToSend = 0;
9366 float time = 0.0f; // default is from start 10085 float time = 0.0f; // default is from start
9367 10086
9368 ScenePresence presence = null; 10087 ScenePresence presence = null;
9369 10088
9370 for (int i = 0; i < commandList.Data.Length; i++) 10089 for (int i = 0; i < commandList.Data.Length; i++)
9371 { 10090 {
9372 ParcelMediaCommandEnum command = (ParcelMediaCommandEnum)commandList.Data[i]; 10091 uint command = (uint)(commandList.GetLSLIntegerItem(i));
9373 switch (command) 10092 switch (command)
9374 { 10093 {
9375 case ParcelMediaCommandEnum.Agent: 10094 case (uint)ParcelMediaCommandEnum.Agent:
9376 // we send only to one agent 10095 // we send only to one agent
9377 if ((i + 1) < commandList.Length) 10096 if ((i + 1) < commandList.Length)
9378 { 10097 {
@@ -9389,25 +10108,25 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9389 } 10108 }
9390 break; 10109 break;
9391 10110
9392 case ParcelMediaCommandEnum.Loop: 10111 case (uint)ParcelMediaCommandEnum.Loop:
9393 loop = 1; 10112 loop = 1;
9394 commandToSend = command; 10113 commandToSend = command;
9395 update = true; //need to send the media update packet to set looping 10114 update = true; //need to send the media update packet to set looping
9396 break; 10115 break;
9397 10116
9398 case ParcelMediaCommandEnum.Play: 10117 case (uint)ParcelMediaCommandEnum.Play:
9399 loop = 0; 10118 loop = 0;
9400 commandToSend = command; 10119 commandToSend = command;
9401 update = true; //need to send the media update packet to make sure it doesn't loop 10120 update = true; //need to send the media update packet to make sure it doesn't loop
9402 break; 10121 break;
9403 10122
9404 case ParcelMediaCommandEnum.Pause: 10123 case (uint)ParcelMediaCommandEnum.Pause:
9405 case ParcelMediaCommandEnum.Stop: 10124 case (uint)ParcelMediaCommandEnum.Stop:
9406 case ParcelMediaCommandEnum.Unload: 10125 case (uint)ParcelMediaCommandEnum.Unload:
9407 commandToSend = command; 10126 commandToSend = command;
9408 break; 10127 break;
9409 10128
9410 case ParcelMediaCommandEnum.Url: 10129 case (uint)ParcelMediaCommandEnum.Url:
9411 if ((i + 1) < commandList.Length) 10130 if ((i + 1) < commandList.Length)
9412 { 10131 {
9413 if (commandList.Data[i + 1] is LSL_String) 10132 if (commandList.Data[i + 1] is LSL_String)
@@ -9420,7 +10139,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9420 } 10139 }
9421 break; 10140 break;
9422 10141
9423 case ParcelMediaCommandEnum.Texture: 10142 case (uint)ParcelMediaCommandEnum.Texture:
9424 if ((i + 1) < commandList.Length) 10143 if ((i + 1) < commandList.Length)
9425 { 10144 {
9426 if (commandList.Data[i + 1] is LSL_String) 10145 if (commandList.Data[i + 1] is LSL_String)
@@ -9433,7 +10152,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9433 } 10152 }
9434 break; 10153 break;
9435 10154
9436 case ParcelMediaCommandEnum.Time: 10155 case (uint)ParcelMediaCommandEnum.Time:
9437 if ((i + 1) < commandList.Length) 10156 if ((i + 1) < commandList.Length)
9438 { 10157 {
9439 if (commandList.Data[i + 1] is LSL_Float) 10158 if (commandList.Data[i + 1] is LSL_Float)
@@ -9445,7 +10164,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9445 } 10164 }
9446 break; 10165 break;
9447 10166
9448 case ParcelMediaCommandEnum.AutoAlign: 10167 case (uint)ParcelMediaCommandEnum.AutoAlign:
9449 if ((i + 1) < commandList.Length) 10168 if ((i + 1) < commandList.Length)
9450 { 10169 {
9451 if (commandList.Data[i + 1] is LSL_Integer) 10170 if (commandList.Data[i + 1] is LSL_Integer)
@@ -9459,7 +10178,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9459 } 10178 }
9460 break; 10179 break;
9461 10180
9462 case ParcelMediaCommandEnum.Type: 10181 case (uint)ParcelMediaCommandEnum.Type:
9463 if ((i + 1) < commandList.Length) 10182 if ((i + 1) < commandList.Length)
9464 { 10183 {
9465 if (commandList.Data[i + 1] is LSL_String) 10184 if (commandList.Data[i + 1] is LSL_String)
@@ -9472,7 +10191,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9472 } 10191 }
9473 break; 10192 break;
9474 10193
9475 case ParcelMediaCommandEnum.Desc: 10194 case (uint)ParcelMediaCommandEnum.Desc:
9476 if ((i + 1) < commandList.Length) 10195 if ((i + 1) < commandList.Length)
9477 { 10196 {
9478 if (commandList.Data[i + 1] is LSL_String) 10197 if (commandList.Data[i + 1] is LSL_String)
@@ -9485,7 +10204,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9485 } 10204 }
9486 break; 10205 break;
9487 10206
9488 case ParcelMediaCommandEnum.Size: 10207 case (uint)ParcelMediaCommandEnum.Size:
9489 if ((i + 2) < commandList.Length) 10208 if ((i + 2) < commandList.Length)
9490 { 10209 {
9491 if (commandList.Data[i + 1] is LSL_Integer) 10210 if (commandList.Data[i + 1] is LSL_Integer)
@@ -9555,7 +10274,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9555 } 10274 }
9556 } 10275 }
9557 10276
9558 if (commandToSend != null) 10277 if (commandToSend != 0)
9559 { 10278 {
9560 // the commandList contained a start/stop/... command, too 10279 // the commandList contained a start/stop/... command, too
9561 if (presence == null) 10280 if (presence == null)
@@ -9592,7 +10311,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9592 10311
9593 if (aList.Data[i] != null) 10312 if (aList.Data[i] != null)
9594 { 10313 {
9595 switch ((ParcelMediaCommandEnum) aList.Data[i]) 10314 switch ((ParcelMediaCommandEnum) Convert.ToInt32(aList.Data[i].ToString()))
9596 { 10315 {
9597 case ParcelMediaCommandEnum.Url: 10316 case ParcelMediaCommandEnum.Url:
9598 list.Add(new LSL_String(World.GetLandData(m_host.AbsolutePosition.X, m_host.AbsolutePosition.Y).MediaURL)); 10317 list.Add(new LSL_String(World.GetLandData(m_host.AbsolutePosition.X, m_host.AbsolutePosition.Y).MediaURL));
@@ -9635,16 +10354,16 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9635 { 10354 {
9636 m_host.AddScriptLPS(1); 10355 m_host.AddScriptLPS(1);
9637 10356
9638 lock (m_host.TaskInventory) 10357 m_host.TaskInventory.LockItemsForRead(true);
10358 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
9639 { 10359 {
9640 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 10360 if (inv.Value.Name == name)
9641 { 10361 {
9642 if (inv.Value.Name == name) 10362 m_host.TaskInventory.LockItemsForRead(false);
9643 { 10363 return inv.Value.Type;
9644 return inv.Value.Type;
9645 }
9646 } 10364 }
9647 } 10365 }
10366 m_host.TaskInventory.LockItemsForRead(false);
9648 10367
9649 return -1; 10368 return -1;
9650 } 10369 }
@@ -9655,15 +10374,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9655 10374
9656 if (quick_pay_buttons.Data.Length < 4) 10375 if (quick_pay_buttons.Data.Length < 4)
9657 { 10376 {
9658 LSLError("List must have at least 4 elements"); 10377 int x;
9659 return; 10378 for (x=quick_pay_buttons.Data.Length; x<= 4; x++)
10379 {
10380 quick_pay_buttons.Add(ScriptBaseClass.PAY_HIDE);
10381 }
9660 } 10382 }
9661 m_host.ParentGroup.RootPart.PayPrice[0]=price; 10383 int[] nPrice = new int[5];
9662 10384 nPrice[0] = price;
9663 m_host.ParentGroup.RootPart.PayPrice[1]=(LSL_Integer)quick_pay_buttons.Data[0]; 10385 nPrice[1] = quick_pay_buttons.GetLSLIntegerItem(0);
9664 m_host.ParentGroup.RootPart.PayPrice[2]=(LSL_Integer)quick_pay_buttons.Data[1]; 10386 nPrice[2] = quick_pay_buttons.GetLSLIntegerItem(1);
9665 m_host.ParentGroup.RootPart.PayPrice[3]=(LSL_Integer)quick_pay_buttons.Data[2]; 10387 nPrice[3] = quick_pay_buttons.GetLSLIntegerItem(2);
9666 m_host.ParentGroup.RootPart.PayPrice[4]=(LSL_Integer)quick_pay_buttons.Data[3]; 10388 nPrice[4] = quick_pay_buttons.GetLSLIntegerItem(3);
10389 m_host.ParentGroup.RootPart.PayPrice = nPrice;
9667 m_host.ParentGroup.HasGroupChanged = true; 10390 m_host.ParentGroup.HasGroupChanged = true;
9668 } 10391 }
9669 10392
@@ -9680,7 +10403,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9680 return new LSL_Vector(); 10403 return new LSL_Vector();
9681 } 10404 }
9682 10405
9683 ScenePresence presence = World.GetScenePresence(m_host.OwnerID); 10406// ScenePresence presence = World.GetScenePresence(m_host.OwnerID);
10407 ScenePresence presence = World.GetScenePresence(m_item.PermsGranter);
9684 if (presence != null) 10408 if (presence != null)
9685 { 10409 {
9686 LSL_Vector pos = new LSL_Vector(presence.CameraPosition.X, presence.CameraPosition.Y, presence.CameraPosition.Z); 10410 LSL_Vector pos = new LSL_Vector(presence.CameraPosition.X, presence.CameraPosition.Y, presence.CameraPosition.Z);
@@ -9702,7 +10426,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9702 return new LSL_Rotation(); 10426 return new LSL_Rotation();
9703 } 10427 }
9704 10428
9705 ScenePresence presence = World.GetScenePresence(m_host.OwnerID); 10429// ScenePresence presence = World.GetScenePresence(m_host.OwnerID);
10430 ScenePresence presence = World.GetScenePresence(m_item.PermsGranter);
9706 if (presence != null) 10431 if (presence != null)
9707 { 10432 {
9708 return new LSL_Rotation(presence.CameraRotation.X, presence.CameraRotation.Y, presence.CameraRotation.Z, presence.CameraRotation.W); 10433 return new LSL_Rotation(presence.CameraRotation.X, presence.CameraRotation.Y, presence.CameraRotation.Z, presence.CameraRotation.W);
@@ -9762,8 +10487,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9762 { 10487 {
9763 m_host.AddScriptLPS(1); 10488 m_host.AddScriptLPS(1);
9764 DetectParams detectedParams = m_ScriptEngine.GetDetectParams(m_item.ItemID, 0); 10489 DetectParams detectedParams = m_ScriptEngine.GetDetectParams(m_item.ItemID, 0);
9765 if (detectedParams == null) return; // only works on the first detected avatar 10490 if (detectedParams == null)
9766 10491 {
10492 if (m_host.ParentGroup.IsAttachment == true)
10493 {
10494 detectedParams = new DetectParams();
10495 detectedParams.Key = m_host.OwnerID;
10496 }
10497 else
10498 {
10499 return;
10500 }
10501 }
10502
9767 ScenePresence avatar = World.GetScenePresence(detectedParams.Key); 10503 ScenePresence avatar = World.GetScenePresence(detectedParams.Key);
9768 if (avatar != null) 10504 if (avatar != null)
9769 { 10505 {
@@ -9771,6 +10507,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9771 new Vector3((float)pos.x, (float)pos.y, (float)pos.z), 10507 new Vector3((float)pos.x, (float)pos.y, (float)pos.z),
9772 new Vector3((float)lookAt.x, (float)lookAt.y, (float)lookAt.z)); 10508 new Vector3((float)lookAt.x, (float)lookAt.y, (float)lookAt.z));
9773 } 10509 }
10510
9774 ScriptSleep(1000); 10511 ScriptSleep(1000);
9775 } 10512 }
9776 10513
@@ -9894,12 +10631,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
9894 10631
9895 SortedDictionary<int, float> parameters = new SortedDictionary<int, float>(); 10632 SortedDictionary<int, float> parameters = new SortedDictionary<int, float>();
9896 object[] data = rules.Data; 10633 object[] data = rules.Data;
9897 for (int i = 0; i < data.Length; ++i) { 10634 for (int i = 0; i < data.Length; ++i)
10635 {
9898 int type = Convert.ToInt32(data[i++].ToString()); 10636 int type = Convert.ToInt32(data[i++].ToString());
9899 if (i >= data.Length) break; // odd number of entries => ignore the last 10637 if (i >= data.Length) break; // odd number of entries => ignore the last
9900 10638
9901 // some special cases: Vector parameters are split into 3 float parameters (with type+1, type+2, type+3) 10639 // some special cases: Vector parameters are split into 3 float parameters (with type+1, type+2, type+3)
9902 switch (type) { 10640 switch (type)
10641 {
9903 case ScriptBaseClass.CAMERA_FOCUS: 10642 case ScriptBaseClass.CAMERA_FOCUS:
9904 case ScriptBaseClass.CAMERA_FOCUS_OFFSET: 10643 case ScriptBaseClass.CAMERA_FOCUS_OFFSET:
9905 case ScriptBaseClass.CAMERA_POSITION: 10644 case ScriptBaseClass.CAMERA_POSITION:
@@ -10005,19 +10744,65 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10005 public LSL_String llXorBase64StringsCorrect(string str1, string str2) 10744 public LSL_String llXorBase64StringsCorrect(string str1, string str2)
10006 { 10745 {
10007 m_host.AddScriptLPS(1); 10746 m_host.AddScriptLPS(1);
10008 string ret = String.Empty; 10747
10009 string src1 = llBase64ToString(str1); 10748 if (str1 == String.Empty)
10010 string src2 = llBase64ToString(str2); 10749 return String.Empty;
10011 int c = 0; 10750 if (str2 == String.Empty)
10012 for (int i = 0; i < src1.Length; i++) 10751 return str1;
10752
10753 int len = str2.Length;
10754 if ((len % 4) != 0) // LL is EVIL!!!!
10755 {
10756 while (str2.EndsWith("="))
10757 str2 = str2.Substring(0, str2.Length - 1);
10758
10759 len = str2.Length;
10760 int mod = len % 4;
10761
10762 if (mod == 1)
10763 str2 = str2.Substring(0, str2.Length - 1);
10764 else if (mod == 2)
10765 str2 += "==";
10766 else if (mod == 3)
10767 str2 += "=";
10768 }
10769
10770 byte[] data1;
10771 byte[] data2;
10772 try
10773 {
10774 data1 = Convert.FromBase64String(str1);
10775 data2 = Convert.FromBase64String(str2);
10776 }
10777 catch (Exception)
10013 { 10778 {
10014 ret += (char) (src1[i] ^ src2[c]); 10779 return new LSL_String(String.Empty);
10780 }
10015 10781
10016 c++; 10782 byte[] d2 = new Byte[data1.Length];
10017 if (c >= src2.Length) 10783 int pos = 0;
10018 c = 0; 10784
10785 if (data1.Length <= data2.Length)
10786 {
10787 Array.Copy(data2, 0, d2, 0, data1.Length);
10788 }
10789 else
10790 {
10791 while (pos < data1.Length)
10792 {
10793 len = data1.Length - pos;
10794 if (len > data2.Length)
10795 len = data2.Length;
10796
10797 Array.Copy(data2, 0, d2, pos, len);
10798 pos += len;
10799 }
10019 } 10800 }
10020 return llStringToBase64(ret); 10801
10802 for (pos = 0 ; pos < data1.Length ; pos++ )
10803 data1[pos] ^= d2[pos];
10804
10805 return Convert.ToBase64String(data1);
10021 } 10806 }
10022 10807
10023 public LSL_String llHTTPRequest(string url, LSL_List parameters, string body) 10808 public LSL_String llHTTPRequest(string url, LSL_List parameters, string body)
@@ -10074,12 +10859,15 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10074 Regex r = new Regex(authregex); 10859 Regex r = new Regex(authregex);
10075 int[] gnums = r.GetGroupNumbers(); 10860 int[] gnums = r.GetGroupNumbers();
10076 Match m = r.Match(url); 10861 Match m = r.Match(url);
10077 if (m.Success) { 10862 if (m.Success)
10078 for (int i = 1; i < gnums.Length; i++) { 10863 {
10864 for (int i = 1; i < gnums.Length; i++)
10865 {
10079 //System.Text.RegularExpressions.Group g = m.Groups[gnums[i]]; 10866 //System.Text.RegularExpressions.Group g = m.Groups[gnums[i]];
10080 //CaptureCollection cc = g.Captures; 10867 //CaptureCollection cc = g.Captures;
10081 } 10868 }
10082 if (m.Groups.Count == 5) { 10869 if (m.Groups.Count == 5)
10870 {
10083 httpHeaders["Authorization"] = String.Format("Basic {0}", Convert.ToBase64String(System.Text.ASCIIEncoding.ASCII.GetBytes(m.Groups[2].ToString() + ":" + m.Groups[3].ToString()))); 10871 httpHeaders["Authorization"] = String.Format("Basic {0}", Convert.ToBase64String(System.Text.ASCIIEncoding.ASCII.GetBytes(m.Groups[2].ToString() + ":" + m.Groups[3].ToString())));
10084 url = m.Groups[1].ToString() + m.Groups[4].ToString(); 10872 url = m.Groups[1].ToString() + m.Groups[4].ToString();
10085 } 10873 }
@@ -10442,15 +11230,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10442 11230
10443 internal UUID ScriptByName(string name) 11231 internal UUID ScriptByName(string name)
10444 { 11232 {
10445 lock (m_host.TaskInventory) 11233 m_host.TaskInventory.LockItemsForRead(true);
11234
11235 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
10446 { 11236 {
10447 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 11237 if (item.Type == 10 && item.Name == name)
10448 { 11238 {
10449 if (item.Type == 10 && item.Name == name) 11239 m_host.TaskInventory.LockItemsForRead(false);
10450 return item.ItemID; 11240 return item.ItemID;
10451 } 11241 }
10452 } 11242 }
10453 11243
11244 m_host.TaskInventory.LockItemsForRead(false);
11245
10454 return UUID.Zero; 11246 return UUID.Zero;
10455 } 11247 }
10456 11248
@@ -10491,6 +11283,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10491 { 11283 {
10492 m_host.AddScriptLPS(1); 11284 m_host.AddScriptLPS(1);
10493 11285
11286 //Clone is thread safe
10494 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 11287 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
10495 11288
10496 UUID assetID = UUID.Zero; 11289 UUID assetID = UUID.Zero;
@@ -10553,6 +11346,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10553 { 11346 {
10554 m_host.AddScriptLPS(1); 11347 m_host.AddScriptLPS(1);
10555 11348
11349 //Clone is thread safe
10556 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone(); 11350 TaskInventoryDictionary itemsDictionary = (TaskInventoryDictionary)m_host.TaskInventory.Clone();
10557 11351
10558 UUID assetID = UUID.Zero; 11352 UUID assetID = UUID.Zero;
@@ -10633,15 +11427,13 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10633 return GetLinkPrimitiveParams(obj, rules); 11427 return GetLinkPrimitiveParams(obj, rules);
10634 } 11428 }
10635 11429
10636 public void print(string str) 11430 public LSL_Integer llGetLinkNumberOfSides(LSL_Integer link)
10637 { 11431 {
10638 // yes, this is a real LSL function. See: http://wiki.secondlife.com/wiki/Print 11432 List<SceneObjectPart> parts = GetLinkParts(link);
10639 IOSSL_Api ossl = (IOSSL_Api)m_ScriptEngine.GetApi(m_item.ItemID, "OSSL"); 11433 if (parts.Count < 1)
10640 if (ossl != null) 11434 return 0;
10641 { 11435
10642 ossl.CheckThreatLevel(ThreatLevel.High, "print"); 11436 return GetNumberOfSides(parts[0]);
10643 m_log.Info("LSL print():" + str);
10644 }
10645 } 11437 }
10646 11438
10647 private string Name2Username(string name) 11439 private string Name2Username(string name)
@@ -10687,155 +11479,482 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10687 return rq.ToString(); 11479 return rq.ToString();
10688 } 11480 }
10689 11481
11482 private void SayShoutTimerElapsed(Object sender, ElapsedEventArgs args)
11483 {
11484 m_SayShoutCount = 0;
11485 }
11486
11487 private struct Tri
11488 {
11489 public Vector3 p1;
11490 public Vector3 p2;
11491 public Vector3 p3;
11492 }
11493
11494 private bool InBoundingBox(ScenePresence avatar, Vector3 point)
11495 {
11496 float height = avatar.Appearance.AvatarHeight;
11497 Vector3 b1 = avatar.AbsolutePosition + new Vector3(-0.22f, -0.22f, -height/2);
11498 Vector3 b2 = avatar.AbsolutePosition + new Vector3(0.22f, 0.22f, height/2);
11499
11500 if (point.X > b1.X && point.X < b2.X &&
11501 point.Y > b1.Y && point.Y < b2.Y &&
11502 point.Z > b1.Z && point.Z < b2.Z)
11503 return true;
11504 return false;
11505 }
11506
11507 private ContactResult[] AvatarIntersection(Vector3 rayStart, Vector3 rayEnd)
11508 {
11509 List<ContactResult> contacts = new List<ContactResult>();
11510
11511 Vector3 ab = rayEnd - rayStart;
11512
11513 World.ForEachScenePresence(delegate(ScenePresence sp)
11514 {
11515 Vector3 ac = sp.AbsolutePosition - rayStart;
11516 Vector3 bc = sp.AbsolutePosition - rayEnd;
11517
11518 double d = Math.Abs(Vector3.Mag(Vector3.Cross(ab, ac)) / Vector3.Distance(rayStart, rayEnd));
11519
11520 if (d > 1.5)
11521 return;
11522
11523 double d2 = Vector3.Dot(Vector3.Negate(ab), ac);
11524
11525 if (d2 > 0)
11526 return;
11527
11528 double dp = Math.Sqrt(Vector3.Mag(ac) * Vector3.Mag(ac) - d * d);
11529 Vector3 p = rayStart + Vector3.Divide(Vector3.Multiply(ab, (float)dp), (float)Vector3.Mag(ab));
11530
11531 if (!InBoundingBox(sp, p))
11532 return;
11533
11534 ContactResult result = new ContactResult ();
11535 result.ConsumerID = sp.LocalId;
11536 result.Depth = Vector3.Distance(rayStart, p);
11537 result.Normal = Vector3.Zero;
11538 result.Pos = p;
11539
11540 contacts.Add(result);
11541 });
11542
11543 return contacts.ToArray();
11544 }
11545
11546 private ContactResult[] ObjectIntersection(Vector3 rayStart, Vector3 rayEnd, bool includePhysical, bool includeNonPhysical, bool includePhantom)
11547 {
11548 Ray ray = new Ray(rayStart, Vector3.Normalize(rayEnd - rayStart));
11549 List<ContactResult> contacts = new List<ContactResult>();
11550
11551 Vector3 ab = rayEnd - rayStart;
11552
11553 World.ForEachSOG(delegate(SceneObjectGroup group)
11554 {
11555 if (m_host.ParentGroup == group)
11556 return;
11557
11558 if (group.IsAttachment)
11559 return;
11560
11561 if (group.RootPart.PhysActor == null)
11562 {
11563 if (!includePhantom)
11564 return;
11565 }
11566 else
11567 {
11568 if (group.RootPart.PhysActor.IsPhysical)
11569 {
11570 if (!includePhysical)
11571 return;
11572 }
11573 else
11574 {
11575 if (!includeNonPhysical)
11576 return;
11577 }
11578 }
11579
11580 // Find the radius ouside of which we don't even need to hit test
11581 float minX;
11582 float maxX;
11583 float minY;
11584 float maxY;
11585 float minZ;
11586 float maxZ;
11587
11588 float radius = 0.0f;
11589
11590 group.GetAxisAlignedBoundingBoxRaw(out minX, out maxX, out minY, out maxY, out minZ, out maxZ);
11591
11592 if (Math.Abs(minX) > radius)
11593 radius = Math.Abs(minX);
11594 if (Math.Abs(minY) > radius)
11595 radius = Math.Abs(minY);
11596 if (Math.Abs(minZ) > radius)
11597 radius = Math.Abs(minZ);
11598 if (Math.Abs(maxX) > radius)
11599 radius = Math.Abs(maxX);
11600 if (Math.Abs(maxY) > radius)
11601 radius = Math.Abs(maxY);
11602 if (Math.Abs(maxZ) > radius)
11603 radius = Math.Abs(maxZ);
11604
11605 Vector3 ac = group.AbsolutePosition - rayStart;
11606 Vector3 bc = group.AbsolutePosition - rayEnd;
11607
11608 double d = Math.Abs(Vector3.Mag(Vector3.Cross(ab, ac)) / Vector3.Distance(rayStart, rayEnd));
11609
11610 // Too far off ray, don't bother
11611 if (d > radius)
11612 return;
11613
11614 // Behind ray, drop
11615 double d2 = Vector3.Dot(Vector3.Negate(ab), ac);
11616 if (d2 > 0)
11617 return;
11618
11619 EntityIntersection intersection = group.TestIntersection(ray, true, false);
11620 // Miss.
11621 if (!intersection.HitTF)
11622 return;
11623
11624 ContactResult result = new ContactResult ();
11625 result.ConsumerID = group.LocalId;
11626 result.Depth = intersection.distance;
11627 result.Normal = intersection.normal;
11628 result.Pos = intersection.ipoint;
11629
11630 contacts.Add(result);
11631 });
11632
11633 return contacts.ToArray();
11634 }
11635
11636 private ContactResult? GroundIntersection(Vector3 rayStart, Vector3 rayEnd)
11637 {
11638 double[,] heightfield = World.Heightmap.GetDoubles();
11639 List<ContactResult> contacts = new List<ContactResult>();
11640
11641 double min = 2048.0;
11642 double max = 0.0;
11643
11644 // Find the min and max of the heightfield
11645 for (int x = 0 ; x < World.Heightmap.Width ; x++)
11646 {
11647 for (int y = 0 ; y < World.Heightmap.Height ; y++)
11648 {
11649 if (heightfield[x, y] > max)
11650 max = heightfield[x, y];
11651 if (heightfield[x, y] < min)
11652 min = heightfield[x, y];
11653 }
11654 }
11655
11656
11657 // A ray extends past rayEnd, but doesn't go back before
11658 // rayStart. If the start is above the highest point of the ground
11659 // and the ray goes up, we can't hit the ground. Ever.
11660 if (rayStart.Z > max && rayEnd.Z >= rayStart.Z)
11661 return null;
11662
11663 // Same for going down
11664 if (rayStart.Z < min && rayEnd.Z <= rayStart.Z)
11665 return null;
11666
11667 List<Tri> trilist = new List<Tri>();
11668
11669 // Create our triangle list
11670 for (int x = 1 ; x < World.Heightmap.Width ; x++)
11671 {
11672 for (int y = 1 ; y < World.Heightmap.Height ; y++)
11673 {
11674 Tri t1 = new Tri();
11675 Tri t2 = new Tri();
11676
11677 Vector3 p1 = new Vector3(x-1, y-1, (float)heightfield[x-1, y-1]);
11678 Vector3 p2 = new Vector3(x, y-1, (float)heightfield[x, y-1]);
11679 Vector3 p3 = new Vector3(x, y, (float)heightfield[x, y]);
11680 Vector3 p4 = new Vector3(x-1, y, (float)heightfield[x-1, y]);
11681
11682 t1.p1 = p1;
11683 t1.p2 = p2;
11684 t1.p3 = p3;
11685
11686 t2.p1 = p3;
11687 t2.p2 = p4;
11688 t2.p3 = p1;
11689
11690 trilist.Add(t1);
11691 trilist.Add(t2);
11692 }
11693 }
11694
11695 // Ray direction
11696 Vector3 rayDirection = rayEnd - rayStart;
11697
11698 foreach (Tri t in trilist)
11699 {
11700 // Compute triangle plane normal and edges
11701 Vector3 u = t.p2 - t.p1;
11702 Vector3 v = t.p3 - t.p1;
11703 Vector3 n = Vector3.Cross(u, v);
11704
11705 if (n == Vector3.Zero)
11706 continue;
11707
11708 Vector3 w0 = rayStart - t.p1;
11709 double a = -Vector3.Dot(n, w0);
11710 double b = Vector3.Dot(n, rayDirection);
11711
11712 // Not intersecting the plane, or in plane (same thing)
11713 // Ignoring this MAY cause the ground to not be detected
11714 // sometimes
11715 if (Math.Abs(b) < 0.000001)
11716 continue;
11717
11718 double r = a / b;
11719
11720 // ray points away from plane
11721 if (r < 0.0)
11722 continue;
11723
11724 Vector3 ip = rayStart + Vector3.Multiply(rayDirection, (float)r);
11725
11726 float uu = Vector3.Dot(u, u);
11727 float uv = Vector3.Dot(u, v);
11728 float vv = Vector3.Dot(v, v);
11729 Vector3 w = ip - t.p1;
11730 float wu = Vector3.Dot(w, u);
11731 float wv = Vector3.Dot(w, v);
11732 float d = uv * uv - uu * vv;
11733
11734 float cs = (uv * wv - vv * wu) / d;
11735 if (cs < 0 || cs > 1.0)
11736 continue;
11737 float ct = (uv * wu - uu * wv) / d;
11738 if (ct < 0 || (cs + ct) > 1.0)
11739 continue;
11740
11741 // Add contact point
11742 ContactResult result = new ContactResult ();
11743 result.ConsumerID = 0;
11744 result.Depth = Vector3.Distance(rayStart, ip);
11745 result.Normal = n;
11746 result.Pos = ip;
11747
11748 contacts.Add(result);
11749 }
11750
11751 if (contacts.Count == 0)
11752 return null;
11753
11754 contacts.Sort(delegate(ContactResult a, ContactResult b)
11755 {
11756 return (int)(a.Depth - b.Depth);
11757 });
11758
11759 return contacts[0];
11760 }
11761/*
11762 // not done:
11763 private ContactResult[] testRay2NonPhysicalPhantom(Vector3 rayStart, Vector3 raydir, float raylenght)
11764 {
11765 ContactResult[] contacts = null;
11766 World.ForEachSOG(delegate(SceneObjectGroup group)
11767 {
11768 if (m_host.ParentGroup == group)
11769 return;
11770
11771 if (group.IsAttachment)
11772 return;
11773
11774 if(group.RootPart.PhysActor != null)
11775 return;
11776
11777 contacts = group.RayCastGroupPartsOBBNonPhysicalPhantom(rayStart, raydir, raylenght);
11778 });
11779 return contacts;
11780 }
11781*/
11782
10690 public LSL_List llCastRay(LSL_Vector start, LSL_Vector end, LSL_List options) 11783 public LSL_List llCastRay(LSL_Vector start, LSL_Vector end, LSL_List options)
10691 { 11784 {
11785 LSL_List list = new LSL_List();
11786
10692 m_host.AddScriptLPS(1); 11787 m_host.AddScriptLPS(1);
10693 11788
10694 Vector3 dir = new Vector3((float)(end-start).x, (float)(end-start).y, (float)(end-start).z); 11789 Vector3 rayStart = new Vector3((float)start.x, (float)start.y, (float)start.z);
10695 Vector3 startvector = new Vector3((float)start.x, (float)start.y, (float)start.z); 11790 Vector3 rayEnd = new Vector3((float)end.x, (float)end.y, (float)end.z);
10696 Vector3 endvector = new Vector3((float)end.x, (float)end.y, (float)end.z); 11791 Vector3 dir = rayEnd - rayStart;
10697 11792
10698 int count = 0; 11793 float dist = Vector3.Mag(dir);
10699// int detectPhantom = 0; 11794
11795 int count = 1;
11796 bool detectPhantom = false;
10700 int dataFlags = 0; 11797 int dataFlags = 0;
10701 int rejectTypes = 0; 11798 int rejectTypes = 0;
10702 11799
10703 for (int i = 0; i < options.Length; i += 2) 11800 for (int i = 0; i < options.Length; i += 2)
10704 { 11801 {
10705 if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_MAX_HITS) 11802 if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_MAX_HITS)
10706 {
10707 count = options.GetLSLIntegerItem(i + 1); 11803 count = options.GetLSLIntegerItem(i + 1);
10708 } 11804 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DETECT_PHANTOM)
10709// else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DETECT_PHANTOM) 11805 detectPhantom = (options.GetLSLIntegerItem(i + 1) > 0);
10710// {
10711// detectPhantom = options.GetLSLIntegerItem(i + 1);
10712// }
10713 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DATA_FLAGS) 11806 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_DATA_FLAGS)
10714 {
10715 dataFlags = options.GetLSLIntegerItem(i + 1); 11807 dataFlags = options.GetLSLIntegerItem(i + 1);
10716 }
10717 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_REJECT_TYPES) 11808 else if (options.GetLSLIntegerItem(i) == ScriptBaseClass.RC_REJECT_TYPES)
10718 {
10719 rejectTypes = options.GetLSLIntegerItem(i + 1); 11809 rejectTypes = options.GetLSLIntegerItem(i + 1);
10720 }
10721 } 11810 }
10722 11811
10723 LSL_List list = new LSL_List(); 11812 if (count > 16)
10724 List<ContactResult> results = World.PhysicsScene.RaycastWorld(startvector, dir, dir.Length(), count); 11813 count = 16;
10725
10726 double distance = Util.GetDistanceTo(startvector, endvector);
10727
10728 if (distance == 0)
10729 distance = 0.001;
10730 11814
10731 Vector3 posToCheck = startvector; 11815 List<ContactResult> results = new List<ContactResult>();
10732 ITerrainChannel channel = World.RequestModuleInterface<ITerrainChannel>();
10733 11816
10734 bool checkTerrain = !((rejectTypes & ScriptBaseClass.RC_REJECT_LAND) == ScriptBaseClass.RC_REJECT_LAND); 11817 bool checkTerrain = !((rejectTypes & ScriptBaseClass.RC_REJECT_LAND) == ScriptBaseClass.RC_REJECT_LAND);
10735 bool checkAgents = !((rejectTypes & ScriptBaseClass.RC_REJECT_AGENTS) == ScriptBaseClass.RC_REJECT_AGENTS); 11818 bool checkAgents = !((rejectTypes & ScriptBaseClass.RC_REJECT_AGENTS) == ScriptBaseClass.RC_REJECT_AGENTS);
10736 bool checkNonPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_NONPHYSICAL) == ScriptBaseClass.RC_REJECT_NONPHYSICAL); 11819 bool checkNonPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_NONPHYSICAL) == ScriptBaseClass.RC_REJECT_NONPHYSICAL);
10737 bool checkPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_PHYSICAL) == ScriptBaseClass.RC_REJECT_PHYSICAL); 11820 bool checkPhysical = !((rejectTypes & ScriptBaseClass.RC_REJECT_PHYSICAL) == ScriptBaseClass.RC_REJECT_PHYSICAL);
10738 11821
10739 for (float i = 0; i <= distance; i += 0.1f) 11822
11823 if (World.SuportsRayCastFiltered())
10740 { 11824 {
10741 posToCheck = startvector + (dir * (i / (float)distance)); 11825 if (dist == 0)
11826 return list;
11827
11828 RayFilterFlags rayfilter = RayFilterFlags.ClosestAndBackCull;
11829 if (checkTerrain)
11830 rayfilter |= RayFilterFlags.land;
11831// if (checkAgents)
11832// rayfilter |= RayFilterFlags.agent;
11833 if (checkPhysical)
11834 rayfilter |= RayFilterFlags.physical;
11835 if (checkNonPhysical)
11836 rayfilter |= RayFilterFlags.nonphysical;
11837 if (detectPhantom)
11838 rayfilter |= RayFilterFlags.LSLPhanton;
11839
11840 Vector3 direction = dir * ( 1/dist);
10742 11841
10743 if (checkTerrain && channel[(int)(posToCheck.X + startvector.X), (int)(posToCheck.Y + startvector.Y)] < posToCheck.Z) 11842 if(rayfilter == 0)
10744 { 11843 {
10745 ContactResult result = new ContactResult(); 11844 list.Add(new LSL_Integer(0));
10746 result.ConsumerID = 0; 11845 return list;
10747 result.Depth = 0;
10748 result.Normal = Vector3.Zero;
10749 result.Pos = posToCheck;
10750 results.Add(result);
10751 checkTerrain = false;
10752 } 11846 }
10753 11847
10754 if (checkAgents) 11848 // get some more contacts to sort ???
11849 int physcount = 4 * count;
11850 if (physcount > 20)
11851 physcount = 20;
11852
11853 object physresults;
11854 physresults = World.RayCastFiltered(rayStart, direction, dist, physcount, rayfilter);
11855
11856 if (physresults == null)
10755 { 11857 {
10756 World.ForEachRootScenePresence(delegate(ScenePresence sp) 11858 list.Add(new LSL_Integer(-3)); // timeout error
10757 { 11859 return list;
10758 if (sp.AbsolutePosition.ApproxEquals(posToCheck, sp.PhysicsActor.Size.X))
10759 {
10760 ContactResult result = new ContactResult ();
10761 result.ConsumerID = sp.LocalId;
10762 result.Depth = 0;
10763 result.Normal = Vector3.Zero;
10764 result.Pos = posToCheck;
10765 results.Add(result);
10766 }
10767 });
10768 } 11860 }
10769 }
10770 11861
10771 int refcount = 0; 11862 results = (List<ContactResult>)physresults;
10772 foreach (ContactResult result in results) 11863
10773 { 11864 // for now physics doesn't detect sitted avatars so do it outside physics
10774 if ((rejectTypes & ScriptBaseClass.RC_REJECT_LAND) 11865 if (checkAgents)
10775 == ScriptBaseClass.RC_REJECT_LAND && result.ConsumerID == 0) 11866 {
10776 continue; 11867 ContactResult[] agentHits = AvatarIntersection(rayStart, rayEnd);
11868 foreach (ContactResult r in agentHits)
11869 results.Add(r);
11870 }
10777 11871
10778 ISceneEntity entity = World.GetSceneObjectPart(result.ConsumerID); 11872 // bug: will not detect phantom unless they are physical
11873 // don't use ObjectIntersection because its also bad
10779 11874
10780 if (entity == null && (rejectTypes & ScriptBaseClass.RC_REJECT_AGENTS) != ScriptBaseClass.RC_REJECT_AGENTS) 11875 }
10781 entity = World.GetScenePresence(result.ConsumerID); //Only check if we should be looking for agents 11876 else
11877 {
11878 if (checkTerrain)
11879 {
11880 ContactResult? groundContact = GroundIntersection(rayStart, rayEnd);
11881 if (groundContact != null)
11882 results.Add((ContactResult)groundContact);
11883 }
10782 11884
10783 if (entity == null) 11885 if (checkAgents)
10784 { 11886 {
10785 list.Add(UUID.Zero); 11887 ContactResult[] agentHits = AvatarIntersection(rayStart, rayEnd);
11888 foreach (ContactResult r in agentHits)
11889 results.Add(r);
11890 }
10786 11891
10787 if ((dataFlags & ScriptBaseClass.RC_GET_LINK_NUM) == ScriptBaseClass.RC_GET_LINK_NUM) 11892 if (checkPhysical || checkNonPhysical || detectPhantom)
10788 list.Add(0); 11893 {
11894 ContactResult[] objectHits = ObjectIntersection(rayStart, rayEnd, checkPhysical, checkNonPhysical, detectPhantom);
11895 foreach (ContactResult r in objectHits)
11896 results.Add(r);
11897 }
11898 }
10789 11899
10790 list.Add(result.Pos); 11900 results.Sort(delegate(ContactResult a, ContactResult b)
11901 {
11902 return a.Depth.CompareTo(b.Depth);
11903 });
11904
11905 int values = 0;
11906 SceneObjectGroup thisgrp = m_host.ParentGroup;
10791 11907
10792 if ((dataFlags & ScriptBaseClass.RC_GET_NORMAL) == ScriptBaseClass.RC_GET_NORMAL) 11908 foreach (ContactResult result in results)
10793 list.Add(result.Normal); 11909 {
11910 if (result.Depth > dist)
11911 continue;
10794 11912
10795 continue; //Can't find it, so add UUID.Zero 11913 // physics ray can return colisions with host prim
10796 } 11914 if (m_host.LocalId == result.ConsumerID)
11915 continue;
10797 11916
10798 /*if (detectPhantom == 0 && intersection.obj is ISceneChildEntity && 11917 UUID itemID = UUID.Zero;
10799 ((ISceneChildEntity)intersection.obj).PhysActor == null) 11918 int linkNum = 0;
10800 continue;*/ //Can't do this ATM, physics engine knows only of non phantom objects
10801 11919
10802 if (entity is SceneObjectPart) 11920 SceneObjectPart part = World.GetSceneObjectPart(result.ConsumerID);
11921 // It's a prim!
11922 if (part != null)
10803 { 11923 {
10804 PhysicsActor pa = ((SceneObjectPart)entity).PhysActor; 11924 // dont detect members of same object ???
11925 if (part.ParentGroup == thisgrp)
11926 continue;
10805 11927
10806 if (pa != null && pa.IsPhysical) 11928 if ((dataFlags & ScriptBaseClass.RC_GET_ROOT_KEY) == ScriptBaseClass.RC_GET_ROOT_KEY)
10807 { 11929 itemID = part.ParentGroup.UUID;
10808 if (!checkPhysical)
10809 continue;
10810 }
10811 else 11930 else
10812 { 11931 itemID = part.UUID;
10813 if (!checkNonPhysical)
10814 continue;
10815 }
10816 }
10817 11932
10818 refcount++; 11933 linkNum = part.LinkNum;
10819 if ((dataFlags & ScriptBaseClass.RC_GET_ROOT_KEY) == ScriptBaseClass.RC_GET_ROOT_KEY && entity is SceneObjectPart) 11934 }
10820 list.Add(((SceneObjectPart)entity).ParentGroup.UUID);
10821 else 11935 else
10822 list.Add(entity.UUID);
10823
10824 if ((dataFlags & ScriptBaseClass.RC_GET_LINK_NUM) == ScriptBaseClass.RC_GET_LINK_NUM)
10825 { 11936 {
10826 if (entity is SceneObjectPart) 11937 ScenePresence sp = World.GetScenePresence(result.ConsumerID);
10827 list.Add(((SceneObjectPart)entity).LinkNum); 11938 /// It it a boy? a girl?
10828 else 11939 if (sp != null)
10829 list.Add(0); 11940 itemID = sp.UUID;
10830 } 11941 }
10831 11942
10832 list.Add(result.Pos); 11943 list.Add(new LSL_String(itemID.ToString()));
11944 list.Add(new LSL_String(result.Pos.ToString()));
11945
11946 if ((dataFlags & ScriptBaseClass.RC_GET_LINK_NUM) == ScriptBaseClass.RC_GET_LINK_NUM)
11947 list.Add(new LSL_Integer(linkNum));
10833 11948
10834 if ((dataFlags & ScriptBaseClass.RC_GET_NORMAL) == ScriptBaseClass.RC_GET_NORMAL) 11949 if ((dataFlags & ScriptBaseClass.RC_GET_NORMAL) == ScriptBaseClass.RC_GET_NORMAL)
10835 list.Add(result.Normal); 11950 list.Add(new LSL_Vector(result.Normal.X, result.Normal.Y, result.Normal.Z));
11951
11952 values++;
11953 if (values >= count)
11954 break;
10836 } 11955 }
10837 11956
10838 list.Add(refcount); //The status code, either the # of contacts, RCERR_SIM_PERF_LOW, or RCERR_CAST_TIME_EXCEEDED 11957 list.Add(new LSL_Integer(values));
10839 11958
10840 return list; 11959 return list;
10841 } 11960 }
@@ -10875,7 +11994,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10875 case ScriptBaseClass.ESTATE_ACCESS_ALLOWED_AGENT_ADD: 11994 case ScriptBaseClass.ESTATE_ACCESS_ALLOWED_AGENT_ADD:
10876 if (!isAccount) return 0; 11995 if (!isAccount) return 0;
10877 if (estate.HasAccess(id)) return 1; 11996 if (estate.HasAccess(id)) return 1;
10878 if (estate.IsBanned(id)) 11997 if (estate.IsBanned(id, World.GetUserFlags(id)))
10879 estate.RemoveBan(id); 11998 estate.RemoveBan(id);
10880 estate.AddEstateUser(id); 11999 estate.AddEstateUser(id);
10881 break; 12000 break;
@@ -10894,14 +12013,14 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10894 break; 12013 break;
10895 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_ADD: 12014 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_ADD:
10896 if (!isAccount) return 0; 12015 if (!isAccount) return 0;
10897 if (estate.IsBanned(id)) return 1; 12016 if (estate.IsBanned(id, World.GetUserFlags(id))) return 1;
10898 EstateBan ban = new EstateBan(); 12017 EstateBan ban = new EstateBan();
10899 ban.EstateID = estate.EstateID; 12018 ban.EstateID = estate.EstateID;
10900 ban.BannedUserID = id; 12019 ban.BannedUserID = id;
10901 estate.AddBan(ban); 12020 estate.AddBan(ban);
10902 break; 12021 break;
10903 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_REMOVE: 12022 case ScriptBaseClass.ESTATE_ACCESS_BANNED_AGENT_REMOVE:
10904 if (!isAccount || !estate.IsBanned(id)) return 0; 12023 if (!isAccount || !estate.IsBanned(id, World.GetUserFlags(id))) return 0;
10905 estate.RemoveBan(id); 12024 estate.RemoveBan(id);
10906 break; 12025 break;
10907 default: return 0; 12026 default: return 0;
@@ -10930,7 +12049,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10930 return 16384; 12049 return 16384;
10931 } 12050 }
10932 12051
10933 public LSL_Integer llGetUsedMemory() 12052 public virtual LSL_Integer llGetUsedMemory()
10934 { 12053 {
10935 m_host.AddScriptLPS(1); 12054 m_host.AddScriptLPS(1);
10936 // The value returned for LSO scripts in SL 12055 // The value returned for LSO scripts in SL
@@ -10958,7 +12077,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10958 public void llSetSoundQueueing(int queue) 12077 public void llSetSoundQueueing(int queue)
10959 { 12078 {
10960 m_host.AddScriptLPS(1); 12079 m_host.AddScriptLPS(1);
10961 NotImplemented("llSetSoundQueueing");
10962 } 12080 }
10963 12081
10964 public void llCollisionSprite(string impact_sprite) 12082 public void llCollisionSprite(string impact_sprite)
@@ -10970,10 +12088,270 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
10970 public void llGodLikeRezObject(string inventory, LSL_Vector pos) 12088 public void llGodLikeRezObject(string inventory, LSL_Vector pos)
10971 { 12089 {
10972 m_host.AddScriptLPS(1); 12090 m_host.AddScriptLPS(1);
10973 NotImplemented("llGodLikeRezObject"); 12091
12092 if (!World.Permissions.IsGod(m_host.OwnerID))
12093 NotImplemented("llGodLikeRezObject");
12094
12095 AssetBase rezAsset = World.AssetService.Get(inventory);
12096 if (rezAsset == null)
12097 {
12098 llSay(0, "Asset not found");
12099 return;
12100 }
12101
12102 SceneObjectGroup group = null;
12103
12104 try
12105 {
12106 string xmlData = Utils.BytesToString(rezAsset.Data);
12107 group = SceneObjectSerializer.FromOriginalXmlFormat(xmlData);
12108 }
12109 catch
12110 {
12111 llSay(0, "Asset not found");
12112 return;
12113 }
12114
12115 if (group == null)
12116 {
12117 llSay(0, "Asset not found");
12118 return;
12119 }
12120
12121 group.RootPart.AttachPoint = group.RootPart.Shape.State;
12122 group.RootPart.AttachOffset = group.AbsolutePosition;
12123
12124 group.ResetIDs();
12125
12126 Vector3 llpos = new Vector3((float)pos.x, (float)pos.y, (float)pos.z);
12127 World.AddNewSceneObject(group, true, llpos, Quaternion.Identity, Vector3.Zero);
12128 group.CreateScriptInstances(0, true, World.DefaultScriptEngine, 3);
12129 group.ScheduleGroupForFullUpdate();
12130
12131 // objects rezzed with this method are die_at_edge by default.
12132 group.RootPart.SetDieAtEdge(true);
12133
12134 group.ResumeScripts();
12135
12136 m_ScriptEngine.PostObjectEvent(m_host.LocalId, new EventParams(
12137 "object_rez", new Object[] {
12138 new LSL_String(
12139 group.RootPart.UUID.ToString()) },
12140 new DetectParams[0]));
12141 }
12142
12143 public LSL_String llTransferLindenDollars(string destination, int amount)
12144 {
12145 UUID txn = UUID.Random();
12146
12147 Util.FireAndForget(delegate(object x)
12148 {
12149 int replycode = 0;
12150 string replydata = destination + "," + amount.ToString();
12151
12152 try
12153 {
12154 TaskInventoryItem item = m_item;
12155 if (item == null)
12156 {
12157 replydata = "SERVICE_ERROR";
12158 return;
12159 }
12160
12161 m_host.AddScriptLPS(1);
12162
12163 if (item.PermsGranter == UUID.Zero)
12164 {
12165 replydata = "MISSING_PERMISSION_DEBIT";
12166 return;
12167 }
12168
12169 if ((item.PermsMask & ScriptBaseClass.PERMISSION_DEBIT) == 0)
12170 {
12171 replydata = "MISSING_PERMISSION_DEBIT";
12172 return;
12173 }
12174
12175 UUID toID = new UUID();
12176
12177 if (!UUID.TryParse(destination, out toID))
12178 {
12179 replydata = "INVALID_AGENT";
12180 return;
12181 }
12182
12183 IMoneyModule money = World.RequestModuleInterface<IMoneyModule>();
12184
12185 if (money == null)
12186 {
12187 replydata = "TRANSFERS_DISABLED";
12188 return;
12189 }
12190
12191 bool result = money.ObjectGiveMoney(
12192 m_host.ParentGroup.RootPart.UUID, m_host.ParentGroup.RootPart.OwnerID, toID, amount);
12193
12194 if (result)
12195 {
12196 replycode = 1;
12197 return;
12198 }
12199
12200 replydata = "LINDENDOLLAR_INSUFFICIENTFUNDS";
12201 }
12202 finally
12203 {
12204 m_ScriptEngine.PostScriptEvent(m_item.ItemID, new EventParams(
12205 "transaction_result", new Object[] {
12206 new LSL_String(txn.ToString()),
12207 new LSL_Integer(replycode),
12208 new LSL_String(replydata) },
12209 new DetectParams[0]));
12210 }
12211 });
12212
12213 return txn.ToString();
10974 } 12214 }
10975 12215
10976 #endregion 12216 #endregion
12217
12218 public void llSetKeyframedMotion(LSL_List frames, LSL_List options)
12219 {
12220 SceneObjectGroup group = m_host.ParentGroup;
12221
12222 if (group.RootPart.PhysActor != null && group.RootPart.PhysActor.IsPhysical)
12223 return;
12224 if (group.IsAttachment)
12225 return;
12226
12227 if (frames.Data.Length > 0) // We are getting a new motion
12228 {
12229 if (group.RootPart.KeyframeMotion != null)
12230 group.RootPart.KeyframeMotion.Stop();
12231 group.RootPart.KeyframeMotion = null;
12232
12233 int idx = 0;
12234
12235 KeyframeMotion.PlayMode mode = KeyframeMotion.PlayMode.Forward;
12236 KeyframeMotion.DataFormat data = KeyframeMotion.DataFormat.Translation | KeyframeMotion.DataFormat.Rotation;
12237
12238 while (idx < options.Data.Length)
12239 {
12240 int option = (int)options.GetLSLIntegerItem(idx++);
12241 int remain = options.Data.Length - idx;
12242
12243 switch (option)
12244 {
12245 case ScriptBaseClass.KFM_MODE:
12246 if (remain < 1)
12247 break;
12248 int modeval = (int)options.GetLSLIntegerItem(idx++);
12249 switch(modeval)
12250 {
12251 case ScriptBaseClass.KFM_FORWARD:
12252 mode = KeyframeMotion.PlayMode.Forward;
12253 break;
12254 case ScriptBaseClass.KFM_REVERSE:
12255 mode = KeyframeMotion.PlayMode.Reverse;
12256 break;
12257 case ScriptBaseClass.KFM_LOOP:
12258 mode = KeyframeMotion.PlayMode.Loop;
12259 break;
12260 case ScriptBaseClass.KFM_PING_PONG:
12261 mode = KeyframeMotion.PlayMode.PingPong;
12262 break;
12263 }
12264 break;
12265 case ScriptBaseClass.KFM_DATA:
12266 if (remain < 1)
12267 break;
12268 int dataval = (int)options.GetLSLIntegerItem(idx++);
12269 data = (KeyframeMotion.DataFormat)dataval;
12270 break;
12271 }
12272 }
12273
12274 group.RootPart.KeyframeMotion = new KeyframeMotion(group, mode, data);
12275
12276 idx = 0;
12277
12278 int elemLength = 2;
12279 if (data == (KeyframeMotion.DataFormat.Translation | KeyframeMotion.DataFormat.Rotation))
12280 elemLength = 3;
12281
12282 List<KeyframeMotion.Keyframe> keyframes = new List<KeyframeMotion.Keyframe>();
12283 while (idx < frames.Data.Length)
12284 {
12285 int remain = frames.Data.Length - idx;
12286
12287 if (remain < elemLength)
12288 break;
12289
12290 KeyframeMotion.Keyframe frame = new KeyframeMotion.Keyframe();
12291 frame.Position = null;
12292 frame.Rotation = null;
12293
12294 if ((data & KeyframeMotion.DataFormat.Translation) != 0)
12295 {
12296 LSL_Types.Vector3 tempv = frames.GetVector3Item(idx++);
12297 frame.Position = new Vector3((float)tempv.x, (float)tempv.y, (float)tempv.z);
12298 }
12299 if ((data & KeyframeMotion.DataFormat.Rotation) != 0)
12300 {
12301 LSL_Types.Quaternion tempq = frames.GetQuaternionItem(idx++);
12302 frame.Rotation = new Quaternion((float)tempq.x, (float)tempq.y, (float)tempq.z, (float)tempq.s);
12303 }
12304
12305 float tempf = (float)frames.GetLSLFloatItem(idx++);
12306 frame.TimeMS = (int)(tempf * 1000.0f);
12307
12308 keyframes.Add(frame);
12309 }
12310
12311 group.RootPart.KeyframeMotion.SetKeyframes(keyframes.ToArray());
12312 group.RootPart.KeyframeMotion.Start();
12313 }
12314 else
12315 {
12316 if (group.RootPart.KeyframeMotion == null)
12317 return;
12318
12319 if (options.Data.Length == 0)
12320 {
12321 group.RootPart.KeyframeMotion.Stop();
12322 return;
12323 }
12324
12325 int code = (int)options.GetLSLIntegerItem(0);
12326
12327 int idx = 0;
12328
12329 while (idx < options.Data.Length)
12330 {
12331 int option = (int)options.GetLSLIntegerItem(idx++);
12332 int remain = options.Data.Length - idx;
12333
12334 switch (option)
12335 {
12336 case ScriptBaseClass.KFM_COMMAND:
12337 int cmd = (int)options.GetLSLIntegerItem(idx++);
12338 switch (cmd)
12339 {
12340 case ScriptBaseClass.KFM_CMD_PLAY:
12341 group.RootPart.KeyframeMotion.Start();
12342 break;
12343 case ScriptBaseClass.KFM_CMD_STOP:
12344 group.RootPart.KeyframeMotion.Stop();
12345 break;
12346 case ScriptBaseClass.KFM_CMD_PAUSE:
12347 group.RootPart.KeyframeMotion.Pause();
12348 break;
12349 }
12350 break;
12351 }
12352 }
12353 }
12354 }
10977 } 12355 }
10978 12356
10979 public class NotecardCache 12357 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 7ea8b7a..8237b60 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/OSSL_Api.cs
@@ -138,6 +138,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
138 internal ThreatLevel m_MaxThreatLevel = ThreatLevel.VeryLow; 138 internal ThreatLevel m_MaxThreatLevel = ThreatLevel.VeryLow;
139 internal float m_ScriptDelayFactor = 1.0f; 139 internal float m_ScriptDelayFactor = 1.0f;
140 internal float m_ScriptDistanceFactor = 1.0f; 140 internal float m_ScriptDistanceFactor = 1.0f;
141 internal bool m_debuggerSafe = false;
141 internal Dictionary<string, FunctionPerms > m_FunctionPerms = new Dictionary<string, FunctionPerms >(); 142 internal Dictionary<string, FunctionPerms > m_FunctionPerms = new Dictionary<string, FunctionPerms >();
142 143
143 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, TaskInventoryItem item) 144 public void Initialize(IScriptEngine ScriptEngine, SceneObjectPart host, TaskInventoryItem item)
@@ -145,6 +146,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
145 m_ScriptEngine = ScriptEngine; 146 m_ScriptEngine = ScriptEngine;
146 m_host = host; 147 m_host = host;
147 m_item = item; 148 m_item = item;
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 /// <summary> 221 /// <summary>
@@ -920,18 +929,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
920 if (target != null) 929 if (target != null)
921 { 930 {
922 UUID animID=UUID.Zero; 931 UUID animID=UUID.Zero;
923 lock (m_host.TaskInventory) 932 m_host.TaskInventory.LockItemsForRead(true);
933 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory)
924 { 934 {
925 foreach (KeyValuePair<UUID, TaskInventoryItem> inv in m_host.TaskInventory) 935 if (inv.Value.Name == animation)
926 { 936 {
927 if (inv.Value.Name == animation) 937 if (inv.Value.Type == (int)AssetType.Animation)
928 { 938 animID = inv.Value.AssetID;
929 if (inv.Value.Type == (int)AssetType.Animation) 939 continue;
930 animID = inv.Value.AssetID;
931 continue;
932 }
933 } 940 }
934 } 941 }
942 m_host.TaskInventory.LockItemsForRead(false);
935 if (animID == UUID.Zero) 943 if (animID == UUID.Zero)
936 target.Animator.AddAnimation(animation, m_host.UUID); 944 target.Animator.AddAnimation(animation, m_host.UUID);
937 else 945 else
@@ -972,6 +980,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
972 else 980 else
973 animID = UUID.Zero; 981 animID = UUID.Zero;
974 } 982 }
983 m_host.TaskInventory.LockItemsForRead(false);
975 984
976 if (animID == UUID.Zero) 985 if (animID == UUID.Zero)
977 target.Animator.RemoveAnimation(animation); 986 target.Animator.RemoveAnimation(animation);
@@ -1792,6 +1801,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1792 1801
1793 if (!UUID.TryParse(notecardNameOrUuid, out assetID)) 1802 if (!UUID.TryParse(notecardNameOrUuid, out assetID))
1794 { 1803 {
1804 m_host.TaskInventory.LockItemsForRead(true);
1795 foreach (TaskInventoryItem item in m_host.TaskInventory.Values) 1805 foreach (TaskInventoryItem item in m_host.TaskInventory.Values)
1796 { 1806 {
1797 if (item.Type == 7 && item.Name == notecardNameOrUuid) 1807 if (item.Type == 7 && item.Name == notecardNameOrUuid)
@@ -1799,6 +1809,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
1799 assetID = item.AssetID; 1809 assetID = item.AssetID;
1800 } 1810 }
1801 } 1811 }
1812 m_host.TaskInventory.LockItemsForRead(false);
1802 } 1813 }
1803 1814
1804 if (assetID == UUID.Zero) 1815 if (assetID == UUID.Zero)
@@ -2266,7 +2277,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2266 CheckThreatLevel(ThreatLevel.High, "osNpcCreate"); 2277 CheckThreatLevel(ThreatLevel.High, "osNpcCreate");
2267 m_host.AddScriptLPS(1); 2278 m_host.AddScriptLPS(1);
2268 2279
2269 return NpcCreate(firstname, lastname, position, notecard, false, false); 2280 return NpcCreate(firstname, lastname, position, notecard, true, false);
2270 } 2281 }
2271 2282
2272 public LSL_Key osNpcCreate(string firstname, string lastname, LSL_Vector position, string notecard, int options) 2283 public LSL_Key osNpcCreate(string firstname, string lastname, LSL_Vector position, string notecard, int options)
@@ -2277,24 +2288,39 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2277 return NpcCreate( 2288 return NpcCreate(
2278 firstname, lastname, position, notecard, 2289 firstname, lastname, position, notecard,
2279 (options & ScriptBaseClass.OS_NPC_NOT_OWNED) == 0, 2290 (options & ScriptBaseClass.OS_NPC_NOT_OWNED) == 0,
2280 (options & ScriptBaseClass.OS_NPC_SENSE_AS_AGENT) != 0); 2291 false);
2292// (options & ScriptBaseClass.OS_NPC_SENSE_AS_AGENT) != 0);
2281 } 2293 }
2282 2294
2283 private LSL_Key NpcCreate( 2295 private LSL_Key NpcCreate(
2284 string firstname, string lastname, LSL_Vector position, string notecard, bool owned, bool senseAsAgent) 2296 string firstname, string lastname, LSL_Vector position, string notecard, bool owned, bool senseAsAgent)
2285 { 2297 {
2298 if (!owned)
2299 OSSLError("Unowned NPCs are unsupported");
2300
2301 string groupTitle = String.Empty;
2302
2303 if (!World.Permissions.CanRezObject(1, m_host.OwnerID, new Vector3((float)position.x, (float)position.y, (float)position.z)))
2304 return new LSL_Key(UUID.Zero.ToString());
2305
2306 if (firstname != String.Empty || lastname != String.Empty)
2307 {
2308 if (firstname != "Shown outfit:")
2309 groupTitle = "- NPC -";
2310 }
2311
2286 INPCModule module = World.RequestModuleInterface<INPCModule>(); 2312 INPCModule module = World.RequestModuleInterface<INPCModule>();
2287 if (module != null) 2313 if (module != null)
2288 { 2314 {
2289 AvatarAppearance appearance = null; 2315 AvatarAppearance appearance = null;
2290 2316
2291 UUID id; 2317// UUID id;
2292 if (UUID.TryParse(notecard, out id)) 2318// if (UUID.TryParse(notecard, out id))
2293 { 2319// {
2294 ScenePresence clonePresence = World.GetScenePresence(id); 2320// ScenePresence clonePresence = World.GetScenePresence(id);
2295 if (clonePresence != null) 2321// if (clonePresence != null)
2296 appearance = clonePresence.Appearance; 2322// appearance = clonePresence.Appearance;
2297 } 2323// }
2298 2324
2299 if (appearance == null) 2325 if (appearance == null)
2300 { 2326 {
@@ -2322,6 +2348,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2322 World, 2348 World,
2323 appearance); 2349 appearance);
2324 2350
2351 ScenePresence sp;
2352 if (World.TryGetScenePresence(x, out sp))
2353 {
2354 sp.Grouptitle = groupTitle;
2355 sp.SendAvatarDataToAllAgents();
2356 }
2325 return new LSL_Key(x.ToString()); 2357 return new LSL_Key(x.ToString());
2326 } 2358 }
2327 2359
@@ -2613,16 +2645,32 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
2613 CheckThreatLevel(ThreatLevel.High, "osNpcRemove"); 2645 CheckThreatLevel(ThreatLevel.High, "osNpcRemove");
2614 m_host.AddScriptLPS(1); 2646 m_host.AddScriptLPS(1);
2615 2647
2616 INPCModule module = World.RequestModuleInterface<INPCModule>(); 2648 ManualResetEvent ev = new ManualResetEvent(false);
2617 if (module != null)
2618 {
2619 UUID npcId = new UUID(npc.m_string);
2620 2649
2621 if (!module.CheckPermissions(npcId, m_host.OwnerID)) 2650 Util.FireAndForget(delegate(object x) {
2622 return; 2651 try
2652 {
2653 INPCModule module = World.RequestModuleInterface<INPCModule>();
2654 if (module != null)
2655 {
2656 UUID npcId = new UUID(npc.m_string);
2623 2657
2624 module.DeleteNPC(npcId, World); 2658 ILandObject l = World.LandChannel.GetLandObject(m_host.GroupPosition.X, m_host.GroupPosition.Y);
2625 } 2659 if (l == null || m_host.OwnerID != l.LandData.OwnerID)
2660 {
2661 if (!module.CheckPermissions(npcId, m_host.OwnerID))
2662 return;
2663 }
2664
2665 module.DeleteNPC(npcId, World);
2666 }
2667 }
2668 finally
2669 {
2670 ev.Set();
2671 }
2672 });
2673 ev.WaitOne();
2626 } 2674 }
2627 2675
2628 public void osNpcPlayAnimation(LSL_Key npc, string animation) 2676 public void osNpcPlayAnimation(LSL_Key npc, string animation)
@@ -3156,4 +3204,4 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api
3156 ((LSL_Api)m_LSL_Api).DetachFromAvatar(); 3204 ((LSL_Api)m_LSL_Api).DetachFromAvatar();
3157 } 3205 }
3158 } 3206 }
3159} \ No newline at end of file 3207}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs
index 3844753..19f3ce1 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Implementation/Plugins/SensorRepeat.cs
@@ -319,7 +319,8 @@ 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.GetWorldRotation(); 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, rotate the sensor cone with the 326 // In attachments, rotate the sensor cone with the
@@ -333,7 +334,8 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Plugins
333 // Position of a sensor in a child prim attached to an avatar 334 // Position of a sensor in a child prim attached to an avatar
334 // will be still wrong. 335 // will be still wrong.
335 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar); 336 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar);
336 q = avatar.Rotation * q; 337 fromRegionPos = avatar.AbsolutePosition;
338 q = avatar.Rotation;
337 } 339 }
338 340
339 LSL_Types.Quaternion r = new LSL_Types.Quaternion(q.X, q.Y, q.Z, q.W); 341 LSL_Types.Quaternion r = new LSL_Types.Quaternion(q.X, q.Y, q.Z, q.W);
@@ -463,7 +465,10 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Plugins
463 // Position of a sensor in a child prim attached to an avatar 465 // Position of a sensor in a child prim attached to an avatar
464 // will be still wrong. 466 // will be still wrong.
465 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar); 467 ScenePresence avatar = m_CmdManager.m_ScriptEngine.World.GetScenePresence(SensePoint.ParentGroup.AttachedAvatar);
466 q = avatar.Rotation * q; 468 if (avatar == null)
469 return sensedEntities;
470 fromRegionPos = avatar.AbsolutePosition;
471 q = avatar.Rotation;
467 } 472 }
468 473
469 LSL_Types.Quaternion r = new LSL_Types.Quaternion(q.X, q.Y, q.Z, q.W); 474 LSL_Types.Quaternion r = new LSL_Types.Quaternion(q.X, q.Y, q.Z, q.W);
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 }
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ICM_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ICM_Api.cs
new file mode 100644
index 0000000..ab215f3
--- /dev/null
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ICM_Api.cs
@@ -0,0 +1,46 @@
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.Collections;
29using OpenSim.Region.ScriptEngine.Interfaces;
30
31using key = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
32using rotation = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Quaternion;
33using vector = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Vector3;
34using LSL_List = OpenSim.Region.ScriptEngine.Shared.LSL_Types.list;
35using LSL_String = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
36using LSL_Integer = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLInteger;
37using LSL_Float = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLFloat;
38
39namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
40{
41 public interface ICM_Api
42 {
43 string cmDetectedCountry(int num);
44 string cmGetAgentCountry(key key);
45 }
46}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ILSL_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ILSL_Api.cs
index 7f5d1fe..048124d 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ILSL_Api.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Interface/ILSL_Api.cs
@@ -126,6 +126,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
126 LSL_Float llGetEnergy(); 126 LSL_Float llGetEnergy();
127 LSL_Vector llGetForce(); 127 LSL_Vector llGetForce();
128 LSL_Integer llGetFreeMemory(); 128 LSL_Integer llGetFreeMemory();
129 LSL_Integer llGetUsedMemory();
129 LSL_Integer llGetFreeURLs(); 130 LSL_Integer llGetFreeURLs();
130 LSL_Vector llGetGeometricCenter(); 131 LSL_Vector llGetGeometricCenter();
131 LSL_Float llGetGMTclock(); 132 LSL_Float llGetGMTclock();
@@ -149,6 +150,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
149 LSL_Vector llGetLocalPos(); 150 LSL_Vector llGetLocalPos();
150 LSL_Rotation llGetLocalRot(); 151 LSL_Rotation llGetLocalRot();
151 LSL_Float llGetMass(); 152 LSL_Float llGetMass();
153 LSL_Float llGetMassMKS();
152 LSL_Integer llGetMemoryLimit(); 154 LSL_Integer llGetMemoryLimit();
153 void llGetNextEmail(string address, string subject); 155 void llGetNextEmail(string address, string subject);
154 LSL_String llGetNotecardLine(string name, int line); 156 LSL_String llGetNotecardLine(string name, int line);
@@ -202,12 +204,12 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
202 LSL_String llGetTimestamp(); 204 LSL_String llGetTimestamp();
203 LSL_Vector llGetTorque(); 205 LSL_Vector llGetTorque();
204 LSL_Integer llGetUnixTime(); 206 LSL_Integer llGetUnixTime();
205 LSL_Integer llGetUsedMemory();
206 LSL_Vector llGetVel(); 207 LSL_Vector llGetVel();
207 LSL_Float llGetWallclock(); 208 LSL_Float llGetWallclock();
208 void llGiveInventory(string destination, string inventory); 209 void llGiveInventory(string destination, string inventory);
209 void llGiveInventoryList(string destination, string category, LSL_List inventory); 210 void llGiveInventoryList(string destination, string category, LSL_List inventory);
210 LSL_Integer llGiveMoney(string destination, int amount); 211 LSL_Integer llGiveMoney(string destination, int amount);
212 LSL_String llTransferLindenDollars(string destination, int amount);
211 void llGodLikeRezObject(string inventory, LSL_Vector pos); 213 void llGodLikeRezObject(string inventory, LSL_Vector pos);
212 LSL_Float llGround(LSL_Vector offset); 214 LSL_Float llGround(LSL_Vector offset);
213 LSL_Vector llGroundContour(LSL_Vector offset); 215 LSL_Vector llGroundContour(LSL_Vector offset);
@@ -355,6 +357,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
355 void llSetParcelMusicURL(string url); 357 void llSetParcelMusicURL(string url);
356 void llSetPayPrice(int price, LSL_List quick_pay_buttons); 358 void llSetPayPrice(int price, LSL_List quick_pay_buttons);
357 void llSetPos(LSL_Vector pos); 359 void llSetPos(LSL_Vector pos);
360 LSL_Integer llSetRegionPos(LSL_Vector pos);
358 LSL_Integer llSetPrimMediaParams(LSL_Integer face, LSL_List rules); 361 LSL_Integer llSetPrimMediaParams(LSL_Integer face, LSL_List rules);
359 void llSetPrimitiveParams(LSL_List rules); 362 void llSetPrimitiveParams(LSL_List rules);
360 void llSetLinkPrimitiveParamsFast(int linknum, LSL_List rules); 363 void llSetLinkPrimitiveParamsFast(int linknum, LSL_List rules);
@@ -403,6 +406,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
403 void llTargetOmega(LSL_Vector axis, double spinrate, double gain); 406 void llTargetOmega(LSL_Vector axis, double spinrate, double gain);
404 void llTargetRemove(int number); 407 void llTargetRemove(int number);
405 void llTeleportAgentHome(string agent); 408 void llTeleportAgentHome(string agent);
409 void llTeleportAgent(string agent, string simname, LSL_Vector pos, LSL_Vector lookAt);
406 void llTextBox(string avatar, string message, int chat_channel); 410 void llTextBox(string avatar, string message, int chat_channel);
407 LSL_String llToLower(string source); 411 LSL_String llToLower(string source);
408 LSL_String llToUpper(string source); 412 LSL_String llToUpper(string source);
@@ -419,9 +423,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
419 LSL_Vector llWind(LSL_Vector offset); 423 LSL_Vector llWind(LSL_Vector offset);
420 LSL_String llXorBase64Strings(string str1, string str2); 424 LSL_String llXorBase64Strings(string str1, string str2);
421 LSL_String llXorBase64StringsCorrect(string str1, string str2); 425 LSL_String llXorBase64StringsCorrect(string str1, string str2);
422 void print(string str); 426 LSL_Integer llGetLinkNumberOfSides(LSL_Integer link);
427 void llSetPhysicsMaterial(int material_bits, float material_gravity_modifier, float material_restitution, float material_friction, float material_density);
423 428
424 void SetPrimitiveParamsEx(LSL_Key prim, LSL_List rules); 429 void SetPrimitiveParamsEx(LSL_Key prim, LSL_List rules);
425 LSL_List GetLinkPrimitiveParamsEx(LSL_Key prim, LSL_List rules); 430 LSL_List GetLinkPrimitiveParamsEx(LSL_Key prim, LSL_List rules);
431 void llSetKeyframedMotion(LSL_List frames, LSL_List options);
426 } 432 }
427} 433}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Interface/IOSSL_Api.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Interface/IOSSL_Api.cs
index e92518d..7382495 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Interface/IOSSL_Api.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Interface/IOSSL_Api.cs
@@ -85,7 +85,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.Api.Interfaces
85 // Avatar Info Commands 85 // Avatar Info Commands
86 string osGetAgentIP(string agent); 86 string osGetAgentIP(string agent);
87 LSL_List osGetAgents(); 87 LSL_List osGetAgents();
88 88
89 // Teleport commands 89 // Teleport commands
90 void osTeleportAgent(string agent, string regionName, LSL_Types.Vector3 position, LSL_Types.Vector3 lookat); 90 void osTeleportAgent(string agent, string regionName, LSL_Types.Vector3 position, LSL_Types.Vector3 lookat);
91 void osTeleportAgent(string agent, int regionX, int regionY, LSL_Types.Vector3 position, LSL_Types.Vector3 lookat); 91 void osTeleportAgent(string agent, int regionX, int regionY, LSL_Types.Vector3 position, LSL_Types.Vector3 lookat);
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/CM_Stub.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/CM_Stub.cs
new file mode 100644
index 0000000..4132dfa
--- /dev/null
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/CM_Stub.cs
@@ -0,0 +1,71 @@
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.Runtime.Remoting.Lifetime;
30using System.Threading;
31using System.Reflection;
32using System.Collections;
33using System.Collections.Generic;
34using OpenSim.Framework;
35using OpenSim.Region.Framework.Interfaces;
36using OpenSim.Region.ScriptEngine.Interfaces;
37using OpenSim.Region.ScriptEngine.Shared.Api.Interfaces;
38using integer = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLInteger;
39using vector = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Vector3;
40using rotation = OpenSim.Region.ScriptEngine.Shared.LSL_Types.Quaternion;
41using key = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
42using LSL_List = OpenSim.Region.ScriptEngine.Shared.LSL_Types.list;
43using LSL_String = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLString;
44using LSL_Float = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLFloat;
45using LSL_Integer = OpenSim.Region.ScriptEngine.Shared.LSL_Types.LSLInteger;
46
47namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
48{
49 public partial class ScriptBaseClass : MarshalByRefObject
50 {
51 public ICM_Api m_CM_Functions;
52
53 public void ApiTypeCM(IScriptApi api)
54 {
55 if (!(api is ICM_Api))
56 return;
57
58 m_CM_Functions = (ICM_Api)api;
59 }
60
61 public string cmDetectedCountry(int num)
62 {
63 return m_CM_Functions.cmDetectedCountry(num);
64 }
65
66 public string cmGetAgentCountry(key key)
67 {
68 return m_CM_Functions.cmGetAgentCountry(key);
69 }
70 }
71}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/Executor.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/Executor.cs
index 9615315..943d7a2 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/Executor.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/Executor.cs
@@ -27,6 +27,7 @@
27 27
28using System; 28using System;
29using System.Collections.Generic; 29using System.Collections.Generic;
30using System.Diagnostics; //for [DebuggerNonUserCode]
30using System.Reflection; 31using System.Reflection;
31using System.Runtime.Remoting.Lifetime; 32using System.Runtime.Remoting.Lifetime;
32using OpenSim.Region.ScriptEngine.Shared; 33using OpenSim.Region.ScriptEngine.Shared;
@@ -132,6 +133,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
132 return (eventFlags); 133 return (eventFlags);
133 } 134 }
134 135
136 [DebuggerNonUserCode]
135 public void ExecuteEvent(string state, string FunctionName, object[] args) 137 public void ExecuteEvent(string state, string FunctionName, object[] args)
136 { 138 {
137 // IMPORTANT: Types and MemberInfo-derived objects require a LOT of memory. 139 // IMPORTANT: Types and MemberInfo-derived objects require a LOT of memory.
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Constants.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Constants.cs
index b6c21e6..5c6ad8a 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Constants.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Constants.cs
@@ -94,6 +94,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
94 public const int AGENT_CROUCHING = 1024; 94 public const int AGENT_CROUCHING = 1024;
95 public const int AGENT_BUSY = 2048; 95 public const int AGENT_BUSY = 2048;
96 public const int AGENT_ALWAYS_RUN = 4096; 96 public const int AGENT_ALWAYS_RUN = 4096;
97 public const int AGENT_MALE = 8192;
97 98
98 //Particle Systems 99 //Particle Systems
99 public const int PSYS_PART_INTERP_COLOR_MASK = 1; 100 public const int PSYS_PART_INTERP_COLOR_MASK = 1;
@@ -282,6 +283,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
282 public const int CHANGED_REGION_START = 1024; //LL Changed the constant from CHANGED_REGION_RESTART 283 public const int CHANGED_REGION_START = 1024; //LL Changed the constant from CHANGED_REGION_RESTART
283 public const int CHANGED_MEDIA = 2048; 284 public const int CHANGED_MEDIA = 2048;
284 public const int CHANGED_ANIMATION = 16384; 285 public const int CHANGED_ANIMATION = 16384;
286 public const int CHANGED_POSITION = 32768;
285 public const int TYPE_INVALID = 0; 287 public const int TYPE_INVALID = 0;
286 public const int TYPE_INTEGER = 1; 288 public const int TYPE_INTEGER = 1;
287 public const int TYPE_FLOAT = 2; 289 public const int TYPE_FLOAT = 2;
@@ -586,6 +588,17 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
586 public const int PRIM_MEDIA_PERM_OWNER = 1; 588 public const int PRIM_MEDIA_PERM_OWNER = 1;
587 public const int PRIM_MEDIA_PERM_GROUP = 2; 589 public const int PRIM_MEDIA_PERM_GROUP = 2;
588 public const int PRIM_MEDIA_PERM_ANYONE = 4; 590 public const int PRIM_MEDIA_PERM_ANYONE = 4;
591
592 public const int PRIM_PHYSICS_SHAPE_TYPE = 30;
593 public const int PRIM_PHYSICS_SHAPE_PRIM = 0;
594 public const int PRIM_PHYSICS_SHAPE_CONVEX = 2;
595 public const int PRIM_PHYSICS_SHAPE_NONE = 1;
596
597 public const int PRIM_PHYSICS_MATERIAL = 31;
598 public const int DENSITY = 1;
599 public const int FRICTION = 2;
600 public const int RESTITUTION = 4;
601 public const int GRAVITY_MULTIPLIER = 8;
589 602
590 // extra constants for llSetPrimMediaParams 603 // extra constants for llSetPrimMediaParams
591 public static readonly LSLInteger LSL_STATUS_OK = new LSLInteger(0); 604 public static readonly LSLInteger LSL_STATUS_OK = new LSLInteger(0);
@@ -659,5 +672,18 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
659 public static readonly LSLInteger RCERR_UNKNOWN = -1; 672 public static readonly LSLInteger RCERR_UNKNOWN = -1;
660 public static readonly LSLInteger RCERR_SIM_PERF_LOW = -2; 673 public static readonly LSLInteger RCERR_SIM_PERF_LOW = -2;
661 public static readonly LSLInteger RCERR_CAST_TIME_EXCEEDED = 3; 674 public static readonly LSLInteger RCERR_CAST_TIME_EXCEEDED = 3;
675
676 public const int KFM_MODE = 1;
677 public const int KFM_LOOP = 1;
678 public const int KFM_REVERSE = 3;
679 public const int KFM_FORWARD = 0;
680 public const int KFM_PING_PONG = 2;
681 public const int KFM_DATA = 2;
682 public const int KFM_TRANSLATION = 2;
683 public const int KFM_ROTATION = 1;
684 public const int KFM_COMMAND = 0;
685 public const int KFM_CMD_PLAY = 0;
686 public const int KFM_CMD_STOP = 1;
687 public const int KFM_CMD_PAUSE = 2;
662 } 688 }
663} 689}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Stub.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Stub.cs
index c0bf29c..2d23d30 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Stub.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LSL_Stub.cs
@@ -26,6 +26,7 @@
26 */ 26 */
27 27
28using System; 28using System;
29using System.Diagnostics; //for [DebuggerNonUserCode]
29using System.Runtime.Remoting.Lifetime; 30using System.Runtime.Remoting.Lifetime;
30using System.Threading; 31using System.Threading;
31using System.Reflection; 32using System.Reflection;
@@ -164,6 +165,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
164 m_LSL_Functions.llBreakLink(linknum); 165 m_LSL_Functions.llBreakLink(linknum);
165 } 166 }
166 167
168 public LSL_List llCastRay(LSL_Vector start, LSL_Vector end, LSL_List options)
169 {
170 return m_LSL_Functions.llCastRay(start, end, options);
171 }
172
167 public LSL_Integer llCeil(double f) 173 public LSL_Integer llCeil(double f)
168 { 174 {
169 return m_LSL_Functions.llCeil(f); 175 return m_LSL_Functions.llCeil(f);
@@ -314,6 +320,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
314 m_LSL_Functions.llDialog(avatar, message, buttons, chat_channel); 320 m_LSL_Functions.llDialog(avatar, message, buttons, chat_channel);
315 } 321 }
316 322
323 [DebuggerNonUserCode]
317 public void llDie() 324 public void llDie()
318 { 325 {
319 m_LSL_Functions.llDie(); 326 m_LSL_Functions.llDie();
@@ -474,6 +481,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
474 return m_LSL_Functions.llGetFreeMemory(); 481 return m_LSL_Functions.llGetFreeMemory();
475 } 482 }
476 483
484 public LSL_Integer llGetUsedMemory()
485 {
486 return m_LSL_Functions.llGetUsedMemory();
487 }
488
477 public LSL_Integer llGetFreeURLs() 489 public LSL_Integer llGetFreeURLs()
478 { 490 {
479 return m_LSL_Functions.llGetFreeURLs(); 491 return m_LSL_Functions.llGetFreeURLs();
@@ -579,6 +591,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
579 return m_LSL_Functions.llGetMass(); 591 return m_LSL_Functions.llGetMass();
580 } 592 }
581 593
594 public LSL_Float llGetMassMKS()
595 {
596 return m_LSL_Functions.llGetMassMKS();
597 }
598
582 public LSL_Integer llGetMemoryLimit() 599 public LSL_Integer llGetMemoryLimit()
583 { 600 {
584 return m_LSL_Functions.llGetMemoryLimit(); 601 return m_LSL_Functions.llGetMemoryLimit();
@@ -844,11 +861,6 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
844 return m_LSL_Functions.llGetUnixTime(); 861 return m_LSL_Functions.llGetUnixTime();
845 } 862 }
846 863
847 public LSL_Integer llGetUsedMemory()
848 {
849 return m_LSL_Functions.llGetUsedMemory();
850 }
851
852 public LSL_Vector llGetVel() 864 public LSL_Vector llGetVel()
853 { 865 {
854 return m_LSL_Functions.llGetVel(); 866 return m_LSL_Functions.llGetVel();
@@ -874,6 +886,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
874 return m_LSL_Functions.llGiveMoney(destination, amount); 886 return m_LSL_Functions.llGiveMoney(destination, amount);
875 } 887 }
876 888
889 public LSL_String llTransferLindenDollars(string destination, int amount)
890 {
891 return m_LSL_Functions.llTransferLindenDollars(destination, amount);
892 }
893
877 public void llGodLikeRezObject(string inventory, LSL_Vector pos) 894 public void llGodLikeRezObject(string inventory, LSL_Vector pos)
878 { 895 {
879 m_LSL_Functions.llGodLikeRezObject(inventory, pos); 896 m_LSL_Functions.llGodLikeRezObject(inventory, pos);
@@ -1598,6 +1615,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
1598 m_LSL_Functions.llSetPos(pos); 1615 m_LSL_Functions.llSetPos(pos);
1599 } 1616 }
1600 1617
1618 public LSL_Integer llSetRegionPos(LSL_Vector pos)
1619 {
1620 return m_LSL_Functions.llSetRegionPos(pos);
1621 }
1622
1601 public void llSetPrimitiveParams(LSL_List rules) 1623 public void llSetPrimitiveParams(LSL_List rules)
1602 { 1624 {
1603 m_LSL_Functions.llSetPrimitiveParams(rules); 1625 m_LSL_Functions.llSetPrimitiveParams(rules);
@@ -1833,6 +1855,11 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
1833 m_LSL_Functions.llTargetRemove(number); 1855 m_LSL_Functions.llTargetRemove(number);
1834 } 1856 }
1835 1857
1858 public void llTeleportAgent(string agent, string simname, LSL_Vector pos, LSL_Vector lookAt)
1859 {
1860 m_LSL_Functions.llTeleportAgent(agent, simname, pos, lookAt);
1861 }
1862
1836 public void llTeleportAgentHome(string agent) 1863 public void llTeleportAgentHome(string agent)
1837 { 1864 {
1838 m_LSL_Functions.llTeleportAgentHome(agent); 1865 m_LSL_Functions.llTeleportAgentHome(agent);
@@ -1948,9 +1975,19 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
1948 return m_LSL_Functions.llClearLinkMedia(link, face); 1975 return m_LSL_Functions.llClearLinkMedia(link, face);
1949 } 1976 }
1950 1977
1951 public void print(string str) 1978 public LSL_Integer llGetLinkNumberOfSides(LSL_Integer link)
1979 {
1980 return m_LSL_Functions.llGetLinkNumberOfSides(link);
1981 }
1982
1983 public void llSetKeyframedMotion(LSL_List frames, LSL_List options)
1984 {
1985 m_LSL_Functions.llSetKeyframedMotion(frames, options);
1986 }
1987
1988 public void llSetPhysicsMaterial(int material_bits, float material_gravity_modifier, float material_restitution, float material_friction, float material_density)
1952 { 1989 {
1953 m_LSL_Functions.print(str); 1990 m_LSL_Functions.llSetPhysicsMaterial(material_bits, material_gravity_modifier, material_restitution, material_friction, material_density);
1954 } 1991 }
1955 } 1992 }
1956} 1993}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LS_Stub.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LS_Stub.cs
index 143b497..2e27f16 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LS_Stub.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/LS_Stub.cs
@@ -72,9 +72,30 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
72 { 72 {
73 return m_LS_Functions.lsSetWindlightSceneTargeted(rules, target); 73 return m_LS_Functions.lsSetWindlightSceneTargeted(rules, target);
74 } 74 }
75
75 public void lsClearWindlightScene() 76 public void lsClearWindlightScene()
76 { 77 {
77 m_LS_Functions.lsClearWindlightScene(); 78 m_LS_Functions.lsClearWindlightScene();
78 } 79 }
80
81 public LSL_List cmGetWindlightScene(LSL_List rules)
82 {
83 return m_LS_Functions.lsGetWindlightScene(rules);
84 }
85
86 public int cmSetWindlightScene(LSL_List rules)
87 {
88 return m_LS_Functions.lsSetWindlightScene(rules);
89 }
90
91 public int cmSetWindlightSceneTargeted(LSL_List rules, key target)
92 {
93 return m_LS_Functions.lsSetWindlightSceneTargeted(rules, target);
94 }
95
96 public void cmClearWindlightScene()
97 {
98 m_LS_Functions.lsClearWindlightScene();
99 }
79 } 100 }
80} 101}
diff --git a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/ScriptBase.cs b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/ScriptBase.cs
index edbbc2a..b138da3 100644
--- a/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/ScriptBase.cs
+++ b/OpenSim/Region/ScriptEngine/Shared/Api/Runtime/ScriptBase.cs
@@ -33,6 +33,7 @@ using System.Threading;
33using System.Reflection; 33using System.Reflection;
34using System.Collections; 34using System.Collections;
35using System.Collections.Generic; 35using System.Collections.Generic;
36using System.Diagnostics; //for [DebuggerNonUserCode]
36using OpenSim.Region.ScriptEngine.Interfaces; 37using OpenSim.Region.ScriptEngine.Interfaces;
37using OpenSim.Region.ScriptEngine.Shared; 38using OpenSim.Region.ScriptEngine.Shared;
38using OpenSim.Region.ScriptEngine.Shared.Api.Runtime; 39using OpenSim.Region.ScriptEngine.Shared.Api.Runtime;
@@ -90,6 +91,7 @@ namespace OpenSim.Region.ScriptEngine.Shared.ScriptBase
90 return (int)m_Executor.GetStateEventFlags(state); 91 return (int)m_Executor.GetStateEventFlags(state);
91 } 92 }
92 93
94 [DebuggerNonUserCode]
93 public void ExecuteEvent(string state, string FunctionName, object[] args) 95 public void ExecuteEvent(string state, string FunctionName, object[] args)
94 { 96 {
95 m_Executor.ExecuteEvent(state, FunctionName, args); 97 m_Executor.ExecuteEvent(state, FunctionName, args);