diff options
Diffstat (limited to '')
-rw-r--r-- | OpenSim/Region/ScriptEngine/XEngine/XEngine.cs | 353 |
1 files changed, 222 insertions, 131 deletions
diff --git a/OpenSim/Region/ScriptEngine/XEngine/XEngine.cs b/OpenSim/Region/ScriptEngine/XEngine/XEngine.cs index 8629674..6bdd4c8 100644 --- a/OpenSim/Region/ScriptEngine/XEngine/XEngine.cs +++ b/OpenSim/Region/ScriptEngine/XEngine/XEngine.cs | |||
@@ -30,6 +30,7 @@ using System.IO; | |||
30 | using System.Threading; | 30 | using System.Threading; |
31 | using System.Collections; | 31 | using System.Collections; |
32 | using System.Collections.Generic; | 32 | using System.Collections.Generic; |
33 | using System.Diagnostics; //for [DebuggerNonUserCode] | ||
33 | using System.Security; | 34 | using System.Security; |
34 | using System.Security.Policy; | 35 | using System.Security.Policy; |
35 | using System.Reflection; | 36 | using System.Reflection; |
@@ -103,6 +104,8 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
103 | private Dictionary<UUID, IScriptInstance> m_Scripts = | 104 | private Dictionary<UUID, IScriptInstance> m_Scripts = |
104 | new Dictionary<UUID, IScriptInstance>(); | 105 | new Dictionary<UUID, IScriptInstance>(); |
105 | 106 | ||
107 | private OpenMetaverse.ReaderWriterLockSlim m_scriptsLock = new OpenMetaverse.ReaderWriterLockSlim(); | ||
108 | |||
106 | // Maps the asset ID to the assembly | 109 | // Maps the asset ID to the assembly |
107 | 110 | ||
108 | private Dictionary<UUID, string> m_Assemblies = | 111 | private Dictionary<UUID, string> m_Assemblies = |
@@ -125,6 +128,71 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
125 | IWorkItemResult m_CurrentCompile = null; | 128 | IWorkItemResult m_CurrentCompile = null; |
126 | private Dictionary<UUID, int> m_CompileDict = new Dictionary<UUID, int>(); | 129 | private Dictionary<UUID, int> m_CompileDict = new Dictionary<UUID, int>(); |
127 | 130 | ||
131 | private void lockScriptsForRead(bool locked) | ||
132 | { | ||
133 | if (locked) | ||
134 | { | ||
135 | if (m_scriptsLock.RecursiveReadCount > 0) | ||
136 | { | ||
137 | m_log.Error("[XEngine.m_Scripts] Recursive read lock requested. This should not happen and means something needs to be fixed. For now though, it's safe to continue."); | ||
138 | m_scriptsLock.ExitReadLock(); | ||
139 | } | ||
140 | if (m_scriptsLock.RecursiveWriteCount > 0) | ||
141 | { | ||
142 | m_log.Error("[XEngine.m_Scripts] Recursive write lock requested. This should not happen and means something needs to be fixed."); | ||
143 | m_scriptsLock.ExitWriteLock(); | ||
144 | } | ||
145 | |||
146 | while (!m_scriptsLock.TryEnterReadLock(60000)) | ||
147 | { | ||
148 | m_log.Error("[XEngine.m_Scripts] Thread lock detected while trying to aquire READ lock of m_scripts in XEngine. I'm going to try to solve the thread lock automatically to preserve region stability, but this needs to be fixed."); | ||
149 | if (m_scriptsLock.IsWriteLockHeld) | ||
150 | { | ||
151 | m_scriptsLock = new OpenMetaverse.ReaderWriterLockSlim(); | ||
152 | } | ||
153 | } | ||
154 | } | ||
155 | else | ||
156 | { | ||
157 | if (m_scriptsLock.RecursiveReadCount > 0) | ||
158 | { | ||
159 | m_scriptsLock.ExitReadLock(); | ||
160 | } | ||
161 | } | ||
162 | } | ||
163 | private void lockScriptsForWrite(bool locked) | ||
164 | { | ||
165 | if (locked) | ||
166 | { | ||
167 | if (m_scriptsLock.RecursiveReadCount > 0) | ||
168 | { | ||
169 | m_log.Error("[XEngine.m_Scripts] Recursive read lock requested. This should not happen and means something needs to be fixed. For now though, it's safe to continue."); | ||
170 | m_scriptsLock.ExitReadLock(); | ||
171 | } | ||
172 | if (m_scriptsLock.RecursiveWriteCount > 0) | ||
173 | { | ||
174 | m_log.Error("[XEngine.m_Scripts] Recursive write lock requested. This should not happen and means something needs to be fixed."); | ||
175 | m_scriptsLock.ExitWriteLock(); | ||
176 | } | ||
177 | |||
178 | while (!m_scriptsLock.TryEnterWriteLock(60000)) | ||
179 | { | ||
180 | m_log.Error("[XEngine.m_Scripts] Thread lock detected while trying to aquire WRITE lock of m_scripts in XEngine. I'm going to try to solve the thread lock automatically to preserve region stability, but this needs to be fixed."); | ||
181 | if (m_scriptsLock.IsWriteLockHeld) | ||
182 | { | ||
183 | m_scriptsLock = new OpenMetaverse.ReaderWriterLockSlim(); | ||
184 | } | ||
185 | } | ||
186 | } | ||
187 | else | ||
188 | { | ||
189 | if (m_scriptsLock.RecursiveWriteCount > 0) | ||
190 | { | ||
191 | m_scriptsLock.ExitWriteLock(); | ||
192 | } | ||
193 | } | ||
194 | } | ||
195 | |||
128 | public string ScriptEngineName | 196 | public string ScriptEngineName |
129 | { | 197 | { |
130 | get { return "XEngine"; } | 198 | get { return "XEngine"; } |
@@ -270,43 +338,45 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
270 | 338 | ||
271 | public void RemoveRegion(Scene scene) | 339 | public void RemoveRegion(Scene scene) |
272 | { | 340 | { |
273 | lock (m_Scripts) | 341 | lockScriptsForRead(true); |
342 | foreach (IScriptInstance instance in m_Scripts.Values) | ||
274 | { | 343 | { |
275 | foreach (IScriptInstance instance in m_Scripts.Values) | 344 | // Force a final state save |
345 | // | ||
346 | if (m_Assemblies.ContainsKey(instance.AssetID)) | ||
276 | { | 347 | { |
277 | // Force a final state save | 348 | string assembly = m_Assemblies[instance.AssetID]; |
278 | // | 349 | instance.SaveState(assembly); |
279 | if (m_Assemblies.ContainsKey(instance.AssetID)) | 350 | } |
280 | { | ||
281 | string assembly = m_Assemblies[instance.AssetID]; | ||
282 | instance.SaveState(assembly); | ||
283 | } | ||
284 | 351 | ||
285 | // Clear the event queue and abort the instance thread | 352 | // Clear the event queue and abort the instance thread |
286 | // | 353 | // |
287 | instance.ClearQueue(); | 354 | instance.ClearQueue(); |
288 | instance.Stop(0); | 355 | instance.Stop(0); |
289 | 356 | ||
290 | // Release events, timer, etc | 357 | // Release events, timer, etc |
291 | // | 358 | // |
292 | instance.DestroyScriptInstance(); | 359 | instance.DestroyScriptInstance(); |
293 | 360 | ||
294 | // Unload scripts and app domains | 361 | // Unload scripts and app domains |
295 | // Must be done explicitly because they have infinite | 362 | // Must be done explicitly because they have infinite |
296 | // lifetime | 363 | // lifetime |
297 | // | 364 | // |
298 | m_DomainScripts[instance.AppDomain].Remove(instance.ItemID); | 365 | m_DomainScripts[instance.AppDomain].Remove(instance.ItemID); |
299 | if (m_DomainScripts[instance.AppDomain].Count == 0) | 366 | if (m_DomainScripts[instance.AppDomain].Count == 0) |
300 | { | 367 | { |
301 | m_DomainScripts.Remove(instance.AppDomain); | 368 | m_DomainScripts.Remove(instance.AppDomain); |
302 | UnloadAppDomain(instance.AppDomain); | 369 | UnloadAppDomain(instance.AppDomain); |
303 | } | ||
304 | } | 370 | } |
305 | m_Scripts.Clear(); | ||
306 | m_PrimObjects.Clear(); | ||
307 | m_Assemblies.Clear(); | ||
308 | m_DomainScripts.Clear(); | ||
309 | } | 371 | } |
372 | lockScriptsForRead(false); | ||
373 | lockScriptsForWrite(true); | ||
374 | m_Scripts.Clear(); | ||
375 | lockScriptsForWrite(false); | ||
376 | m_PrimObjects.Clear(); | ||
377 | m_Assemblies.Clear(); | ||
378 | m_DomainScripts.Clear(); | ||
379 | |||
310 | lock (m_ScriptEngines) | 380 | lock (m_ScriptEngines) |
311 | { | 381 | { |
312 | m_ScriptEngines.Remove(this); | 382 | m_ScriptEngines.Remove(this); |
@@ -365,22 +435,20 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
365 | 435 | ||
366 | List<IScriptInstance> instances = new List<IScriptInstance>(); | 436 | List<IScriptInstance> instances = new List<IScriptInstance>(); |
367 | 437 | ||
368 | lock (m_Scripts) | 438 | lockScriptsForRead(true); |
369 | { | 439 | foreach (IScriptInstance instance in m_Scripts.Values) |
370 | foreach (IScriptInstance instance in m_Scripts.Values) | ||
371 | instances.Add(instance); | 440 | instances.Add(instance); |
372 | } | 441 | lockScriptsForRead(false); |
373 | 442 | ||
374 | foreach (IScriptInstance i in instances) | 443 | foreach (IScriptInstance i in instances) |
375 | { | 444 | { |
376 | string assembly = String.Empty; | 445 | string assembly = String.Empty; |
377 | 446 | ||
378 | lock (m_Scripts) | 447 | |
379 | { | ||
380 | if (!m_Assemblies.ContainsKey(i.AssetID)) | 448 | if (!m_Assemblies.ContainsKey(i.AssetID)) |
381 | continue; | 449 | continue; |
382 | assembly = m_Assemblies[i.AssetID]; | 450 | assembly = m_Assemblies[i.AssetID]; |
383 | } | 451 | |
384 | 452 | ||
385 | i.SaveState(assembly); | 453 | i.SaveState(assembly); |
386 | } | 454 | } |
@@ -709,92 +777,95 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
709 | } | 777 | } |
710 | 778 | ||
711 | ScriptInstance instance = null; | 779 | ScriptInstance instance = null; |
712 | lock (m_Scripts) | 780 | // Create the object record |
781 | lockScriptsForRead(true); | ||
782 | if ((!m_Scripts.ContainsKey(itemID)) || | ||
783 | (m_Scripts[itemID].AssetID != assetID)) | ||
713 | { | 784 | { |
714 | // Create the object record | 785 | lockScriptsForRead(false); |
715 | 786 | ||
716 | if ((!m_Scripts.ContainsKey(itemID)) || | 787 | UUID appDomain = assetID; |
717 | (m_Scripts[itemID].AssetID != assetID)) | ||
718 | { | ||
719 | UUID appDomain = assetID; | ||
720 | 788 | ||
721 | if (part.ParentGroup.IsAttachment) | 789 | if (part.ParentGroup.IsAttachment) |
722 | appDomain = part.ParentGroup.RootPart.UUID; | 790 | appDomain = part.ParentGroup.RootPart.UUID; |
723 | 791 | ||
724 | if (!m_AppDomains.ContainsKey(appDomain)) | 792 | if (!m_AppDomains.ContainsKey(appDomain)) |
793 | { | ||
794 | try | ||
725 | { | 795 | { |
726 | try | 796 | AppDomainSetup appSetup = new AppDomainSetup(); |
727 | { | 797 | appSetup.PrivateBinPath = Path.Combine( |
728 | AppDomainSetup appSetup = new AppDomainSetup(); | 798 | m_ScriptEnginesPath, |
729 | appSetup.PrivateBinPath = Path.Combine( | 799 | m_Scene.RegionInfo.RegionID.ToString()); |
730 | m_ScriptEnginesPath, | 800 | |
731 | m_Scene.RegionInfo.RegionID.ToString()); | 801 | Evidence baseEvidence = AppDomain.CurrentDomain.Evidence; |
732 | 802 | Evidence evidence = new Evidence(baseEvidence); | |
733 | Evidence baseEvidence = AppDomain.CurrentDomain.Evidence; | 803 | |
734 | Evidence evidence = new Evidence(baseEvidence); | 804 | AppDomain sandbox; |
735 | 805 | if (m_AppDomainLoading) | |
736 | AppDomain sandbox; | 806 | sandbox = AppDomain.CreateDomain( |
737 | if (m_AppDomainLoading) | 807 | m_Scene.RegionInfo.RegionID.ToString(), |
738 | sandbox = AppDomain.CreateDomain( | 808 | evidence, appSetup); |
739 | m_Scene.RegionInfo.RegionID.ToString(), | 809 | else |
740 | evidence, appSetup); | 810 | sandbox = AppDomain.CurrentDomain; |
741 | else | 811 | |
742 | sandbox = AppDomain.CurrentDomain; | 812 | //PolicyLevel sandboxPolicy = PolicyLevel.CreateAppDomainLevel(); |
743 | 813 | //AllMembershipCondition sandboxMembershipCondition = new AllMembershipCondition(); | |
744 | //PolicyLevel sandboxPolicy = PolicyLevel.CreateAppDomainLevel(); | 814 | //PermissionSet sandboxPermissionSet = sandboxPolicy.GetNamedPermissionSet("Internet"); |
745 | //AllMembershipCondition sandboxMembershipCondition = new AllMembershipCondition(); | 815 | //PolicyStatement sandboxPolicyStatement = new PolicyStatement(sandboxPermissionSet); |
746 | //PermissionSet sandboxPermissionSet = sandboxPolicy.GetNamedPermissionSet("Internet"); | 816 | //CodeGroup sandboxCodeGroup = new UnionCodeGroup(sandboxMembershipCondition, sandboxPolicyStatement); |
747 | //PolicyStatement sandboxPolicyStatement = new PolicyStatement(sandboxPermissionSet); | 817 | //sandboxPolicy.RootCodeGroup = sandboxCodeGroup; |
748 | //CodeGroup sandboxCodeGroup = new UnionCodeGroup(sandboxMembershipCondition, sandboxPolicyStatement); | 818 | //sandbox.SetAppDomainPolicy(sandboxPolicy); |
749 | //sandboxPolicy.RootCodeGroup = sandboxCodeGroup; | 819 | |
750 | //sandbox.SetAppDomainPolicy(sandboxPolicy); | 820 | m_AppDomains[appDomain] = sandbox; |
751 | 821 | ||
752 | m_AppDomains[appDomain] = sandbox; | 822 | m_AppDomains[appDomain].AssemblyResolve += |
753 | 823 | new ResolveEventHandler( | |
754 | m_AppDomains[appDomain].AssemblyResolve += | 824 | AssemblyResolver.OnAssemblyResolve); |
755 | new ResolveEventHandler( | 825 | m_DomainScripts[appDomain] = new List<UUID>(); |
756 | AssemblyResolver.OnAssemblyResolve); | 826 | } |
757 | m_DomainScripts[appDomain] = new List<UUID>(); | 827 | catch (Exception e) |
758 | } | 828 | { |
759 | catch (Exception e) | 829 | m_log.ErrorFormat("[XEngine] Exception creating app domain:\n {0}", e.ToString()); |
830 | m_ScriptErrorMessage += "Exception creating app domain:\n"; | ||
831 | m_ScriptFailCount++; | ||
832 | lock (m_AddingAssemblies) | ||
760 | { | 833 | { |
761 | m_log.ErrorFormat("[XEngine] Exception creating app domain:\n {0}", e.ToString()); | 834 | m_AddingAssemblies[assembly]--; |
762 | m_ScriptErrorMessage += "Exception creating app domain:\n"; | ||
763 | m_ScriptFailCount++; | ||
764 | lock (m_AddingAssemblies) | ||
765 | { | ||
766 | m_AddingAssemblies[assembly]--; | ||
767 | } | ||
768 | return false; | ||
769 | } | 835 | } |
836 | return false; | ||
770 | } | 837 | } |
771 | m_DomainScripts[appDomain].Add(itemID); | 838 | } |
772 | 839 | m_DomainScripts[appDomain].Add(itemID); | |
773 | instance = new ScriptInstance(this, part, | 840 | |
774 | itemID, assetID, assembly, | 841 | instance = new ScriptInstance(this, part, |
775 | m_AppDomains[appDomain], | 842 | itemID, assetID, assembly, |
776 | part.ParentGroup.RootPart.Name, | 843 | m_AppDomains[appDomain], |
777 | item.Name, startParam, postOnRez, | 844 | part.ParentGroup.RootPart.Name, |
778 | stateSource, m_MaxScriptQueue); | 845 | item.Name, startParam, postOnRez, |
779 | 846 | stateSource, m_MaxScriptQueue); | |
780 | m_log.DebugFormat( | 847 | |
848 | m_log.DebugFormat( | ||
781 | "[XEngine] Loaded script {0}.{1}, script UUID {2}, prim UUID {3} @ {4}.{5}", | 849 | "[XEngine] Loaded script {0}.{1}, script UUID {2}, prim UUID {3} @ {4}.{5}", |
782 | part.ParentGroup.RootPart.Name, item.Name, assetID, part.UUID, | 850 | part.ParentGroup.RootPart.Name, item.Name, assetID, part.UUID, |
783 | part.ParentGroup.RootPart.AbsolutePosition, part.ParentGroup.Scene.RegionInfo.RegionName); | 851 | part.ParentGroup.RootPart.AbsolutePosition, part.ParentGroup.Scene.RegionInfo.RegionName); |
784 | 852 | ||
785 | if (presence != null) | 853 | if (presence != null) |
786 | { | 854 | { |
787 | ShowScriptSaveResponse(item.OwnerID, | 855 | ShowScriptSaveResponse(item.OwnerID, |
788 | assetID, "Compile successful", true); | 856 | assetID, "Compile successful", true); |
789 | } | ||
790 | |||
791 | instance.AppDomain = appDomain; | ||
792 | instance.LineMap = linemap; | ||
793 | |||
794 | m_Scripts[itemID] = instance; | ||
795 | } | 857 | } |
796 | } | ||
797 | 858 | ||
859 | instance.AppDomain = appDomain; | ||
860 | instance.LineMap = linemap; | ||
861 | lockScriptsForWrite(true); | ||
862 | m_Scripts[itemID] = instance; | ||
863 | lockScriptsForWrite(false); | ||
864 | } | ||
865 | else | ||
866 | { | ||
867 | lockScriptsForRead(false); | ||
868 | } | ||
798 | lock (m_PrimObjects) | 869 | lock (m_PrimObjects) |
799 | { | 870 | { |
800 | if (!m_PrimObjects.ContainsKey(localID)) | 871 | if (!m_PrimObjects.ContainsKey(localID)) |
@@ -813,9 +884,9 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
813 | m_AddingAssemblies[assembly]--; | 884 | m_AddingAssemblies[assembly]--; |
814 | } | 885 | } |
815 | 886 | ||
816 | if (instance != null) | 887 | if (instance!=null) |
817 | instance.Init(); | 888 | instance.Init(); |
818 | 889 | ||
819 | return true; | 890 | return true; |
820 | } | 891 | } |
821 | 892 | ||
@@ -828,20 +899,23 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
828 | m_CompileDict.Remove(itemID); | 899 | m_CompileDict.Remove(itemID); |
829 | } | 900 | } |
830 | 901 | ||
831 | IScriptInstance instance = null; | 902 | lockScriptsForRead(true); |
832 | 903 | // Do we even have it? | |
833 | lock (m_Scripts) | 904 | if (!m_Scripts.ContainsKey(itemID)) |
834 | { | 905 | { |
835 | // Do we even have it? | 906 | lockScriptsForRead(false); |
836 | if (!m_Scripts.ContainsKey(itemID)) | 907 | return; |
837 | return; | ||
838 | |||
839 | instance=m_Scripts[itemID]; | ||
840 | m_Scripts.Remove(itemID); | ||
841 | } | 908 | } |
909 | |||
842 | 910 | ||
911 | IScriptInstance instance=m_Scripts[itemID]; | ||
912 | lockScriptsForRead(false); | ||
913 | lockScriptsForWrite(true); | ||
914 | m_Scripts.Remove(itemID); | ||
915 | lockScriptsForWrite(false); | ||
843 | instance.ClearQueue(); | 916 | instance.ClearQueue(); |
844 | instance.Stop(0); | 917 | instance.Stop(0); |
918 | |||
845 | // bool objectRemoved = false; | 919 | // bool objectRemoved = false; |
846 | 920 | ||
847 | lock (m_PrimObjects) | 921 | lock (m_PrimObjects) |
@@ -877,11 +951,12 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
877 | ObjectRemoved handlerObjectRemoved = OnObjectRemoved; | 951 | ObjectRemoved handlerObjectRemoved = OnObjectRemoved; |
878 | if (handlerObjectRemoved != null) | 952 | if (handlerObjectRemoved != null) |
879 | { | 953 | { |
880 | SceneObjectPart part = m_Scene.GetSceneObjectPart(localID); | 954 | SceneObjectPart part = m_Scene.GetSceneObjectPart(localID); |
881 | handlerObjectRemoved(part.UUID); | 955 | handlerObjectRemoved(part.UUID); |
882 | } | 956 | } |
883 | 957 | ||
884 | 958 | CleanAssemblies(); | |
959 | |||
885 | ScriptRemoved handlerScriptRemoved = OnScriptRemoved; | 960 | ScriptRemoved handlerScriptRemoved = OnScriptRemoved; |
886 | if (handlerScriptRemoved != null) | 961 | if (handlerScriptRemoved != null) |
887 | handlerScriptRemoved(itemID); | 962 | handlerScriptRemoved(itemID); |
@@ -1023,7 +1098,7 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1023 | return false; | 1098 | return false; |
1024 | 1099 | ||
1025 | uuids = m_PrimObjects[localID]; | 1100 | uuids = m_PrimObjects[localID]; |
1026 | } | 1101 | |
1027 | 1102 | ||
1028 | foreach (UUID itemID in uuids) | 1103 | foreach (UUID itemID in uuids) |
1029 | { | 1104 | { |
@@ -1041,6 +1116,7 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1041 | result = true; | 1116 | result = true; |
1042 | } | 1117 | } |
1043 | } | 1118 | } |
1119 | } | ||
1044 | 1120 | ||
1045 | return result; | 1121 | return result; |
1046 | } | 1122 | } |
@@ -1140,12 +1216,14 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1140 | private IScriptInstance GetInstance(UUID itemID) | 1216 | private IScriptInstance GetInstance(UUID itemID) |
1141 | { | 1217 | { |
1142 | IScriptInstance instance; | 1218 | IScriptInstance instance; |
1143 | lock (m_Scripts) | 1219 | lockScriptsForRead(true); |
1220 | if (!m_Scripts.ContainsKey(itemID)) | ||
1144 | { | 1221 | { |
1145 | if (!m_Scripts.ContainsKey(itemID)) | 1222 | lockScriptsForRead(false); |
1146 | return null; | 1223 | return null; |
1147 | instance = m_Scripts[itemID]; | ||
1148 | } | 1224 | } |
1225 | instance = m_Scripts[itemID]; | ||
1226 | lockScriptsForRead(false); | ||
1149 | return instance; | 1227 | return instance; |
1150 | } | 1228 | } |
1151 | 1229 | ||
@@ -1169,6 +1247,7 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1169 | return false; | 1247 | return false; |
1170 | } | 1248 | } |
1171 | 1249 | ||
1250 | [DebuggerNonUserCode] | ||
1172 | public void ApiResetScript(UUID itemID) | 1251 | public void ApiResetScript(UUID itemID) |
1173 | { | 1252 | { |
1174 | IScriptInstance instance = GetInstance(itemID); | 1253 | IScriptInstance instance = GetInstance(itemID); |
@@ -1220,6 +1299,7 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1220 | return UUID.Zero; | 1299 | return UUID.Zero; |
1221 | } | 1300 | } |
1222 | 1301 | ||
1302 | [DebuggerNonUserCode] | ||
1223 | public void SetState(UUID itemID, string newState) | 1303 | public void SetState(UUID itemID, string newState) |
1224 | { | 1304 | { |
1225 | IScriptInstance instance = GetInstance(itemID); | 1305 | IScriptInstance instance = GetInstance(itemID); |
@@ -1240,11 +1320,10 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1240 | { | 1320 | { |
1241 | List<IScriptInstance> instances = new List<IScriptInstance>(); | 1321 | List<IScriptInstance> instances = new List<IScriptInstance>(); |
1242 | 1322 | ||
1243 | lock (m_Scripts) | 1323 | lockScriptsForRead(true); |
1244 | { | 1324 | foreach (IScriptInstance instance in m_Scripts.Values) |
1245 | foreach (IScriptInstance instance in m_Scripts.Values) | ||
1246 | instances.Add(instance); | 1325 | instances.Add(instance); |
1247 | } | 1326 | lockScriptsForRead(false); |
1248 | 1327 | ||
1249 | foreach (IScriptInstance i in instances) | 1328 | foreach (IScriptInstance i in instances) |
1250 | { | 1329 | { |
@@ -1616,5 +1695,17 @@ namespace OpenSim.Region.ScriptEngine.XEngine | |||
1616 | 1695 | ||
1617 | instance.Resume(); | 1696 | instance.Resume(); |
1618 | } | 1697 | } |
1698 | |||
1699 | public bool HasScript(UUID itemID, out bool running) | ||
1700 | { | ||
1701 | running = true; | ||
1702 | |||
1703 | IScriptInstance instance = GetInstance(itemID); | ||
1704 | if (instance == null) | ||
1705 | return false; | ||
1706 | |||
1707 | running = instance.Running; | ||
1708 | return true; | ||
1709 | } | ||
1619 | } | 1710 | } |
1620 | } | 1711 | } |